Pull to refresh

Пишем интерпретатор Brainfuck на Lua

Reading time 5 min
Views 7.6K
Lua Logo
Каждый программист за свою жизнь успевает изучить множество языков, в нескольких из них специализируется и продолжает работать продолжительное время, а остальные проходят мимо. По разным причинам. Стоит ли тратить время на изучение новых языков, когда уже определился с областью в которой будешь работать? Лично я уверен что стоит, хотя, быть может, многие скажут что важны фундаментальные знания в computer science, а на каком языке писать код не критично. В сущности так и есть. И тем не менее изучать языки интересно и полезно.

Lua. Краткая история языка.

Своё начало язык Lua ([луа], порт. «луна») берёт в относительно далёком 1993 году. Его создали Роберто Иерусалимши (Roberto Ierusalimschy), Луис Энрике де Фигуэйредо (Luiz Henrique de Figueiredo) и Вальдемар Селес (Waldemar Celes), в то время члены группы разработки технологии компьютерной графики (Tecgraf) Епископального католического университета Рио-де-Жанейро (Pontifical Catholic University of Rio de Janeiro) в Бразилии. Это скриптовый язык, сочетающий свойства императивных и функциональных языков и обладающий объектно-ориентированными свойствами. Испытал влияние Scheme, SNOBOL, JavaScript, C/C++ и других. В результате получился встраиваемый, легко расширяемый скриптовый язык с простым синтаксисом.
За годы существования Lua обрёл популярность именно как встраиваемый язык: множество программ, но ещё больше игр используют его. Например Vim (с версии 7.3), World of Warcraft, Ragnarok Online и многие другие

Немного о языке

Лучше всего написано тут www.lua.ru/doc (rus) и тут www.lua.org/manual/5.1 (eng)

Устанавливаем Lua

Скачать Lua можно тут luabinaries.sourceforge.net/download.html
Под Linux (правда в репозитории Ubuntu 10.04 есть аж три версии)
sudo apt-get install lua5.1
sudo apt-get install lua50
sudo apt-get install lua40


Либо собрать из исходников (название пакета lua5.1 было совсем не очевидно, поэтому пришлось собрать)
cd /tmp
wget http://www.lua.org/ftp/lua-5.1.4.tar.gz
tar -xf lua-5.1.4.tar.gz
cd lua-5.1.4
sudo apt-get install build-essential libreadline5-dev
make linux test
sudo checkinstall --fstrans=no --install=no --pkgname=lua --pkgversion "5.1.4" --default
sudo dpkg -i lua_5.1.4-1_i386.deb

lua -v
>>> Lua 5.1.4  Copyright (C) 1994-2008 Lua.org, PUC-Rio


Замечательно, теперь можно начинать развлекаться.

Hello World!

С чего начать изучение языка? С Hello world! не интересно. Давайте напишем интерпретатор. Есть такой замечательный язык Brainfuck, очень простой и очень интересный. Помогает размять мозг.

Ставим задачу:
  • Чтение brainfuck кода из файла
  • Базовая валидация кода
  • Исполнение Brainfuck кода


Описание Brainfuck

В «классическом» Brainfuck, описанном Урбаном Мюллером, размер ячейки — один байт, количество ячеек 30 000, ввод/вывод происходит побайтово, количество инструкций 8 шт. ниже краткое описание:

  • ">" перейти к следующей ячейке
  • "<" перейти к предыдущей ячейке
  • "+" увеличить значение в текущей ячейке на 1
  • "-" уменьшить значение в текущей ячейке на 1
  • "." напечатать значение из текущей ячейки
  • "," ввести извне значение и сохранить в текущей ячейке
  • "[" если значение текущей ячейки нуль, перейти вперёд по тексту программы на ячейку, следующую за соответствующей "]" (с учётом вложенности)
  • "]" если значение текущей ячейки не нуль, перейти назад по тексту программы на символ "[" (с учётом вложенности)


Пишем

Начнём с малого: создадим каталог для работы, файл brainfuck.lua в нём, сделаем его исполняемым

#!/usr/bin/env lua
-- Lua Brainfuck Interpreter

Brainfuck =
{
    -- validate source
    -- Return 1 if closing bracket(s) missing.
    -- Return 2 if opening bracket(s) missing.
    -- Return 0 otherwise.
    validate = function (self, source)
        return 0
    end,

    -- debug function
    showError = function (self, errorCode)

    end,

    -- brainfuck function
    brainfuck = function (self, source)

    end,
}


Также создадим файл hello.b (код на brainfuck. выводит Hello World! на экран. нужен для тестов)
++++++++++[>+++++++>++++++++++>+++>+<<<<-]>++.>+.+++++++..+++.>++.<<+++++++++++++++.>.+++.------.--------.>+.>.

