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

Diff for /mandoc/roff.c between version 1.26 and 1.258

version 1.26, 2008/12/01 09:25:18 version 1.258, 2015/01/28 17:32:07
Line 1 
Line 1 
 /* $Id$ */  /*      $Id$ */
 /*  /*
  * Copyright (c) 2008 Kristaps Dzonsons <kristaps@kth.se>   * Copyright (c) 2010, 2011, 2012, 2014 Kristaps Dzonsons <kristaps@bsd.lv>
    * Copyright (c) 2010-2015 Ingo Schwarze <schwarze@openbsd.org>
  *   *
  * Permission to use, copy, modify, and distribute this software for any   * Permission to use, copy, modify, and distribute this software for any
  * purpose with or without fee is hereby granted, provided that the   * purpose with or without fee is hereby granted, provided that the above
  * above copyright notice and this permission notice appear in all   * copyright notice and this permission notice appear in all copies.
  * copies.  
  *   *
  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL   * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES
  * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED   * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE   * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR
  * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL   * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR   * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER   * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR   * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  * PERFORMANCE OF THIS SOFTWARE.  
  */   */
   #include "config.h"
   
   #include <sys/types.h>
   
 #include <assert.h>  #include <assert.h>
 #include <ctype.h>  #include <ctype.h>
 #include <err.h>  #include <limits.h>
 #include <stdarg.h>  
 #include <stdlib.h>  
 #include <stdio.h>  #include <stdio.h>
   #include <stdlib.h>
 #include <string.h>  #include <string.h>
 #include <time.h>  
   
 #include "libmdocml.h"  #include "mandoc.h"
 #include "private.h"  #include "mandoc_aux.h"
   #include "libmandoc.h"
   #include "libroff.h"
   
 /* FIXME: warn if Pp occurs before/after Sh etc. (see mdoc.samples). */  /* Maximum number of nested if-else conditionals. */
   #define RSTACK_MAX      128
   
 /* FIXME: warn about "X section only" macros. */  /* Maximum number of string expansions per line, to break infinite loops. */
   #define EXPAND_LIMIT    1000
   
 /* FIXME: warn about empty lists. */  enum    rofft {
           ROFF_ab,
           ROFF_ad,
           ROFF_af,
           ROFF_aln,
           ROFF_als,
           ROFF_am,
           ROFF_am1,
           ROFF_ami,
           ROFF_ami1,
           ROFF_as,
           ROFF_as1,
           ROFF_asciify,
           ROFF_backtrace,
           ROFF_bd,
           ROFF_bleedat,
           ROFF_blm,
           ROFF_box,
           ROFF_boxa,
           ROFF_bp,
           ROFF_BP,
           /* MAN_br, MDOC_br */
           ROFF_break,
           ROFF_breakchar,
           ROFF_brnl,
           ROFF_brp,
           ROFF_brpnl,
           ROFF_c2,
           ROFF_cc,
           ROFF_ce,
           ROFF_cf,
           ROFF_cflags,
           ROFF_ch,
           ROFF_char,
           ROFF_chop,
           ROFF_class,
           ROFF_close,
           ROFF_CL,
           ROFF_color,
           ROFF_composite,
           ROFF_continue,
           ROFF_cp,
           ROFF_cropat,
           ROFF_cs,
           ROFF_cu,
           ROFF_da,
           ROFF_dch,
           ROFF_Dd,
           ROFF_de,
           ROFF_de1,
           ROFF_defcolor,
           ROFF_dei,
           ROFF_dei1,
           ROFF_device,
           ROFF_devicem,
           ROFF_di,
           ROFF_do,
           ROFF_ds,
           ROFF_ds1,
           ROFF_dwh,
           ROFF_dt,
           ROFF_ec,
           ROFF_ecr,
           ROFF_ecs,
           ROFF_el,
           ROFF_em,
           ROFF_EN,
           ROFF_eo,
           ROFF_EP,
           ROFF_EQ,
           ROFF_errprint,
           ROFF_ev,
           ROFF_evc,
           ROFF_ex,
           ROFF_fallback,
           ROFF_fam,
           ROFF_fc,
           ROFF_fchar,
           ROFF_fcolor,
           ROFF_fdeferlig,
           ROFF_feature,
           /* MAN_fi; ignored in mdoc(7) */
           ROFF_fkern,
           ROFF_fl,
           ROFF_flig,
           ROFF_fp,
           ROFF_fps,
           ROFF_fschar,
           ROFF_fspacewidth,
           ROFF_fspecial,
           /* MAN_ft; ignored in mdoc(7) */
           ROFF_ftr,
           ROFF_fzoom,
           ROFF_gcolor,
           ROFF_hc,
           ROFF_hcode,
           ROFF_hidechar,
           ROFF_hla,
           ROFF_hlm,
           ROFF_hpf,
           ROFF_hpfa,
           ROFF_hpfcode,
           ROFF_hw,
           ROFF_hy,
           ROFF_hylang,
           ROFF_hylen,
           ROFF_hym,
           ROFF_hypp,
           ROFF_hys,
           ROFF_ie,
           ROFF_if,
           ROFF_ig,
           /* MAN_in; ignored in mdoc(7) */
           ROFF_index,
           ROFF_it,
           ROFF_itc,
           ROFF_IX,
           ROFF_kern,
           ROFF_kernafter,
           ROFF_kernbefore,
           ROFF_kernpair,
           ROFF_lc,
           ROFF_lc_ctype,
           ROFF_lds,
           ROFF_length,
           ROFF_letadj,
           ROFF_lf,
           ROFF_lg,
           ROFF_lhang,
           ROFF_linetabs,
           /* MAN_ll, MDOC_ll */
           ROFF_lnr,
           ROFF_lnrf,
           ROFF_lpfx,
           ROFF_ls,
           ROFF_lsm,
           ROFF_lt,
           ROFF_mc,
           ROFF_mediasize,
           ROFF_minss,
           ROFF_mk,
           ROFF_mso,
           ROFF_na,
           ROFF_ne,
           /* MAN_nf; ignored in mdoc(7) */
           ROFF_nh,
           ROFF_nhychar,
           ROFF_nm,
           ROFF_nn,
           ROFF_nop,
           ROFF_nr,
           ROFF_nrf,
           ROFF_nroff,
           ROFF_ns,
           ROFF_nx,
           ROFF_open,
           ROFF_opena,
           ROFF_os,
           ROFF_output,
           ROFF_padj,
           ROFF_papersize,
           ROFF_pc,
           ROFF_pev,
           ROFF_pi,
           ROFF_PI,
           ROFF_pl,
           ROFF_pm,
           ROFF_pn,
           ROFF_pnr,
           ROFF_po,
           ROFF_ps,
           ROFF_psbb,
           ROFF_pshape,
           ROFF_pso,
           ROFF_ptr,
           ROFF_pvs,
           ROFF_rchar,
           ROFF_rd,
           ROFF_recursionlimit,
           ROFF_return,
           ROFF_rfschar,
           ROFF_rhang,
           ROFF_rj,
           ROFF_rm,
           ROFF_rn,
           ROFF_rnn,
           ROFF_rr,
           ROFF_rs,
           ROFF_rt,
           ROFF_schar,
           ROFF_sentchar,
           ROFF_shc,
           ROFF_shift,
           ROFF_sizes,
           ROFF_so,
           /* MAN_sp, MDOC_sp */
           ROFF_spacewidth,
           ROFF_special,
           ROFF_spreadwarn,
           ROFF_ss,
           ROFF_sty,
           ROFF_substring,
           ROFF_sv,
           ROFF_sy,
           ROFF_T_,
           ROFF_ta,
           ROFF_tc,
           ROFF_TE,
           ROFF_TH,
           ROFF_ti,
           ROFF_tkf,
           ROFF_tl,
           ROFF_tm,
           ROFF_tm1,
           ROFF_tmc,
           ROFF_tr,
           ROFF_track,
           ROFF_transchar,
           ROFF_trf,
           ROFF_trimat,
           ROFF_trin,
           ROFF_trnt,
           ROFF_troff,
           ROFF_TS,
           ROFF_uf,
           ROFF_ul,
           ROFF_unformat,
           ROFF_unwatch,
           ROFF_unwatchn,
           ROFF_vpt,
           ROFF_vs,
           ROFF_warn,
           ROFF_warnscale,
           ROFF_watch,
           ROFF_watchlength,
           ROFF_watchn,
           ROFF_wh,
           ROFF_while,
           ROFF_write,
           ROFF_writec,
           ROFF_writem,
           ROFF_xflag,
           ROFF_cblock,
           ROFF_USERDEF,
           ROFF_MAX
   };
   
 /* FIXME: roff_layout and roff_text have identical-ish lower bodies. */  /*
    * An incredibly-simple string buffer.
    */
   struct  roffstr {
           char            *p; /* nil-terminated buffer */
           size_t           sz; /* saved strlen(p) */
   };
   
 /* FIXME: NAME section needs specific elements. */  /*
    * A key-value roffstr pair as part of a singly-linked list.
    */
   struct  roffkv {
           struct roffstr   key;
           struct roffstr   val;
           struct roffkv   *next; /* next in list */
   };
   
 #define ROFF_MAXARG       32  /*
    * A single number register as part of a singly-linked list.
    */
   struct  roffreg {
           struct roffstr   key;
           int              val;
           struct roffreg  *next;
   };
   
 enum    roffd {  struct  roff {
         ROFF_ENTER = 0,          struct mparse   *parse; /* parse point */
         ROFF_EXIT          const struct mchars *mchars; /* character table */
           struct roffnode *last; /* leaf of stack */
           int             *rstack; /* stack of inverted `ie' values */
           struct roffreg  *regtab; /* number registers */
           struct roffkv   *strtab; /* user-defined strings & macros */
           struct roffkv   *xmbtab; /* multi-byte trans table (`tr') */
           struct roffstr  *xtab; /* single-byte trans table (`tr') */
           const char      *current_string; /* value of last called user macro */
           struct tbl_node *first_tbl; /* first table parsed */
           struct tbl_node *last_tbl; /* last table parsed */
           struct tbl_node *tbl; /* current table being parsed */
           struct eqn_node *last_eqn; /* last equation parsed */
           struct eqn_node *first_eqn; /* first equation 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 */
 };  };
   
 enum    rofftype {  struct  roffnode {
         ROFF_COMMENT,          enum rofft       tok; /* type of node */
         ROFF_TEXT,          struct roffnode *parent; /* up one in stack */
         ROFF_LAYOUT,          int              line; /* parse line */
         ROFF_SPECIAL          int              col; /* parse col */
           char            *name; /* node name, e.g. macro name */
           char            *end; /* end-rules: custom token */
           int              endspan; /* end-rules: next-line or infty */
           int              rule; /* current evaluation rule */
 };  };
   
 #define ROFFCALL_ARGS \  #define ROFF_ARGS        struct roff *r, /* parse ctx */ \
         int tok, struct rofftree *tree, \                           enum rofft tok, /* tok of macro */ \
         char *argv[], enum roffd type                           struct buf *buf, /* input buffer */ \
                            int ln, /* parse line */ \
                            int ppos, /* original pos in buffer */ \
                            int pos, /* current pos in buffer */ \
                            int *offs /* reset offset of buffer data */
   
 struct  rofftree;  typedef enum rofferr (*roffproc)(ROFF_ARGS);
   
 struct  rofftok {  struct  roffmac {
         int             (*cb)(ROFFCALL_ARGS);   /* Callback. */          const char      *name; /* macro name */
         const int        *args;                 /* Args (or NULL). */          roffproc         proc; /* process new macro */
         const int        *parents;          roffproc         text; /* process as child text of macro */
         const int        *children;          roffproc         sub; /* process as child of macro */
         int               ctx;          int              flags;
         enum rofftype     type;                 /* Type of macro. */  #define ROFFMAC_STRUCT  (1 << 0) /* always interpret */
         int               flags;          struct roffmac  *next;
 #define ROFF_PARSED      (1 << 0)               /* "Parsed". */  
 #define ROFF_CALLABLE    (1 << 1)               /* "Callable". */  
 #define ROFF_SHALLOW     (1 << 2)               /* Nesting block. */  
 };  };
   
 struct  roffarg {  struct  predef {
         int               flags;          const char      *name; /* predefined input name */
 #define ROFF_VALUE       (1 << 0)               /* Has a value. */          const char      *str; /* replacement symbol */
 };  };
   
 struct  roffnode {  #define PREDEF(__name, __str) \
         int               tok;                  /* Token id. */          { (__name), (__str) },
         struct roffnode  *parent;               /* Parent (or NULL). */  
 };  
   
 struct  rofftree {  static  enum rofft       roffhash_find(const char *, size_t);
         struct roffnode  *last;                 /* Last parsed node. */  static  void             roffhash_init(void);
         char             *cur;  static  void             roffnode_cleanscope(struct roff *);
   static  void             roffnode_pop(struct roff *);
   static  void             roffnode_push(struct roff *, enum rofft,
                                   const char *, int, int);
   static  enum rofferr     roff_block(ROFF_ARGS);
   static  enum rofferr     roff_block_text(ROFF_ARGS);
   static  enum rofferr     roff_block_sub(ROFF_ARGS);
   static  enum rofferr     roff_brp(ROFF_ARGS);
   static  enum rofferr     roff_cblock(ROFF_ARGS);
   static  enum rofferr     roff_cc(ROFF_ARGS);
   static  void             roff_ccond(struct roff *, int, int);
   static  enum rofferr     roff_cond(ROFF_ARGS);
   static  enum rofferr     roff_cond_text(ROFF_ARGS);
   static  enum rofferr     roff_cond_sub(ROFF_ARGS);
   static  enum rofferr     roff_ds(ROFF_ARGS);
   static  enum rofferr     roff_eqndelim(struct roff *, struct buf *, 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  void             roff_free1(struct roff *);
   static  void             roff_freereg(struct roffreg *);
   static  void             roff_freestr(struct roffkv *);
   static  size_t           roff_getname(struct roff *, char **, int, int);
   static  int              roff_getnum(const char *, int *, int *);
   static  int              roff_getop(const char *, int *, char *);
   static  int              roff_getregn(const struct roff *,
                                   const char *, size_t);
   static  int              roff_getregro(const char *name);
   static  const char      *roff_getstrn(const struct roff *,
                                   const char *, size_t);
   static  enum rofferr     roff_insec(ROFF_ARGS);
   static  enum rofferr     roff_it(ROFF_ARGS);
   static  enum rofferr     roff_line_ignore(ROFF_ARGS);
   static  enum rofferr     roff_nr(ROFF_ARGS);
   static  enum rofft       roff_parse(struct roff *, char *, int *,
                                   int, int);
   static  enum rofferr     roff_parsetext(struct buf *, int, int *);
   static  enum rofferr     roff_res(struct roff *, struct buf *, int, int);
   static  enum rofferr     roff_rm(ROFF_ARGS);
   static  enum rofferr     roff_rr(ROFF_ARGS);
   static  void             roff_setstr(struct roff *,
                                   const char *, const char *, int);
   static  void             roff_setstrn(struct roffkv **, const char *,
                                   size_t, const char *, size_t, int);
   static  enum rofferr     roff_so(ROFF_ARGS);
   static  enum rofferr     roff_tr(ROFF_ARGS);
   static  enum rofferr     roff_Dd(ROFF_ARGS);
   static  enum rofferr     roff_TH(ROFF_ARGS);
   static  enum rofferr     roff_TE(ROFF_ARGS);
   static  enum rofferr     roff_TS(ROFF_ARGS);
   static  enum rofferr     roff_EQ(ROFF_ARGS);
   static  enum rofferr     roff_EN(ROFF_ARGS);
   static  enum rofferr     roff_T_(ROFF_ARGS);
   static  enum rofferr     roff_unsupp(ROFF_ARGS);
   static  enum rofferr     roff_userdef(ROFF_ARGS);
   
         time_t            date;                 /* `Dd' results. */  /* See roffhash_find() */
         char              os[64];               /* `Os' results. */  
         char              title[64];            /* `Dt' results. */  
         char              section[64];          /* `Dt' results. */  
         char              volume[64];           /* `Dt' results. */  
   
         int               state;  #define ASCII_HI         126
 #define ROFF_PRELUDE     (1 << 1)               /* In roff prelude. */  #define ASCII_LO         33
 #define ROFF_PRELUDE_Os  (1 << 2)               /* `Os' is parsed. */  #define HASHWIDTH       (ASCII_HI - ASCII_LO + 1)
 #define ROFF_PRELUDE_Dt  (1 << 3)               /* `Dt' is parsed. */  
 #define ROFF_PRELUDE_Dd  (1 << 4)               /* `Dd' is parsed. */  
 #define ROFF_BODY        (1 << 5)               /* In roff body. */  
   
         struct roffcb     cb;  static  struct roffmac  *hash[HASHWIDTH];
         void             *arg;  
   static  struct roffmac   roffs[ROFF_MAX] = {
           { "ab", roff_unsupp, NULL, NULL, 0, NULL },
           { "ad", roff_line_ignore, NULL, NULL, 0, NULL },
           { "af", roff_line_ignore, NULL, NULL, 0, NULL },
           { "aln", roff_unsupp, NULL, NULL, 0, NULL },
           { "als", roff_unsupp, NULL, NULL, 0, NULL },
           { "am", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "am1", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "ami", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "ami1", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "as", roff_ds, NULL, NULL, 0, NULL },
           { "as1", roff_ds, NULL, NULL, 0, NULL },
           { "asciify", roff_unsupp, NULL, NULL, 0, NULL },
           { "backtrace", roff_line_ignore, NULL, NULL, 0, NULL },
           { "bd", roff_line_ignore, NULL, NULL, 0, NULL },
           { "bleedat", roff_line_ignore, NULL, NULL, 0, NULL },
           { "blm", roff_unsupp, NULL, NULL, 0, NULL },
           { "box", roff_unsupp, NULL, NULL, 0, NULL },
           { "boxa", roff_unsupp, NULL, NULL, 0, NULL },
           { "bp", roff_line_ignore, NULL, NULL, 0, NULL },
           { "BP", roff_unsupp, NULL, NULL, 0, NULL },
           { "break", roff_unsupp, NULL, NULL, 0, NULL },
           { "breakchar", roff_line_ignore, NULL, NULL, 0, NULL },
           { "brnl", roff_line_ignore, NULL, NULL, 0, NULL },
           { "brp", roff_brp, NULL, NULL, 0, NULL },
           { "brpnl", roff_line_ignore, NULL, NULL, 0, NULL },
           { "c2", roff_unsupp, NULL, NULL, 0, NULL },
           { "cc", roff_cc, NULL, NULL, 0, NULL },
           { "ce", roff_line_ignore, NULL, NULL, 0, NULL },
           { "cf", roff_insec, NULL, NULL, 0, NULL },
           { "cflags", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ch", roff_line_ignore, NULL, NULL, 0, NULL },
           { "char", roff_unsupp, NULL, NULL, 0, NULL },
           { "chop", roff_unsupp, NULL, NULL, 0, NULL },
           { "class", roff_line_ignore, NULL, NULL, 0, NULL },
           { "close", roff_insec, NULL, NULL, 0, NULL },
           { "CL", roff_unsupp, NULL, NULL, 0, NULL },
           { "color", roff_line_ignore, NULL, NULL, 0, NULL },
           { "composite", roff_unsupp, NULL, NULL, 0, NULL },
           { "continue", roff_unsupp, NULL, NULL, 0, NULL },
           { "cp", roff_line_ignore, NULL, NULL, 0, NULL },
           { "cropat", roff_line_ignore, NULL, NULL, 0, NULL },
           { "cs", roff_line_ignore, NULL, NULL, 0, NULL },
           { "cu", roff_line_ignore, NULL, NULL, 0, NULL },
           { "da", roff_unsupp, NULL, NULL, 0, NULL },
           { "dch", roff_unsupp, NULL, NULL, 0, NULL },
           { "Dd", roff_Dd, NULL, NULL, 0, NULL },
           { "de", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "de1", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "defcolor", roff_line_ignore, NULL, NULL, 0, NULL },
           { "dei", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "dei1", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "device", roff_unsupp, NULL, NULL, 0, NULL },
           { "devicem", roff_unsupp, NULL, NULL, 0, NULL },
           { "di", roff_unsupp, NULL, NULL, 0, NULL },
           { "do", roff_unsupp, NULL, NULL, 0, NULL },
           { "ds", roff_ds, NULL, NULL, 0, NULL },
           { "ds1", roff_ds, NULL, NULL, 0, NULL },
           { "dwh", roff_unsupp, NULL, NULL, 0, NULL },
           { "dt", roff_unsupp, NULL, NULL, 0, NULL },
           { "ec", roff_unsupp, NULL, NULL, 0, NULL },
           { "ecr", roff_unsupp, NULL, NULL, 0, NULL },
           { "ecs", roff_unsupp, NULL, NULL, 0, NULL },
           { "el", roff_cond, roff_cond_text, roff_cond_sub, ROFFMAC_STRUCT, NULL },
           { "em", roff_unsupp, NULL, NULL, 0, NULL },
           { "EN", roff_EN, NULL, NULL, 0, NULL },
           { "eo", roff_unsupp, NULL, NULL, 0, NULL },
           { "EP", roff_unsupp, NULL, NULL, 0, NULL },
           { "EQ", roff_EQ, NULL, NULL, 0, NULL },
           { "errprint", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ev", roff_unsupp, NULL, NULL, 0, NULL },
           { "evc", roff_unsupp, NULL, NULL, 0, NULL },
           { "ex", roff_unsupp, NULL, NULL, 0, NULL },
           { "fallback", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fam", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fc", roff_unsupp, NULL, NULL, 0, NULL },
           { "fchar", roff_unsupp, NULL, NULL, 0, NULL },
           { "fcolor", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fdeferlig", roff_line_ignore, NULL, NULL, 0, NULL },
           { "feature", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fkern", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fl", roff_line_ignore, NULL, NULL, 0, NULL },
           { "flig", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fp", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fps", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fschar", roff_unsupp, NULL, NULL, 0, NULL },
           { "fspacewidth", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fspecial", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ftr", roff_line_ignore, NULL, NULL, 0, NULL },
           { "fzoom", roff_line_ignore, NULL, NULL, 0, NULL },
           { "gcolor", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hc", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hcode", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hidechar", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hla", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hlm", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hpf", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hpfa", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hpfcode", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hw", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hy", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hylang", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hylen", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hym", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hypp", roff_line_ignore, NULL, NULL, 0, NULL },
           { "hys", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ie", roff_cond, roff_cond_text, roff_cond_sub, ROFFMAC_STRUCT, NULL },
           { "if", roff_cond, roff_cond_text, roff_cond_sub, ROFFMAC_STRUCT, NULL },
           { "ig", roff_block, roff_block_text, roff_block_sub, 0, NULL },
           { "index", roff_unsupp, NULL, NULL, 0, NULL },
           { "it", roff_it, NULL, NULL, 0, NULL },
           { "itc", roff_unsupp, NULL, NULL, 0, NULL },
           { "IX", roff_line_ignore, NULL, NULL, 0, NULL },
           { "kern", roff_line_ignore, NULL, NULL, 0, NULL },
           { "kernafter", roff_line_ignore, NULL, NULL, 0, NULL },
           { "kernbefore", roff_line_ignore, NULL, NULL, 0, NULL },
           { "kernpair", roff_line_ignore, NULL, NULL, 0, NULL },
           { "lc", roff_unsupp, NULL, NULL, 0, NULL },
           { "lc_ctype", roff_unsupp, NULL, NULL, 0, NULL },
           { "lds", roff_unsupp, NULL, NULL, 0, NULL },
           { "length", roff_unsupp, NULL, NULL, 0, NULL },
           { "letadj", roff_line_ignore, NULL, NULL, 0, NULL },
           { "lf", roff_insec, NULL, NULL, 0, NULL },
           { "lg", roff_line_ignore, NULL, NULL, 0, NULL },
           { "lhang", roff_line_ignore, NULL, NULL, 0, NULL },
           { "linetabs", roff_unsupp, NULL, NULL, 0, NULL },
           { "lnr", roff_unsupp, NULL, NULL, 0, NULL },
           { "lnrf", roff_unsupp, NULL, NULL, 0, NULL },
           { "lpfx", roff_unsupp, NULL, NULL, 0, NULL },
           { "ls", roff_line_ignore, NULL, NULL, 0, NULL },
           { "lsm", roff_unsupp, NULL, NULL, 0, NULL },
           { "lt", roff_line_ignore, NULL, NULL, 0, NULL },
           { "mc", roff_line_ignore, NULL, NULL, 0, NULL },
           { "mediasize", roff_line_ignore, NULL, NULL, 0, NULL },
           { "minss", roff_line_ignore, NULL, NULL, 0, NULL },
           { "mk", roff_line_ignore, NULL, NULL, 0, NULL },
           { "mso", roff_insec, NULL, NULL, 0, NULL },
           { "na", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ne", roff_line_ignore, NULL, NULL, 0, NULL },
           { "nh", roff_line_ignore, NULL, NULL, 0, NULL },
           { "nhychar", roff_line_ignore, NULL, NULL, 0, NULL },
           { "nm", roff_unsupp, NULL, NULL, 0, NULL },
           { "nn", roff_unsupp, NULL, NULL, 0, NULL },
           { "nop", roff_unsupp, NULL, NULL, 0, NULL },
           { "nr", roff_nr, NULL, NULL, 0, NULL },
           { "nrf", roff_unsupp, NULL, NULL, 0, NULL },
           { "nroff", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ns", roff_line_ignore, NULL, NULL, 0, NULL },
           { "nx", roff_insec, NULL, NULL, 0, NULL },
           { "open", roff_insec, NULL, NULL, 0, NULL },
           { "opena", roff_insec, NULL, NULL, 0, NULL },
           { "os", roff_line_ignore, NULL, NULL, 0, NULL },
           { "output", roff_unsupp, NULL, NULL, 0, NULL },
           { "padj", roff_line_ignore, NULL, NULL, 0, NULL },
           { "papersize", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pc", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pev", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pi", roff_insec, NULL, NULL, 0, NULL },
           { "PI", roff_unsupp, NULL, NULL, 0, NULL },
           { "pl", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pm", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pn", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pnr", roff_line_ignore, NULL, NULL, 0, NULL },
           { "po", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ps", roff_line_ignore, NULL, NULL, 0, NULL },
           { "psbb", roff_unsupp, NULL, NULL, 0, NULL },
           { "pshape", roff_unsupp, NULL, NULL, 0, NULL },
           { "pso", roff_insec, NULL, NULL, 0, NULL },
           { "ptr", roff_line_ignore, NULL, NULL, 0, NULL },
           { "pvs", roff_line_ignore, NULL, NULL, 0, NULL },
           { "rchar", roff_unsupp, NULL, NULL, 0, NULL },
           { "rd", roff_line_ignore, NULL, NULL, 0, NULL },
           { "recursionlimit", roff_line_ignore, NULL, NULL, 0, NULL },
           { "return", roff_unsupp, NULL, NULL, 0, NULL },
           { "rfschar", roff_unsupp, NULL, NULL, 0, NULL },
           { "rhang", roff_line_ignore, NULL, NULL, 0, NULL },
           { "rj", roff_line_ignore, NULL, NULL, 0, NULL },
           { "rm", roff_rm, NULL, NULL, 0, NULL },
           { "rn", roff_unsupp, NULL, NULL, 0, NULL },
           { "rnn", roff_unsupp, NULL, NULL, 0, NULL },
           { "rr", roff_rr, NULL, NULL, 0, NULL },
           { "rs", roff_line_ignore, NULL, NULL, 0, NULL },
           { "rt", roff_line_ignore, NULL, NULL, 0, NULL },
           { "schar", roff_unsupp, NULL, NULL, 0, NULL },
           { "sentchar", roff_line_ignore, NULL, NULL, 0, NULL },
           { "shc", roff_line_ignore, NULL, NULL, 0, NULL },
           { "shift", roff_unsupp, NULL, NULL, 0, NULL },
           { "sizes", roff_line_ignore, NULL, NULL, 0, NULL },
           { "so", roff_so, NULL, NULL, 0, NULL },
           { "spacewidth", roff_line_ignore, NULL, NULL, 0, NULL },
           { "special", roff_line_ignore, NULL, NULL, 0, NULL },
           { "spreadwarn", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ss", roff_line_ignore, NULL, NULL, 0, NULL },
           { "sty", roff_line_ignore, NULL, NULL, 0, NULL },
           { "substring", roff_unsupp, NULL, NULL, 0, NULL },
           { "sv", roff_line_ignore, NULL, NULL, 0, NULL },
           { "sy", roff_insec, NULL, NULL, 0, NULL },
           { "T&", roff_T_, NULL, NULL, 0, NULL },
           { "ta", roff_unsupp, NULL, NULL, 0, NULL },
           { "tc", roff_unsupp, NULL, NULL, 0, NULL },
           { "TE", roff_TE, NULL, NULL, 0, NULL },
           { "TH", roff_TH, NULL, NULL, 0, NULL },
           { "ti", roff_unsupp, NULL, NULL, 0, NULL },
           { "tkf", roff_line_ignore, NULL, NULL, 0, NULL },
           { "tl", roff_unsupp, NULL, NULL, 0, NULL },
           { "tm", roff_line_ignore, NULL, NULL, 0, NULL },
           { "tm1", roff_line_ignore, NULL, NULL, 0, NULL },
           { "tmc", roff_line_ignore, NULL, NULL, 0, NULL },
           { "tr", roff_tr, NULL, NULL, 0, NULL },
           { "track", roff_line_ignore, NULL, NULL, 0, NULL },
           { "transchar", roff_line_ignore, NULL, NULL, 0, NULL },
           { "trf", roff_insec, NULL, NULL, 0, NULL },
           { "trimat", roff_line_ignore, NULL, NULL, 0, NULL },
           { "trin", roff_unsupp, NULL, NULL, 0, NULL },
           { "trnt", roff_unsupp, NULL, NULL, 0, NULL },
           { "troff", roff_line_ignore, NULL, NULL, 0, NULL },
           { "TS", roff_TS, NULL, NULL, 0, NULL },
           { "uf", roff_line_ignore, NULL, NULL, 0, NULL },
           { "ul", roff_line_ignore, NULL, NULL, 0, NULL },
           { "unformat", roff_unsupp, NULL, NULL, 0, NULL },
           { "unwatch", roff_line_ignore, NULL, NULL, 0, NULL },
           { "unwatchn", roff_line_ignore, NULL, NULL, 0, NULL },
           { "vpt", roff_line_ignore, NULL, NULL, 0, NULL },
           { "vs", roff_line_ignore, NULL, NULL, 0, NULL },
           { "warn", roff_line_ignore, NULL, NULL, 0, NULL },
           { "warnscale", roff_line_ignore, NULL, NULL, 0, NULL },
           { "watch", roff_line_ignore, NULL, NULL, 0, NULL },
           { "watchlength", roff_line_ignore, NULL, NULL, 0, NULL },
           { "watchn", roff_line_ignore, NULL, NULL, 0, NULL },
           { "wh", roff_unsupp, NULL, NULL, 0, NULL },
           { "while", roff_unsupp, NULL, NULL, 0, NULL },
           { "write", roff_insec, NULL, NULL, 0, NULL },
           { "writec", roff_insec, NULL, NULL, 0, NULL },
           { "writem", roff_insec, NULL, NULL, 0, NULL },
           { "xflag", roff_line_ignore, NULL, NULL, 0, NULL },
           { ".", roff_cblock, NULL, NULL, 0, NULL },
           { NULL, roff_userdef, NULL, NULL, 0, NULL },
 };  };
   
 static  int               roff_Dd(ROFFCALL_ARGS);  /* not currently implemented: Ds em Eq LP Me PP pp Or Rd Sf SH */
 static  int               roff_Dt(ROFFCALL_ARGS);  const   char *const __mdoc_reserved[] = {
 static  int               roff_Os(ROFFCALL_ARGS);          "Ac", "Ad", "An", "Ao", "Ap", "Aq", "Ar", "At",
 #ifdef notyet          "Bc", "Bd", "Bf", "Bk", "Bl", "Bo", "Bq",
 static  int               roff_Ns(ROFFCALL_ARGS);          "Brc", "Bro", "Brq", "Bsx", "Bt", "Bx",
 #endif          "Cd", "Cm", "Db", "Dc", "Dd", "Dl", "Do", "Dq",
           "Dt", "Dv", "Dx", "D1",
           "Ec", "Ed", "Ef", "Ek", "El", "Em",
           "En", "Eo", "Er", "Es", "Ev", "Ex",
           "Fa", "Fc", "Fd", "Fl", "Fn", "Fo", "Fr", "Ft", "Fx",
           "Hf", "Ic", "In", "It", "Lb", "Li", "Lk", "Lp",
           "Ms", "Mt", "Nd", "Nm", "No", "Ns", "Nx",
           "Oc", "Oo", "Op", "Os", "Ot", "Ox",
           "Pa", "Pc", "Pf", "Po", "Pp", "Pq",
           "Qc", "Ql", "Qo", "Qq", "Re", "Rs", "Rv",
           "Sc", "Sh", "Sm", "So", "Sq",
           "Ss", "St", "Sx", "Sy",
           "Ta", "Tn", "Ud", "Ux", "Va", "Vt", "Xc", "Xo", "Xr",
           "%A", "%B", "%C", "%D", "%I", "%J", "%N", "%O",
           "%P", "%Q", "%R", "%T", "%U", "%V",
           NULL
   };
   
 static  int               roff_layout(ROFFCALL_ARGS);  /* not currently implemented: BT DE DS ME MT PT SY TQ YS */
 static  int               roff_text(ROFFCALL_ARGS);  const   char *const __man_reserved[] = {
 static  int               roff_comment(ROFFCALL_ARGS);          "AT", "B", "BI", "BR", "DT",
 static  int               roff_close(ROFFCALL_ARGS);          "EE", "EN", "EQ", "EX", "HP", "I", "IB", "IP", "IR",
           "LP", "OP", "P", "PD", "PP",
           "R", "RB", "RE", "RI", "RS", "SB", "SH", "SM", "SS",
           "TE", "TH", "TP", "TS", "T&", "UC", "UE", "UR",
           NULL
   };
   
 static  struct roffnode  *roffnode_new(int, struct rofftree *);  /* Array of injected predefined strings. */
 static  void              roffnode_free(struct rofftree *);  #define PREDEFS_MAX      38
   static  const struct predef predefs[PREDEFS_MAX] = {
   #include "predefs.in"
   };
   
 static  void              roff_warn(const struct rofftree *,  /* See roffhash_find() */
                                 const char *, char *, ...);  #define ROFF_HASH(p)    (p[0] - ASCII_LO)
 static  void              roff_err(const struct rofftree *,  
                                 const char *, char *, ...);  
   
 static  int               roffscan(int, const int *);  static  int      roffit_lines;  /* number of lines to delay */
 static  int               rofffindtok(const char *);  static  char    *roffit_macro;  /* nil-terminated macro line */
 static  int               rofffindarg(const char *);  
 static  int               rofffindcallable(const char *);  
 static  int               roffargs(const struct rofftree *,  
                                 int, char *, char **);  
 static  int               roffargok(int, int);  
 static  int               roffnextopt(const struct rofftree *,  
                                 int, char ***, char **);  
 static  int               roffparse(struct rofftree *, char *);  
 static  int               textparse(const struct rofftree *, char *);  
   
   
 static  const int roffarg_An[] = { ROFF_Split, ROFF_Nosplit,  static void
         ROFF_ARGMAX };  roffhash_init(void)
 static  const int roffarg_Bd[] = { ROFF_Ragged, ROFF_Unfilled,  {
         ROFF_Literal, ROFF_File, ROFF_Offset, ROFF_Filled,          struct roffmac   *n;
         ROFF_Compact, ROFF_ARGMAX };          int               buc, i;
 static  const int roffarg_Bk[] = { ROFF_Words, ROFF_ARGMAX };  
 static  const int roffarg_Ex[] = { ROFF_Std, ROFF_ARGMAX };  
 static  const int roffarg_Rv[] = { ROFF_Std, ROFF_ARGMAX };  
 static  const int roffarg_Bl[] = { ROFF_Bullet, ROFF_Dash,  
         ROFF_Hyphen, ROFF_Item, ROFF_Enum, ROFF_Tag, ROFF_Diag,  
         ROFF_Hang, ROFF_Ohang, ROFF_Inset, ROFF_Column, ROFF_Offset,  
         ROFF_Width, ROFF_Compact, ROFF_ARGMAX };  
 static  const int roffarg_St[] = {  
         ROFF_p1003_1_88, ROFF_p1003_1_90, ROFF_p1003_1_96,  
         ROFF_p1003_1_2001, ROFF_p1003_1_2004, ROFF_p1003_1,  
         ROFF_p1003_1b, ROFF_p1003_1b_93, ROFF_p1003_1c_95,  
         ROFF_p1003_1g_2000, ROFF_p1003_2_92, ROFF_p1387_2_95,  
         ROFF_p1003_2, ROFF_p1387_2, ROFF_isoC_90, ROFF_isoC_amd1,  
         ROFF_isoC_tcor1, ROFF_isoC_tcor2, ROFF_isoC_99, ROFF_ansiC,  
         ROFF_ansiC_89, ROFF_ansiC_99, ROFF_ieee754, ROFF_iso8802_3,  
         ROFF_xpg3, ROFF_xpg4, ROFF_xpg4_2, ROFF_xpg4_3, ROFF_xbd5,  
         ROFF_xcu5, ROFF_xsh5, ROFF_xns5, ROFF_xns5_2d2_0,  
         ROFF_xcurses4_2, ROFF_susv2, ROFF_susv3, ROFF_svid4,  
         ROFF_ARGMAX };  
   
 static  const int roffchild_Bl[] = { ROFF_It, ROFF_El, ROFF_MAX };          for (i = 0; i < (int)ROFF_USERDEF; i++) {
 static  const int roffchild_Fo[] = { ROFF_Fa, ROFF_Fc, ROFF_MAX };                  assert(roffs[i].name[0] >= ASCII_LO);
 static  const int roffchild_Oo[] = { ROFF_Op, ROFF_Oc, ROFF_MAX };                  assert(roffs[i].name[0] <= ASCII_HI);
 static  const int roffchild_Rs[] = { ROFF_Re, ROFF__A, ROFF__B,  
         ROFF__D, ROFF__I, ROFF__J, ROFF__N, ROFF__O, ROFF__P,  
         ROFF__R, ROFF__T, ROFF__V, ROFF_MAX };  
   
 static  const int roffparent_El[] = { ROFF_Bl, ROFF_It, ROFF_MAX };                  buc = ROFF_HASH(roffs[i].name);
 static  const int roffparent_Fc[] = { ROFF_Fo, ROFF_Fa, ROFF_MAX };  
 static  const int roffparent_Oc[] = { ROFF_Oo, ROFF_Oc, ROFF_MAX };  
 static  const int roffparent_It[] = { ROFF_Bl, ROFF_It, ROFF_MAX };  
 static  const int roffparent_Re[] = { ROFF_Rs, ROFF_MAX };  
   
 /* Table of all known tokens. */                  if (NULL != (n = hash[buc])) {
 static  const struct rofftok tokens[ROFF_MAX] = {                          for ( ; n->next; n = n->next)
         {roff_comment, NULL, NULL, NULL, 0, ROFF_COMMENT, 0 }, /* \" */                                  /* Do nothing. */ ;
         {     roff_Dd, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Dd */                          n->next = &roffs[i];
         {     roff_Dt, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Dt */                  } else
         {     roff_Os, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Os */                          hash[buc] = &roffs[i];
         { roff_layout, NULL, NULL, NULL, ROFF_Sh, ROFF_LAYOUT, 0 }, /* Sh */          }
         { roff_layout, NULL, NULL, NULL, ROFF_Ss, ROFF_LAYOUT, 0 }, /* Ss */  }
         {   roff_text, NULL, NULL, NULL, ROFF_Pp, ROFF_TEXT, 0 }, /* Pp */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* D1 */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Dl */  
         { roff_layout, roffarg_Bd, NULL, NULL, 0, ROFF_LAYOUT, 0 },     /* Bd */  
         {  roff_close, NULL, NULL, NULL, ROFF_Bd, ROFF_LAYOUT, 0 }, /* Ed */  
         { roff_layout, roffarg_Bl, NULL, roffchild_Bl, 0, ROFF_LAYOUT, 0 }, /* Bl */  
         {  roff_close, NULL, roffparent_El, NULL, ROFF_Bl, ROFF_LAYOUT, 0 }, /* El */  
         { roff_layout, NULL, roffparent_It, NULL, ROFF_It, ROFF_LAYOUT, ROFF_PARSED | ROFF_SHALLOW }, /* It */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ad */ /* FIXME */  
         {   roff_text, roffarg_An, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* An */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ar */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Cd */ /* XXX man.4 only */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Cm */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Dv */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Er */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ev */ /* XXX needs arg */  
         {   roff_text, roffarg_Ex, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Ex */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Fa */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Fd */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Fl */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Fn */ /* XXX needs arg */ /* FIXME */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Ft */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ic */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* In */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Li */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Nd */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Nm */ /* FIXME */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Op */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Ot */ /* XXX deprecated */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Pa */  
         {   roff_text, roffarg_Rv, NULL, NULL, 0, ROFF_TEXT, 0 }, /* Rv */  
         {   roff_text, roffarg_St, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* St */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Va */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Vt */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Xr */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* %A */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE}, /* %B */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %D */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE}, /* %I */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE}, /* %J */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %N */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %O */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %P */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %R */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* %T */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* %V */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ac */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ao */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Aq */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, 0 }, /* At */ /* XXX at most 2 args */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Bc */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, 0 },  /* Bf */ /* FIXME */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Bo */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Bq */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Bsx */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Bx */  
         {        NULL, NULL, NULL, NULL, 0, ROFF_SPECIAL, 0 },  /* Db */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Dc */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Do */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Dq */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ec */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, 0 },  /* Ef */ /* FIXME */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Em */ /* XXX needs arg */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Eo */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Fx */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Ms */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* No */  
         {        NULL, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ns */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Nx */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Ox */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Pc */  
         {        NULL, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Pf */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_LAYOUT, ROFF_PARSED | ROFF_CALLABLE }, /* Po */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Pq */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Qc */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Ql */  
         { roff_layout, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Qo */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Qq */  
         {  roff_close, NULL, roffparent_Re, NULL, ROFF_Rs, ROFF_LAYOUT, 0 }, /* Re */  
         { roff_layout, NULL, NULL, roffchild_Rs, 0, ROFF_LAYOUT, 0 },   /* Rs */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Sc */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* So */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Sq */  
         {        NULL, NULL, NULL, NULL, 0, ROFF_SPECIAL, 0 }, /* Sm */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Sx */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Sy */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Tn */  
         {   roff_text, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED }, /* Ux */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Xc */  
         {   NULL, NULL, NULL, NULL, 0, ROFF_TEXT, ROFF_PARSED | ROFF_CALLABLE }, /* Xo */  
         { roff_layout, NULL, NULL, roffchild_Fo, 0, ROFF_LAYOUT, 0 }, /* Fo */  
         {  roff_close, NULL, roffparent_Fc, NULL, ROFF_Fo, ROFF_LAYOUT, 0 }, /* Fc */  
         { roff_layout, NULL, NULL, roffchild_Oo, 0, ROFF_LAYOUT, 0 }, /* Oo */  
         {  roff_close, NULL, roffparent_Oc, NULL, ROFF_Oo, ROFF_LAYOUT, 0 }, /* Oc */  
         { roff_layout, roffarg_Bk, NULL, NULL, 0, ROFF_LAYOUT, 0 }, /* Bk */  
         {  roff_close, NULL, NULL, NULL, ROFF_Bk, ROFF_LAYOUT, 0 }, /* Ek */  
         };  
   
 /* Table of all known token arguments. */  /*
 static  const int tokenargs[ROFF_ARGMAX] = {   * Look up a roff token by its name.  Returns ROFF_MAX if no macro by
         0,              0,              0,              0,   * the nil-terminated string name could be found.
         0,              ROFF_VALUE,     ROFF_VALUE,     0,   */
         0,              0,              0,              0,  static enum rofft
         0,              0,              0,              0,  roffhash_find(const char *p, size_t s)
         0,              0,              ROFF_VALUE,     0,  {
         0,              0,              0,              0,          int              buc;
         0,              0,              0,              0,          struct roffmac  *n;
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         0,              0,              0,              0,  
         };  
   
 const   char *const toknamesp[ROFF_MAX] = {          /*
         "\\\"",         "Dd",           "Dt",           "Os",           * libroff has an extremely simple hashtable, for the time
         "Sh",           "Ss",           "Pp",           "D1",           * being, which simply keys on the first character, which must
         "Dl",           "Bd",           "Ed",           "Bl",           * be printable, then walks a chain.  It works well enough until
         "El",           "It",           "Ad",           "An",           * optimised.
         "Ar",           "Cd",           "Cm",           "Dv",           */
         "Er",           "Ev",           "Ex",           "Fa",  
         "Fd",           "Fl",           "Fn",           "Ft",  
         "Ic",           "In",           "Li",           "Nd",  
         "Nm",           "Op",           "Ot",           "Pa",  
         "Rv",           "St",           "Va",           "Vt",  
         /* LINTED */  
         "Xr",           "\%A",          "\%B",          "\%D",  
         /* LINTED */  
         "\%I",          "\%J",          "\%N",          "\%O",  
         /* LINTED */  
         "\%P",          "\%R",          "\%T",          "\%V",  
         "Ac",           "Ao",           "Aq",           "At",  
         "Bc",           "Bf",           "Bo",           "Bq",  
         "Bsx",          "Bx",           "Db",           "Dc",  
         "Do",           "Dq",           "Ec",           "Ef",  
         "Em",           "Eo",           "Fx",           "Ms",  
         "No",           "Ns",           "Nx",           "Ox",  
         "Pc",           "Pf",           "Po",           "Pq",  
         "Qc",           "Ql",           "Qo",           "Qq",  
         "Re",           "Rs",           "Sc",           "So",  
         "Sq",           "Sm",           "Sx",           "Sy",  
         "Tn",           "Ux",           "Xc",           "Xo",  
         "Fo",           "Fc",           "Oo",           "Oc",  
         "Bk",           "Ek",  
         };  
   
 const   char *const tokargnamesp[ROFF_ARGMAX] = {          if (p[0] < ASCII_LO || p[0] > ASCII_HI)
         "split",                "nosplit",              "ragged",                  return(ROFF_MAX);
         "unfilled",             "literal",              "file",  
         "offset",               "bullet",               "dash",  
         "hyphen",               "item",                 "enum",  
         "tag",                  "diag",                 "hang",  
         "ohang",                "inset",                "column",  
         "width",                "compact",              "std",  
         "p1003.1-88",           "p1003.1-90",           "p1003.1-96",  
         "p1003.1-2001",         "p1003.1-2004",         "p1003.1",  
         "p1003.1b",             "p1003.1b-93",          "p1003.1c-95",  
         "p1003.1g-2000",        "p1003.2-92",           "p1387.2-95",  
         "p1003.2",              "p1387.2",              "isoC-90",  
         "isoC-amd1",            "isoC-tcor1",           "isoC-tcor2",  
         "isoC-99",              "ansiC",                "ansiC-89",  
         "ansiC-99",             "ieee754",              "iso8802-3",  
         "xpg3",                 "xpg4",                 "xpg4.2",  
         "xpg4.3",               "xbd5",                 "xcu5",  
         "xsh5",                 "xns5",                 "xns5.2d2.0",  
         "xcurses4.2",           "susv2",                "susv3",  
         "svid4",                "filled",               "words",  
         };  
   
 const   char *const *toknames = toknamesp;          buc = ROFF_HASH(p);
 const   char *const *tokargnames = tokargnamesp;  
   
           if (NULL == (n = hash[buc]))
                   return(ROFF_MAX);
           for ( ; n; n = n->next)
                   if (0 == strncmp(n->name, p, s) && '\0' == n->name[(int)s])
                           return((enum rofft)(n - roffs));
   
 int          return(ROFF_MAX);
 roff_free(struct rofftree *tree, int flush)  }
   
   /*
    * Pop the current node off of the stack of roff instructions currently
    * pending.
    */
   static void
   roffnode_pop(struct roff *r)
 {  {
         int              error, t;          struct roffnode *p;
         struct roffnode *n;  
   
         error = 0;          assert(r->last);
           p = r->last;
   
         if ( ! flush)          r->last = r->last->parent;
                 goto end;          free(p->name);
           free(p->end);
           free(p);
   }
   
         error = 1;  /*
    * Push a roff node onto the instruction stack.  This must later be
    * removed with roffnode_pop().
    */
   static void
   roffnode_push(struct roff *r, enum rofft tok, const char *name,
                   int line, int col)
   {
           struct roffnode *p;
   
         if (ROFF_PRELUDE & tree->state) {          p = mandoc_calloc(1, sizeof(struct roffnode));
                 roff_warn(tree, NULL, "prelude never finished");          p->tok = tok;
                 goto end;          if (name)
         }                  p->name = mandoc_strdup(name);
           p->parent = r->last;
           p->line = line;
           p->col = col;
           p->rule = p->parent ? p->parent->rule : 0;
   
         for (n = tree->last; n->parent; n = n->parent) {          r->last = p;
                 if (0 != tokens[n->tok].ctx)  }
                         continue;  
                 roff_warn(tree, NULL, "closing explicit scope `%s'",  static void
                                 toknames[n->tok]);  roff_free1(struct roff *r)
                 goto end;  {
           struct tbl_node *tbl;
           struct eqn_node *e;
           int              i;
   
           while (NULL != (tbl = r->first_tbl)) {
                   r->first_tbl = tbl->next;
                   tbl_free(tbl);
         }          }
           r->first_tbl = r->last_tbl = r->tbl = NULL;
   
         while (tree->last) {          while (NULL != (e = r->first_eqn)) {
                 t = tree->last->tok;                  r->first_eqn = e->next;
                 if ( ! (*tokens[t].cb)(t, tree, NULL, ROFF_EXIT))                  eqn_free(e);
                         goto end;  
         }          }
           r->first_eqn = r->last_eqn = r->eqn = NULL;
   
         error = 0;          while (r->last)
                   roffnode_pop(r);
   
 end:          free (r->rstack);
           r->rstack = NULL;
           r->rstacksz = 0;
           r->rstackpos = -1;
   
         while (tree->last)          roff_freereg(r->regtab);
                 roffnode_free(tree);          r->regtab = NULL;
   
         free(tree);          roff_freestr(r->strtab);
           roff_freestr(r->xmbtab);
           r->strtab = r->xmbtab = NULL;
   
         return(error ? 0 : 1);          if (r->xtab)
                   for (i = 0; i < 128; i++)
                           free(r->xtab[i].p);
           free(r->xtab);
           r->xtab = NULL;
 }  }
   
   void
   roff_reset(struct roff *r)
   {
   
 struct rofftree *          roff_free1(r);
 roff_alloc(const struct roffcb *cb, void *args)          r->format = r->options & (MPARSE_MDOC | MPARSE_MAN);
           r->control = 0;
   }
   
   void
   roff_free(struct roff *r)
 {  {
         struct rofftree *tree;  
   
         assert(args);          roff_free1(r);
         assert(cb);          free(r);
   }
   
         if (NULL == (tree = calloc(1, sizeof(struct rofftree))))  struct roff *
                 err(1, "calloc");  roff_alloc(struct mparse *parse, const struct mchars *mchars, int options)
   {
           struct roff     *r;
   
         tree->state = ROFF_PRELUDE;          r = mandoc_calloc(1, sizeof(struct roff));
         tree->arg = args;          r->parse = parse;
           r->mchars = mchars;
           r->options = options;
           r->format = options & (MPARSE_MDOC | MPARSE_MAN);
           r->rstackpos = -1;
   
         (void)memcpy(&tree->cb, cb, sizeof(struct roffcb));          roffhash_init();
   
         return(tree);          return(r);
 }  }
   
   /*
 int   * In the current line, expand escape sequences that tend to get
 roff_engine(struct rofftree *tree, char *buf)   * used in numerical expressions and conditional requests.
    * Also check the syntax of the remaining escape sequences.
    */
   static enum rofferr
   roff_res(struct roff *r, struct buf *buf, int ln, int pos)
 {  {
           char             ubuf[24]; /* buffer to print the number */
           const char      *start; /* start of the string to process */
           char            *stesc; /* start of an escape sequence ('\\') */
           const char      *stnam; /* start of the name, after "[(*" */
           const char      *cp;    /* end of the name, e.g. before ']' */
           const char      *res;   /* the string to be substituted */
           char            *nbuf;  /* new buffer to copy buf->buf to */
           size_t           maxl;  /* expected length of the escape name */
           size_t           naml;  /* actual length of the escape name */
           enum mandoc_esc  esc;   /* type of the escape sequence */
           int              inaml; /* length returned from mandoc_escape() */
           int              expand_count;  /* to avoid infinite loops */
           int              npos;  /* position in numeric expression */
           int              arg_complete; /* argument not interrupted by eol */
           char             term;  /* character terminating the escape */
   
         tree->cur = buf;          expand_count = 0;
         assert(buf);          start = buf->buf + pos;
           stesc = strchr(start, '\0') - 1;
           while (stesc-- > start) {
   
         if (0 == *buf) {                  /* Search backwards for the next backslash. */
                 roff_warn(tree, buf, "blank line");  
                 return(0);  
         } else if ('.' != *buf)  
                 return(textparse(tree, buf));  
   
         return(roffparse(tree, buf));                  if (*stesc != '\\')
 }                          continue;
   
                   /* If it is escaped, skip it. */
   
 static int                  for (cp = stesc - 1; cp >= start; cp--)
 textparse(const struct rofftree *tree, char *buf)                          if (*cp != '\\')
 {                                  break;
   
         return((*tree->cb.roffdata)(tree->arg, 1, buf));                  if ((stesc - cp) % 2 == 0) {
 }                          stesc = (char *)cp;
                           continue;
                   }
   
                   /* Decide whether to expand or to check only. */
   
 static int                  term = '\0';
 roffargs(const struct rofftree *tree,                  cp = stesc + 1;
                 int tok, char *buf, char **argv)                  switch (*cp) {
 {                  case '*':
         int              i;                          res = NULL;
         char            *p;                          break;
                   case 'B':
                           /* FALLTHROUGH */
                   case 'w':
                           term = cp[1];
                           /* FALLTHROUGH */
                   case 'n':
                           res = ubuf;
                           break;
                   default:
                           esc = mandoc_escape(&cp, &stnam, &inaml);
                           if (esc == ESCAPE_ERROR ||
                               (esc == ESCAPE_SPECIAL &&
                                mchars_spec2cp(r->mchars, stnam, inaml) < 0))
                                   mandoc_vmsg(MANDOCERR_ESC_BAD,
                                       r->parse, ln, (int)(stesc - buf->buf),
                                       "%.*s", (int)(cp - stesc), stesc);
                           continue;
                   }
   
         assert(tok >= 0 && tok < ROFF_MAX);                  if (EXPAND_LIMIT < ++expand_count) {
         assert('.' == *buf);                          mandoc_msg(MANDOCERR_ROFFLOOP, r->parse,
                               ln, (int)(stesc - buf->buf), NULL);
                           return(ROFF_IGN);
                   }
   
         p = buf;                  /*
                    * The third character decides the length
                    * of the name of the string or register.
                    * Save a pointer to the name.
                    */
   
         /* LINTED */                  if (term == '\0') {
         for (i = 0; *buf && i < ROFF_MAXARG; i++) {                          switch (*++cp) {
                 if ('\"' == *buf) {                          case '\0':
                         argv[i] = ++buf;                                  maxl = 0;
                         while (*buf && '\"' != *buf)                                  break;
                                 buf++;                          case '(':
                         if (0 == *buf) {                                  cp++;
                                 roff_err(tree, argv[i], "unclosed "                                  maxl = 2;
                                                 "quote in argument "                                  break;
                                                 "list for `%s'",                          case '[':
                                                 toknames[tok]);                                  cp++;
                                 return(0);                                  term = ']';
                                   maxl = 0;
                                   break;
                           default:
                                   maxl = 1;
                                   break;
                         }                          }
                 } else {                  } else {
                         argv[i] = buf++;                          cp += 2;
                         while (*buf && ! isspace(*buf))                          maxl = 0;
                                 buf++;                  }
                         if (0 == *buf)                  stnam = cp;
   
                   /* Advance to the end of the name. */
   
                   naml = 0;
                   arg_complete = 1;
                   while (maxl == 0 || naml < maxl) {
                           if (*cp == '\0') {
                                   mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                                       ln, (int)(stesc - buf->buf), stesc);
                                   arg_complete = 0;
                                   break;
                           }
                           if (maxl == 0 && *cp == term) {
                                   cp++;
                                   break;
                           }
                           if (*cp++ != '\\' || stesc[1] != 'w') {
                                   naml++;
                                 continue;                                  continue;
                           }
                           switch (mandoc_escape(&cp, NULL, NULL)) {
                           case ESCAPE_SPECIAL:
                                   /* FALLTHROUGH */
                           case ESCAPE_UNICODE:
                                   /* FALLTHROUGH */
                           case ESCAPE_NUMBERED:
                                   /* FALLTHROUGH */
                           case ESCAPE_OVERSTRIKE:
                                   naml++;
                                   break;
                           default:
                                   break;
                           }
                 }                  }
                 *buf++ = 0;  
                 while (*buf && isspace(*buf))  
                         buf++;  
         }  
   
         assert(i > 0);  
         if (ROFF_MAXARG == i && *buf) {  
                 roff_err(tree, p, "too many arguments for `%s'", toknames  
                                 [tok]);  
                 return(0);  
         }  
   
         argv[i] = NULL;                  /*
         return(1);                   * Retrieve the replacement string; if it is
 }                   * undefined, resume searching for escapes.
                    */
   
                   switch (stesc[1]) {
                   case '*':
                           if (arg_complete)
                                   res = roff_getstrn(r, stnam, naml);
                           break;
                   case 'B':
                           npos = 0;
                           ubuf[0] = arg_complete &&
                               roff_evalnum(r, ln, stnam, &npos, NULL, 0) &&
                               stnam + npos + 1 == cp ? '1' : '0';
                           ubuf[1] = '\0';
                           break;
                   case 'n':
                           if (arg_complete)
                                   (void)snprintf(ubuf, sizeof(ubuf), "%d",
                                       roff_getregn(r, stnam, naml));
                           else
                                   ubuf[0] = '\0';
                           break;
                   case 'w':
                           /* use even incomplete args */
                           (void)snprintf(ubuf, sizeof(ubuf), "%d",
                               24 * (int)naml);
                           break;
                   }
   
 static int                  if (res == NULL) {
 roffscan(int tok, const int *tokv)                          mandoc_vmsg(MANDOCERR_STR_UNDEF,
 {                              r->parse, ln, (int)(stesc - buf->buf),
                               "%.*s", (int)naml, stnam);
                           res = "";
                   } else if (buf->sz + strlen(res) > SHRT_MAX) {
                           mandoc_msg(MANDOCERR_ROFFLOOP, r->parse,
                               ln, (int)(stesc - buf->buf), NULL);
                           return(ROFF_IGN);
                   }
   
         if (NULL == tokv)                  /* Replace the escape sequence by the string. */
                 return(1);  
   
         for ( ; ROFF_MAX != *tokv; tokv++)                  *stesc = '\0';
                 if (tok == *tokv)                  buf->sz = mandoc_asprintf(&nbuf, "%s%s%s",
                         return(1);                      buf->buf, res, cp) + 1;
   
         return(0);                  /* Prepare for the next replacement. */
   
                   start = nbuf + pos;
                   stesc = nbuf + (stesc - buf->buf) + strlen(res);
                   free(buf->buf);
                   buf->buf = nbuf;
           }
           return(ROFF_CONT);
 }  }
   
   /*
    * Process text streams:
    * Convert all breakable hyphens into ASCII_HYPH.
    * Decrement and spring input line trap.
    */
   static enum rofferr
   roff_parsetext(struct buf *buf, int pos, int *offs)
   {
           size_t           sz;
           const char      *start;
           char            *p;
           int              isz;
           enum mandoc_esc  esc;
   
 static int          start = p = buf->buf + pos;
 roffparse(struct rofftree *tree, char *buf)  
           while (*p != '\0') {
                   sz = strcspn(p, "-\\");
                   p += sz;
   
                   if (*p == '\0')
                           break;
   
                   if (*p == '\\') {
                           /* Skip over escapes. */
                           p++;
                           esc = mandoc_escape((const char **)&p, NULL, NULL);
                           if (esc == ESCAPE_ERROR)
                                   break;
                           continue;
                   } else if (p == start) {
                           p++;
                           continue;
                   }
   
                   if (isalpha((unsigned char)p[-1]) &&
                       isalpha((unsigned char)p[1]))
                           *p = ASCII_HYPH;
                   p++;
           }
   
           /* Spring the input line trap. */
           if (roffit_lines == 1) {
                   isz = mandoc_asprintf(&p, "%s\n.%s", buf->buf, roffit_macro);
                   free(buf->buf);
                   buf->buf = p;
                   buf->sz = isz + 1;
                   *offs = 0;
                   free(roffit_macro);
                   roffit_lines = 0;
                   return(ROFF_REPARSE);
           } else if (roffit_lines > 1)
                   --roffit_lines;
           return(ROFF_CONT);
   }
   
   enum rofferr
   roff_parseln(struct roff *r, int ln, struct buf *buf, int *offs)
 {  {
         int               tok, t;          enum rofft       t;
         struct roffnode  *n;          enum rofferr     e;
         char             *argv[ROFF_MAXARG];          int              pos;   /* parse point */
         char            **argvp;          int              spos;  /* saved parse point for messages */
           int              ppos;  /* original offset in buf->buf */
           int              ctl;   /* macro line (boolean) */
   
         if (0 != *buf && 0 != *(buf + 1) && 0 != *(buf + 2))          ppos = pos = *offs;
                 if (0 == strncmp(buf, ".\\\"", 3))  
                         return(1);  
   
         if (ROFF_MAX == (tok = rofffindtok(buf + 1))) {          /* Handle in-line equation delimiters. */
                 roff_err(tree, buf + 1, "bogus line macro");  
                 return(0);          if (r->tbl == NULL &&
         } else if (NULL == tokens[tok].cb) {              r->last_eqn != NULL && r->last_eqn->delim &&
                 roff_err(tree, buf + 1, "unsupported macro `%s'",              (r->eqn == NULL || r->eqn_inline)) {
                                 toknames[tok]);                  e = roff_eqndelim(r, buf, pos);
                 return(0);                  if (e == ROFF_REPARSE)
                           return(e);
                   assert(e == ROFF_CONT);
         }          }
   
         assert(ROFF___ != tok);          /* Expand some escape sequences. */
         if ( ! roffargs(tree, tok, buf, argv))  
                 return(0);  
   
         argvp = (char **)argv;          e = roff_res(r, buf, ln, pos);
           if (e == ROFF_IGN)
                   return(e);
           assert(e == ROFF_CONT);
   
         /*          ctl = roff_getcontrol(r, buf->buf, &pos);
          * Prelude macros break some assumptions, so branch now.  
           /*
            * First, if a scope is open and we're not a macro, pass the
            * text through the macro's filter.
            * Equations process all content themselves.
            * Tables process almost all content themselves, but we want
            * to warn about macros before passing it there.
          */           */
   
         if (ROFF_PRELUDE & tree->state) {  
                 assert(NULL == tree->last);  
                 return((*tokens[tok].cb)(tok, tree, argvp, ROFF_ENTER));  
         } else  
                 assert(tree->last);  
   
         assert(ROFF_BODY & tree->state);          if (r->last != NULL && ! ctl) {
                   t = r->last->tok;
                   assert(roffs[t].text);
                   e = (*roffs[t].text)(r, t, buf, ln, pos, pos, offs);
                   assert(e == ROFF_IGN || e == ROFF_CONT);
                   if (e != ROFF_CONT)
                           return(e);
           }
           if (r->eqn != NULL)
                   return(eqn_read(&r->eqn, ln, buf->buf, ppos, offs));
           if (r->tbl != NULL && ( ! ctl || buf->buf[pos] == '\0'))
                   return(tbl_read(r->tbl, ln, buf->buf, pos));
           if ( ! ctl)
                   return(roff_parsetext(buf, pos, offs));
   
         /*          /* Skip empty request lines. */
          * First check that our possible parents and parent's possible  
          * children are satisfied.          if (buf->buf[pos] == '"') {
                   mandoc_msg(MANDOCERR_COMMENT_BAD, r->parse,
                       ln, pos, NULL);
                   return(ROFF_IGN);
           } else if (buf->buf[pos] == '\0')
                   return(ROFF_IGN);
   
           /*
            * If a scope is open, go to the child handler for that macro,
            * as it may want to preprocess before doing anything with it.
            * Don't do so if an equation is open.
          */           */
   
         if ( ! roffscan(tree->last->tok, tokens[tok].parents)) {          if (r->last) {
                 roff_err(tree, *argvp, "`%s' has invalid parent `%s'",                  t = r->last->tok;
                                 toknames[tok],                  assert(roffs[t].sub);
                                 toknames[tree->last->tok]);                  return((*roffs[t].sub)(r, t, buf, ln, ppos, pos, offs));
                 return(0);          }
         }  
   
         if ( ! roffscan(tok, tokens[tree->last->tok].children)) {          /* No scope is open.  This is a new request or macro. */
                 roff_err(tree, *argvp, "`%s' is invalid child of `%s'",  
                                 toknames[tok],          spos = pos;
                                 toknames[tree->last->tok]);          t = roff_parse(r, buf->buf, &pos, ln, ppos);
                 return(0);  
           /* Tables ignore most macros. */
   
           if (r->tbl != NULL && (t == ROFF_MAX || t == ROFF_TS)) {
                   mandoc_msg(MANDOCERR_TBLMACRO, r->parse,
                       ln, pos, buf->buf + spos);
                   if (t == ROFF_TS)
                           return(ROFF_IGN);
                   while (buf->buf[pos] != '\0' && buf->buf[pos] != ' ')
                           pos++;
                   while (buf->buf[pos] != '\0' && buf->buf[pos] == ' ')
                           pos++;
                   return(tbl_read(r->tbl, ln, buf->buf, pos));
         }          }
   
         /*          /*
          * Branch if we're not a layout token.           * This is neither a roff request nor a user-defined macro.
            * Let the standard macro set parsers handle it.
          */           */
   
         if (ROFF_LAYOUT != tokens[tok].type)          if (t == ROFF_MAX)
                 return((*tokens[tok].cb)(tok, tree, argvp, ROFF_ENTER));                  return(ROFF_CONT);
   
         /*          /* Execute a roff request or a user defined macro. */
          * Check our scope rules.  
           assert(roffs[t].proc);
           return((*roffs[t].proc)(r, t, buf, ln, ppos, pos, offs));
   }
   
   void
   roff_endparse(struct roff *r)
   {
   
           if (r->last)
                   mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                       r->last->line, r->last->col,
                       roffs[r->last->tok].name);
   
           if (r->eqn) {
                   mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                       r->eqn->eqn.ln, r->eqn->eqn.pos, "EQ");
                   eqn_end(&r->eqn);
           }
   
           if (r->tbl) {
                   mandoc_msg(MANDOCERR_BLK_NOEND, r->parse,
                       r->tbl->line, r->tbl->pos, "TS");
                   tbl_end(&r->tbl);
           }
   }
   
   /*
    * Parse a roff node's type from the input buffer.  This must be in the
    * form of ".foo xxx" in the usual way.
    */
   static enum rofft
   roff_parse(struct roff *r, char *buf, int *pos, int ln, int ppos)
   {
           char            *cp;
           const char      *mac;
           size_t           maclen;
           enum rofft       t;
   
           cp = buf + *pos;
   
           if ('\0' == *cp || '"' == *cp || '\t' == *cp || ' ' == *cp)
                   return(ROFF_MAX);
   
           mac = cp;
           maclen = roff_getname(r, &cp, ln, ppos);
   
           t = (r->current_string = roff_getstrn(r, mac, maclen))
               ? ROFF_USERDEF : roffhash_find(mac, maclen);
   
           if (ROFF_MAX != t)
                   *pos = cp - buf;
   
           return(t);
   }
   
   static enum rofferr
   roff_cblock(ROFF_ARGS)
   {
   
           /*
            * A block-close `..' should only be invoked as a child of an
            * ignore macro, otherwise raise a warning and just ignore it.
          */           */
   
         if (0 == tokens[tok].ctx)          if (r->last == NULL) {
                 return((*tokens[tok].cb)(tok, tree, argvp, ROFF_ENTER));                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "..");
                   return(ROFF_IGN);
           }
   
           switch (r->last->tok) {
           case ROFF_am:
                   /* ROFF_am1 is remapped to ROFF_am in roff_block(). */
                   /* FALLTHROUGH */
           case ROFF_ami:
                   /* FALLTHROUGH */
           case ROFF_de:
                   /* ROFF_de1 is remapped to ROFF_de in roff_block(). */
                   /* FALLTHROUGH */
           case ROFF_dei:
                   /* FALLTHROUGH */
           case ROFF_ig:
                   break;
           default:
                   mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "..");
                   return(ROFF_IGN);
           }
   
           if (buf->buf[pos] != '\0')
                   mandoc_vmsg(MANDOCERR_ARG_SKIP, r->parse, ln, pos,
                       ".. %s", buf->buf + pos);
   
           roffnode_pop(r);
           roffnode_cleanscope(r);
           return(ROFF_IGN);
   
   }
   
   static void
   roffnode_cleanscope(struct roff *r)
   {
   
           while (r->last) {
                   if (--r->last->endspan != 0)
                           break;
                   roffnode_pop(r);
           }
   }
   
   static void
   roff_ccond(struct roff *r, int ln, int ppos)
   {
   
           if (NULL == r->last) {
                   mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                   return;
           }
   
           switch (r->last->tok) {
           case ROFF_el:
                   /* FALLTHROUGH */
           case ROFF_ie:
                   /* FALLTHROUGH */
           case ROFF_if:
                   break;
           default:
                   mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                   return;
           }
   
           if (r->last->endspan > -1) {
                   mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "\\}");
                   return;
           }
   
           roffnode_pop(r);
           roffnode_cleanscope(r);
           return;
   }
   
   static enum rofferr
   roff_block(ROFF_ARGS)
   {
           const char      *name;
           char            *iname, *cp;
           size_t           namesz;
   
           /* Ignore groff compatibility mode for now. */
   
           if (tok == ROFF_de1)
                   tok = ROFF_de;
           else if (tok == ROFF_dei1)
                   tok = ROFF_dei;
           else if (tok == ROFF_am1)
                   tok = ROFF_am;
           else if (tok == ROFF_ami1)
                   tok = ROFF_ami;
   
           /* Parse the macro name argument. */
   
           cp = buf->buf + pos;
           if (tok == ROFF_ig) {
                   iname = NULL;
                   namesz = 0;
           } else {
                   iname = cp;
                   namesz = roff_getname(r, &cp, ln, ppos);
                   iname[namesz] = '\0';
           }
   
           /* Resolve the macro name argument if it is indirect. */
   
           if (namesz && (tok == ROFF_dei || tok == ROFF_ami)) {
                   if ((name = roff_getstrn(r, iname, namesz)) == NULL) {
                           mandoc_vmsg(MANDOCERR_STR_UNDEF,
                               r->parse, ln, (int)(iname - buf->buf),
                               "%.*s", (int)namesz, iname);
                           namesz = 0;
                   } else
                           namesz = strlen(name);
           } else
                   name = iname;
   
           if (namesz == 0 && tok != ROFF_ig) {
                   mandoc_msg(MANDOCERR_REQ_EMPTY, r->parse,
                       ln, ppos, roffs[tok].name);
                   return(ROFF_IGN);
           }
   
           roffnode_push(r, tok, name, ln, ppos);
   
         /*          /*
          * First consider implicit-end tags, like as follows:           * At the beginning of a `de' macro, clear the existing string
          *      .Sh SECTION 1           * with the same name, if there is one.  New content will be
          *      .Sh SECTION 2           * appended from roff_block_text() in multiline mode.
          * In this, we want to close the scope of the NAME section.  If  
          * there's an intermediary implicit-end tag, such as  
          *      .Sh SECTION 1  
          *      .Ss Subsection 1  
          *      .Sh SECTION 2  
          * then it must be closed as well.  
          */           */
   
         if (tok == tokens[tok].ctx) {          if (tok == ROFF_de || tok == ROFF_dei)
                 /*                  roff_setstrn(&r->strtab, name, namesz, "", 0, 0);
                  * First search up to the point where we must close.  
                  * If one doesn't exist, then we can open a new scope.  
                  */  
   
                 for (n = tree->last; n; n = n->parent) {          if (*cp == '\0')
                         assert(0 == tokens[n->tok].ctx ||                  return(ROFF_IGN);
                                         n->tok == tokens[n->tok].ctx);  
                         if (n->tok == tok)  
                                 break;  
                         if (ROFF_SHALLOW & tokens[tok].flags) {  
                                 n = NULL;  
                                 break;  
                         }  
                 }  
   
                 /*          /* Get the custom end marker. */
                  * Create a new scope, as no previous one exists to  
                  * close out.  
                  */  
   
                 if (NULL == n)          iname = cp;
                         return((*tokens[tok].cb)(tok, tree, argvp, ROFF_ENTER));          namesz = roff_getname(r, &cp, ln, ppos);
   
                 /*          /* Resolve the end marker if it is indirect. */
                  * Close out all intermediary scoped blocks, then hang  
                  * the current scope from our predecessor's parent.  
                  */  
   
                 do {          if (namesz && (tok == ROFF_dei || tok == ROFF_ami)) {
                         t = tree->last->tok;                  if ((name = roff_getstrn(r, iname, namesz)) == NULL) {
                         if ( ! (*tokens[t].cb)(t, tree, NULL, ROFF_EXIT))                          mandoc_vmsg(MANDOCERR_STR_UNDEF,
                                 return(0);                              r->parse, ln, (int)(iname - buf->buf),
                 } while (t != tok);                              "%.*s", (int)namesz, iname);
                           namesz = 0;
                   } else
                           namesz = strlen(name);
           } else
                   name = iname;
   
                 return((*tokens[tok].cb)(tok, tree, argvp, ROFF_ENTER));          if (namesz)
                   r->last->end = mandoc_strndup(name, namesz);
   
           if (*cp != '\0')
                   mandoc_vmsg(MANDOCERR_ARG_EXCESS, r->parse,
                       ln, pos, ".%s ... %s", roffs[tok].name, cp);
   
           return(ROFF_IGN);
   }
   
   static enum rofferr
   roff_block_sub(ROFF_ARGS)
   {
           enum rofft      t;
           int             i, j;
   
           /*
            * First check whether a custom macro exists at this level.  If
            * it does, then check against it.  This is some of groff's
            * stranger behaviours.  If we encountered a custom end-scope
            * tag and that tag also happens to be a "real" macro, then we
            * need to try interpreting it again as a real macro.  If it's
            * not, then return ignore.  Else continue.
            */
   
           if (r->last->end) {
                   for (i = pos, j = 0; r->last->end[j]; j++, i++)
                           if (buf->buf[i] != r->last->end[j])
                                   break;
   
                   if (r->last->end[j] == '\0' &&
                       (buf->buf[i] == '\0' ||
                        buf->buf[i] == ' ' ||
                        buf->buf[i] == '\t')) {
                           roffnode_pop(r);
                           roffnode_cleanscope(r);
   
                           while (buf->buf[i] == ' ' || buf->buf[i] == '\t')
                                   i++;
   
                           pos = i;
                           if (roff_parse(r, buf->buf, &pos, ln, ppos) !=
                               ROFF_MAX)
                                   return(ROFF_RERUN);
                           return(ROFF_IGN);
                   }
         }          }
   
         /*          /*
          * Now consider explicit-end tags, where we want to close back           * If we have no custom end-query or lookup failed, then try
          * to a specific tag.  Example:           * pulling it out of the hashtable.
          *      .Bl  
          *      .It Item.  
          *      .El  
          * In this, the `El' tag closes out the scope of `Bl'.  
          */           */
   
         assert(tree->last);          t = roff_parse(r, buf->buf, &pos, ln, ppos);
         assert(tok != tokens[tok].ctx && 0 != tokens[tok].ctx);  
   
         /* LINTED */          if (t != ROFF_cblock) {
         do {                  if (tok != ROFF_ig)
                 t = tree->last->tok;                          roff_setstr(r, r->last->name, buf->buf + ppos, 2);
                 if ( ! (*tokens[t].cb)(t, tree, NULL, ROFF_EXIT))                  return(ROFF_IGN);
                         return(0);          }
         } while (t != tokens[tok].ctx);  
   
         assert(tree->last);          assert(roffs[t].proc);
         return(1);          return((*roffs[t].proc)(r, t, buf, ln, ppos, pos, offs));
 }  }
   
   static enum rofferr
   roff_block_text(ROFF_ARGS)
   {
   
 static int          if (tok != ROFF_ig)
 rofffindarg(const char *name)                  roff_setstr(r, r->last->name, buf->buf + pos, 2);
   
           return(ROFF_IGN);
   }
   
   static enum rofferr
   roff_cond_sub(ROFF_ARGS)
 {  {
         size_t           i;          enum rofft       t;
           char            *ep;
           int              rr;
   
         /* FIXME: use a table, this is slow but ok for now. */          rr = r->last->rule;
           roffnode_cleanscope(r);
           t = roff_parse(r, buf->buf, &pos, ln, ppos);
   
         /* LINTED */          /*
         for (i = 0; i < ROFF_ARGMAX; i++)           * Fully handle known macros when they are structurally
                 /* LINTED */           * required or when the conditional evaluated to true.
                 if (0 == strcmp(name, tokargnames[i]))           */
                         return((int)i);  
           if ((t != ROFF_MAX) &&
         return(ROFF_ARGMAX);              (rr || roffs[t].flags & ROFFMAC_STRUCT)) {
                   assert(roffs[t].proc);
                   return((*roffs[t].proc)(r, t, buf, ln, ppos, pos, offs));
           }
   
           /*
            * If `\}' occurs on a macro line without a preceding macro,
            * drop the line completely.
            */
   
           ep = buf->buf + pos;
           if (ep[0] == '\\' && ep[1] == '}')
                   rr = 0;
   
           /* Always check for the closing delimiter `\}'. */
   
           while ((ep = strchr(ep, '\\')) != NULL) {
                   if (*(++ep) == '}') {
                           *ep = '&';
                           roff_ccond(r, ln, ep - buf->buf - 1);
                   }
                   if (*ep != '\0')
                           ++ep;
           }
           return(rr ? ROFF_CONT : ROFF_IGN);
 }  }
   
   static enum rofferr
   roff_cond_text(ROFF_ARGS)
   {
           char            *ep;
           int              rr;
   
           rr = r->last->rule;
           roffnode_cleanscope(r);
   
           ep = buf->buf + pos;
           while ((ep = strchr(ep, '\\')) != NULL) {
                   if (*(++ep) == '}') {
                           *ep = '&';
                           roff_ccond(r, ln, ep - buf->buf - 1);
                   }
                   if (*ep != '\0')
                           ++ep;
           }
           return(rr ? ROFF_CONT : ROFF_IGN);
   }
   
   /*
    * Parse a single signed integer number.  Stop at the first non-digit.
    * If there is at least one digit, return success and advance the
    * parse point, else return failure and let the parse point unchanged.
    * Ignore overflows, treat them just like the C language.
    */
 static int  static int
 rofffindtok(const char *buf)  roff_getnum(const char *v, int *pos, int *res)
 {  {
         char             token[4];          int      myres, n, p;
         int              i;  
   
         for (i = 0; *buf && ! isspace(*buf) && i < 3; i++, buf++)          if (NULL == res)
                 token[i] = *buf;                  res = &myres;
   
         if (i == 3)          p = *pos;
                 return(ROFF_MAX);          n = v[p] == '-';
           if (n)
                   p++;
   
         token[i] = 0;          for (*res = 0; isdigit((unsigned char)v[p]); p++)
                   *res = 10 * *res + v[p] - '0';
           if (p == *pos + n)
                   return 0;
   
         /* FIXME: use a table, this is slow but ok for now. */          if (n)
                   *res = -*res;
   
         /* LINTED */          /* Each number may be followed by one optional scaling unit. */
         for (i = 0; i < ROFF_MAX; i++)  
                 /* LINTED */  
                 if (0 == strcmp(toknames[i], token))  
                         return((int)i);  
   
         return(ROFF_MAX);          switch (v[p]) {
           case 'f':
                   *res *= 65536;
                   break;
           case 'i':
                   *res *= 240;
                   break;
           case 'c':
                   *res *= 240;
                   *res /= 2.54;
                   break;
           case 'v':
                   /* FALLTROUGH */
           case 'P':
                   *res *= 40;
                   break;
           case 'm':
                   /* FALLTROUGH */
           case 'n':
                   *res *= 24;
                   break;
           case 'p':
                   *res *= 10;
                   *res /= 3;
                   break;
           case 'u':
                   break;
           case 'M':
                   *res *= 6;
                   *res /= 25;
                   break;
           default:
                   p--;
                   break;
           }
   
           *pos = p + 1;
           return(1);
 }  }
   
   /*
    * Evaluate a string comparison condition.
    * The first character is the delimiter.
    * Succeed if the string up to its second occurrence
    * matches the string up to its third occurence.
    * Advance the cursor after the third occurrence
    * or lacking that, to the end of the line.
    */
   static int
   roff_evalstrcond(const char *v, int *pos)
   {
           const char      *s1, *s2, *s3;
           int              match;
   
           match = 0;
           s1 = v + *pos;          /* initial delimiter */
           s2 = s1 + 1;            /* for scanning the first string */
           s3 = strchr(s2, *s1);   /* for scanning the second string */
   
           if (NULL == s3)         /* found no middle delimiter */
                   goto out;
   
           while ('\0' != *++s3) {
                   if (*s2 != *s3) {  /* mismatch */
                           s3 = strchr(s3, *s1);
                           break;
                   }
                   if (*s3 == *s1) {  /* found the final delimiter */
                           match = 1;
                           break;
                   }
                   s2++;
           }
   
   out:
           if (NULL == s3)
                   s3 = strchr(s2, '\0');
           else if (*s3 != '\0')
                   s3++;
           *pos = s3 - v;
           return(match);
   }
   
   /*
    * Evaluate an optionally negated single character, numerical,
    * or string condition.
    */
 static int  static int
 roffispunct(const char *p)  roff_evalcond(struct roff *r, int ln, const char *v, int *pos)
 {  {
           int      number, savepos, wanttrue;
   
         if (0 == *p)          if ('!' == v[*pos]) {
                 return(0);                  wanttrue = 0;
         if (0 != *(p + 1))                  (*pos)++;
                 return(0);          } else
                   wanttrue = 1;
   
         switch (*p) {          switch (v[*pos]) {
         case('{'):          case '\0':
                   return(0);
           case 'n':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case('.'):          case 'o':
                   (*pos)++;
                   return(wanttrue);
           case 'c':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case(','):          case 'd':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case(';'):          case 'e':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case(':'):          case 'r':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case('?'):          case 't':
                 /* FALLTHROUGH */                  /* FALLTHROUGH */
         case('!'):          case 'v':
                 /* FALLTHROUGH */                  (*pos)++;
         case('('):                  return(!wanttrue);
                 /* FALLTHROUGH */  
         case(')'):  
                 /* FALLTHROUGH */  
         case('['):  
                 /* FALLTHROUGH */  
         case(']'):  
                 /* FALLTHROUGH */  
         case('}'):  
                 return(1);  
         default:          default:
                 break;                  break;
         }          }
   
         return(0);          savepos = *pos;
           if (roff_evalnum(r, ln, v, pos, &number, 0))
                   return((number > 0) == wanttrue);
           else if (*pos == savepos)
                   return(roff_evalstrcond(v, pos) == wanttrue);
           else
                   return (0);
 }  }
   
   static enum rofferr
   roff_line_ignore(ROFF_ARGS)
   {
   
 static int          return(ROFF_IGN);
 rofffindcallable(const char *name)  }
   
   static enum rofferr
   roff_insec(ROFF_ARGS)
 {  {
         int              c;  
   
         if (ROFF_MAX == (c = rofffindtok(name)))          mandoc_msg(MANDOCERR_REQ_INSEC, r->parse,
                 return(ROFF_MAX);              ln, ppos, roffs[tok].name);
         assert(c >= 0 && c < ROFF_MAX);          return(ROFF_IGN);
         return(ROFF_CALLABLE & tokens[c].flags ? c : ROFF_MAX);  
 }  }
   
   static enum rofferr
   roff_unsupp(ROFF_ARGS)
   {
   
 static struct roffnode *          mandoc_msg(MANDOCERR_REQ_UNSUPP, r->parse,
 roffnode_new(int tokid, struct rofftree *tree)              ln, ppos, roffs[tok].name);
           return(ROFF_IGN);
   }
   
   static enum rofferr
   roff_cond(ROFF_ARGS)
 {  {
         struct roffnode *p;  
   
         if (NULL == (p = malloc(sizeof(struct roffnode))))  
                 err(1, "malloc");  
   
         p->tok = tokid;          roffnode_push(r, tok, NULL, ln, ppos);
         p->parent = tree->last;  
         tree->last = p;  
   
         return(p);          /*
            * An `.el' has no conditional body: it will consume the value
            * of the current rstack entry set in prior `ie' calls or
            * defaults to DENY.
            *
            * If we're not an `el', however, then evaluate the conditional.
            */
   
           r->last->rule = tok == ROFF_el ?
               (r->rstackpos < 0 ? 0 : r->rstack[r->rstackpos--]) :
               roff_evalcond(r, ln, buf->buf, &pos);
   
           /*
            * An if-else will put the NEGATION of the current evaluated
            * conditional into the stack of rules.
            */
   
           if (tok == ROFF_ie) {
                   if (r->rstackpos + 1 == r->rstacksz) {
                           r->rstacksz += 16;
                           r->rstack = mandoc_reallocarray(r->rstack,
                               r->rstacksz, sizeof(int));
                   }
                   r->rstack[++r->rstackpos] = !r->last->rule;
           }
   
           /* If the parent has false as its rule, then so do we. */
   
           if (r->last->parent && !r->last->parent->rule)
                   r->last->rule = 0;
   
           /*
            * Determine scope.
            * If there is nothing on the line after the conditional,
            * not even whitespace, use next-line scope.
            */
   
           if (buf->buf[pos] == '\0') {
                   r->last->endspan = 2;
                   goto out;
           }
   
           while (buf->buf[pos] == ' ')
                   pos++;
   
           /* An opening brace requests multiline scope. */
   
           if (buf->buf[pos] == '\\' && buf->buf[pos + 1] == '{') {
                   r->last->endspan = -1;
                   pos += 2;
                   goto out;
           }
   
           /*
            * Anything else following the conditional causes
            * single-line scope.  Warn if the scope contains
            * nothing but trailing whitespace.
            */
   
           if (buf->buf[pos] == '\0')
                   mandoc_msg(MANDOCERR_COND_EMPTY, r->parse,
                       ln, ppos, roffs[tok].name);
   
           r->last->endspan = 1;
   
   out:
           *offs = pos;
           return(ROFF_RERUN);
 }  }
   
   static enum rofferr
   roff_ds(ROFF_ARGS)
   {
           char            *string;
           const char      *name;
           size_t           namesz;
   
           /* Ignore groff compatibility mode for now. */
   
           if (tok == ROFF_ds1)
                   tok = ROFF_ds;
           else if (tok == ROFF_as1)
                   tok = ROFF_as;
   
           /*
            * The first word is the name of the string.
            * If it is empty or terminated by an escape sequence,
            * abort the `ds' request without defining anything.
            */
   
           name = string = buf->buf + pos;
           if (*name == '\0')
                   return(ROFF_IGN);
   
           namesz = roff_getname(r, &string, ln, pos);
           if (name[namesz] == '\\')
                   return(ROFF_IGN);
   
           /* Read past the initial double-quote, if any. */
           if (*string == '"')
                   string++;
   
           /* The rest is the value. */
           roff_setstrn(&r->strtab, name, namesz, string, strlen(string),
               ROFF_as == tok);
           return(ROFF_IGN);
   }
   
   /*
    * Parse a single operator, one or two characters long.
    * If the operator is recognized, return success and advance the
    * parse point, else return failure and let the parse point unchanged.
    */
 static int  static int
 roffargok(int tokid, int argid)  roff_getop(const char *v, int *pos, char *res)
 {  {
         const int       *c;  
   
         if (NULL == (c = tokens[tokid].args))          *res = v[*pos];
   
           switch (*res) {
           case '+':
                   /* FALLTHROUGH */
           case '-':
                   /* FALLTHROUGH */
           case '*':
                   /* FALLTHROUGH */
           case '/':
                   /* FALLTHROUGH */
           case '%':
                   /* FALLTHROUGH */
           case '&':
                   /* FALLTHROUGH */
           case ':':
                   break;
           case '<':
                   switch (v[*pos + 1]) {
                   case '=':
                           *res = 'l';
                           (*pos)++;
                           break;
                   case '>':
                           *res = '!';
                           (*pos)++;
                           break;
                   case '?':
                           *res = 'i';
                           (*pos)++;
                           break;
                   default:
                           break;
                   }
                   break;
           case '>':
                   switch (v[*pos + 1]) {
                   case '=':
                           *res = 'g';
                           (*pos)++;
                           break;
                   case '?':
                           *res = 'a';
                           (*pos)++;
                           break;
                   default:
                           break;
                   }
                   break;
           case '=':
                   if ('=' == v[*pos + 1])
                           (*pos)++;
                   break;
           default:
                 return(0);                  return(0);
           }
           (*pos)++;
   
         for ( ; ROFF_ARGMAX != *c; c++)          return(*res);
                 if (argid == *c)  
                         return(1);  
   
         return(0);  
 }  }
   
   /*
 static void   * Evaluate either a parenthesized numeric expression
 roffnode_free(struct rofftree *tree)   * or a single signed integer number.
    */
   static int
   roff_evalpar(struct roff *r, int ln,
           const char *v, int *pos, int *res)
 {  {
         struct roffnode *p;  
   
         assert(tree->last);          if ('(' != v[*pos])
                   return(roff_getnum(v, pos, res));
   
         p = tree->last;          (*pos)++;
         tree->last = tree->last->parent;          if ( ! roff_evalnum(r, ln, v, pos, res, 1))
         free(p);                  return(0);
 }  
   
           /*
            * Omission of the closing parenthesis
            * is an error in validation mode,
            * but ignored in evaluation mode.
            */
   
           if (')' == v[*pos])
                   (*pos)++;
           else if (NULL == res)
                   return(0);
   
           return(1);
   }
   
   /*
    * Evaluate a complete numeric expression.
    * Proceed left to right, there is no concept of precedence.
    */
 static int  static int
 roffnextopt(const struct rofftree *tree, int tok,  roff_evalnum(struct roff *r, int ln, const char *v,
                 char ***in, char **val)          int *pos, int *res, int skipwhite)
 {  {
         char            *arg, **argv;          int              mypos, operand2;
         int              v;          char             operator;
   
         *val = NULL;          if (NULL == pos) {
         argv = *in;                  mypos = 0;
         assert(argv);                  pos = &mypos;
           }
   
         if (NULL == (arg = *argv))          if (skipwhite)
                 return(-1);                  while (isspace((unsigned char)v[*pos]))
         if ('-' != *arg)                          (*pos)++;
                 return(-1);  
   
         if (ROFF_ARGMAX == (v = rofffindarg(arg + 1))) {          if ( ! roff_evalpar(r, ln, v, pos, res))
                 roff_warn(tree, arg, "argument-like parameter `%s' to "                  return(0);
                                 "`%s'", &arg[1], toknames[tok]);  
                 return(-1);  
         }  
   
         if ( ! roffargok(tok, v)) {  
                 roff_warn(tree, arg, "invalid argument parameter `%s' to "  
                                 "`%s'", tokargnames[v], toknames[tok]);  
                 return(-1);  
         }  
   
         if ( ! (ROFF_VALUE & tokenargs[v]))  
                 return(v);  
   
         *in = ++argv;          while (1) {
                   if (skipwhite)
                           while (isspace((unsigned char)v[*pos]))
                                   (*pos)++;
   
         if (NULL == *argv) {                  if ( ! roff_getop(v, pos, &operator))
                 roff_err(tree, arg, "empty value of `%s' for `%s'",                          break;
                                 tokargnames[v], toknames[tok]);  
                 return(ROFF_ARGMAX);  
         }  
   
         return(v);                  if (skipwhite)
 }                          while (isspace((unsigned char)v[*pos]))
                                   (*pos)++;
   
                   if ( ! roff_evalpar(r, ln, v, pos, &operand2))
                           return(0);
   
 /* ARGSUSED */                  if (skipwhite)
 static  int                          while (isspace((unsigned char)v[*pos]))
 roff_Dd(ROFFCALL_ARGS)                                  (*pos)++;
   
                   if (NULL == res)
                           continue;
   
                   switch (operator) {
                   case '+':
                           *res += operand2;
                           break;
                   case '-':
                           *res -= operand2;
                           break;
                   case '*':
                           *res *= operand2;
                           break;
                   case '/':
                           if (operand2 == 0) {
                                   mandoc_msg(MANDOCERR_DIVZERO,
                                           r->parse, ln, *pos, v);
                                   *res = 0;
                                   break;
                           }
                           *res /= operand2;
                           break;
                   case '%':
                           if (operand2 == 0) {
                                   mandoc_msg(MANDOCERR_DIVZERO,
                                           r->parse, ln, *pos, v);
                                   *res = 0;
                                   break;
                           }
                           *res %= operand2;
                           break;
                   case '<':
                           *res = *res < operand2;
                           break;
                   case '>':
                           *res = *res > operand2;
                           break;
                   case 'l':
                           *res = *res <= operand2;
                           break;
                   case 'g':
                           *res = *res >= operand2;
                           break;
                   case '=':
                           *res = *res == operand2;
                           break;
                   case '!':
                           *res = *res != operand2;
                           break;
                   case '&':
                           *res = *res && operand2;
                           break;
                   case ':':
                           *res = *res || operand2;
                           break;
                   case 'i':
                           if (operand2 < *res)
                                   *res = operand2;
                           break;
                   case 'a':
                           if (operand2 > *res)
                                   *res = operand2;
                           break;
                   default:
                           abort();
                   }
           }
           return(1);
   }
   
   void
   roff_setreg(struct roff *r, const char *name, int val, char sign)
 {  {
           struct roffreg  *reg;
   
         if (ROFF_BODY & tree->state) {          /* Search for an existing register with the same name. */
                 assert( ! (ROFF_PRELUDE & tree->state));          reg = r->regtab;
                 assert(ROFF_PRELUDE_Dd & tree->state);  
                 return(roff_text(tok, tree, argv, type));          while (reg && strcmp(name, reg->key.p))
                   reg = reg->next;
   
           if (NULL == reg) {
                   /* Create a new register. */
                   reg = mandoc_malloc(sizeof(struct roffreg));
                   reg->key.p = mandoc_strdup(name);
                   reg->key.sz = strlen(name);
                   reg->val = 0;
                   reg->next = r->regtab;
                   r->regtab = reg;
         }          }
   
         assert(ROFF_PRELUDE & tree->state);          if ('+' == sign)
         assert( ! (ROFF_BODY & tree->state));                  reg->val += val;
           else if ('-' == sign)
                   reg->val -= val;
           else
                   reg->val = val;
   }
   
         if (ROFF_PRELUDE_Dd & tree->state) {  /*
                 roff_err(tree, *argv, "repeated `Dd' in prelude");   * Handle some predefined read-only number registers.
    * For now, return -1 if the requested register is not predefined;
    * in case a predefined read-only register having the value -1
    * were to turn up, another special value would have to be chosen.
    */
   static int
   roff_getregro(const char *name)
   {
   
           switch (*name) {
           case 'A':  /* ASCII approximation mode is always off. */
                 return(0);                  return(0);
         } else if (ROFF_PRELUDE_Dt & tree->state) {          case 'g':  /* Groff compatibility mode is always on. */
                 roff_err(tree, *argv, "out-of-order `Dd' in prelude");                  return(1);
           case 'H':  /* Fixed horizontal resolution. */
                   return (24);
           case 'j':  /* Always adjust left margin only. */
                 return(0);                  return(0);
           case 'T':  /* Some output device is always defined. */
                   return(1);
           case 'V':  /* Fixed vertical resolution. */
                   return (40);
           default:
                   return (-1);
         }          }
   }
   
         /* TODO: parse date. */  int
   roff_getreg(const struct roff *r, const char *name)
   {
           struct roffreg  *reg;
           int              val;
   
         assert(NULL == tree->last);          if ('.' == name[0] && '\0' != name[1] && '\0' == name[2]) {
         tree->state |= ROFF_PRELUDE_Dd;                  val = roff_getregro(name + 1);
                   if (-1 != val)
                           return (val);
           }
   
         return(1);          for (reg = r->regtab; reg; reg = reg->next)
                   if (0 == strcmp(name, reg->key.p))
                           return(reg->val);
   
           return(0);
 }  }
   
   static int
 /* ARGSUSED */  roff_getregn(const struct roff *r, const char *name, size_t len)
 static  int  
 roff_Dt(ROFFCALL_ARGS)  
 {  {
           struct roffreg  *reg;
           int              val;
   
         if (ROFF_BODY & tree->state) {          if ('.' == name[0] && 2 == len) {
                 assert( ! (ROFF_PRELUDE & tree->state));                  val = roff_getregro(name + 1);
                 assert(ROFF_PRELUDE_Dt & tree->state);                  if (-1 != val)
                 return(roff_text(tok, tree, argv, type));                          return (val);
         }          }
   
         assert(ROFF_PRELUDE & tree->state);          for (reg = r->regtab; reg; reg = reg->next)
         assert( ! (ROFF_BODY & tree->state));                  if (len == reg->key.sz &&
                       0 == strncmp(name, reg->key.p, len))
                           return(reg->val);
   
         if ( ! (ROFF_PRELUDE_Dd & tree->state)) {          return(0);
                 roff_err(tree, *argv, "out-of-order `Dt' in prelude");  }
                 return(0);  
         } else if (ROFF_PRELUDE_Dt & tree->state) {  static void
                 roff_err(tree, *argv, "repeated `Dt' in prelude");  roff_freereg(struct roffreg *reg)
                 return(0);  {
           struct roffreg  *old_reg;
   
           while (NULL != reg) {
                   free(reg->key.p);
                   old_reg = reg;
                   reg = reg->next;
                   free(old_reg);
         }          }
   }
   
         /* TODO: parse date. */  static enum rofferr
   roff_nr(ROFF_ARGS)
   {
           char            *key, *val;
           size_t           keysz;
           int              iv;
           char             sign;
   
         assert(NULL == tree->last);          key = val = buf->buf + pos;
         tree->state |= ROFF_PRELUDE_Dt;          if (*key == '\0')
                   return(ROFF_IGN);
   
         return(1);          keysz = roff_getname(r, &val, ln, pos);
           if (key[keysz] == '\\')
                   return(ROFF_IGN);
           key[keysz] = '\0';
   
           sign = *val;
           if (sign == '+' || sign == '-')
                   val++;
   
           if (roff_evalnum(r, ln, val, NULL, &iv, 0))
                   roff_setreg(r, key, iv, sign);
   
           return(ROFF_IGN);
 }  }
   
   static enum rofferr
   roff_rr(ROFF_ARGS)
   {
           struct roffreg  *reg, **prev;
           char            *name, *cp;
           size_t           namesz;
   
 /* ARGSUSED */          name = cp = buf->buf + pos;
 static  int          if (*name == '\0')
 roff_Os(ROFFCALL_ARGS)                  return(ROFF_IGN);
           namesz = roff_getname(r, &cp, ln, pos);
           name[namesz] = '\0';
   
           prev = &r->regtab;
           while (1) {
                   reg = *prev;
                   if (reg == NULL || !strcmp(name, reg->key.p))
                           break;
                   prev = &reg->next;
           }
           if (reg != NULL) {
                   *prev = reg->next;
                   free(reg->key.p);
                   free(reg);
           }
           return(ROFF_IGN);
   }
   
   static enum rofferr
   roff_rm(ROFF_ARGS)
 {  {
           const char       *name;
           char             *cp;
           size_t            namesz;
   
         if (ROFF_EXIT == type) {          cp = buf->buf + pos;
                 roffnode_free(tree);          while (*cp != '\0') {
                 return((*tree->cb.rofftail)(tree->arg));                  name = cp;
         } else if (ROFF_BODY & tree->state) {                  namesz = roff_getname(r, &cp, ln, (int)(cp - buf->buf));
                 assert( ! (ROFF_PRELUDE & tree->state));                  roff_setstrn(&r->strtab, name, namesz, NULL, 0, 0);
                 assert(ROFF_PRELUDE_Os & tree->state);                  if (name[namesz] == '\\')
                 return(roff_text(tok, tree, argv, type));                          break;
         }          }
           return(ROFF_IGN);
   }
   
         assert(ROFF_PRELUDE & tree->state);  static enum rofferr
         if ( ! (ROFF_PRELUDE_Dt & tree->state) ||  roff_it(ROFF_ARGS)
                         ! (ROFF_PRELUDE_Dd & tree->state)) {  {
                 roff_err(tree, *argv, "out-of-order `Os' in prelude");          char            *cp;
                 return(0);          size_t           len;
           int              iv;
   
           /* Parse the number of lines. */
           cp = buf->buf + pos;
           len = strcspn(cp, " \t");
           cp[len] = '\0';
           if ((iv = mandoc_strntoi(cp, len, 10)) <= 0) {
                   mandoc_msg(MANDOCERR_IT_NONUM, r->parse,
                       ln, ppos, buf->buf + 1);
                   return(ROFF_IGN);
         }          }
           cp += len + 1;
   
         /* TODO: extract OS. */          /* Arm the input line trap. */
           roffit_lines = iv;
           roffit_macro = mandoc_strdup(cp);
           return(ROFF_IGN);
   }
   
         tree->state |= ROFF_PRELUDE_Os;  static enum rofferr
         tree->state &= ~ROFF_PRELUDE;  roff_Dd(ROFF_ARGS)
         tree->state |= ROFF_BODY;  {
           const char *const       *cp;
   
         assert(NULL == tree->last);          if ((r->options & (MPARSE_MDOC | MPARSE_QUICK)) == 0)
                   for (cp = __mdoc_reserved; *cp; cp++)
                           roff_setstr(r, *cp, NULL, 0);
   
         if (NULL == roffnode_new(tok, tree))          if (r->format == 0)
                 return(0);                  r->format = MPARSE_MDOC;
   
         return((*tree->cb.roffhead)(tree->arg));          return(ROFF_CONT);
 }  }
   
   static enum rofferr
 /* ARGSUSED */  roff_TH(ROFF_ARGS)
 static int  
 roff_layout(ROFFCALL_ARGS)  
 {  {
         int              i, c, argcp[ROFF_MAXARG];          const char *const       *cp;
         char            *v, *argvp[ROFF_MAXARG];  
   
         if (ROFF_PRELUDE & tree->state) {          if ((r->options & MPARSE_QUICK) == 0)
                 roff_err(tree, *argv, "`%s' disallowed in prelude",                  for (cp = __man_reserved; *cp; cp++)
                                 toknames[tok]);                          roff_setstr(r, *cp, NULL, 0);
                 return(0);  
         }  
   
         if (ROFF_EXIT == type) {          if (r->format == 0)
                 roffnode_free(tree);                  r->format = MPARSE_MAN;
                 return((*tree->cb.roffblkout)(tree->arg, tok));  
         }  
   
         i = 0;          return(ROFF_CONT);
         argv++;  }
   
         while (-1 != (c = roffnextopt(tree, tok, &argv, &v))) {  static enum rofferr
                 if (ROFF_ARGMAX == c)  roff_TE(ROFF_ARGS)
                         return(0);  {
   
                 argcp[i] = c;          if (NULL == r->tbl)
                 argvp[i] = v;                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                 i++;                      ln, ppos, "TE");
                 argv++;          else if ( ! tbl_end(&r->tbl)) {
                   free(buf->buf);
                   buf->buf = mandoc_strdup(".sp");
                   buf->sz = 4;
                   return(ROFF_REPARSE);
         }          }
           return(ROFF_IGN);
   }
   
         argcp[i] = ROFF_ARGMAX;  static enum rofferr
         argvp[i] = NULL;  roff_T_(ROFF_ARGS)
   {
   
         if (NULL == roffnode_new(tok, tree))          if (NULL == r->tbl)
                 return(0);                  mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse,
                       ln, ppos, "T&");
           else
                   tbl_restart(ppos, ln, r->tbl);
   
         if ( ! (*tree->cb.roffblkin)(tree->arg, tok, argcp, argvp))          return(ROFF_IGN);
                 return(0);  }
   
         if (NULL == *argv)  /*
                 return(1);   * Handle in-line equation delimiters.
    */
   static enum rofferr
   roff_eqndelim(struct roff *r, struct buf *buf, int pos)
   {
           char            *cp1, *cp2;
           const char      *bef_pr, *bef_nl, *mac, *aft_nl, *aft_pr;
   
         if ( ! (*tree->cb.roffin)(tree->arg, tok, argcp, argvp))          /*
                 return(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.
            */
   
         if ( ! (ROFF_PARSED & tokens[tok].flags)) {          cp1 = buf->buf + pos;
                 i = 0;          cp2 = strchr(cp1, r->eqn == NULL ?
                 while (*argv) {              r->last_eqn->odelim : r->last_eqn->cdelim);
                         if ( ! (*tree->cb.roffdata)(tree->arg, i, *argv++))          if (cp2 == NULL)
                                 return(0);                  return(ROFF_CONT);
                         i = 1;  
                 }          *cp2++ = '\0';
                 return((*tree->cb.roffout)(tree->arg, tok));          bef_pr = bef_nl = aft_nl = aft_pr = "";
   
           /* Handle preceding text, protecting whitespace. */
   
           if (*buf->buf != '\0') {
                   if (r->eqn == NULL)
                           bef_pr = "\\&";
                   bef_nl = "\n";
         }          }
   
         i = 0;          /*
         while (*argv) {           * Prepare replacing the delimiter with an equation macro
                 if (ROFF_MAX == (c = rofffindcallable(*argv))) {           * and drop leading white space from the equation.
                         assert(tree->arg);           */
                         if ( ! (*tree->cb.roffdata)  
                                         (tree->arg, i, *argv++))  
                                 return(0);  
                         i = 1;  
                         continue;  
                 }  
   
                 if (NULL == tokens[c].cb) {          if (r->eqn == NULL) {
                         roff_err(tree, *argv, "unsupported macro `%s'",                  while (*cp2 == ' ')
                                         toknames[c]);                          cp2++;
                         return(0);                  mac = ".EQ";
                 }          } else
                   mac = ".EN";
   
                 if ( ! (*tokens[c].cb)(c, tree, argv, ROFF_ENTER))          /* Handle following text, protecting whitespace. */
                         return(0);  
   
                 break;          if (*cp2 != '\0') {
                   aft_nl = "\n";
                   if (r->eqn != NULL)
                           aft_pr = "\\&";
         }          }
   
         /*          /* Do the actual replacement. */
          * If we're the first parser (*argv == tree->cur) then purge out  
          * any additional punctuation, should there be any remaining at  
          * the end of line.  
          */  
   
         if ( ! (ROFF_PARSED & tokens[tok].flags && *argv))          buf->sz = mandoc_asprintf(&cp1, "%s%s%s%s%s%s%s", buf->buf,
                 return((*tree->cb.roffout)(tree->arg, tok));              bef_pr, bef_nl, mac, aft_nl, aft_pr, cp2) + 1;
           free(buf->buf);
           buf->buf = cp1;
   
         i = 0;          /* Toggle the in-line state of the eqn subsystem. */
         while (argv[i])  
                 i++;  
   
         assert(i > 0);          r->eqn_inline = r->eqn == NULL;
         if ( ! roffispunct(argv[--i]))          return(ROFF_REPARSE);
                 return((*tree->cb.roffout)(tree->arg, tok));  }
   
         while (i >= 0 && roffispunct(argv[i]))  static enum rofferr
                 i--;  roff_EQ(ROFF_ARGS)
   {
           struct eqn_node *e;
   
         assert(0 != i);          assert(r->eqn == NULL);
         i++;          e = eqn_alloc(ppos, ln, r->parse);
   
         /* LINTED */          if (r->last_eqn) {
         while (argv[i])                  r->last_eqn->next = e;
                 if ( ! (*tree->cb.roffdata)(tree->arg, 0, argv[i++]))                  e->delim = r->last_eqn->delim;
                         return(0);                  e->odelim = r->last_eqn->odelim;
                   e->cdelim = r->last_eqn->cdelim;
           } else
                   r->first_eqn = r->last_eqn = e;
   
         return((*tree->cb.roffout)(tree->arg, tok));          r->eqn = r->last_eqn = e;
   
           if (buf->buf[pos] != '\0')
                   mandoc_vmsg(MANDOCERR_ARG_SKIP, r->parse, ln, pos,
                       ".EQ %s", buf->buf + pos);
   
           return(ROFF_IGN);
 }  }
   
   static enum rofferr
   roff_EN(ROFF_ARGS)
   {
   
 /* ARGSUSED */          mandoc_msg(MANDOCERR_BLK_NOTOPEN, r->parse, ln, ppos, "EN");
 static int          return(ROFF_IGN);
 roff_text(ROFFCALL_ARGS)  }
   
   static enum rofferr
   roff_TS(ROFF_ARGS)
 {  {
         int              i, j, first, c, argcp[ROFF_MAXARG];          struct tbl_node *tbl;
         char            *v, *argvp[ROFF_MAXARG];  
   
         if (ROFF_PRELUDE & tree->state) {          if (r->tbl) {
                 roff_err(tree, *argv, "`%s' disallowed in prelude",                  mandoc_msg(MANDOCERR_BLK_BROKEN, r->parse,
                                 toknames[tok]);                      ln, ppos, "TS breaks TS");
                 return(0);                  tbl_end(&r->tbl);
         }          }
   
         /* FIXME: breaks if passed from roff_layout. */          tbl = tbl_alloc(ppos, ln, r->parse);
         first = *argv == tree->cur;  
   
         i = 0;          if (r->last_tbl)
         argv++;                  r->last_tbl->next = tbl;
           else
                   r->first_tbl = r->last_tbl = tbl;
   
         while (-1 != (c = roffnextopt(tree, tok, &argv, &v))) {          r->tbl = r->last_tbl = tbl;
                 if (ROFF_ARGMAX == c)          return(ROFF_IGN);
                         return(0);  }
   
                 argcp[i] = c;  static enum rofferr
                 argvp[i] = v;  roff_brp(ROFF_ARGS)
                 i++;  {
                 argv++;  
           buf->buf[pos - 1] = '\0';
           return(ROFF_CONT);
   }
   
   static enum rofferr
   roff_cc(ROFF_ARGS)
   {
           const char      *p;
   
           p = buf->buf + pos;
   
           if (*p == '\0' || (r->control = *p++) == '.')
                   r->control = 0;
   
           if (*p != '\0')
                   mandoc_msg(MANDOCERR_ARGCOUNT, r->parse, ln, ppos, NULL);
   
           return(ROFF_IGN);
   }
   
   static enum rofferr
   roff_tr(ROFF_ARGS)
   {
           const char      *p, *first, *second;
           size_t           fsz, ssz;
           enum mandoc_esc  esc;
   
           p = buf->buf + pos;
   
           if (*p == '\0') {
                   mandoc_msg(MANDOCERR_ARGCOUNT, r->parse, ln, ppos, NULL);
                   return(ROFF_IGN);
         }          }
   
         argcp[i] = ROFF_ARGMAX;          while (*p != '\0') {
         argvp[i] = NULL;                  fsz = ssz = 1;
   
         if ( ! (*tree->cb.roffin)(tree->arg, tok, argcp, argvp))                  first = p++;
                 return(0);                  if (*first == '\\') {
                           esc = mandoc_escape(&p, NULL, NULL);
                           if (esc == ESCAPE_ERROR) {
                                   mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                                       ln, (int)(p - buf->buf), first);
                                   return(ROFF_IGN);
                           }
                           fsz = (size_t)(p - first);
                   }
   
         if ( ! (ROFF_PARSED & tokens[tok].flags)) {                  second = p++;
                 i = 0;                  if (*second == '\\') {
                 while (*argv) {                          esc = mandoc_escape(&p, NULL, NULL);
                         if ( ! (*tree->cb.roffdata)(tree->arg, i, *argv++))                          if (esc == ESCAPE_ERROR) {
                                 return(0);                                  mandoc_msg(MANDOCERR_ESC_BAD, r->parse,
                         i = 1;                                      ln, (int)(p - buf->buf), second);
                                   return(ROFF_IGN);
                           }
                           ssz = (size_t)(p - second);
                   } else if (*second == '\0') {
                           mandoc_msg(MANDOCERR_ARGCOUNT, r->parse,
                               ln, (int)(p - buf->buf), NULL);
                           second = " ";
                           p--;
                 }                  }
                 return((*tree->cb.roffout)(tree->arg, tok));  
                   if (fsz > 1) {
                           roff_setstrn(&r->xmbtab, first, fsz,
                               second, ssz, 0);
                           continue;
                   }
   
                   if (r->xtab == NULL)
                           r->xtab = mandoc_calloc(128,
                               sizeof(struct roffstr));
   
                   free(r->xtab[(int)*first].p);
                   r->xtab[(int)*first].p = mandoc_strndup(second, ssz);
                   r->xtab[(int)*first].sz = ssz;
         }          }
   
         i = 0;          return(ROFF_IGN);
         while (*argv) {  }
                 if (ROFF_MAX == (c = rofffindcallable(*argv))) {  
                         /*  
                          * If all that remains is roff punctuation, then  
                          * close out our scope and return.  
                          */  
                         if (roffispunct(*argv)) {  
                                 for (j = 0; argv[j]; j++)  
                                         if ( ! roffispunct(argv[j]))  
                                                 break;  
                                 if (NULL == argv[j])  
                                         break;  
                                 i = 1;  
                         }  
   
                         if ( ! (*tree->cb.roffdata)  
                                         (tree->arg, i, *argv++))  
                                 return(0);  
   
                         i = 1;  static enum rofferr
   roff_so(ROFF_ARGS)
   {
           char *name, *cp;
   
           name = buf->buf + pos;
           mandoc_vmsg(MANDOCERR_SO, r->parse, ln, ppos, "so %s", name);
   
           /*
            * Handle `so'.  Be EXTREMELY careful, as we shouldn't be
            * opening anything that's not in our cwd or anything beneath
            * it.  Thus, explicitly disallow traversing up the file-system
            * or using absolute paths.
            */
   
           if (*name == '/' || strstr(name, "../") || strstr(name, "/..")) {
                   mandoc_vmsg(MANDOCERR_SO_PATH, r->parse, ln, ppos,
                       ".so %s", name);
                   buf->sz = mandoc_asprintf(&cp,
                       ".sp\nSee the file %s.\n.sp", name) + 1;
                   free(buf->buf);
                   buf->buf = cp;
                   *offs = 0;
                   return(ROFF_REPARSE);
           }
   
           *offs = pos;
           return(ROFF_SO);
   }
   
   static enum rofferr
   roff_userdef(ROFF_ARGS)
   {
           const char       *arg[9];
           char             *cp, *n1, *n2;
           int               i;
   
           /*
            * Collect pointers to macro argument strings
            * and NUL-terminate them.
            */
           cp = buf->buf + pos;
           for (i = 0; i < 9; i++)
                   arg[i] = *cp == '\0' ? "" :
                       mandoc_getarg(r->parse, &cp, ln, &pos);
   
           /*
            * Expand macro arguments.
            */
           buf->sz = 0;
           n1 = cp = mandoc_strdup(r->current_string);
           while ((cp = strstr(cp, "\\$")) != NULL) {
                   i = cp[2] - '1';
                   if (0 > i || 8 < i) {
                           /* Not an argument invocation. */
                           cp += 2;
                         continue;                          continue;
                 }                  }
                   *cp = '\0';
                   buf->sz = mandoc_asprintf(&n2, "%s%s%s",
                       n1, arg[i], cp + 3) + 1;
                   cp = n2 + (cp - n1);
                   free(n1);
                   n1 = n2;
           }
   
                 /*          /*
                  * A sub-command has been found.  Execute it and           * Replace the macro invocation
                  * discontinue parsing for arguments.           * by the expanded macro.
                  */           */
           free(buf->buf);
           buf->buf = n1;
           if (buf->sz == 0)
                   buf->sz = strlen(buf->buf) + 1;
           *offs = 0;
   
                 if (NULL == tokens[c].cb) {          return(buf->sz > 1 && buf->buf[buf->sz - 2] == '\n' ?
                         roff_err(tree, *argv, "unsupported macro `%s'",             ROFF_REPARSE : ROFF_APPEND);
                                         toknames[c]);  }
                         return(0);  
                 }  
   
                 if ( ! (*tokens[c].cb)(c, tree, argv, ROFF_ENTER))  
                         return(0);  
   
   static size_t
   roff_getname(struct roff *r, char **cpp, int ln, int pos)
   {
           char     *name, *cp;
           size_t    namesz;
   
           name = *cpp;
           if ('\0' == *name)
                   return(0);
   
           /* Read until end of name and terminate it with NUL. */
           for (cp = name; 1; cp++) {
                   if ('\0' == *cp || ' ' == *cp) {
                           namesz = cp - name;
                           break;
                   }
                   if ('\\' != *cp)
                           continue;
                   namesz = cp - name;
                   if ('{' == cp[1] || '}' == cp[1])
                           break;
                   cp++;
                   if ('\\' == *cp)
                           continue;
                   mandoc_vmsg(MANDOCERR_NAMESC, r->parse, ln, pos,
                       "%.*s", (int)(cp - name + 1), name);
                   mandoc_escape((const char **)&cp, NULL, NULL);
                 break;                  break;
         }          }
   
         if ( ! (*tree->cb.roffout)(tree->arg, tok))          /* Read past spaces. */
                 return(0);          while (' ' == *cp)
                   cp++;
   
         /*          *cpp = cp;
          * If we're the first parser (*argv == tree->cur) then purge out          return(namesz);
          * any additional punctuation, should there be any remaining at  }
          * the end of line.  
   /*
    * Store *string into the user-defined string called *name.
    * To clear an existing entry, call with (*r, *name, NULL, 0).
    * append == 0: replace mode
    * append == 1: single-line append mode
    * append == 2: multiline append mode, append '\n' after each call
    */
   static void
   roff_setstr(struct roff *r, const char *name, const char *string,
           int append)
   {
   
           roff_setstrn(&r->strtab, name, strlen(name), string,
               string ? strlen(string) : 0, append);
   }
   
   static void
   roff_setstrn(struct roffkv **r, const char *name, size_t namesz,
                   const char *string, size_t stringsz, int append)
   {
           struct roffkv   *n;
           char            *c;
           int              i;
           size_t           oldch, newch;
   
           /* Search for an existing string with the same name. */
           n = *r;
   
           while (n && (namesz != n->key.sz ||
                           strncmp(n->key.p, name, namesz)))
                   n = n->next;
   
           if (NULL == n) {
                   /* Create a new string table entry. */
                   n = mandoc_malloc(sizeof(struct roffkv));
                   n->key.p = mandoc_strndup(name, namesz);
                   n->key.sz = namesz;
                   n->val.p = NULL;
                   n->val.sz = 0;
                   n->next = *r;
                   *r = n;
           } else if (0 == append) {
                   free(n->val.p);
                   n->val.p = NULL;
                   n->val.sz = 0;
           }
   
           if (NULL == string)
                   return;
   
           /*
            * One additional byte for the '\n' in multiline mode,
            * and one for the terminating '\0'.
          */           */
           newch = stringsz + (1 < append ? 2u : 1u);
   
         if ( ! (first && *argv))          if (NULL == n->val.p) {
                 return(1);                  n->val.p = mandoc_malloc(newch);
                   *n->val.p = '\0';
                   oldch = 0;
           } else {
                   oldch = n->val.sz;
                   n->val.p = mandoc_realloc(n->val.p, oldch + newch);
           }
   
           /* Skip existing content in the destination buffer. */
           c = n->val.p + (int)oldch;
   
           /* Append new content to the destination buffer. */
         i = 0;          i = 0;
         while (argv[i])          while (i < (int)stringsz) {
                 i++;                  /*
                    * Rudimentary roff copy mode:
                    * Handle escaped backslashes.
                    */
                   if ('\\' == string[i] && '\\' == string[i + 1])
                           i++;
                   *c++ = string[i++];
           }
   
         assert(i > 0);          /* Append terminating bytes. */
         if ( ! roffispunct(argv[--i]))          if (1 < append)
                 return(1);                  *c++ = '\n';
   
         while (i >= 0 && roffispunct(argv[i]))          *c = '\0';
                 i--;          n->val.sz = (int)(c - n->val.p);
   }
   
         assert(0 != i);  static const char *
         i++;  roff_getstrn(const struct roff *r, const char *name, size_t len)
   {
           const struct roffkv *n;
           int i;
   
         /* LINTED */          for (n = r->strtab; n; n = n->next)
         while (argv[i])                  if (0 == strncmp(name, n->key.p, len) &&
                 if ( ! (*tree->cb.roffdata)(tree->arg, 0, argv[i++]))                      '\0' == n->key.p[(int)len])
                         return(0);                          return(n->val.p);
   
         return(1);          for (i = 0; i < PREDEFS_MAX; i++)
                   if (0 == strncmp(name, predefs[i].name, len) &&
                                   '\0' == predefs[i].name[(int)len])
                           return(predefs[i].str);
   
           return(NULL);
 }  }
   
   static void
 /* ARGSUSED */  roff_freestr(struct roffkv *r)
 static int  
 roff_comment(ROFFCALL_ARGS)  
 {  {
           struct roffkv    *n, *nn;
   
         return(1);          for (n = r; n; n = nn) {
                   free(n->key.p);
                   free(n->val.p);
                   nn = n->next;
                   free(n);
           }
 }  }
   
   const struct tbl_span *
 /* ARGSUSED */  roff_span(const struct roff *r)
 static int  
 roff_close(ROFFCALL_ARGS)  
 {  {
   
         return(1);          return(r->tbl ? tbl_span(r->tbl) : NULL);
 }  }
   
   const struct eqn *
   roff_eqn(const struct roff *r)
   {
   
 #if notyet          return(r->last_eqn ? &r->last_eqn->eqn : NULL);
 /* ARGSUSED */  }
 static int  
 roff_Ns(ROFFCALL_ARGS)  /*
    * Duplicate an input string, making the appropriate character
    * conversations (as stipulated by `tr') along the way.
    * Returns a heap-allocated string with all the replacements made.
    */
   char *
   roff_strdup(const struct roff *r, const char *p)
 {  {
         int              c;          const struct roffkv *cp;
           char            *res;
           const char      *pp;
           size_t           ssz, sz;
           enum mandoc_esc  esc;
   
         argv++;          if (NULL == r->xmbtab && NULL == r->xtab)
                   return(mandoc_strdup(p));
           else if ('\0' == *p)
                   return(mandoc_strdup(""));
   
         if (ROFF_MAX != (c = rofffindcallable(*argv))) {          /*
                 if (NULL == tokens[c].cb) {           * Step through each character looking for term matches
                         roff_err(tree, *argv, "unsupported macro `%s'",           * (remember that a `tr' can be invoked with an escape, which is
                                         toknames[c]);           * a glyph but the escape is multi-character).
                         return(0);           * We only do this if the character hash has been initialised
            * and the string is >0 length.
            */
   
           res = NULL;
           ssz = 0;
   
           while ('\0' != *p) {
                   if ('\\' != *p && r->xtab && r->xtab[(int)*p].p) {
                           sz = r->xtab[(int)*p].sz;
                           res = mandoc_realloc(res, ssz + sz + 1);
                           memcpy(res + ssz, r->xtab[(int)*p].p, sz);
                           ssz += sz;
                           p++;
                           continue;
                   } else if ('\\' != *p) {
                           res = mandoc_realloc(res, ssz + 2);
                           res[ssz++] = *p++;
                           continue;
                 }                  }
                 if ( ! (*tree->cb.roffspecial)(tree->arg, tok))  
                         return(0);  
                 if ( ! (*tokens[c].cb)(c, tree, argv, ROFF_ENTER))  
                         return(0);  
   
                 return(1);                  /* Search for term matches. */
         } else if ( ! (*tree->cb.roffdata)(tree->arg, 0, *argv++))                  for (cp = r->xmbtab; cp; cp = cp->next)
                 return(0);                          if (0 == strncmp(p, cp->key.p, cp->key.sz))
                                   break;
         while (*argv) {  
                 if (ROFF_MAX == (c = rofffindcallable(*argv))) {                  if (NULL != cp) {
                         assert(tree->arg);                          /*
                         if ( ! (*tree->cb.roffdata)                           * A match has been found.
                                         (tree->arg, 1, *argv++))                           * Append the match to the array and move
                                 return(0);                           * forward by its keysize.
                            */
                           res = mandoc_realloc(res,
                               ssz + cp->val.sz + 1);
                           memcpy(res + ssz, cp->val.p, cp->val.sz);
                           ssz += cp->val.sz;
                           p += (int)cp->key.sz;
                         continue;                          continue;
                 }                  }
                 if (NULL == tokens[c].cb) {  
                         roff_err(tree, *argv, "unsupported macro `%s'",  
                                         toknames[c]);  
                         return(0);  
                 }  
                 if ( ! (*tokens[c].cb)(c, tree, argv, ROFF_ENTER))  
                         return(0);  
   
                 break;                  /*
                    * Handle escapes carefully: we need to copy
                    * over just the escape itself, or else we might
                    * do replacements within the escape itself.
                    * Make sure to pass along the bogus string.
                    */
                   pp = p++;
                   esc = mandoc_escape(&p, NULL, NULL);
                   if (ESCAPE_ERROR == esc) {
                           sz = strlen(pp);
                           res = mandoc_realloc(res, ssz + sz + 1);
                           memcpy(res + ssz, pp, sz);
                           break;
                   }
                   /*
                    * We bail out on bad escapes.
                    * No need to warn: we already did so when
                    * roff_res() was called.
                    */
                   sz = (int)(p - pp);
                   res = mandoc_realloc(res, ssz + sz + 1);
                   memcpy(res + ssz, pp, sz);
                   ssz += sz;
         }          }
   
         return(1);          res[(int)ssz] = '\0';
           return(res);
 }  }
 #endif  
   
   int
 static void  roff_getformat(const struct roff *r)
 roff_warn(const struct rofftree *tree, const char *pos, char *fmt, ...)  
 {  {
         va_list          ap;  
         char             buf[128];  
   
         va_start(ap, fmt);          return(r->format);
         (void)vsnprintf(buf, sizeof(buf), fmt, ap);  
         va_end(ap);  
   
         (*tree->cb.roffmsg)(tree->arg,  
                         ROFF_WARN, tree->cur, pos, buf);  
 }  }
   
   /*
 static void   * Find out whether a line is a macro line or not.
 roff_err(const struct rofftree *tree, const char *pos, char *fmt, ...)   * If it is, adjust the current position and return one; if it isn't,
    * return zero and don't change the current position.
    * If the control character has been set with `.cc', then let that grain
    * precedence.
    * This is slighly contrary to groff, where using the non-breaking
    * control character when `cc' has been invoked will cause the
    * non-breaking macro contents to be printed verbatim.
    */
   int
   roff_getcontrol(const struct roff *r, const char *cp, int *ppos)
 {  {
         va_list          ap;          int             pos;
         char             buf[128];  
   
         va_start(ap, fmt);          pos = *ppos;
         (void)vsnprintf(buf, sizeof(buf), fmt, ap);  
         va_end(ap);  
   
         (*tree->cb.roffmsg)(tree->arg,          if (0 != r->control && cp[pos] == r->control)
                         ROFF_ERROR, tree->cur, pos, buf);                  pos++;
           else if (0 != r->control)
                   return(0);
           else if ('\\' == cp[pos] && '.' == cp[pos + 1])
                   pos += 2;
           else if ('.' == cp[pos] || '\'' == cp[pos])
                   pos++;
           else
                   return(0);
   
           while (' ' == cp[pos] || '\t' == cp[pos])
                   pos++;
   
           *ppos = pos;
           return(1);
 }  }

Legend:
Removed from v.1.26  
changed lines
  Added in v.1.258

CVSweb