Компиляторы, 3 курс, 5 семестр, 2016/17
Лектор: Булычев Дмитрий Юрьевич.
https://github.com/dboulytchev/sample-compiler
Образ с настроенным софтом для виртуальной машины: OVA (логин внутри: me, пароль: me).
Вся информация на этой странице сдублирована из почты!
Содержание
Домашние задания
Домашнее задание от 13.09.2016
- Дописать интерпретатор стековой машины с символическим стеком (там надо поддержать оставшиеся 5 инструкций).
- Реализовать печать инструкций в правильном формате ассемблера x86 (см., например, https://en.wikibooks.org/wiki/X86_Assembly/GAS_Syntax).
- Реализовать правильное "обрамление" порожденной программы:
.text
.globl main
main:
<тут идет текст ассемблерного порождения>
ret
Конструкции управления
Базовые:
-
if E then S1 else S2 fi, гдеE— выражение, аS1иS2— операторы. Если значениеE вычисляется не в ноль, то исполняется <code>S1, иначе —S2. -
while E do S od</code, где <code>E — выражение, <code>S— оператор. Если значениеEвычисляется в не ноль, то исполняетсяS, а потом весьwhileзаново; иначе не делается ничего.
Дополнительные:
-
if E1 then S1 {elif E2 then S2 elif E3 then S3} ... [else Sk] fiэквивалентно:
if E1 then S1
else if E2 then S2
else if E3 then S3
....
[else Sk]
fi
repeat S until EэквивалентноS; while E == 0 do S odfor S1, E, S2 do S odэквивалентноS1; while E do S; S2 od
Поддержка функций
На уровне выражений язык расширяется конструкцией вызова функции: <ident> (<expr>, <expr>, ...). Список аргументов может быть пустым, но скобки нужны в любом случае.
На уровне операторов язык расширяется двумя конструкциями:
-
<ident> (<expr> , <expr>, ... )— вызов функции. Список аргументов может быть пустым, но скобки нужны в любом случае. -
return <expr>— возврат значения. Выполнение функции должно всегда заканчиваться выполнением оператораreturn; если функция вызывается как оператор, то её результат теряется.
Кроме того, теперь перед самой программой может идти одно или несколько описаний функций. Каждое описание имеет следующий вид:
fun <ident> (<ident> , <ident>, ... )
begin
<stmt>
end
Список формальных параметров може быть пуст, но скобки нужны в любом случае.
Правила видимости для переменных внутри функции: каждая функция "видит" свои параметры; все остальные используемые переменные считаются локальными для данного вызова функции. Глобальные переменные не видны. Функции могут быть рекурсивными, в том числе взаимно-рекурсивными.
Расширение системы команд стековой машины:
-
CALL of string * string list— инструкция вызова. Получает имя функции и список имен её параметров. -
RET— возврат из функции. -
END— конец программы (завершает выполнение, даже если "хвост" инструкций не пуст).
Пример:
fun fact (n)
begin
if n < 2
then return 1
else return n * fact (n-1)
fi
end
read (n);
write (fact (n))
Документация
Система сборки (новая)
Используется opam - это пакетный менеджер для OCaml.
Подготовка (альтернативно можно запустить скрипт от Димы):
- Требуется установить opam версии хотя бы 1.2.2. Есть в репозиториях Debian/Ubuntu (
sudo apt-get install opam). - Должен автоматически поставится компилятор ocaml. Проверяем версию:
ocaml -version. - Если версия хотя бы 4.02 - запускаем из домашнего каталога
opam init. Если версия древнее — запускаемopam init --comp 4.03.0. - Чтобы подцепить изменения в окружении в текущей сессии, надо запустить
eval `opam config env`. Теперь версия компилятора точно должна быть хотя бы 4.02. - Выполняем команды:
opam pin add logger https://github.com/dboulytchev/logger.git -y -n opam pin add GT https://github.com/dboulytchev/GT.git -y -n opam pin add ostap https://github.com/dboulytchev/ostap.git -y -n opam pin add ocanren https://github.com/dboulytchev/ocanren.git -y -n opam install ostap opam install ocanren
- Также для сборки
runtime.oна Debian/Ubuntu может потребоваться пакетg++-multilib.
Как собирать:
- Если вы находитесь в корне репозитория:
make -f Makefile.ob. - Если вы находитесь не в корне, а, скажем, в папке src, то надо добавить параметр
-C <путь-до-корня>:make -C .. -f Makefile.ob.