improve macro error messages
[pspp] / src / language / lexer / macro.c
index 6d7decb40faccf0e97e85a40ca04379885c068c4..bee23c999cb1be7f6254d69cdf315eea6206319c 100644 (file)
@@ -32,6 +32,7 @@
 #include "libpspp/str.h"
 #include "libpspp/string-array.h"
 #include "libpspp/string-map.h"
+#include "libpspp/stringi-set.h"
 
 #include "gl/c-ctype.h"
 #include "gl/ftoastr.h"
@@ -59,6 +60,41 @@ macro_token_to_representation (struct macro_token *mt, struct string *s)
   ds_put_substring (s, mt->representation);
 }
 
+bool
+is_macro_keyword (struct substring s)
+{
+  static struct stringi_set keywords = STRINGI_SET_INITIALIZER (keywords);
+  if (stringi_set_is_empty (&keywords))
+    {
+      static const char *kws[] = {
+        "BREAK",
+        "CHAREND",
+        "CMDEND",
+        "DEFAULT",
+        "DO",
+        "DOEND",
+        "ELSE",
+        "ENCLOSE",
+        "ENDDEFINE",
+        "IF",
+        "IFEND",
+        "IN",
+        "LET",
+        "NOEXPAND",
+        "OFFEXPAND",
+        "ONEXPAND",
+        "POSITIONAL",
+        "THEN",
+        "TOKENS",
+      };
+      for (size_t i = 0; i < sizeof kws / sizeof *kws; i++)
+        stringi_set_insert (&keywords, kws[i]);
+    }
+
+  ss_ltrim (&s, ss_cstr ("!"));
+  return stringi_set_contains_len (&keywords, s.string, s.length);
+}
+
 void
 macro_tokens_copy (struct macro_tokens *dst, const struct macro_tokens *src)
 {
@@ -251,28 +287,37 @@ classify_token (enum token_type type)
 }
 
 void
-macro_tokens_to_representation (struct macro_tokens *mts, struct string *s)
+macro_tokens_to_representation (struct macro_tokens *mts, struct string *s,
+                                size_t *ofs, size_t *len)
 {
+  assert ((ofs != NULL) == (len != NULL));
+
   if (!mts->n)
     return;
 
-  macro_token_to_representation (&mts->mts[0], s);
-  for (size_t i = 1; i < mts->n; i++)
+  for (size_t i = 0; i < mts->n; i++)
     {
-      enum token_type prev = mts->mts[i - 1].token.type;
-      enum token_type next = mts->mts[i].token.type;
-
-      if (prev == T_ENDCMD)
-        ds_put_byte (s, '\n');
-      else
+      if (i > 0)
         {
-          enum token_class pc = classify_token (prev);
-          enum token_class nc = classify_token (next);
-          if (needs_space (pc, nc))
-            ds_put_byte (s, ' ');
+          enum token_type prev = mts->mts[i - 1].token.type;
+          enum token_type next = mts->mts[i].token.type;
+
+          if (prev == T_ENDCMD)
+            ds_put_byte (s, '\n');
+          else
+            {
+              enum token_class pc = classify_token (prev);
+              enum token_class nc = classify_token (next);
+              if (needs_space (pc, nc))
+                ds_put_byte (s, ' ');
+            }
         }
 
+      if (ofs)
+        ofs[i] = s->ss.length;
       macro_token_to_representation (&mts->mts[i], s);
+      if (len)
+        len[i] = s->ss.length - ofs[i];
     }
 }
 
