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

Diff for /mandoc/mdoc.c between version 1.138 and 1.155

version 1.138, 2010/05/25 12:37:20 version 1.155, 2010/07/01 23:01:47
Line 1 
Line 1 
 /*      $Id$ */  /*      $Id$ */
 /*  /*
  * Copyright (c) 2008, 2009 Kristaps Dzonsons <kristaps@kth.se>   * Copyright (c) 2008, 2009 Kristaps Dzonsons <kristaps@bsd.lv>
  *   *
  * 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 above   * purpose with or without fee is hereby granted, provided that the above
Line 29 
Line 29 
 #include <time.h>  #include <time.h>
   
 #include "mandoc.h"  #include "mandoc.h"
   #include "regs.h"
 #include "libmdoc.h"  #include "libmdoc.h"
 #include "libmandoc.h"  #include "libmandoc.h"
   
Line 69  const char *const __mdoc_macronames[MDOC_MAX] = {   
Line 70  const char *const __mdoc_macronames[MDOC_MAX] = {   
         /* LINTED */          /* LINTED */
         "Dx",           "%Q",           "br",           "sp",          "Dx",           "%Q",           "br",           "sp",
         /* LINTED */          /* LINTED */
         "%U"          "%U",           "Ta"
         };          };
   
 const   char *const __mdoc_argnames[MDOC_ARG_MAX] = {  const   char *const __mdoc_argnames[MDOC_ARG_MAX] = {
Line 191  mdoc_free(struct mdoc *mdoc)
Line 192  mdoc_free(struct mdoc *mdoc)
  * Allocate volatile and non-volatile parse resources.   * Allocate volatile and non-volatile parse resources.
  */   */
 struct mdoc *  struct mdoc *
 mdoc_alloc(void *data, int pflags, mandocmsg msg)  mdoc_alloc(struct regset *regs, void *data,
                   int pflags, mandocmsg msg)
 {  {
         struct mdoc     *p;          struct mdoc     *p;
   
Line 200  mdoc_alloc(void *data, int pflags, mandocmsg msg)
Line 202  mdoc_alloc(void *data, int pflags, mandocmsg msg)
         p->msg = msg;          p->msg = msg;
         p->data = data;          p->data = data;
         p->pflags = pflags;          p->pflags = pflags;
           p->regs = regs;
   
         mdoc_hash_init();          mdoc_hash_init();
         mdoc_alloc1(p);          mdoc_alloc1(p);
Line 236  mdoc_parseln(struct mdoc *m, int ln, char *buf, int of
Line 239  mdoc_parseln(struct mdoc *m, int ln, char *buf, int of
                 return(0);                  return(0);
   
         m->flags |= MDOC_NEWLINE;          m->flags |= MDOC_NEWLINE;
   
           /*
            * Let the roff nS register switch SYNOPSIS mode early,
            * such that the parser knows at all times
            * whether this mode is on or off.
            * Note that this mode is also switched by the Sh macro.
            */
           if (m->regs->regs[(int)REG_nS].set) {
                   if (m->regs->regs[(int)REG_nS].v.u)
                           m->flags |= MDOC_SYNOPSIS;
                   else
                           m->flags &= ~MDOC_SYNOPSIS;
           }
   
         return(('.' == buf[offs] || '\'' == buf[offs]) ?          return(('.' == buf[offs] || '\'' == buf[offs]) ?
                         mdoc_pmacro(m, ln, buf, offs) :                          mdoc_pmacro(m, ln, buf, offs) :
                         mdoc_ptext(m, ln, buf, offs));                          mdoc_ptext(m, ln, buf, offs));
Line 258  mdoc_vmsg(struct mdoc *mdoc, enum mandocerr t, 
Line 275  mdoc_vmsg(struct mdoc *mdoc, enum mandocerr t, 
   
   
 int  int
 mdoc_macro(struct mdoc *m, enum mdoct tok,  mdoc_macro(MACRO_PROT_ARGS)
                 int ln, int pp, int *pos, char *buf)  
 {  {
         assert(tok < MDOC_MAX);          assert(tok < MDOC_MAX);
   
Line 267  mdoc_macro(struct mdoc *m, enum mdoct tok, 
Line 283  mdoc_macro(struct mdoc *m, enum mdoct tok, 
   
         if (MDOC_PROLOGUE & mdoc_macros[tok].flags &&          if (MDOC_PROLOGUE & mdoc_macros[tok].flags &&
                         MDOC_PBODY & m->flags)                          MDOC_PBODY & m->flags)
                 return(mdoc_pmsg(m, ln, pp, MANDOCERR_BADBODY));                  return(mdoc_pmsg(m, line, ppos, MANDOCERR_BADBODY));
   
         /* If we're in the prologue, deny "body" macros.  */          /* If we're in the prologue, deny "body" macros.  */
   
         if ( ! (MDOC_PROLOGUE & mdoc_macros[tok].flags) &&          if ( ! (MDOC_PROLOGUE & mdoc_macros[tok].flags) &&
                         ! (MDOC_PBODY & m->flags)) {                          ! (MDOC_PBODY & m->flags)) {
                 if ( ! mdoc_pmsg(m, ln, pp, MANDOCERR_BADPROLOG))                  if ( ! mdoc_pmsg(m, line, ppos, MANDOCERR_BADPROLOG))
                         return(0);                          return(0);
                 if (NULL == m->meta.title)                  if (NULL == m->meta.title)
                         m->meta.title = mandoc_strdup("unknown");                          m->meta.title = mandoc_strdup("UNKNOWN");
                 if (NULL == m->meta.vol)                  if (NULL == m->meta.vol)
                         m->meta.vol = mandoc_strdup("local");                          m->meta.vol = mandoc_strdup("LOCAL");
                 if (NULL == m->meta.os)                  if (NULL == m->meta.os)
                         m->meta.os = mandoc_strdup("local");                          m->meta.os = mandoc_strdup("LOCAL");
                 if (0 == m->meta.date)                  if (0 == m->meta.date)
                         m->meta.date = time(NULL);                          m->meta.date = time(NULL);
                 m->flags |= MDOC_PBODY;                  m->flags |= MDOC_PBODY;
         }          }
   
         return((*mdoc_macros[tok].fp)(m, tok, ln, pp, pos, buf));          return((*mdoc_macros[tok].fp)(m, tok, line, ppos, pos, buf));
 }  }
   
   
Line 330  node_append(struct mdoc *mdoc, struct mdoc_node *p)
Line 346  node_append(struct mdoc *mdoc, struct mdoc_node *p)
                 p->parent->tail = p;                  p->parent->tail = p;
                 break;                  break;
         case (MDOC_BODY):          case (MDOC_BODY):
                   if (p->end)
                           break;
                 assert(MDOC_BLOCK == p->parent->type);                  assert(MDOC_BLOCK == p->parent->type);
                 p->parent->body = p;                  p->parent->body = p;
                 break;                  break;
Line 366  node_alloc(struct mdoc *m, int line, int pos, 
Line 384  node_alloc(struct mdoc *m, int line, int pos, 
         p->pos = pos;          p->pos = pos;
         p->tok = tok;          p->tok = tok;
         p->type = type;          p->type = type;
   
           /* Flag analysis. */
   
           if (MDOC_SYNOPSIS & m->flags)
                   p->flags |= MDOC_SYNPRETTY;
           else
                   p->flags &= ~MDOC_SYNPRETTY;
         if (MDOC_NEWLINE & m->flags)          if (MDOC_NEWLINE & m->flags)
                 p->flags |= MDOC_LINE;                  p->flags |= MDOC_LINE;
         m->flags &= ~MDOC_NEWLINE;          m->flags &= ~MDOC_NEWLINE;
   
         return(p);          return(p);
 }  }
   
Line 416  mdoc_body_alloc(struct mdoc *m, int line, int pos, enu
Line 442  mdoc_body_alloc(struct mdoc *m, int line, int pos, enu
   
   
 int  int
   mdoc_endbody_alloc(struct mdoc *m, int line, int pos, enum mdoct tok,
                   struct mdoc_node *body, enum mdoc_endbody end)
   {
           struct mdoc_node *p;
   
           p = node_alloc(m, line, pos, tok, MDOC_BODY);
           p->pending = body;
           p->end = end;
           if ( ! node_append(m, p))
                   return(0);
           m->next = MDOC_NEXT_SIBLING;
           return(1);
   }
   
   
   int
 mdoc_block_alloc(struct mdoc *m, int line, int pos,  mdoc_block_alloc(struct mdoc *m, int line, int pos,
                 enum mdoct tok, struct mdoc_arg *args)                  enum mdoct tok, struct mdoc_arg *args)
 {  {
Line 472  mdoc_word_alloc(struct mdoc *m, int line, int pos, con
Line 514  mdoc_word_alloc(struct mdoc *m, int line, int pos, con
 }  }
   
   
 void  static void
 mdoc_node_free(struct mdoc_node *p)  mdoc_node_free(struct mdoc_node *p)
 {  {
   
           if (MDOC_Bd == p->tok && MDOC_BLOCK == p->type)
                   if (p->data.Bd)
                           free(p->data.Bd);
           if (MDOC_Bl == p->tok && MDOC_BLOCK == p->type)
                   if (p->data.Bl)
                           free(p->data.Bl);
   
         if (p->string)          if (p->string)
                 free(p->string);                  free(p->string);
         if (p->args)          if (p->args)
Line 542  mdoc_node_delete(struct mdoc *m, struct mdoc_node *p)
Line 591  mdoc_node_delete(struct mdoc *m, struct mdoc_node *p)
 static int  static int
 mdoc_ptext(struct mdoc *m, int line, char *buf, int offs)  mdoc_ptext(struct mdoc *m, int line, char *buf, int offs)
 {  {
         char            *c, *ws, *end;          char             *c, *ws, *end;
           struct mdoc_node *n;
   
         /* Ignore bogus comments. */          /* Ignore bogus comments. */
   
Line 556  mdoc_ptext(struct mdoc *m, int line, char *buf, int of
Line 606  mdoc_ptext(struct mdoc *m, int line, char *buf, int of
         if (SEC_NONE == m->lastnamed)          if (SEC_NONE == m->lastnamed)
                 return(mdoc_pmsg(m, line, offs, MANDOCERR_NOTEXT));                  return(mdoc_pmsg(m, line, offs, MANDOCERR_NOTEXT));
   
           assert(m->last);
           n = m->last;
   
         /*          /*
            * Divert directly to list processing if we're encountering a
            * columnar MDOC_BLOCK with or without a prior MDOC_BLOCK entry
            * (a MDOC_BODY means it's already open, in which case we should
            * process within its context in the normal way).
            */
   
           if (MDOC_Bl == n->tok && MDOC_BODY == n->type &&
                           LIST_column == n->data.Bl->type) {
                   /* `Bl' is open without any children. */
                   m->flags |= MDOC_FREECOL;
                   return(mdoc_macro(m, MDOC_It, line, offs, &offs, buf));
           }
   
           if (MDOC_It == n->tok && MDOC_BLOCK == n->type &&
                           NULL != n->parent &&
                           MDOC_Bl == n->parent->tok &&
                           LIST_column == n->parent->data.Bl->type) {
                   /* `Bl' has block-level `It' children. */
                   m->flags |= MDOC_FREECOL;
                   return(mdoc_macro(m, MDOC_It, line, offs, &offs, buf));
           }
   
           /*
          * Search for the beginning of unescaped trailing whitespace (ws)           * Search for the beginning of unescaped trailing whitespace (ws)
          * and for the first character not to be output (end).           * and for the first character not to be output (end).
          */           */
   
           /* FIXME: replace with strcspn(). */
         ws = NULL;          ws = NULL;
         for (c = end = buf + offs; *c; c++) {          for (c = end = buf + offs; *c; c++) {
                 switch (*c) {                  switch (*c) {
                 case '-':                  case '-':
                         if (mandoc_hyph(buf + offs, c))                          if (mandoc_hyph(buf + offs, c))
                                 *c = ASCII_HYPH;                                  *c = ASCII_HYPH;
                           ws = NULL;
                         break;                          break;
                 case ' ':                  case ' ':
                         if (NULL == ws)                          if (NULL == ws)
Line 650  macrowarn(struct mdoc *m, int ln, const char *buf, int
Line 729  macrowarn(struct mdoc *m, int ln, const char *buf, int
                         buf, strlen(buf) > 3 ? "..." : "");                          buf, strlen(buf) > 3 ? "..." : "");
   
         /* FIXME: logic should be in driver. */          /* FIXME: logic should be in driver. */
           /* FIXME: broken, will error out and not omit a message. */
         return(MDOC_IGN_MACRO & m->pflags ? rc : 0);          return(MDOC_IGN_MACRO & m->pflags ? rc : 0);
 }  }
   
Line 658  macrowarn(struct mdoc *m, int ln, const char *buf, int
Line 738  macrowarn(struct mdoc *m, int ln, const char *buf, int
  * Parse a macro line, that is, a line beginning with the control   * Parse a macro line, that is, a line beginning with the control
  * character.   * character.
  */   */
 int  static int
 mdoc_pmacro(struct mdoc *m, int ln, char *buf, int offs)  mdoc_pmacro(struct mdoc *m, int ln, char *buf, int offs)
 {  {
         enum mdoct      tok;          enum mdoct        tok;
         int             i, j, sv;          int               i, j, sv;
         char            mac[5];          char              mac[5];
           struct mdoc_node *n;
   
         /* Empty lines are ignored. */          /* Empty lines are ignored. */
   
Line 731  mdoc_pmacro(struct mdoc *m, int ln, char *buf, int off
Line 812  mdoc_pmacro(struct mdoc *m, int ln, char *buf, int off
                 if ( ! mdoc_pmsg(m, ln, i - 1, MANDOCERR_EOLNSPACE))                  if ( ! mdoc_pmsg(m, ln, i - 1, MANDOCERR_EOLNSPACE))
                         goto err;                          goto err;
   
         /*          /*
          * Begin recursive parse sequence.  Since we're at the start of           * If an initial macro or a list invocation, divert directly
          * the line, we don't need to do callable/parseable checks.           * into macro processing.
          */           */
   
           if (NULL == m->last || MDOC_It == tok || MDOC_El == tok) {
                   if ( ! mdoc_macro(m, tok, ln, sv, &i, buf))
                           goto err;
                   return(1);
           }
   
           n = m->last;
           assert(m->last);
   
           /*
            * If the first macro of a `Bl -column', open an `It' block
            * context around the parsed macro.
            */
   
           if (MDOC_Bl == n->tok && MDOC_BODY == n->type &&
                           LIST_column == n->data.Bl->type) {
                   m->flags |= MDOC_FREECOL;
                   if ( ! mdoc_macro(m, MDOC_It, ln, sv, &sv, buf))
                           goto err;
                   return(1);
           }
   
           /*
            * If we're following a block-level `It' within a `Bl -column'
            * context (perhaps opened in the above block or in ptext()),
            * then open an `It' block context around the parsed macro.
            */
   
           if (MDOC_It == n->tok && MDOC_BLOCK == n->type &&
                           NULL != n->parent &&
                           MDOC_Bl == n->parent->tok &&
                           LIST_column == n->parent->data.Bl->type) {
                   m->flags |= MDOC_FREECOL;
                   if ( ! mdoc_macro(m, MDOC_It, ln, sv, &sv, buf))
                           goto err;
                   return(1);
           }
   
           /* Normal processing of a macro. */
   
         if ( ! mdoc_macro(m, tok, ln, sv, &i, buf))          if ( ! mdoc_macro(m, tok, ln, sv, &i, buf))
                 goto err;                  goto err;
   

Legend:
Removed from v.1.138  
changed lines
  Added in v.1.155

CVSweb