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

Diff for /mandoc/man_term.c between version 1.234 and 1.243

version 1.234, 2020/02/27 01:43:52 version 1.243, 2023/10/24 20:53:12
Line 1 
Line 1 
 /*      $Id$ */  /* $Id$ */
 /*  /*
    * Copyright (c) 2010-15,2017-20,2022-23 Ingo Schwarze <schwarze@openbsd.org>
  * Copyright (c) 2008-2012 Kristaps Dzonsons <kristaps@bsd.lv>   * Copyright (c) 2008-2012 Kristaps Dzonsons <kristaps@bsd.lv>
  * Copyright (c) 2010-2015, 2017-2020 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 above   * purpose with or without fee is hereby granted, provided that the above
Line 14 
Line 14 
  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN   * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF   * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.   * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
    *
    * Plain text formatter for man(7), used by mandoc(1)
    * for ASCII, UTF-8, PostScript, and PDF output.
  */   */
 #include "config.h"  #include "config.h"
   
Line 32 
Line 35 
 #include "man.h"  #include "man.h"
 #include "out.h"  #include "out.h"
 #include "term.h"  #include "term.h"
 #include "tag.h"  #include "term_tag.h"
 #include "main.h"  #include "main.h"
   
 #define MAXMARGINS        64 /* maximum number of indented scopes */  #define MAXMARGINS        64 /* maximum number of indented scopes */
Line 71  static int    pre_DT(DECL_ARGS);
Line 74  static int    pre_DT(DECL_ARGS);
 static  int               pre_HP(DECL_ARGS);  static  int               pre_HP(DECL_ARGS);
 static  int               pre_I(DECL_ARGS);  static  int               pre_I(DECL_ARGS);
 static  int               pre_IP(DECL_ARGS);  static  int               pre_IP(DECL_ARGS);
   static  int               pre_MR(DECL_ARGS);
 static  int               pre_OP(DECL_ARGS);  static  int               pre_OP(DECL_ARGS);
 static  int               pre_PD(DECL_ARGS);  static  int               pre_PD(DECL_ARGS);
 static  int               pre_PP(DECL_ARGS);  static  int               pre_PP(DECL_ARGS);
Line 80  static int    pre_SS(DECL_ARGS);
Line 84  static int    pre_SS(DECL_ARGS);
 static  int               pre_SY(DECL_ARGS);  static  int               pre_SY(DECL_ARGS);
 static  int               pre_TP(DECL_ARGS);  static  int               pre_TP(DECL_ARGS);
 static  int               pre_UR(DECL_ARGS);  static  int               pre_UR(DECL_ARGS);
 static  int               pre_abort(DECL_ARGS);  
 static  int               pre_alternate(DECL_ARGS);  static  int               pre_alternate(DECL_ARGS);
 static  int               pre_ign(DECL_ARGS);  static  int               pre_ign(DECL_ARGS);
 static  int               pre_in(DECL_ARGS);  static  int               pre_in(DECL_ARGS);