Ну, теперь всё готово.

1. Чтение brainfuck кода из файла.

Пусть наш интерпретатор выполняет код из файла имя которого передано в командной строке
./brainfuck.lua hello.b
Документация скажет нам что Lua помещает параметры в массив arg

-- start here
if arg[1] then
    -- read source from file in arg[1]
    source = io.input(arg[1]):read("*a")
    -- get error code (0 == no error)
    errorCode = Brainfuck:validate(source)
    -- if no error run source else show error
    if errorCode == 0 then
        Brainfuck:brainfuck(source)
    else
        Brainfuck:showError(errorCode)
    end
else
    print("Usage: ./brainfuck.lua script")
    Brainfuck:showError(3)
end


2. Базовая валидация кода


-- validate source
-- Return 1 if closing bracket(s) missing.
-- Return 2 if opening bracket(s) missing.
-- Return 0 otherwise.
validate = function (self, source)
    local i, errorCode, l = 0, 0, 0

    for i = 1, string.len(source), 1 do
        -- [ 91
        if string.byte(source, i) == 91 then
            l = l + 1
            -- ] 93
        elseif string.byte(source, i) == 93 then
            l = l - 1
            if l < 0 then return 2 end
        end
    end

    if l > 0 then
        return 1
    elseif l < 0 then
        return 2
    else
        return 0
    end
end,

-- debug function
showError = function (self, errorCode)
    if errorCode == 1 then
        print("Error: Closing bracket(s) missing.")
    elseif errorCode == 2 then
        print("Error: Opening bracket(s) missing.")
    elseif errorCode == 3 then
        print("Error: No source file.")
    else
        print("Error: Unknown error code.")
    end
end,


3. Исполнение Brainfuck кода


-- brainfuck function
brainfuck = function (self, source)
-- memSize: Brainfuck memory size (30k)
-- maxVal: Max memory value (255) byte
-- mem: Memory table (array)
-- pointer: default 0
-- l: default 0. braket level counter
    local memSize, maxVal, mem, pointer, l = 30000, 255, {}, 0, 0

    -- clear memory
    for i = 0, memSize, 1 do mem[i] = 0 end

    -- execute program
    i = 0
    while i <= string.len(source) do
        i = i + 1
        -- + 43 C eqv ++(*p);
        if string.byte(source, i) == 43 then
            if mem[pointer] < maxVal then
                mem[pointer] = mem[pointer] + 1
            end

            -- - 45 C eqv --(*p);
        elseif string.byte(source, i) == 45 then
            if mem[pointer] > 0 then
                mem[pointer] = mem[pointer] - 1
            end

            -- , 44 C eqv *p = getchar();
        elseif string.byte(source, i) == 44 then
            mem[pointer] = string.byte(io.stdin:read('*l'), 1)

            -- . 46 C eqv putchar(*p);
        elseif string.byte(source, i) == 46 then
            io.write(string.char(mem[pointer]))

            -- < 60 C eqv --p;
        elseif string.byte(source, i) == 60 then
            pointer = pointer - 1
            if pointer < 0 then pointer = 0 end

            -- > 62 C eqv ++p;
        elseif string.byte(source, i) == 62 then
            pointer = pointer + 1
            if pointer > memSize then pointer = memSize end

            -- [ 91 C eqv while (*p) {
        elseif string.byte(source, i) == 91 then
            if mem[pointer] == 0 then
                while (string.byte(source, i) ~= 93) or (l > 0) do
                    i = i + 1
                    if string.byte(source, i) == 91 then l = l + 1 end
                    if string.byte(source, i) == 93 then l = l - 1 end
                end
            end

            -- ] 93 C eqv }
        elseif string.byte(source, i) == 93 then
            if mem[pointer] ~= 0 then
                while (string.byte(source, i) ~= 91) or (l > 0) do
                    i = i - 1
                    if string.byte(source, i) == 91 then l = l - 1 end
                    if string.byte(source, i) == 93 then l = l + 1 end
                end
            end
        else
            -- print("Unknown symbol")
            -- return
        end
        -- print("Debug: l="..l.." cmd="..string.char(string.byte(source, i)))
    end

end,


upd: Source code

Готово. Теперь можно запустить пример и убедиться что всё работает. Как видите Lua вполне пригоден для standalone использования)

Прошу прощения за отсутствие отступов. Утеряны при подстветке.
Блога о Lua не нашел, поэтому размещу в Ненормальное программирование.
Если статьи о Lua интересны сообществу могу написать ещё

Полезная литература

Tags:
Hubs:
+29
Comments 15
Comments Comments 15

Articles