DEADSOFTWARE

Добавлена конструкция REPEAT-UNTIL
authorDeaDDooMER <deaddoomer@deadsoftware.ru>
Thu, 3 Aug 2017 10:58:37 +0000 (13:58 +0300)
committerDeaDDooMER <deaddoomer@deadsoftware.ru>
Thu, 3 Aug 2017 10:58:37 +0000 (13:58 +0300)
notes
src/oberon.c
src/test.c

diff --git a/notes b/notes
index 0675bf305a87efbe7d223f9c0021289588676a9d..7279e3fcdb420b31f3efda7f9bccf2d185766c5c 100644 (file)
--- a/notes
+++ b/notes
@@ -1,16 +1,22 @@
-- нету типа set
-- нету операторов if, while и т.д.
-
 - Нужен тип представляющий типы
     Требуется для оператора IS и некоторых встраиваемых функций
+
+- Нет типа SET
 - Нет оператора IS
-- не реализованы все встроенные функции
-- не реализована свёртка констант
+- Нет конструкции CASE
+- Нет конструкции FOR
+- Нет конструкции LOOP/EXIT
+- Нет конструкции WITH
+- Нет модуля SYSTEM
+- Нет функций ASH CAP CHR ENTIER LEN LONG MAX MIN ODD ORD SHORT SIZE
+- Нет процедур ASSERT COPY DEC EXCL HALT INC INCL
+- Не реализована свёртка констант
+- Не счёта строк / столбцов
 
-- нету счёта строк / столбцов
-- любая ошибка фатальна
+- Нужно пробежаться по стандарту и всё перепроверить.
 
 - Нужны средства создания биндингов. На данный момент реализуемо как заглушки для модулей.
-- нет проверок переполнения в компилтайме.
+- Любая ошибка фатальна
+- Нет проверок переполнения в компилтайме.
     Возможно можно заюзать это:
       https://gcc.gnu.org/onlinedocs/gcc/Integer-Overflow-Builtins.html
index 8a84e1359dcd48f368f41cd945a76bb92a11b599..e8cbf428c12a7261b90dd057c8dc96381f465a3b 100644 (file)
@@ -67,7 +67,9 @@ enum {
        ELSE,
        ELSIF,
        WHILE,
-       DO
+       DO,
+       REPEAT,
+       UNTIL
 };
 
 // =======================================================================
@@ -410,6 +412,14 @@ oberon_read_ident(oberon_context_t * ctx)
        {
                ctx -> token = DO;
        }
+       else if(strcmp(ident, "REPEAT") == 0)
+       {
+               ctx -> token = REPEAT;
+       }
+       else if(strcmp(ident, "UNTIL") == 0)
+       {
+               ctx -> token = UNTIL;
+       }
 }
 
 static void
@@ -2986,6 +2996,27 @@ oberon_statement(oberon_context_t * ctx)
                oberon_assert_token(ctx, END);
                oberon_generate_label(ctx, end);
        }
+       else if(ctx -> token == REPEAT)
+       {
+               gen_label_t * begin;
+               oberon_expr_t * cond;
+
+               begin = oberon_generator_reserve_label(ctx);
+               oberon_generate_label(ctx, begin);
+               oberon_assert_token(ctx, REPEAT);
+
+               oberon_statement_seq(ctx);
+
+               oberon_assert_token(ctx, UNTIL);
+
+               cond = oberon_expr(ctx);
+               if(cond -> result -> class != OBERON_TYPE_BOOLEAN)
+               {
+                       oberon_error(ctx, "condition must be boolean");
+               }
+
+               oberon_generate_branch(ctx, cond, true, begin);
+       }
        else if(ctx -> token == RETURN)
        {
                oberon_assert_token(ctx, RETURN);
index 549e5eaa69e890f6c70b2c053474152028f2f571..059af717a19c98da40c7ebade7168065ce8a2f9c 100644 (file)
@@ -15,10 +15,10 @@ static char source_test[] =
        "BEGIN"
        "  i := 0;"
        "  Out.Open();"
-       "  WHILE i < 4 DO"
+       "  REPEAT"
        "    Out.String('Count '); Out.Int(i, 0); Out.Ln;"
        "    i := i + 1;"
-       "  END;"
+       "  UNTIL i < 4;"
        "  Out.String('end'); Out.Ln;"
        "END Test."
 ;