Line 94  static void    post_SY(DECL_ARGS);
Line 97  static void    post_SY(DECL_ARGS);
 static  void              post_TP(DECL_ARGS);  static  void              post_TP(DECL_ARGS);
 static  void              post_UR(DECL_ARGS);  static  void              post_UR(DECL_ARGS);
   
 static  void              tag_man(struct termp *, struct roff_node *);  
   
 static const struct man_term_act man_term_acts[MAN_MAX - MAN_TH] = {  static const struct man_term_act man_term_acts[MAN_MAX - MAN_TH] = {
         { NULL, NULL, 0 }, /* TH */          { NULL, NULL, 0 }, /* TH */
         { pre_SH, post_SH, 0 }, /* SH */          { pre_SH, post_SH, 0 }, /* SH */
         { pre_SS, post_SH, 0 }, /* SS */          { pre_SS, post_SH, 0 }, /* SS */
         { pre_TP, post_TP, 0 }, /* TP */          { pre_TP, post_TP, 0 }, /* TP */
         { pre_TP, post_TP, 0 }, /* TQ */          { pre_TP, post_TP, 0 }, /* TQ */
         { pre_abort, NULL, 0 }, /* LP */          { pre_PP, NULL, 0 }, /* LP */
         { pre_PP, NULL, 0 }, /* PP */          { pre_PP, NULL, 0 }, /* PP */
         { pre_abort, NULL, 0 }, /* P */          { pre_PP, NULL, 0 }, /* P */
         { pre_IP, post_IP, 0 }, /* IP */          { pre_IP, post_IP, 0 }, /* IP */
         { pre_HP, post_HP, 0 }, /* HP */          { pre_HP, post_HP, 0 }, /* HP */
         { NULL, NULL, 0 }, /* SM */          { NULL, NULL, 0 }, /* SM */
Line 120  static const struct man_term_act man_term_acts[MAN_MAX
Line 121  static const struct man_term_act man_term_acts[MAN_MAX
         { pre_alternate, NULL, 0 }, /* RI */          { pre_alternate, NULL, 0 }, /* RI */
         { NULL, NULL, 0 }, /* RE */          { NULL, NULL, 0 }, /* RE */
         { pre_RS, post_RS, 0 }, /* RS */          { pre_RS, post_RS, 0 }, /* RS */
         { pre_DT, NULL, 0 }, /* DT */          { pre_DT, NULL, MAN_NOTEXT }, /* DT */
         { pre_ign, NULL, MAN_NOTEXT }, /* UC */          { pre_ign, NULL, MAN_NOTEXT }, /* UC */
         { pre_PD, NULL, MAN_NOTEXT }, /* PD */          { pre_PD, NULL, MAN_NOTEXT }, /* PD */
         { pre_ign, NULL, 0 }, /* AT */          { pre_ign, NULL, MAN_NOTEXT }, /* AT */
         { pre_in, NULL, MAN_NOTEXT }, /* in */          { pre_in, NULL, MAN_NOTEXT }, /* in */
         { pre_SY, post_SY, 0 }, /* SY */          { pre_SY, post_SY, 0 }, /* SY */
         { NULL, NULL, 0 }, /* YS */          { NULL, NULL, 0 }, /* YS */
Line 134  static const struct man_term_act man_term_acts[MAN_MAX
Line 135  static const struct man_term_act man_term_acts[MAN_MAX
         { NULL, NULL, 0 }, /* UE */          { NULL, NULL, 0 }, /* UE */
         { pre_UR, post_UR, 0 }, /* MT */          { pre_UR, post_UR, 0 }, /* MT */
         { NULL, NULL, 0 }, /* ME */          { NULL, NULL, 0 }, /* ME */
           { pre_MR, NULL, 0 }, /* MR */
 };  };
 static const struct man_term_act *man_term_act(enum roff_tok);  static const struct man_term_act *man_term_act(enum roff_tok);
   
Line 224  print_bvspace(struct termp *p, struct roff_node *n, in
Line 226  print_bvspace(struct termp *p, struct roff_node *n, in
                 term_vspace(p);                  term_vspace(p);
 }  }
   
   
 static int  static int
 pre_abort(DECL_ARGS)  
 {  
         abort();  
 }  
   
 static int  
 pre_ign(DECL_ARGS)  pre_ign(DECL_ARGS)
 {  {
         return 0;          return 0;
Line 335  pre_B(DECL_ARGS)
Line 330  pre_B(DECL_ARGS)
 }  }
   
 static int  static int
   pre_MR(DECL_ARGS)
   {
           term_fontrepl(p, TERMFONT_NONE);
           n = n->child;
           if (n != NULL) {
                   term_word(p, n->string);   /* name */
                   p->flags |= TERMP_NOSPACE;
           }
           term_word(p, "(");
           p->flags |= TERMP_NOSPACE;
           if (n != NULL && (n = n->next) != NULL) {
                   term_word(p, n->string);   /* section */
                   p->flags |= TERMP_NOSPACE;
           }
           term_word(p, ")");
           if (n != NULL && (n = n->next) != NULL) {
                   p->flags |= TERMP_NOSPACE;
                   term_word(p, n->string);   /* suffix */
           }
           return 0;
   }
   
   static int
 pre_OP(DECL_ARGS)  pre_OP(DECL_ARGS)
 {  {
         term_word(p, "[");          term_word(p, "[");
Line 516  pre_IP(DECL_ARGS)
Line 534  pre_IP(DECL_ARGS)
                 p->trailspace = 1;                  p->trailspace = 1;
                 break;                  break;
         case ROFFT_BODY:          case ROFFT_BODY:
                 p->flags |= TERMP_NOSPACE;                  p->flags |= TERMP_NOSPACE | TERMP_NONEWLINE;
                 break;                  break;
         default:          default:
                 abort();                  abort();
Line 539  pre_IP(DECL_ARGS)
Line 557  pre_IP(DECL_ARGS)
         case ROFFT_HEAD:          case ROFFT_HEAD:
                 p->tcol->offset = mt->offset;                  p->tcol->offset = mt->offset;
                 p->tcol->rmargin = mt->offset + len;                  p->tcol->rmargin = mt->offset + len;
                 if (n->child != NULL) {                  if (n->child != NULL)
                         print_man_node(p, mt, n->child, meta);                          print_man_node(p, mt, n->child, meta);
                         tag_man(p, n->child);  
                 }  
                 return 0;                  return 0;
         case ROFFT_BODY:          case ROFFT_BODY:
                 p->tcol->offset = mt->offset + len;                  p->tcol->offset = mt->offset + len;
Line 592  pre_TP(DECL_ARGS)
Line 608  pre_TP(DECL_ARGS)
                 p->trailspace = 1;                  p->trailspace = 1;
                 break;                  break;
         case ROFFT_BODY:          case ROFFT_BODY:
                 p->flags |= TERMP_NOSPACE;                  p->flags |= TERMP_NOSPACE | TERMP_NONEWLINE;
                 break;                  break;
         default:          default:
                 abort();                  abort();
Line 622  pre_TP(DECL_ARGS)
Line 638  pre_TP(DECL_ARGS)
                 while (nn != NULL && (nn->flags & NODE_LINE) == 0)                  while (nn != NULL && (nn->flags & NODE_LINE) == 0)
                         nn = nn->next;                          nn = nn->next;
   
                 if (nn == NULL)  
                         return 0;  
   
                 if (nn->type == ROFFT_TEXT)  
                         tag_man(p, nn);  
                 else if (nn->child != NULL &&  
                     nn->child->type == ROFFT_TEXT &&  
                     (nn->tok == MAN_B || nn->tok == MAN_BI ||  
                      nn->tok == MAN_BR || nn->tok == MAN_I ||  
                      nn->tok == MAN_IB || nn->tok == MAN_IR))  
                         tag_man(p, nn->child);  
   
                 while (nn != NULL) {                  while (nn != NULL) {
                         print_man_node(p, mt, nn, meta);                          print_man_node(p, mt, nn, meta);
                         nn = nn->next;                          nn = nn->next;
Line 913  print_man_node(DECL_ARGS)
Line 917  print_man_node(DECL_ARGS)
         const struct man_term_act *act;          const struct man_term_act *act;
         int c;          int c;
   
           /*
            * In no-fill mode, break the output line at the beginning
            * of new input lines except after \c, and nowhere else.
            */
   
           if (n->flags & NODE_NOFILL) {
                   if (n->flags & NODE_LINE &&
                       (p->flags & TERMP_NONEWLINE) == 0)
                           term_newln(p);
                   p->flags |= TERMP_BRNEVER;
           } else {
                   if (n->flags & NODE_LINE)
                           term_tab_ref(p);
                   p->flags &= ~TERMP_BRNEVER;
           }
   
           if (n->flags & NODE_ID)
                   term_tag_write(n, p->line);
   
         switch (n->type) {          switch (n->type) {
         case ROFFT_TEXT:          case ROFFT_TEXT:
                 /*                  /*
Line 945  print_man_node(DECL_ARGS)
Line 968  print_man_node(DECL_ARGS)
                 return;                  return;
         case ROFFT_TBL:          case ROFFT_TBL:
                 if (p->tbl.cols == NULL)                  if (p->tbl.cols == NULL)
                         term_vspace(p);                          term_newln(p);
                 term_tbl(p, n->span);                  term_tbl(p, n->span);
                 return;                  return;
         default:          default:
Line 974  print_man_node(DECL_ARGS)
Line 997  print_man_node(DECL_ARGS)
                 term_fontrepl(p, TERMFONT_NONE);                  term_fontrepl(p, TERMFONT_NONE);
   
 out:  out:
         /*          if (n->parent->tok == MAN_HP && n->parent->type == ROFFT_BODY &&
          * If we're in a literal context, make sure that words              n->prev == NULL && n->flags & NODE_NOFILL) {
          * together on the same line stay together.  This is a                  term_newln(p);
          * POST-printing call, so we check the NEXT word.  Since                  p->tcol->offset = p->tcol->rmargin;
          * -man doesn't have nested macros, we don't need to be                  p->tcol->rmargin = p->maxrmargin;
          * more specific than this.  
          */  
         if (n->flags & NODE_NOFILL &&  
             ! (p->flags & (TERMP_NOBREAK | TERMP_NONEWLINE)) &&  
             (n->next == NULL || n->next->flags & NODE_LINE)) {  
                 p->flags |= TERMP_BRNEVER | TERMP_NOSPACE;  
                 if (n->string != NULL && *n->string != '\0')  
                         term_flushln(p);  
                 else  
                         term_newln(p);  
                 p->flags &= ~TERMP_BRNEVER;  
                 if (p->tcol->rmargin < p->maxrmargin &&  
                     n->parent->tok == MAN_HP) {  
                         p->tcol->offset = p->tcol->rmargin;  
                         p->tcol->rmargin = p->maxrmargin;  
                 }  
         }          }
         if (n->flags & NODE_EOS)          if (n->flags & NODE_EOS)
                 p->flags |= TERMP_SENTENCE;                  p->flags |= TERMP_SENTENCE;
Line 1031  print_man_foot(struct termp *p, const struct roff_meta
Line 1038  print_man_foot(struct termp *p, const struct roff_meta
          */           */
   
         if ( ! p->mdocstyle) {          if ( ! p->mdocstyle) {
                 if (meta->hasbody) {  
                         term_vspace(p);  
                         term_vspace(p);  
                 }  
                 mandoc_asprintf(&title, "%s(%s)",                  mandoc_asprintf(&title, "%s(%s)",
                     meta->title, meta->msec);                      meta->title, meta->msec);
         } else if (meta->os != NULL) {          } else if (meta->os != NULL) {
Line 1153  print_man_head(struct termp *p, const struct roff_meta
Line 1156  print_man_head(struct termp *p, const struct roff_meta
          */           */
   
         term_vspace(p);          term_vspace(p);
         if ( ! p->mdocstyle) {  
                 term_vspace(p);  
                 term_vspace(p);  
         }  
         free(title);          free(title);
 }  
   
 /*  
  * Skip leading whitespace, dashes, backslashes, and font escapes,  
  * then create a tag if the first following byte is a letter.  
  * Priority is high unless whitespace is present.  
  */  
 static void  
 tag_man(struct termp *p, struct roff_node *n)  
 {  
         const char      *cp, *arg;  
         int              prio, sz;  
   
         assert(n->type == ROFFT_TEXT);  
         cp = n->string;  
         prio = TAG_STRONG;  
         for (;;) {  
                 switch (*cp) {  
                 case ' ':  
                 case '\t':  
                         prio = TAG_WEAK;  
                         /* FALLTHROUGH */  
                 case '-':  
                         cp++;  
                         break;  
                 case '\\':  
                         cp++;  
                         switch (mandoc_escape(&cp, &arg, &sz)) {  
                         case ESCAPE_FONT:  
                         case ESCAPE_FONTROMAN:  
                         case ESCAPE_FONTITALIC:  
                         case ESCAPE_FONTBOLD:  
                         case ESCAPE_FONTPREV:  
                         case ESCAPE_FONTBI:  
                                 break;  
                         case ESCAPE_SPECIAL:  
                                 if (sz != 1)  
                                         return;  
                                 switch (*arg) {  
                                 case '&':  
                                 case '-':  
                                 case 'e':  
                                         break;  
                                 default:  
                                         return;  
                                 }  
                                 break;  
                         default:  
                                 return;  
                         }  
                         break;  
                 default:  
                         if (isalpha((unsigned char)*cp))  
                                 tag_put(cp, prio, p->line);  
                         return;  
                 }  
         }  
 }  }

Legend:
Removed from v.1.234  
changed lines
  Added in v.1.243

CVSweb