[BACK]Return to roff.c CVS log [TXT][DIR] Up to [cvsweb.bsd.lv] / mandoc

Diff for /mandoc/roff.c between version 1.208 and 1.237

version 1.208, 2014/04/20 19:40:13 version 1.237, 2014/10/28 17:36:19
Line 15 
Line 15 
  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF   * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.   * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  */   */
 #ifdef HAVE_CONFIG_H  
 #include "config.h"  #include "config.h"
 #endif  
   
   #include <sys/types.h>
   
 #include <assert.h>  #include <assert.h>
 #include <ctype.h>  #include <ctype.h>
 #include <stdio.h>  #include <stdio.h>
Line 27 
Line 27 
   
 #include "mandoc.h"  #include "mandoc.h"
 #include "mandoc_aux.h"  #include "mandoc_aux.h"
 #include "libroff.h"  
 #include "libmandoc.h"  #include "libmandoc.h"
   #include "libroff.h"
   
 /* Maximum number of nested if-else conditionals. */  /* Maximum number of nested if-else conditionals. */
 #define RSTACK_MAX      128  #define RSTACK_MAX      128
Line 60  enum rofft {
Line 60  enum rofft {
         ROFF_nh,          ROFF_nh,
         ROFF_nr,          ROFF_nr,
         ROFF_ns,          ROFF_ns,
           ROFF_pl,
         ROFF_ps,          ROFF_ps,
         ROFF_rm,          ROFF_rm,
         ROFF_rr,          ROFF_rr,
Line 106  struct roffreg {
Line 107  struct roffreg {
   
 struct  roff {  struct  roff {
         struct mparse   *parse; /* parse point */          struct mparse   *parse; /* parse point */
         int              options; /* parse options */          const struct mchars *mchars; /* character table */
         struct roffnode *last; /* leaf of stack */          struct roffnode *last; /* leaf of stack */
         int              rstack[RSTACK_MAX]; /* stack of !`ie' rules */          int             *rstack; /* stack of inverted `ie' values */
         char             control; /* control character */  
         int              rstackpos; /* position in rstack */  
         struct roffreg  *regtab; /* number registers */          struct roffreg  *regtab; /* number registers */
         struct roffkv   *strtab; /* user-defined strings & macros */          struct roffkv   *strtab; /* user-defined strings & macros */
         struct roffkv   *xmbtab; /* multi-byte trans table (`tr') */          struct roffkv   *xmbtab; /* multi-byte trans table (`tr') */
Line 122  struct roff {
Line 121  struct roff {
         struct eqn_node *last_eqn; /* last equation parsed */          struct eqn_node *last_eqn; /* last equation parsed */
         struct eqn_node *first_eqn; /* first equation parsed */          struct eqn_node *first_eqn; /* first equation parsed */
         struct eqn_node *eqn; /* current equation being parsed */          struct eqn_node *eqn; /* current equation being parsed */
           int              eqn_inline; /* current equation is inline */
           int              options; /* parse options */
           int              rstacksz; /* current size limit of rstack */
           int              rstackpos; /* position in rstack */
           int              format; /* current file in mdoc or man format */
           char             control; /* control character */
 };  };
   
 struct  roffnode {  struct  roffnode {
Line 180  static enum rofferr  roff_cond(ROFF_ARGS);
Line 185  static enum rofferr  roff_cond(ROFF_ARGS);
 static  enum rofferr     roff_cond_text(ROFF_ARGS);  static  enum rofferr     roff_cond_text(ROFF_ARGS);
 static  enum rofferr     roff_cond_sub(ROFF_ARGS);  static  enum rofferr     roff_cond_sub(ROFF_ARGS);
 static  enum rofferr     roff_ds(ROFF_ARGS);  static  enum rofferr     roff_ds(ROFF_ARGS);
 static  int              roff_evalcond(const char *, int *);  static  enum rofferr     roff_eqndelim(struct roff *,
 static  int              roff_evalnum(const char *, int *, int *, int);                                  char **, size_t *, int);
 static  int              roff_evalpar(const char *, int *, int *);  static  int              roff_evalcond(struct roff *r, int,
                                   const char *, int *);
   static  int              roff_evalnum(struct roff *, int,
                                   const char *, int *, int *, int);
   static  int              roff_evalpar(struct roff *, int,
                                   const char *, int *, int *);
 static  int              roff_evalstrcond(const char *, int *);  static  int              roff_evalstrcond(const char *, int *);
 static  void             roff_free1(struct roff *);  static  void             roff_free1(struct roff *);
 static  void             roff_freereg(struct roffreg *);  static  void             roff_freereg(struct roffreg *);
 static  void             roff_freestr(struct roffkv *);  static  void             roff_freestr(struct roffkv *);
 static  char            *roff_getname(struct roff *, char **, int, int);  static  size_t           roff_getname(struct roff *, char **, int, int);
 static  int              roff_getnum(const char *, int *, int *);  static  int              roff_getnum(const char *, int *, int *);
 static  int              roff_getop(const char *, int *, char *);  static  int              roff_getop(const char *, int *, char *);
 static  int              roff_getregn(const struct roff *,  static  int              roff_getregn(const struct roff *,
Line 198  static const char *roff_getstrn(const struct roff *,
Line 208  static const char *roff_getstrn(const struct roff *,
 static  enum rofferr     roff_it(ROFF_ARGS);  static  enum rofferr     roff_it(ROFF_ARGS);
 static  enum rofferr     roff_line_ignore(ROFF_ARGS);  static  enum rofferr     roff_line_ignore(ROFF_ARGS);
 static  enum rofferr     roff_nr(ROFF_ARGS);  static  enum rofferr     roff_nr(ROFF_ARGS);
 static  void             roff_openeqn(struct roff *, const char *,  static  enum rofft       roff_parse(struct roff *, char *, int *,
                                 int, int, const char *);                                  int, int);
 static  enum rofft       roff_parse(struct roff *, const char *, int *);  
 static  enum rofferr     roff_parsetext(char **, size_t *, int, int *);  static  enum rofferr     roff_parsetext(char **, size_t *, int, int *);
 static  enum rofferr     roff_res(struct roff *,  static  enum rofferr     roff_res(struct roff *,
                                 char **, size_t *, int, int);                                  char **, size_t *, int, int);
Line 253  static struct roffmac  roffs[ROFF_MAX] = {
Line 262  static struct roffmac  roffs[ROFF_MAX] = {
         { "nh", roff_line_ignore, NULL, NULL, 0, NULL },          { "nh", roff_line_ignore, NULL, NULL, 0, NULL },
         { "nr", roff_nr, NULL, NULL, 0, NULL },          { "nr", roff_nr, NULL, NULL, 0, NULL },
         { "ns", roff_line_ignore, NULL, NULL, 0, NULL },          { "ns", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pl", roff_line_ignore, NULL, NULL, 0, NULL },
         { "ps", roff_line_ignore, NULL, NULL, 0, NULL },          { "ps", roff_line_ignore, NULL, NULL, 0, NULL },
         { "rm", roff_rm, NULL, NULL, 0, NULL },          { "rm", roff_rm, NULL, NULL, 0, NULL },
         { "rr", roff_rr, NULL, NULL, 0, NULL },          { "rr", roff_rr, NULL, NULL, 0, NULL },
Line 419  roff_free1(struct roff *r)
Line 429  roff_free1(struct roff *r)
                 r->first_tbl = tbl->next;                  r->first_tbl = tbl->next;
                 tbl_free(tbl);                  tbl_free(tbl);
         }          }
   
         r->first_tbl = r->last_tbl = r->tbl = NULL;          r->first_tbl = r->last_tbl = r->tbl = NULL;
   
         while (NULL != (e = r->first_eqn)) {          while (NULL != (e = r->first_eqn)) {
                 r->first_eqn = e->next;                  r->first_eqn = e->next;
                 eqn_free(e);                  eqn_free(e);
         }          }
   
         r->first_eqn = r->last_eqn = r->eqn = NULL;          r->first_eqn = r->last_eqn = r->eqn = NULL;
   
         while (r->last)          while (r->last)
                 roffnode_pop(r);                  roffnode_pop(r);
   
         roff_freestr(r->strtab);          free (r->rstack);
         roff_freestr(r->xmbtab);          r->rstack = NULL;
           r->rstacksz = 0;
           r->rstackpos = -1;
   
         r->strtab = r->xmbtab = NULL;  
   
         roff_freereg(r->regtab);          roff_freereg(r->regtab);
   
         r->regtab = NULL;          r->regtab = NULL;
   
           roff_freestr(r->strtab);
           roff_freestr(r->xmbtab);
           r->strtab = r->xmbtab = NULL;
   
         if (r->xtab)          if (r->xtab)
                 for (i = 0; i < 128; i++)                  for (i = 0; i < 128; i++)
                         free(r->xtab[i].p);                          free(r->xtab[i].p);
   
         free(r->xtab);          free(r->xtab);
         r->xtab = NULL;          r->xtab = NULL;
 }  }
Line 454  roff_reset(struct roff *r)
Line 464  roff_reset(struct roff *r)
 {  {
   
         roff_free1(r);          roff_free1(r);
           r->format = r->options & (MPARSE_MDOC | MPARSE_MAN);
         r->control = 0;          r->control = 0;
 }  }
   
Line 466  roff_free(struct roff *r)
Line 477  roff_free(struct roff *r)
 }  }
   
 struct roff *  struct roff *
 roff_alloc(struct mparse *parse, int options)  roff_alloc(struct mparse *parse, const struct mchars *mchars, int options)
 {  {
         struct roff     *r;          struct roff     *r;
   
         r = mandoc_calloc(1, sizeof(struct roff));          r = mandoc_calloc(1, sizeof(struct roff));
         r->parse = parse;          r->parse = parse;
           r->mchars = mchars;
         r->options = options;          r->options = options;
           r->format = options & (MPARSE_MDOC | MPARSE_MAN);
         r->rstackpos = -1;          r->rstackpos = -1;
   
         roffhash_init();          roffhash_init();
Line 490  roff_res(struct roff *r, char **bufp, size_t *szp, int
Line 503  roff_res(struct roff *r, char **bufp, size_t *szp, int
 {  {
         char             ubuf[24]; /* buffer to print the number */          char             ubuf[24]; /* buffer to print the number */
         const char      *start; /* start of the string to process */          const char      *start; /* start of the string to process */
         const char      *stesc; /* start of an escape sequence ('\\') */          char            *stesc; /* start of an escape sequence ('\\') */
         const char      *stnam; /* start of the name, after "[(*" */          const char      *stnam; /* start of the name, after "[(*" */
         const char      *cp;    /* end of the name, e.g. before ']' */          const char      *cp;    /* end of the name, e.g. before ']' */
         const char      *res;   /* the string to be substituted */          const char      *res;   /* the string to be substituted */
         char            *nbuf;  /* new buffer to copy bufp to */          char            *nbuf;  /* new buffer to copy bufp to */
         size_t           maxl;  /* expected length of the escape name */          size_t           maxl;  /* expected length of the escape name */
         size_t           naml;  /* actual length of the escape name */          size_t           naml;  /* actual length of the escape name */
         size_t           ressz; /* size of the replacement string */          enum mandoc_esc  esc;   /* type of the escape sequence */
           int              inaml; /* length returned from mandoc_escape() */
         int              expand_count;  /* to avoid infinite loops */          int              expand_count;  /* to avoid infinite loops */
         int              npos;  /* position in numeric expression */          int              npos;  /* position in numeric expression */
         int              irc;   /* return code from roff_evalnum() */          int              arg_complete; /* argument not interrupted by eol */
         char             term;  /* character terminating the escape */          char             term;  /* character terminating the escape */
   
         expand_count = 0;          expand_count = 0;
Line 520  roff_res(struct roff *r, char **bufp, size_t *szp, int
Line 534  roff_res(struct roff *r, char **bufp, size_t *szp, int
                                 break;                                  break;
   
                 if (0 == (stesc - cp) % 2) {                  if (0 == (stesc - cp) % 2) {
                         stesc = cp;                          stesc = (char *)cp;
                         continue;                          continue;
                 }                  }
   
Line 541  roff_res(struct roff *r, char **bufp, size_t *szp, int
Line 555  roff_res(struct roff *r, char **bufp, size_t *szp, int
                         res = ubuf;                          res = ubuf;
                         break;                          break;
                 default:                  default:
                         if (ESCAPE_ERROR == mandoc_escape(&cp, NULL, NULL))                          esc = mandoc_escape(&cp, &stnam, &inaml);
                                 mandoc_msg(MANDOCERR_BADESCAPE, r->parse,                          if (esc == ESCAPE_ERROR ||
                                     ln, (int)(stesc - *bufp), NULL);                              (esc == ESCAPE_SPECIAL &&
                                mchars_spec2cp(r->mchars, stnam, inaml) < 0))
                                   mandoc_vmsg(MANDOCERR_ESC_BAD,
                                       r->parse, ln, (int)(stesc - *bufp),
                                       "%.*s", (int)(cp - stesc), stesc);
                         continue;                          continue;
                 }                  }
   
Line 585  roff_res(struct roff *r, char **bufp, size_t *szp, int
Line 603  roff_res(struct roff *r, char **bufp, size_t *szp, int
   
                 /* Advance to the end of the name. */                  /* Advance to the end of the name. */
   
                   arg_complete = 1;
                 for (naml = 0; 0 == maxl || naml < maxl; naml++, cp++) {                  for (naml = 0; 0 == maxl || naml < maxl; naml++, cp++) {
                         if ('\0' == *cp) {                          if ('\0' == *cp) {
                                 mandoc_msg(MANDOCERR_BADESCAPE, r->parse,                                  mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                                     ln, (int)(stesc - *bufp), NULL);                                      ln, (int)(stesc - *bufp), stesc);
                                   arg_complete = 0;
                                 break;                                  break;
                         }                          }
                         if (0 == maxl && *cp == term) {                          if (0 == maxl && *cp == term) {
Line 604  roff_res(struct roff *r, char **bufp, size_t *szp, int
Line 624  roff_res(struct roff *r, char **bufp, size_t *szp, int
   
                 switch (stesc[1]) {                  switch (stesc[1]) {
                 case '*':                  case '*':
                         res = roff_getstrn(r, stnam, naml);                          if (arg_complete)
                                   res = roff_getstrn(r, stnam, naml);
                         break;                          break;
                 case 'B':                  case 'B':
                         npos = 0;                          npos = 0;
                         irc = roff_evalnum(stnam, &npos, NULL, 0);                          ubuf[0] = arg_complete &&
                         ubuf[0] = irc && stnam + npos + 1 == cp                              roff_evalnum(r, ln, stnam, &npos, NULL, 0) &&
                             ? '1' : '0';                              stnam + npos + 1 == cp ? '1' : '0';
                         ubuf[1] = '\0';                          ubuf[1] = '\0';
                         break;                          break;
                 case 'n':                  case 'n':
                         (void)snprintf(ubuf, sizeof(ubuf), "%d",                          if (arg_complete)
                             roff_getregn(r, stnam, naml));                                  (void)snprintf(ubuf, sizeof(ubuf), "%d",
                                       roff_getregn(r, stnam, naml));
                           else
                                   ubuf[0] = '\0';
                         break;                          break;
                 case 'w':                  case 'w':
                           /* use even incomplete args */
                         (void)snprintf(ubuf, sizeof(ubuf), "%d",                          (void)snprintf(ubuf, sizeof(ubuf), "%d",
                             24 * (int)naml);                              24 * (int)naml);
                         break;                          break;
                 }                  }
   
                 if (NULL == res) {                  if (NULL == res) {
                         mandoc_msg(MANDOCERR_BADESCAPE, r->parse,                          mandoc_vmsg(MANDOCERR_STR_UNDEF,
                             ln, (int)(stesc - *bufp), NULL);                              r->parse, ln, (int)(stesc - *bufp),
                               "%.*s", (int)naml, stnam);
                         res = "";                          res = "";
                 }                  }
                 ressz = strlen(res);  
   
                 /* Replace the escape sequence by the string. */                  /* Replace the escape sequence by the string. */
   
                 *szp += ressz + 1;                  *stesc = '\0';
                 nbuf = mandoc_malloc(*szp);                  *szp = mandoc_asprintf(&nbuf, "%s%s%s",
                       *bufp, res, cp) + 1;
   
                 strlcpy(nbuf, *bufp, (size_t)(stesc - *bufp + 1));  
                 strlcat(nbuf, res, *szp);  
                 strlcat(nbuf, cp, *szp);  
   
                 /* Prepare for the next replacement. */                  /* Prepare for the next replacement. */
   
                 start = nbuf + pos;                  start = nbuf + pos;
                 stesc = nbuf + (stesc - *bufp) + ressz;                  stesc = nbuf + (stesc - *bufp) + strlen(res);
                 free(*bufp);                  free(*bufp);
                 *bufp = nbuf;                  *bufp = nbuf;
         }          }
Line 713  roff_parseln(struct roff *r, int ln, char **bufp,
Line 735  roff_parseln(struct roff *r, int ln, char **bufp,
         enum rofferr     e;          enum rofferr     e;
         int              ppos, ctl;          int              ppos, ctl;
   
         /*          /* Handle in-line equation delimiters. */
          * Run the reserved-word filter only if we have some reserved  
          * words to fill in.  
          */  
   
           if (r->tbl == NULL &&
               r->last_eqn != NULL && r->last_eqn->delim &&
               (r->eqn == NULL || r->eqn_inline)) {
                   e = roff_eqndelim(r, bufp, szp, pos);
                   if (e == ROFF_REPARSE)
                           return(e);
                   assert(e == ROFF_CONT);
           }
   
           /* Expand some escape sequences. */
   
         e = roff_res(r, bufp, szp, ln, pos);          e = roff_res(r, bufp, szp, ln, pos);
         if (ROFF_IGN == e)          if (ROFF_IGN == e)
                 return(e);                  return(e);
Line 750  roff_parseln(struct roff *r, int ln, char **bufp,
Line 780  roff_parseln(struct roff *r, int ln, char **bufp,
                 return(roff_parsetext(bufp, szp, pos, offs));                  return(roff_parsetext(bufp, szp, pos, offs));
         }          }
   
           /* Skip empty request lines. */
   
           if ((*bufp)[pos] == '"') {
                   mandoc_msg(MANDOCERR_COMMENT_BAD, r->parse,
                       ln, pos, NULL);
                   return(ROFF_IGN);
           } else if ((*bufp)[pos] == '\0')
                   return(ROFF_IGN);
   
         /*          /*
          * If a scope is open, go to the child handler for that macro,           * If a scope is open, go to the child handler for that macro,
          * as it may want to preprocess before doing anything with it.           * as it may want to preprocess before doing anything with it.
Line 769  roff_parseln(struct roff *r, int ln, char **bufp,
Line 808  roff_parseln(struct roff *r, int ln, char **bufp,
          * the compilers handle it.           * the compilers handle it.
          */           */
   
         if (ROFF_MAX == (t = roff_parse(r, *bufp, &pos)))          if (ROFF_MAX == (t = roff_parse(r, *bufp, &pos, ln, ppos)))
                 return(ROFF_CONT);                  return(ROFF_CONT);
   
         assert(roffs[t].proc);          assert(roffs[t].proc);
Line 781  roff_endparse(struct roff *r)
Line 820  roff_endparse(struct roff *r)
 {  {
   
         if (r->last)          if (r->last)
                 mandoc_msg(MANDOCERR_SCOPEEXIT, r->parse,                  mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                     r->last->line, r->last->col, NULL);                      r->last->line, r->last->col,
                       roffs[r->last->tok].name);
   
         if (r->eqn) {          if (r->eqn) {
                 mandoc_msg(MANDOCERR_SCOPEEXIT, r->parse,                  mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                     r->eqn->eqn.ln, r->eqn->eqn.pos, NULL);                      r->eqn->eqn.ln, r->eqn->eqn.pos, "EQ");
                 eqn_end(&r->eqn);                  eqn_end(&r->eqn);
         }          }
   
         if (r->tbl) {          if (r->tbl) {
                 mandoc_msg(MANDOCERR_SCOPEEXIT, r->parse,                  mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                     r->tbl->line, r->tbl->pos, NULL);                      r->tbl->line, r->tbl->pos, "TS");
                 tbl_end(&r->tbl);                  tbl_end(&r->tbl);
         }          }
 }  }
Line 802  roff_endparse(struct roff *r)
Line 842  roff_endparse(struct roff *r)
  * form of ".foo xxx" in the usual way.   * form of ".foo xxx" in the usual way.
  */   */
 static enum rofft  static enum rofft
 roff_parse(struct roff *r, const char *buf, int *pos)  roff_parse(struct roff *r, char *buf, int *pos, int ln, int ppos)
 {  {
           char            *cp;
         const char      *mac;          const char      *mac;
         size_t           maclen;          size_t           maclen;
         enum rofft       t;          enum rofft       t;
   
         if ('\0' == buf[*pos] || '"' == buf[*pos] ||          cp = buf + *pos;
             '\t' == buf[*pos] || ' ' == buf[*pos])  
           if ('\0' == *cp || '"' == *cp || '\t' == *cp || ' ' == *cp)
                 return(ROFF_MAX);                  return(ROFF_MAX);
   
         /* We stop the macro parse at an escape, tab, space, or nil. */          mac = cp;
           maclen = roff_getname(r, &cp, ln, ppos);
   
         mac = buf + *pos;  
         maclen = strcspn(mac, " \\\t\0");  
   
         t = (r->current_string = roff_getstrn(r, mac, maclen))          t = (r->current_string = roff_getstrn(r, mac, maclen))
             ? ROFF_USERDEF : roffhash_find(mac, maclen);              ? ROFF_USERDEF : roffhash_find(mac, maclen);
   
         *pos += (int)maclen;          if (ROFF_MAX != t)
                   *pos = cp - buf;
   
         while (buf[*pos] && ' ' == buf[*pos])  
                 (*pos)++;  
   
         return(t);          return(t);
 }  }
   
Line 838  roff_cblock(ROFF_ARGS)
Line 876  roff_cblock(ROFF_ARGS)
          */           */
   
         if (NULL == r->last) {          if (NULL == r->last) {
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "..");
                 return(ROFF_IGN);                  return(ROFF_IGN);
         }          }
   
         switch (r->last->tok) {          switch (r->last->tok) {
         case ROFF_am:          case ROFF_am:
                   /* ROFF_am1 is remapped to ROFF_am in roff_block(). */
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case ROFF_ami:          case ROFF_ami:
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case ROFF_am1:  
                 /* FALLTHROUGH */  
         case ROFF_de:          case ROFF_de:
                 /* ROFF_de1 is remapped to ROFF_de in roff_block(). */                  /* ROFF_de1 is remapped to ROFF_de in roff_block(). */
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
Line 857  roff_cblock(ROFF_ARGS)
Line 895  roff_cblock(ROFF_ARGS)
         case ROFF_ig:          case ROFF_ig:
                 break;                  break;
         default:          default:
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "..");
                 return(ROFF_IGN);                  return(ROFF_IGN);
         }          }
   
         if ((*bufp)[pos])          if ((*bufp)[pos])
                 mandoc_msg(MANDOCERR_ARGSLOST, r->parse, ln, pos, NULL);                  mandoc_vmsg(MANDOCERR_ARG_SKIP, r->parse, ln, pos,
                       ".. %s", *bufp + pos);
   
         roffnode_pop(r);          roffnode_pop(r);
         roffnode_cleanscope(r);          roffnode_cleanscope(r);
Line 886  roff_ccond(struct roff *r, int ln, int ppos)
Line 926  roff_ccond(struct roff *r, int ln, int ppos)
 {  {
   
         if (NULL == r->last) {          if (NULL == r->last) {
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                 return;                  return;
         }          }
   
Line 898  roff_ccond(struct roff *r, int ln, int ppos)
Line 939  roff_ccond(struct roff *r, int ln, int ppos)
         case ROFF_if:          case ROFF_if:
                 break;                  break;
         default:          default:
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                 return;                  return;
         }          }
   
         if (r->last->endspan > -1) {          if (r->last->endspan > -1) {
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                 return;                  return;
         }          }
   
Line 915  roff_ccond(struct roff *r, int ln, int ppos)
Line 958  roff_ccond(struct roff *r, int ln, int ppos)
 static enum rofferr  static enum rofferr
 roff_block(ROFF_ARGS)  roff_block(ROFF_ARGS)
 {  {
         int             sv;          const char      *name;
         size_t          sz;          char            *iname, *cp;
         char            *name;          size_t           namesz;
   
         name = NULL;          /* Ignore groff compatibility mode for now. */
   
         if (ROFF_ig != tok) {          if (ROFF_de1 == tok)
                 if ('\0' == (*bufp)[pos]) {                  tok = ROFF_de;
                         mandoc_msg(MANDOCERR_NOARGS, r->parse, ln, ppos, NULL);          else if (ROFF_am1 == tok)
                         return(ROFF_IGN);                  tok = ROFF_am;
                 }  
   
                 /*          /* Parse the macro name argument. */
                  * Re-write `de1', since we don't really care about  
                  * groff's strange compatibility mode, into `de'.  
                  */  
   
                 if (ROFF_de1 == tok)          cp = *bufp + pos;
                         tok = ROFF_de;          if (ROFF_ig == tok) {
                 if (ROFF_de == tok)                  iname = NULL;
                         name = *bufp + pos;                  namesz = 0;
                 else          } else {
                         mandoc_msg(MANDOCERR_REQUEST, r->parse, ln, ppos,                  iname = cp;
                             roffs[tok].name);                  namesz = roff_getname(r, &cp, ln, ppos);
                   iname[namesz] = '\0';
           }
   
                 while ((*bufp)[pos] && ! isspace((unsigned char)(*bufp)[pos]))          /* Resolve the macro name argument if it is indirect. */
                         pos++;  
   
                 while (isspace((unsigned char)(*bufp)[pos]))          if (namesz && (ROFF_dei == tok || ROFF_ami == tok)) {
                         (*bufp)[pos++] = '\0';                  if (NULL == (name = roff_getstrn(r, iname, namesz))) {
                           mandoc_vmsg(MANDOCERR_STR_UNDEF,
                               r->parse, ln, (int)(iname - *bufp),
                               "%.*s", (int)namesz, iname);
                           namesz = 0;
                   } else
                           namesz = strlen(name);
           } else
                   name = iname;
   
           if (0 == namesz && ROFF_ig != tok) {
                   mandoc_msg(MANDOCERR_REQ_EMPTY, r->parse,
                       ln, ppos, roffs[tok].name);
                   return(ROFF_IGN);
         }          }
   
         roffnode_push(r, tok, name, ln, ppos);          roffnode_push(r, tok, name, ln, ppos);
Line 955  roff_block(ROFF_ARGS)
Line 1008  roff_block(ROFF_ARGS)
          * appended from roff_block_text() in multiline mode.           * appended from roff_block_text() in multiline mode.
          */           */
   
         if (ROFF_de == tok)          if (ROFF_de == tok || ROFF_dei == tok)
                 roff_setstr(r, name, "", 0);                  roff_setstrn(&r->strtab, name, namesz, "", 0, 0);
   
         if ('\0' == (*bufp)[pos])          if ('\0' == *cp)
                 return(ROFF_IGN);                  return(ROFF_IGN);
   
         /* If present, process the custom end-of-line marker. */          /* Get the custom end marker. */
   
         sv = pos;          iname = cp;
         while ((*bufp)[pos] && ! isspace((unsigned char)(*bufp)[pos]))          namesz = roff_getname(r, &cp, ln, ppos);
                 pos++;  
   
         /*          /* Resolve the end marker if it is indirect. */
          * Note: groff does NOT like escape characters in the input.  
          * Instead of detecting this, we're just going to let it fly and  
          * to hell with it.  
          */  
   
         assert(pos > sv);          if (namesz && (ROFF_dei == tok || ROFF_ami == tok)) {
         sz = (size_t)(pos - sv);                  if (NULL == (name = roff_getstrn(r, iname, namesz))) {
                           mandoc_vmsg(MANDOCERR_STR_UNDEF,
                               r->parse, ln, (int)(iname - *bufp),
                               "%.*s", (int)namesz, iname);
                           namesz = 0;
                   } else
                           namesz = strlen(name);
           } else
                   name = iname;
   
         if (1 == sz && '.' == (*bufp)[sv])          if (namesz)
                 return(ROFF_IGN);                  r->last->end = mandoc_strndup(name, namesz);
   
         r->last->end = mandoc_malloc(sz + 1);          if ('\0' != *cp)
                   mandoc_vmsg(MANDOCERR_ARG_EXCESS, r->parse,
                       ln, pos, ".%s ... %s", roffs[tok].name, cp);
   
         memcpy(r->last->end, *bufp + sv, sz);  
         r->last->end[(int)sz] = '\0';  
   
         if ((*bufp)[pos])  
                 mandoc_msg(MANDOCERR_ARGSLOST, r->parse, ln, pos, NULL);  
   
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
   
Line 1021  roff_block_sub(ROFF_ARGS)
Line 1073  roff_block_sub(ROFF_ARGS)
                                 i++;                                  i++;
   
                         pos = i;                          pos = i;
                         if (ROFF_MAX != roff_parse(r, *bufp, &pos))                          if (ROFF_MAX != roff_parse(r, *bufp, &pos, ln, ppos))
                                 return(ROFF_RERUN);                                  return(ROFF_RERUN);
                         return(ROFF_IGN);                          return(ROFF_IGN);
                 }                  }
Line 1032  roff_block_sub(ROFF_ARGS)
Line 1084  roff_block_sub(ROFF_ARGS)
          * pulling it out of the hashtable.           * pulling it out of the hashtable.
          */           */
   
         t = roff_parse(r, *bufp, &pos);          t = roff_parse(r, *bufp, &pos, ln, ppos);
   
         /*  
          * Macros other than block-end are only significant  
          * in `de' blocks; elsewhere, simply throw them away.  
          */  
         if (ROFF_cblock != t) {          if (ROFF_cblock != t) {
                 if (ROFF_de == tok)                  if (ROFF_ig != tok)
                         roff_setstr(r, r->last->name, *bufp + ppos, 2);                          roff_setstr(r, r->last->name, *bufp + ppos, 2);
                 return(ROFF_IGN);                  return(ROFF_IGN);
         }          }
Line 1052  static enum rofferr
Line 1100  static enum rofferr
 roff_block_text(ROFF_ARGS)  roff_block_text(ROFF_ARGS)
 {  {
   
         if (ROFF_de == tok)          if (ROFF_ig != tok)
                 roff_setstr(r, r->last->name, *bufp + pos, 2);                  roff_setstr(r, r->last->name, *bufp + pos, 2);
   
         return(ROFF_IGN);          return(ROFF_IGN);
Line 1067  roff_cond_sub(ROFF_ARGS)
Line 1115  roff_cond_sub(ROFF_ARGS)
   
         rr = r->last->rule;          rr = r->last->rule;
         roffnode_cleanscope(r);          roffnode_cleanscope(r);
         t = roff_parse(r, *bufp, &pos);          t = roff_parse(r, *bufp, &pos, ln, ppos);
   
         /*          /*
          * Fully handle known macros when they are structurally           * Fully handle known macros when they are structurally
Line 1201  out:
Line 1249  out:
  * or string condition.   * or string condition.
  */   */
 static int  static int
 roff_evalcond(const char *v, int *pos)  roff_evalcond(struct roff *r, int ln, const char *v, int *pos)
 {  {
         int      wanttrue, number;          int      wanttrue, number;
   
Line 1232  roff_evalcond(const char *v, int *pos)
Line 1280  roff_evalcond(const char *v, int *pos)
                 break;                  break;
         }          }
   
         if (roff_evalnum(v, pos, &number, 0))          if (roff_evalnum(r, ln, v, pos, &number, 0))
                 return((number > 0) == wanttrue);                  return((number > 0) == wanttrue);
         else          else
                 return(roff_evalstrcond(v, pos) == wanttrue);                  return(roff_evalstrcond(v, pos) == wanttrue);
Line 1261  roff_cond(ROFF_ARGS)
Line 1309  roff_cond(ROFF_ARGS)
   
         r->last->rule = ROFF_el == tok ?          r->last->rule = ROFF_el == tok ?
             (r->rstackpos < 0 ? 0 : r->rstack[r->rstackpos--]) :              (r->rstackpos < 0 ? 0 : r->rstack[r->rstackpos--]) :
             roff_evalcond(*bufp, &pos);              roff_evalcond(r, ln, *bufp, &pos);
   
         /*          /*
          * An if-else will put the NEGATION of the current evaluated           * An if-else will put the NEGATION of the current evaluated
Line 1269  roff_cond(ROFF_ARGS)
Line 1317  roff_cond(ROFF_ARGS)
          */           */
   
         if (ROFF_ie == tok) {          if (ROFF_ie == tok) {
                 if (r->rstackpos == RSTACK_MAX - 1) {                  if (r->rstackpos + 1 == r->rstacksz) {
                         mandoc_msg(MANDOCERR_MEM,                          r->rstacksz += 16;
                             r->parse, ln, ppos, NULL);                          r->rstack = mandoc_reallocarray(r->rstack,
                         return(ROFF_ERR);                              r->rstacksz, sizeof(int));
                 }                  }
                 r->rstack[++r->rstackpos] = !r->last->rule;                  r->rstack[++r->rstackpos] = !r->last->rule;
         }          }
Line 1311  roff_cond(ROFF_ARGS)
Line 1359  roff_cond(ROFF_ARGS)
          */           */
   
         if ('\0' == (*bufp)[pos])          if ('\0' == (*bufp)[pos])
                 mandoc_msg(MANDOCERR_NOARGS, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_COND_EMPTY, r->parse,
                       ln, ppos, roffs[tok].name);
   
         r->last->endspan = 1;          r->last->endspan = 1;
   
Line 1323  out:
Line 1372  out:
 static enum rofferr  static enum rofferr
 roff_ds(ROFF_ARGS)  roff_ds(ROFF_ARGS)
 {  {
         char            *name, *string;          char            *string;
           const char      *name;
           size_t           namesz;
   
         /*          /*
          * A symbol is named by the first word following the macro           * The first word is the name of the string.
          * invocation up to a space.  Its value is anything after the           * If it is empty or terminated by an escape sequence,
          * name's trailing whitespace and optional double-quote.  Thus,           * abort the `ds' request without defining anything.
          *  
          *  [.ds foo "bar  "     ]  
          *  
          * will have `bar  "     ' as its value.  
          */           */
   
         string = *bufp + pos;          name = string = *bufp + pos;
         name = roff_getname(r, &string, ln, pos);  
         if ('\0' == *name)          if ('\0' == *name)
                 return(ROFF_IGN);                  return(ROFF_IGN);
   
         /* Read past initial double-quote. */          namesz = roff_getname(r, &string, ln, pos);
           if ('\\' == name[namesz])
                   return(ROFF_IGN);
   
           /* Read past the initial double-quote, if any. */
         if ('"' == *string)          if ('"' == *string)
                 string++;                  string++;
   
         /* The rest is the value. */          /* The rest is the value. */
         roff_setstr(r, name, string, ROFF_as == tok);          roff_setstrn(&r->strtab, name, namesz, string, strlen(string),
               ROFF_as == tok);
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
   
Line 1424  roff_getop(const char *v, int *pos, char *res)
Line 1475  roff_getop(const char *v, int *pos, char *res)
  * or a single signed integer number.   * or a single signed integer number.
  */   */
 static int  static int
 roff_evalpar(const char *v, int *pos, int *res)  roff_evalpar(struct roff *r, int ln,
           const char *v, int *pos, int *res)
 {  {
   
         if ('(' != v[*pos])          if ('(' != v[*pos])
                 return(roff_getnum(v, pos, res));                  return(roff_getnum(v, pos, res));
   
         (*pos)++;          (*pos)++;
         if ( ! roff_evalnum(v, pos, res, 1))          if ( ! roff_evalnum(r, ln, v, pos, res, 1))
                 return(0);                  return(0);
   
         /*          /*
Line 1453  roff_evalpar(const char *v, int *pos, int *res)
Line 1505  roff_evalpar(const char *v, int *pos, int *res)
  * Proceed left to right, there is no concept of precedence.   * Proceed left to right, there is no concept of precedence.
  */   */
 static int  static int
 roff_evalnum(const char *v, int *pos, int *res, int skipwhite)  roff_evalnum(struct roff *r, int ln, const char *v,
           int *pos, int *res, int skipwhite)
 {  {
         int              mypos, operand2;          int              mypos, operand2;
         char             operator;          char             operator;
Line 1467  roff_evalnum(const char *v, int *pos, int *res, int sk
Line 1520  roff_evalnum(const char *v, int *pos, int *res, int sk
                 while (isspace((unsigned char)v[*pos]))                  while (isspace((unsigned char)v[*pos]))
                         (*pos)++;                          (*pos)++;
   
         if ( ! roff_evalpar(v, pos, res))          if ( ! roff_evalpar(r, ln, v, pos, res))
                 return(0);                  return(0);
   
         while (1) {          while (1) {
Line 1482  roff_evalnum(const char *v, int *pos, int *res, int sk
Line 1535  roff_evalnum(const char *v, int *pos, int *res, int sk
                         while (isspace((unsigned char)v[*pos]))                          while (isspace((unsigned char)v[*pos]))
                                 (*pos)++;                                  (*pos)++;
   
                 if ( ! roff_evalpar(v, pos, &operand2))                  if ( ! roff_evalpar(r, ln, v, pos, &operand2))
                         return(0);                          return(0);
   
                 if (skipwhite)                  if (skipwhite)
Line 1503  roff_evalnum(const char *v, int *pos, int *res, int sk
Line 1556  roff_evalnum(const char *v, int *pos, int *res, int sk
                         *res *= operand2;                          *res *= operand2;
                         break;                          break;
                 case '/':                  case '/':
                           if (0 == operand2) {
                                   mandoc_msg(MANDOCERR_DIVZERO,
                                           r->parse, ln, *pos, v);
                                   *res = 0;
                                   break;
                           }
                         *res /= operand2;                          *res /= operand2;
                         break;                          break;
                 case '%':                  case '%':
Line 1659  roff_freereg(struct roffreg *reg)
Line 1718  roff_freereg(struct roffreg *reg)
 static enum rofferr  static enum rofferr
 roff_nr(ROFF_ARGS)  roff_nr(ROFF_ARGS)
 {  {
         const char      *key;          char            *key, *val;
         char            *val;          size_t           keysz;
         int              iv;          int              iv;
         char             sign;          char             sign;
   
         val = *bufp + pos;          key = val = *bufp + pos;
         key = roff_getname(r, &val, ln, pos);          if ('\0' == *key)
                   return(ROFF_IGN);
   
           keysz = roff_getname(r, &val, ln, pos);
           if ('\\' == key[keysz])
                   return(ROFF_IGN);
           key[keysz] = '\0';
   
         sign = *val;          sign = *val;
         if ('+' == sign || '-' == sign)          if ('+' == sign || '-' == sign)
                 val++;                  val++;
   
         if (roff_evalnum(val, NULL, &iv, 0))          if (roff_evalnum(r, ln, val, NULL, &iv, 0))
                 roff_setreg(r, key, iv, sign);                  roff_setreg(r, key, iv, sign);
   
         return(ROFF_IGN);          return(ROFF_IGN);
Line 1681  static enum rofferr
Line 1746  static enum rofferr
 roff_rr(ROFF_ARGS)  roff_rr(ROFF_ARGS)
 {  {
         struct roffreg  *reg, **prev;          struct roffreg  *reg, **prev;
         const char      *name;          char            *name, *cp;
         char            *cp;          size_t           namesz;
   
         cp = *bufp + pos;          name = cp = *bufp + pos;
         name = roff_getname(r, &cp, ln, pos);          if ('\0' == *name)
                   return(ROFF_IGN);
           namesz = roff_getname(r, &cp, ln, pos);
           name[namesz] = '\0';
   
         prev = &r->regtab;          prev = &r->regtab;
         while (1) {          while (1) {
Line 1707  roff_rm(ROFF_ARGS)
Line 1775  roff_rm(ROFF_ARGS)
 {  {
         const char       *name;          const char       *name;
         char             *cp;          char             *cp;
           size_t            namesz;
   
         cp = *bufp + pos;          cp = *bufp + pos;
         while ('\0' != *cp) {          while ('\0' != *cp) {
                 name = roff_getname(r, &cp, ln, (int)(cp - *bufp));                  name = cp;
                 if ('\0' != *name)                  namesz = roff_getname(r, &cp, ln, (int)(cp - *bufp));
                         roff_setstr(r, name, NULL, 0);                  roff_setstrn(&r->strtab, name, namesz, NULL, 0, 0);
                   if ('\\' == name[namesz])
                           break;
         }          }
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
Line 1729  roff_it(ROFF_ARGS)
Line 1800  roff_it(ROFF_ARGS)
         len = strcspn(cp, " \t");          len = strcspn(cp, " \t");
         cp[len] = '\0';          cp[len] = '\0';
         if ((iv = mandoc_strntoi(cp, len, 10)) <= 0) {          if ((iv = mandoc_strntoi(cp, len, 10)) <= 0) {
                 mandoc_msg(MANDOCERR_NUMERIC, r->parse,                  mandoc_msg(MANDOCERR_IT_NONUM, r->parse,
                     ln, ppos, *bufp + 1);                      ln, ppos, *bufp + 1);
                 return(ROFF_IGN);                  return(ROFF_IGN);
         }          }
Line 1746  roff_Dd(ROFF_ARGS)
Line 1817  roff_Dd(ROFF_ARGS)
 {  {
         const char *const       *cp;          const char *const       *cp;
   
         if (0 == ((MPARSE_MDOC | MPARSE_QUICK) & r->options))          if ((r->options & (MPARSE_MDOC | MPARSE_QUICK)) == 0)
                 for (cp = __mdoc_reserved; *cp; cp++)                  for (cp = __mdoc_reserved; *cp; cp++)
                         roff_setstr(r, *cp, NULL, 0);                          roff_setstr(r, *cp, NULL, 0);
   
           if (r->format == 0)
                   r->format = MPARSE_MDOC;
   
         return(ROFF_CONT);          return(ROFF_CONT);
 }  }
   
Line 1758  roff_TH(ROFF_ARGS)
Line 1832  roff_TH(ROFF_ARGS)
 {  {
         const char *const       *cp;          const char *const       *cp;
   
         if (0 == (MPARSE_QUICK & r->options))          if ((r->options & MPARSE_QUICK) == 0)
                 for (cp = __man_reserved; *cp; cp++)                  for (cp = __man_reserved; *cp; cp++)
                         roff_setstr(r, *cp, NULL, 0);                          roff_setstr(r, *cp, NULL, 0);
   
           if (r->format == 0)
                   r->format = MPARSE_MAN;
   
         return(ROFF_CONT);          return(ROFF_CONT);
 }  }
   
Line 1770  roff_TE(ROFF_ARGS)
Line 1847  roff_TE(ROFF_ARGS)
 {  {
   
         if (NULL == r->tbl)          if (NULL == r->tbl)
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "TE");
         else          else
                 tbl_end(&r->tbl);                  tbl_end(&r->tbl);
   
Line 1782  roff_T_(ROFF_ARGS)
Line 1860  roff_T_(ROFF_ARGS)
 {  {
   
         if (NULL == r->tbl)          if (NULL == r->tbl)
                 mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "T&");
         else          else
                 tbl_restart(ppos, ln, r->tbl);                  tbl_restart(ppos, ln, r->tbl);
   
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
   
 #if 0  /*
 static int   * Handle in-line equation delimiters.
 roff_closeeqn(struct roff *r)   */
   static enum rofferr
   roff_eqndelim(struct roff *r, char **bufp, size_t *szp, int pos)
 {  {
           char            *cp1, *cp2;
           const char      *bef_pr, *bef_nl, *mac, *aft_nl, *aft_pr;
   
         return(r->eqn && ROFF_EQN == eqn_end(&r->eqn) ? 1 : 0);          /*
            * Outside equations, look for an opening delimiter.
            * If we are inside an equation, we already know it is
            * in-line, or this function wouldn't have been called;
            * so look for a closing delimiter.
            */
   
           cp1 = *bufp + pos;
           cp2 = strchr(cp1, r->eqn == NULL ?
               r->last_eqn->odelim : r->last_eqn->cdelim);
           if (cp2 == NULL)
                   return(ROFF_CONT);
   
           *cp2++ = '\0';
           bef_pr = bef_nl = aft_nl = aft_pr = "";
   
           /* Handle preceding text, protecting whitespace. */
   
           if (**bufp != '\0') {
                   if (r->eqn == NULL)
                           bef_pr = "\\&";
                   bef_nl = "\n";
           }
   
           /*
            * Prepare replacing the delimiter with an equation macro
            * and drop leading white space from the equation.
            */
   
           if (r->eqn == NULL) {
                   while (*cp2 == ' ')
                           cp2++;
                   mac = ".EQ";
           } else
                   mac = ".EN";
   
           /* Handle following text, protecting whitespace. */
   
           if (*cp2 != '\0') {
                   aft_nl = "\n";
                   if (r->eqn != NULL)
                           aft_pr = "\\&";
           }
   
           /* Do the actual replacement. */
   
           *szp = mandoc_asprintf(&cp1, "%s%s%s%s%s%s%s", *bufp,
               bef_pr, bef_nl, mac, aft_nl, aft_pr, cp2) + 1;
           free(*bufp);
           *bufp = cp1;
   
           /* Toggle the in-line state of the eqn subsystem. */
   
           r->eqn_inline = r->eqn == NULL;
           return(ROFF_REPARSE);
 }  }
 #endif  
   
 static void  static enum rofferr
 roff_openeqn(struct roff *r, const char *name, int line,  roff_EQ(ROFF_ARGS)
                 int offs, const char *buf)  
 {  {
         struct eqn_node *e;          struct eqn_node *e;
         int              poff;  
   
         assert(NULL == r->eqn);          assert(NULL == r->eqn);
         e = eqn_alloc(name, offs, line, r->parse);          e = eqn_alloc(ppos, ln, r->parse);
   
         if (r->last_eqn)          if (r->last_eqn) {
                 r->last_eqn->next = e;                  r->last_eqn->next = e;
         else                  e->delim = r->last_eqn->delim;
                   e->odelim = r->last_eqn->odelim;
                   e->cdelim = r->last_eqn->cdelim;
           } else
                 r->first_eqn = r->last_eqn = e;                  r->first_eqn = r->last_eqn = e;
   
         r->eqn = r->last_eqn = e;          r->eqn = r->last_eqn = e;
   
         if (buf) {          if ((*bufp)[pos])
                 poff = 0;                  mandoc_vmsg(MANDOCERR_ARG_SKIP, r->parse, ln, pos,
                 eqn_read(&r->eqn, line, buf, offs, &poff);                      ".EQ %s", *bufp + pos);
         }  
 }  
   
 static enum rofferr  
 roff_EQ(ROFF_ARGS)  
 {  
   
         roff_openeqn(r, *bufp + pos, ln, ppos, NULL);  
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
   
Line 1833  static enum rofferr
Line 1963  static enum rofferr
 roff_EN(ROFF_ARGS)  roff_EN(ROFF_ARGS)
 {  {
   
         mandoc_msg(MANDOCERR_NOSCOPE, r->parse, ln, ppos, NULL);          mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse, ln, ppos, "EN");
         return(ROFF_IGN);          return(ROFF_IGN);
 }  }
   
Line 1843  roff_TS(ROFF_ARGS)
Line 1973  roff_TS(ROFF_ARGS)
         struct tbl_node *tbl;          struct tbl_node *tbl;
   
         if (r->tbl) {          if (r->tbl) {
                 mandoc_msg(MANDOCERR_SCOPEBROKEN, r->parse, ln, ppos, NULL);                  mandoc_msg(MANDOCERR_BLK_BROKEN, r->parse,
                       ln, ppos, "TS breaks TS");
                 tbl_end(&r->tbl);                  tbl_end(&r->tbl);
         }          }
   
Line 1895  roff_tr(ROFF_ARGS)
Line 2026  roff_tr(ROFF_ARGS)
                 if ('\\' == *first) {                  if ('\\' == *first) {
                         esc = mandoc_escape(&p, NULL, NULL);                          esc = mandoc_escape(&p, NULL, NULL);
                         if (ESCAPE_ERROR == esc) {                          if (ESCAPE_ERROR == esc) {
                                 mandoc_msg(MANDOCERR_BADESCAPE,                                  mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                                     r->parse, ln,                                      ln, (int)(p - *bufp), first);
                                     (int)(p - *bufp), NULL);  
                                 return(ROFF_IGN);                                  return(ROFF_IGN);
                         }                          }
                         fsz = (size_t)(p - first);                          fsz = (size_t)(p - first);
Line 1907  roff_tr(ROFF_ARGS)
Line 2037  roff_tr(ROFF_ARGS)
                 if ('\\' == *second) {                  if ('\\' == *second) {
                         esc = mandoc_escape(&p, NULL, NULL);                          esc = mandoc_escape(&p, NULL, NULL);
                         if (ESCAPE_ERROR == esc) {                          if (ESCAPE_ERROR == esc) {
                                 mandoc_msg(MANDOCERR_BADESCAPE,                                  mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                                     r->parse, ln,                                      ln, (int)(p - *bufp), second);
                                     (int)(p - *bufp), NULL);  
                                 return(ROFF_IGN);                                  return(ROFF_IGN);
                         }                          }
                         ssz = (size_t)(p - second);                          ssz = (size_t)(p - second);
Line 1943  roff_so(ROFF_ARGS)
Line 2072  roff_so(ROFF_ARGS)
 {  {
         char *name;          char *name;
   
         mandoc_msg(MANDOCERR_SO, r->parse, ln, ppos, NULL);          name = *bufp + pos;
           mandoc_vmsg(MANDOCERR_SO, r->parse, ln, ppos, "so %s", name);
   
         /*          /*
          * Handle `so'.  Be EXTREMELY careful, as we shouldn't be           * Handle `so'.  Be EXTREMELY careful, as we shouldn't be
Line 1952  roff_so(ROFF_ARGS)
Line 2082  roff_so(ROFF_ARGS)
          * or using absolute paths.           * or using absolute paths.
          */           */
   
         name = *bufp + pos;  
         if ('/' == *name || strstr(name, "../") || strstr(name, "/..")) {          if ('/' == *name || strstr(name, "../") || strstr(name, "/..")) {
                 mandoc_msg(MANDOCERR_SOPATH, r->parse, ln, pos, NULL);                  mandoc_vmsg(MANDOCERR_SO_PATH, r->parse, ln, ppos,
                       ".so %s", name);
                 return(ROFF_ERR);                  return(ROFF_ERR);
         }          }
   
Line 1990  roff_userdef(ROFF_ARGS)
Line 2120  roff_userdef(ROFF_ARGS)
                         cp += 2;                          cp += 2;
                         continue;                          continue;
                 }                  }
                   *cp = '\0';
                 *szp = strlen(n1) - 3 + strlen(arg[i]) + 1;                  *szp = mandoc_asprintf(&n2, "%s%s%s",
                 n2 = mandoc_malloc(*szp);                      n1, arg[i], cp + 3) + 1;
   
                 strlcpy(n2, n1, (size_t)(cp - n1 + 1));  
                 strlcat(n2, arg[i], *szp);  
                 strlcat(n2, cp + 3, *szp);  
   
                 cp = n2 + (cp - n1);                  cp = n2 + (cp - n1);
                 free(n1);                  free(n1);
                 n1 = n2;                  n1 = n2;
Line 2016  roff_userdef(ROFF_ARGS)
Line 2141  roff_userdef(ROFF_ARGS)
            ROFF_REPARSE : ROFF_APPEND);             ROFF_REPARSE : ROFF_APPEND);
 }  }
   
 static char *  static size_t
 roff_getname(struct roff *r, char **cpp, int ln, int pos)  roff_getname(struct roff *r, char **cpp, int ln, int pos)
 {  {
         char     *name, *cp;          char     *name, *cp;
           size_t    namesz;
   
         name = *cpp;          name = *cpp;
         if ('\0' == *name)          if ('\0' == *name)
                 return(name);                  return(0);
   
         /* Read until end of name. */          /* Read until end of name and terminate it with NUL. */
         for (cp = name; '\0' != *cp && ' ' != *cp; cp++) {          for (cp = name; 1; cp++) {
                   if ('\0' == *cp || ' ' == *cp) {
                           namesz = cp - name;
                           break;
                   }
                 if ('\\' != *cp)                  if ('\\' != *cp)
                         continue;                          continue;
                   namesz = cp - name;
                   if ('{' == cp[1] || '}' == cp[1])
                           break;
                 cp++;                  cp++;
                 if ('\\' == *cp)                  if ('\\' == *cp)
                         continue;                          continue;
                 mandoc_msg(MANDOCERR_NAMESC, r->parse, ln, pos, NULL);                  mandoc_vmsg(MANDOCERR_NAMESC, r->parse, ln, pos,
                 *cp = '\0';                      "%.*s", (int)(cp - name + 1), name);
                 name = cp;                  mandoc_escape((const char **)&cp, NULL, NULL);
                   break;
         }          }
   
         /* Nil-terminate name. */  
         if ('\0' != *cp)  
                 *(cp++) = '\0';  
   
         /* Read past spaces. */          /* Read past spaces. */
         while (' ' == *cp)          while (' ' == *cp)
                 cp++;                  cp++;
   
         *cpp = cp;          *cpp = cp;
         return(name);          return(namesz);
 }  }
   
 /*  /*
Line 2077  roff_setstrn(struct roffkv **r, const char *name, size
Line 2207  roff_setstrn(struct roffkv **r, const char *name, size
         /* Search for an existing string with the same name. */          /* Search for an existing string with the same name. */
         n = *r;          n = *r;
   
         while (n && strcmp(name, n->key.p))          while (n && (namesz != n->key.sz ||
                           strncmp(n->key.p, name, namesz)))
                 n = n->next;                  n = n->next;
   
         if (NULL == n) {          if (NULL == n) {
Line 2272  roff_strdup(const struct roff *r, const char *p)
Line 2403  roff_strdup(const struct roff *r, const char *p)
   
         res[(int)ssz] = '\0';          res[(int)ssz] = '\0';
         return(res);          return(res);
   }
   
   int
   roff_getformat(const struct roff *r)
   {
   
           return(r->format);
 }  }
   
 /*  /*

Legend:
Removed from v.1.208  
changed lines
  Added in v.1.237

CVSweb