]> git.cameronkatri.com Git - mandoc.git/blobdiff - mdoc_validate.c
`Bl -column' now correctly handles tail entries (Bl -column -more... arg0...).
[mandoc.git] / mdoc_validate.c
index 1da8a28d45e11956a82d7b70f07d7d3139ffe0cb..3111f8642dffc7c04eb5c4c359a886875eb65c5b 100644 (file)
@@ -1,4 +1,4 @@
-/*     $Id: mdoc_validate.c,v 1.8 2009/06/11 10:34:31 kristaps Exp $ */
+/*     $Id: mdoc_validate.c,v 1.14 2009/06/17 14:08:47 kristaps Exp $ */
 /*
  * Copyright (c) 2008, 2009 Kristaps Dzonsons <kristaps@kth.se>
  *
@@ -42,9 +42,11 @@ enum merr {
        ELISTTYPE,
        EDISPTYPE,
        EMULTIDISP,
+       ESECNAME,
        EMULTILIST,
        EARGREP,
        EBOOL,
+       ECOLMIS,
        ENESTDISP
 };
 
@@ -53,6 +55,7 @@ enum  mwarn {
        WNOWIDTH,
        WMISSWIDTH,
        WESCAPE,
+       WDEPESC,
        WDEPCOL,
        WWRONGMSEC,
        WSECOOO,
@@ -115,7 +118,6 @@ static      int     pre_fd(PRE_ARGS);
 static int     pre_it(PRE_ARGS);
 static int     pre_lb(PRE_ARGS);
 static int     pre_os(PRE_ARGS);
-static int     pre_prologue(PRE_ARGS);
 static int     pre_rv(PRE_ARGS);
 static int     pre_sh(PRE_ARGS);
 static int     pre_ss(PRE_ARGS);
@@ -137,6 +139,7 @@ static      int     post_args(POST_ARGS);
 static int     post_at(POST_ARGS);
 static int     post_bf(POST_ARGS);
 static int     post_bl(POST_ARGS);
+static int     post_bl_head(POST_ARGS);
 static int     post_it(POST_ARGS);
 static int     post_nm(POST_ARGS);
 static int     post_root(POST_ARGS);
@@ -154,15 +157,15 @@ static    v_pre   pres_an[] = { pre_an, NULL };
 static v_pre   pres_bd[] = { pre_display, pre_bd, NULL };
 static v_pre   pres_bl[] = { pre_bl, NULL };
 static v_pre   pres_cd[] = { pre_cd, NULL };
-static v_pre   pres_dd[] = { pre_prologue, pre_dd, NULL };
+static v_pre   pres_dd[] = { pre_dd, NULL };
 static v_pre   pres_d1[] = { pre_display, NULL };
-static v_pre   pres_dt[] = { pre_prologue, pre_dt, NULL };
+static v_pre   pres_dt[] = { pre_dt, NULL };
 static v_pre   pres_er[] = { pre_er, NULL };
 static v_pre   pres_ex[] = { pre_ex, NULL };
 static v_pre   pres_fd[] = { pre_fd, NULL };
 static v_pre   pres_it[] = { pre_it, NULL };
 static v_pre   pres_lb[] = { pre_lb, NULL };
-static v_pre   pres_os[] = { pre_prologue, pre_os, NULL };
+static v_pre   pres_os[] = { pre_os, NULL };
 static v_pre   pres_rv[] = { pre_rv, NULL };
 static v_pre   pres_sh[] = { pre_sh, NULL };
 static v_pre   pres_ss[] = { pre_ss, NULL };
@@ -173,7 +176,7 @@ static      v_post  posts_wtext[] = { ewarn_ge1, NULL };
 static v_post  posts_notext[] = { eerr_eq0, NULL };
 static v_post  posts_wline[] = { bwarn_ge1, herr_eq0, NULL };
 static v_post  posts_sh[] = { herr_ge1, bwarn_ge1, post_sh, NULL };
-static v_post  posts_bl[] = { herr_eq0, bwarn_ge1, post_bl, NULL };
+static v_post  posts_bl[] = { bwarn_ge1, post_bl, NULL };
 static v_post  posts_it[] = { post_it, NULL };
 static v_post  posts_in[] = { ewarn_eq1, NULL };
 static v_post  posts_ss[] = { herr_ge1, NULL };
@@ -191,7 +194,7 @@ static      v_post  posts_bf[] = { hwarn_le1, post_bf, NULL };
 static v_post  posts_fo[] = { hwarn_eq1, bwarn_ge1, NULL };
 
 const  struct valids mdoc_valids[MDOC_MAX] = {
-       { NULL, NULL },                         /* \" */
+       { NULL, NULL },                         /* Ap */
        { pres_dd, posts_text },                /* Dd */
        { pres_dt, NULL },                      /* Dt */
        { pres_os, NULL },                      /* Os */
@@ -298,9 +301,8 @@ const       struct valids mdoc_valids[MDOC_MAX] = {
        { NULL, NULL },                         /* Fr */
        { NULL, posts_notext },                 /* Ud */
        { pres_lb, posts_lb },                  /* Lb */
-       { NULL, NULL },                         /* Ap */
        { NULL, posts_pp },                     /* Lp */ 
-       { NULL, posts_text },                   /* Lk */ 
+       { NULL, NULL },                         /* Lk */ 
        { NULL, posts_text },                   /* Mt */ 
        { NULL, posts_wline },                  /* Brq */ 
        { NULL, NULL },                         /* Bro */ 
@@ -413,6 +415,9 @@ perr(struct mdoc *m, int line, int pos, enum merr type)
        case (EDISPTYPE):
                p = "missing display type";
                break;
+       case (ESECNAME):
+               p = "the NAME section must come first";
+               break;
        case (ELINE):
                p = "expected line arguments";
                break;
@@ -422,6 +427,9 @@ perr(struct mdoc *m, int line, int pos, enum merr type)
        case (ENODATA):
                p = "document has no data";
                break;
+       case (ECOLMIS):
+               p = "column syntax style mismatch";
+               break;
        case (EATT):
                p = "expected valid AT&T symbol";
                break;
@@ -479,6 +487,9 @@ pwarn(struct mdoc *m, int line, int pos, enum mwarn type)
        case (WESCAPE):
                p = "invalid escape sequence";
                break;
+       case (WDEPESC):
+               p = "deprecated special-character escape";
+               break;
        case (WNOLINE):
                p = "suggested no line arguments";
                break;
@@ -718,8 +729,6 @@ check_text(struct mdoc *mdoc, int line, int pos, const char *p)
 {
        size_t           c;
 
-       /* FIXME: indicate deprecated escapes \*(xx and \*x. */
-
        for ( ; *p; p++) {
                if ('\t' == *p) {
                        if ( ! (MDOC_LITERAL & mdoc->flags))
@@ -734,6 +743,10 @@ check_text(struct mdoc *mdoc, int line, int pos, const char *p)
 
                c = mdoc_isescape(p);
                if (c) {
+                       /* See if form is deprecated. */
+                       if ('*' == p[1]) 
+                               if ( ! pwarn(mdoc, line, pos, WDEPESC))
+                                       return(0);
                        p += (int)c - 1;
                        continue;
                }
@@ -789,7 +802,7 @@ pre_display(PRE_ARGS)
 static int
 pre_bl(PRE_ARGS)
 {
-       int              pos, type, width, offset;
+       int              pos, col, type, width, offset;
 
        if (MDOC_BLOCK != n->type)
                return(1);
@@ -798,7 +811,7 @@ pre_bl(PRE_ARGS)
 
        /* Make sure that only one type of list is specified.  */
 
-       type = offset = width = -1;
+       type = offset = width = col = -1;
 
        /* LINTED */
        for (pos = 0; pos < (int)n->args->argc; pos++)
@@ -827,6 +840,7 @@ pre_bl(PRE_ARGS)
                        if (-1 != type) 
                                return(nerr(mdoc, n, EMULTILIST));
                        type = n->args->argv[pos].arg;
+                       col = pos;
                        break;
                case (MDOC_Width):
                        if (-1 != width)
@@ -870,21 +884,6 @@ pre_bl(PRE_ARGS)
                break;
        }
 
-       /*
-        * General validation of fields.
-        */
-
-       switch (type) {
-       case (MDOC_Column):
-               if (0 == n->args->argv[pos].sz)
-                       break;
-               if ( ! nwarn(mdoc, n, WDEPCOL))
-                       return(0);
-               break;
-       default:
-               break;
-       }
-
        return(1);
 }
 
@@ -1011,14 +1010,6 @@ pre_cd(PRE_ARGS)
 }
 
 
-static int
-pre_prologue(PRE_ARGS)
-{
-
-       return(check_sec(mdoc, n, SEC_PROLOGUE, SEC_CUSTOM));
-}
-
-
 static int
 pre_dt(PRE_ARGS)
 {
@@ -1072,25 +1063,24 @@ post_bf(POST_ARGS)
 
        head = mdoc->last->head;
 
-       if (NULL == mdoc->last->args) {
-               if (NULL == head->child || 
-                               MDOC_TEXT != head->child->type)
-                       return(mdoc_err(mdoc, "text argument expected"));
+       if (mdoc->last->args && head->child)
+               return(mdoc_err(mdoc, "one argument expected"));
+       else if (mdoc->last->args)
+               return(1);
 
-               p = head->child->string;
-               if (0 == strcmp(p, "Em"))
-                       return(1);
-               else if (0 == strcmp(p, "Li"))
-                       return(1);
-               else if (0 == strcmp(p, "Sm"))
-                       return(1);
-               return(mdoc_nerr(mdoc, head->child, "invalid font"));
-       }
+       if (NULL == head->child || MDOC_TEXT != head->child->type)
+               return(mdoc_err(mdoc, "text argument expected"));
 
-       if (head->child)
-               return(mdoc_err(mdoc, "one argument expected"));
+       p = head->child->string;
 
-       return(1);
+       if (0 == strcmp(p, "Em"))
+               return(1);
+       else if (0 == strcmp(p, "Li"))
+               return(1);
+       else if (0 == strcmp(p, "Sm"))
+               return(1);
+
+       return(mdoc_nerr(mdoc, head->child, "invalid font mode"));
 }
 
 
@@ -1255,11 +1245,36 @@ post_it(POST_ARGS)
 }
 
 
+static int
+post_bl_head(POST_ARGS) 
+{
+       int                     i;
+       const struct mdoc_node *n;
+
+       n = mdoc->last->parent;
+       assert(n->args);
+
+       for (i = 0; i < (int)n->args->argc; i++)
+               if (n->args->argv[i].arg == MDOC_Column)
+                       break;
+
+       if (i == (int)n->args->argc)
+               return(1);
+
+       if (n->args->argv[i].sz && mdoc->last->child)
+               return(nerr(mdoc, n, ECOLMIS));
+
+       return(1);
+}
+
+
 static int
 post_bl(POST_ARGS)
 {
        struct mdoc_node        *n;
 
+       if (MDOC_HEAD == mdoc->last->type) 
+               return(post_bl_head(mdoc));
        if (MDOC_BODY != mdoc->last->type)
                return(1);
        if (NULL == mdoc->last->child)
@@ -1306,7 +1321,7 @@ post_root(POST_ARGS)
 
        if (NULL == mdoc->first->child)
                return(verr(mdoc, ENODATA));
-       if (SEC_PROLOGUE == mdoc->lastnamed)
+       if ( ! (MDOC_PBODY & mdoc->flags))
                return(verr(mdoc, ENOPROLOGUE));
 
        if (MDOC_BLOCK != mdoc->first->child->type)
@@ -1387,13 +1402,12 @@ post_sh_head(POST_ARGS)
         * certain manual sections.
         */
 
-       assert(MDOC_Sh == mdoc->last->tok);
-
-       /* This is just concat() inlined, which is irritating. */
-
        buf[0] = 0;
+
        for (n = mdoc->last->child; n; n = n->next) {
+               /* XXX - copied from compact(). */
                assert(MDOC_TEXT == n->type);
+
                if (strlcat(buf, n->string, 64) >= 64)
                        return(nerr(mdoc, n, ETOOLONG));
                if (NULL == n->next)
@@ -1404,21 +1418,24 @@ post_sh_head(POST_ARGS)
 
        sec = mdoc_atosec(buf);
 
-       /* The NAME section should always be first. */
+       /* 
+        * Check: NAME should always be first, CUSTOM has no roles,
+        * non-CUSTOM has a conventional order to be followed.
+        */
 
-       if (SEC_BODY == mdoc->lastnamed && SEC_NAME != sec)
-               return(vwarn(mdoc, WSECOOO));
+       if (SEC_NAME != sec && SEC_NONE == mdoc->lastnamed)
+               return(verr(mdoc, ESECNAME));
        if (SEC_CUSTOM == sec)
                return(1);
-
-       /* Check for repeated or out-of-order sections. */
-
        if (sec == mdoc->lastnamed)
                return(vwarn(mdoc, WSECREP));
        if (sec < mdoc->lastnamed)
                return(vwarn(mdoc, WSECOOO));
 
-       /* Check particular section/manual section conventions. */
+       /* 
+        * Check particular section/manual conventions.  LIBRARY can
+        * only occur in msec 2, 3 (TODO: are there more of these?).
+        */
 
        switch (sec) {
        case (SEC_LIBRARY):