@@ -543,8 +588,7 @@ me_enclose (struct macro_expander *me, const struct macro_token *mt)
 static const struct macro_param *
 macro_find_parameter_by_name (const struct macro *m, struct substring name)
 {
-  if (ss_first (name) == '!')
-    ss_advance (&name, 1);
+  ss_ltrim (&name, ss_cstr ("!"));
 
   for (size_t i = 0; i < m->n_params; i++)
     {
@@ -1025,7 +1069,7 @@ expand_macro_function (struct parse_macro_function_ctx *ctx,
       if (mts.n > 1)
         {
           struct macro_tokens tail = { .mts = mts.mts + 1, .n = mts.n - 1 };
-          macro_tokens_to_representation (&tail, output);
+          macro_tokens_to_representation (&tail, output, NULL, NULL);
         }
       macro_tokens_uninit (&mts);
       ds_destroy (&tmp);
@@ -1055,7 +1099,7 @@ expand_macro_function (struct parse_macro_function_ctx *ctx,
       struct macro_tokens exp = { .n = 0 };
       macro_expand (&mts, ctx->nesting_countdown - 1, ctx->macros, ctx->me,
                     ctx->vars, ctx->expand, NULL, &exp);
-      macro_tokens_to_representation (&exp, output);
+      macro_tokens_to_representation (&exp, output, NULL, NULL);
       macro_tokens_uninit (&exp);
       macro_tokens_uninit (&mts);
     }
@@ -1478,6 +1522,12 @@ macro_parse_let (const struct macro_token *tokens, size_t n_tokens,
       return 0;
     }
   const struct substring var_name = p->token.string;
+  if (is_macro_keyword (var_name)
+      || macro_find_parameter_by_name (me->macro, var_name))
+    {
+      printf ("cannot use argument name or macro keyword as !LET variable\n");
+      return 0;
+    }
   p++;
 
   if (p >= end || p->token.type != T_EQUALS)
@@ -1538,8 +1588,15 @@ macro_expand_do (const struct macro_token *tokens, size_t n_tokens,
       return 0;
     }
   const struct substring var_name = p->token.string;
+  if (is_macro_keyword (var_name)
+      || macro_find_parameter_by_name (me->macro, var_name))
+    {
+      printf ("cannot use argument name or macro keyword as !DO variable\n");
+      return 0;
+    }
   p++;
 
+  int miterate = settings_get_miterate ();
   if (p < end && p->token.type == T_MACRO_ID
       && ss_equals_case (p->token.string, ss_cstr ("!IN")))
     {
@@ -1568,6 +1625,11 @@ macro_expand_do (const struct macro_token *tokens, size_t n_tokens,
       };
       for (size_t i = 0; i < items.n; i++)
         {
+          if (i >= miterate)
+            {
+              printf ("exceeded maximum number of iterations %d\n", miterate);
+              break;
+            }
           string_map_replace_nocopy (vars, ss_xstrdup (var_name),
                                      ss_xstrdup (items.mts[i].representation));
 
@@ -1625,21 +1687,31 @@ macro_expand_do (const struct macro_token *tokens, size_t n_tokens,
       };
 
       if ((by > 0 && first <= last) || (by < 0 && first >= last))
-        for (double index = first;
-             by > 0 ? (index <= last) : (index >= last);
-             index += by)
-          {
-            char index_s[DBL_BUFSIZE_BOUND];
-            c_dtoastr (index_s, sizeof index_s, 0, 0, index);
-            string_map_replace_nocopy (vars, ss_xstrdup (var_name),
-                                       xstrdup (index_s));
-
-            bool break_ = false;
-            macro_expand (&inner, nesting_countdown, macros,
-                          me, vars, expand, &break_, exp);
-            if (break_)
-              break;
-          }
+        {
+          int i = 0;
+          for (double index = first;
+               by > 0 ? (index <= last) : (index >= last);
+               index += by)
+            {
+              if (i++ > miterate)
+                {
+                  printf ("exceeded maximum number of iterations %d\n",
+                          miterate);
+                  break;
+                }
+
+              char index_s[DBL_BUFSIZE_BOUND];
+              c_dtoastr (index_s, sizeof index_s, 0, 0, index);
+              string_map_replace_nocopy (vars, ss_xstrdup (var_name),
+                                         xstrdup (index_s));
+
+              bool break_ = false;
+              macro_expand (&inner, nesting_countdown, macros,
+                            me, vars, expand, &break_, exp);
+              if (break_)
+                break;
+            }
+        }
 
       return do_end - tokens + 1;
     }