]> git.ozlabs.org Git - ccan/blobdiff - tools/ccanlint/idempotent.c
Enhance file_analysis preprocessor a little more, use in idempotent test.
[ccan] / tools / ccanlint / idempotent.c
index 38d48d8319676e9044ca631a423559d7657e3d11..ffe3d7380483bc3e4ee7b9fc1faf01ffd1cdbe9f 100644 (file)
@@ -22,81 +22,92 @@ static const char explain[]
   "...\n"
   "#endif /* MY_HEADER_H */\n";
 
-static char *get_ifndef_sym(char *line)
+static char *report_idem(struct ccan_file *f, char *sofar)
 {
-       line += strspn(line, SPACE_CHARS);
-       if (line[0] == '#')
-       {
-               line++;
-               line += strspn(line, SPACE_CHARS);
-               if (strstarts(line, "ifndef") && isspace(line[6]))
-                       return line+6+strspn(line+6, SPACE_CHARS);
-               else if (strstarts(line, "if"))
-               {
-                       line += 2;
-                       line += strspn(line, SPACE_CHARS);
-                       if (line[0] == '!')
-                       {
-                               line++;
-                               line += strspn(line, SPACE_CHARS);
-                               if (strstarts(line, "defined"))
-                               {
-                                       line += 7;
-                                       line += strspn(line, SPACE_CHARS);
-                                       if (line[0] == '(')
-                                       {
-                                               line++;
-                                               line += strspn(line,
-                                                       SPACE_CHARS);
-                                       }
-                                       return line;
-                               }
-                       }
-               }
+       struct line_info *line_info;
+       unsigned int i, first_preproc_line;
+       const char *line, *sym;
+
+       line_info = get_ccan_line_info(f);
+       if (f->num_lines < 3)
+               /* FIXME: We assume small headers probably uninteresting. */
+               return sofar;
+
+       for (i = 0; i < f->num_lines; i++) {
+               if (line_info[i].type == DOC_LINE
+                   || line_info[i].type == COMMENT_LINE)
+                       continue;
+               if (line_info[i].type == CODE_LINE)
+                       return talloc_asprintf_append(sofar,
+                             "%s:%u:expect first non-comment line to be #ifndef.\n", f->name, i+1);
+               else if (line_info[i].type == PREPROC_LINE)
+                       break;
        }
-       return NULL;
-}
 
-static int is_define(char *line, char *id, size_t id_len)
-{
-       line += strspn(line, SPACE_CHARS);
-       if (line[0] == '#')
-       {
-               line++;
-               line += strspn(line, SPACE_CHARS);
-               if (strstarts(line, "define") && isspace(line[6]))
-               {
-                       line += 6;
-                       line += strspn(line, SPACE_CHARS);
-                       if (strspn(line, IDENT_CHARS) == id_len &&
-                           memcmp(id, line, id_len) == 0)
-                               return 1;
-               }
+       /* No code at all?  Don't complain. */
+       if (i == f->num_lines)
+               return sofar;
+
+       first_preproc_line = i;
+       for (i = first_preproc_line+1; i < f->num_lines; i++) {
+               if (line_info[i].type == DOC_LINE
+                   || line_info[i].type == COMMENT_LINE)
+                       continue;
+               if (line_info[i].type == CODE_LINE)
+                       return talloc_asprintf_append(sofar,
+                             "%s:%u:expect second line to be #define.\n", f->name, i+1);
+               else if (line_info[i].type == PREPROC_LINE)
+                       break;
        }
-       return 0;
-}
 
-static char *report_idem(struct ccan_file *f, char *sofar)
-{
-       char **lines;
-       char *id;
-       size_t id_len;
+       /* No code at all?  Weird. */
+       if (i == f->num_lines)
+               return sofar;
 
-       lines = get_ccan_file_lines(f);
-       if (f->num_lines < 3)
-               /* FIXME: We assume small headers probably uninteresting. */
-               return NULL;
+       /* We expect a condition on this line. */
+       if (!line_info[i].cond) {
+               return talloc_asprintf_append(sofar,
+                                             "%s:%u:expected #ifndef.\n",
+                                             f->name, first_preproc_line+1);
+       }
+
+       line = f->lines[i];
 
-       id = get_ifndef_sym(lines[0]);
-       if (!id)
+       /* We expect the condition to be ! IFDEF <symbol>. */
+       if (line_info[i].cond->type != PP_COND_IFDEF
+           || !line_info[i].cond->inverse) {
                return talloc_asprintf_append(sofar,
-                       "%s:1:expect first line to be #ifndef.\n", f->name);
-       id_len = strspn(id, IDENT_CHARS);
+                                             "%s:%u:expected #ifndef.\n",
+                                             f->name, first_preproc_line+1);
+       }
 
-       if (!is_define(lines[1], id, id_len))
+       /* And this to be #define <symbol> */
+       if (!get_token(&line, "#"))
+               abort();
+       if (!get_token(&line, "define")) {
                return talloc_asprintf_append(sofar,
-                       "%s:2:expect second line to be '#define %.*s'.\n",
-                       f->name, (int)id_len, id);
+                             "%s:%u:expected '#define %s'.\n",
+                             f->name, i+1, line_info[i].cond->symbol);
+       }
+       sym = get_symbol_token(f, &line);
+       if (!sym || !streq(sym, line_info[i].cond->symbol)) {
+               return talloc_asprintf_append(sofar,
+                             "%s:%u:expected '#define %s'.\n",
+                             f->name, i+1, line_info[i].cond->symbol);
+       }
+
+       /* Rest of code should all be covered by that conditional. */
+       for (i++; i < f->num_lines; i++) {
+               unsigned int val = 0;
+               if (line_info[i].type == DOC_LINE
+                   || line_info[i].type == COMMENT_LINE)
+                       continue;
+               if (get_ccan_line_pp(line_info[i].cond, sym, &val)
+                   != NOT_COMPILED)
+                       return talloc_asprintf_append(sofar,
+                             "%s:%u:code outside idempotent region.\n",
+                             f->name, i+1);
+       }
 
        return sofar;
 }