1 /* $Id: mdoc_validate.c,v 1.352 2017/08/02 13:29:04 schwarze Exp $ */ 2 /* 3 * Copyright (c) 2008-2012 Kristaps Dzonsons <kristaps@bsd.lv> 4 * Copyright (c) 2010-2017 Ingo Schwarze <schwarze@openbsd.org> 5 * Copyright (c) 2010 Joerg Sonnenberger <joerg@netbsd.org> 6 * 7 * Permission to use, copy, modify, and distribute this software for any 8 * purpose with or without fee is hereby granted, provided that the above 9 * copyright notice and this permission notice appear in all copies. 10 * 11 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHORS DISCLAIM ALL WARRANTIES 12 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 13 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR 14 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 15 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN 16 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF 17 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 18 */ 19 #include "config.h" 20 21 #include <sys/types.h> 22 #ifndef OSNAME 23 #include <sys/utsname.h> 24 #endif 25 26 #include <assert.h> 27 #include <ctype.h> 28 #include <limits.h> 29 #include <stdio.h> 30 #include <stdlib.h> 31 #include <string.h> 32 #include <time.h> 33 34 #include "mandoc_aux.h" 35 #include "mandoc.h" 36 #include "mandoc_xr.h" 37 #include "roff.h" 38 #include "mdoc.h" 39 #include "libmandoc.h" 40 #include "roff_int.h" 41 #include "libmdoc.h" 42 43 /* FIXME: .Bl -diag can't have non-text children in HEAD. */ 44 45 #define POST_ARGS struct roff_man *mdoc 46 47 enum check_ineq { 48 CHECK_LT, 49 CHECK_GT, 50 CHECK_EQ 51 }; 52 53 typedef void (*v_post)(POST_ARGS); 54 55 static int build_list(struct roff_man *, int); 56 static void check_text(struct roff_man *, int, int, char *); 57 static void check_argv(struct roff_man *, 58 struct roff_node *, struct mdoc_argv *); 59 static void check_args(struct roff_man *, struct roff_node *); 60 static void check_toptext(struct roff_man *, int, int, const char *); 61 static int child_an(const struct roff_node *); 62 static size_t macro2len(enum roff_tok); 63 static void rewrite_macro2len(struct roff_man *, char **); 64 static int similar(const char *, const char *); 65 66 static void post_an(POST_ARGS); 67 static void post_an_norm(POST_ARGS); 68 static void post_at(POST_ARGS); 69 static void post_bd(POST_ARGS); 70 static void post_bf(POST_ARGS); 71 static void post_bk(POST_ARGS); 72 static void post_bl(POST_ARGS); 73 static void post_bl_block(POST_ARGS); 74 static void post_bl_head(POST_ARGS); 75 static void post_bl_norm(POST_ARGS); 76 static void post_bx(POST_ARGS); 77 static void post_defaults(POST_ARGS); 78 static void post_display(POST_ARGS); 79 static void post_dd(POST_ARGS); 80 static void post_delim(POST_ARGS); 81 static void post_delim_nb(POST_ARGS); 82 static void post_dt(POST_ARGS); 83 static void post_en(POST_ARGS); 84 static void post_es(POST_ARGS); 85 static void post_eoln(POST_ARGS); 86 static void post_ex(POST_ARGS); 87 static void post_fa(POST_ARGS); 88 static void post_fn(POST_ARGS); 89 static void post_fname(POST_ARGS); 90 static void post_fo(POST_ARGS); 91 static void post_hyph(POST_ARGS); 92 static void post_ignpar(POST_ARGS); 93 static void post_it(POST_ARGS); 94 static void post_lb(POST_ARGS); 95 static void post_nd(POST_ARGS); 96 static void post_nm(POST_ARGS); 97 static void post_ns(POST_ARGS); 98 static void post_obsolete(POST_ARGS); 99 static void post_os(POST_ARGS); 100 static void post_par(POST_ARGS); 101 static void post_prevpar(POST_ARGS); 102 static void post_root(POST_ARGS); 103 static void post_rs(POST_ARGS); 104 static void post_rv(POST_ARGS); 105 static void post_sh(POST_ARGS); 106 static void post_sh_head(POST_ARGS); 107 static void post_sh_name(POST_ARGS); 108 static void post_sh_see_also(POST_ARGS); 109 static void post_sh_authors(POST_ARGS); 110 static void post_sm(POST_ARGS); 111 static void post_st(POST_ARGS); 112 static void post_std(POST_ARGS); 113 static void post_sx(POST_ARGS); 114 static void post_useless(POST_ARGS); 115 static void post_xr(POST_ARGS); 116 static void post_xx(POST_ARGS); 117 118 static const v_post __mdoc_valids[MDOC_MAX - MDOC_Dd] = { 119 post_dd, /* Dd */ 120 post_dt, /* Dt */ 121 post_os, /* Os */ 122 post_sh, /* Sh */ 123 post_ignpar, /* Ss */ 124 post_par, /* Pp */ 125 post_display, /* D1 */ 126 post_display, /* Dl */ 127 post_display, /* Bd */ 128 NULL, /* Ed */ 129 post_bl, /* Bl */ 130 NULL, /* El */ 131 post_it, /* It */ 132 post_delim_nb, /* Ad */ 133 post_an, /* An */ 134 NULL, /* Ap */ 135 post_defaults, /* Ar */ 136 NULL, /* Cd */ 137 post_delim_nb, /* Cm */ 138 post_delim_nb, /* Dv */ 139 post_delim_nb, /* Er */ 140 post_delim_nb, /* Ev */ 141 post_ex, /* Ex */ 142 post_fa, /* Fa */ 143 NULL, /* Fd */ 144 post_delim_nb, /* Fl */ 145 post_fn, /* Fn */ 146 post_delim_nb, /* Ft */ 147 post_delim_nb, /* Ic */ 148 post_delim_nb, /* In */ 149 post_defaults, /* Li */ 150 post_nd, /* Nd */ 151 post_nm, /* Nm */ 152 post_delim_nb, /* Op */ 153 post_obsolete, /* Ot */ 154 post_defaults, /* Pa */ 155 post_rv, /* Rv */ 156 post_st, /* St */ 157 post_delim_nb, /* Va */ 158 post_delim_nb, /* Vt */ 159 post_xr, /* Xr */ 160 NULL, /* %A */ 161 post_hyph, /* %B */ /* FIXME: can be used outside Rs/Re. */ 162 NULL, /* %D */ 163 NULL, /* %I */ 164 NULL, /* %J */ 165 post_hyph, /* %N */ 166 post_hyph, /* %O */ 167 NULL, /* %P */ 168 post_hyph, /* %R */ 169 post_hyph, /* %T */ /* FIXME: can be used outside Rs/Re. */ 170 NULL, /* %V */ 171 NULL, /* Ac */ 172 post_delim_nb, /* Ao */ 173 post_delim_nb, /* Aq */ 174 post_at, /* At */ 175 NULL, /* Bc */ 176 post_bf, /* Bf */ 177 post_delim_nb, /* Bo */ 178 NULL, /* Bq */ 179 post_xx, /* Bsx */ 180 post_bx, /* Bx */ 181 post_obsolete, /* Db */ 182 NULL, /* Dc */ 183 NULL, /* Do */ 184 NULL, /* Dq */ 185 NULL, /* Ec */ 186 NULL, /* Ef */ 187 post_delim_nb, /* Em */ 188 NULL, /* Eo */ 189 post_xx, /* Fx */ 190 post_delim_nb, /* Ms */ 191 NULL, /* No */ 192 post_ns, /* Ns */ 193 post_xx, /* Nx */ 194 post_xx, /* Ox */ 195 NULL, /* Pc */ 196 NULL, /* Pf */ 197 post_delim_nb, /* Po */ 198 post_delim_nb, /* Pq */ 199 NULL, /* Qc */ 200 post_delim_nb, /* Ql */ 201 post_delim_nb, /* Qo */ 202 post_delim_nb, /* Qq */ 203 NULL, /* Re */ 204 post_rs, /* Rs */ 205 NULL, /* Sc */ 206 post_delim_nb, /* So */ 207 post_delim_nb, /* Sq */ 208 post_sm, /* Sm */ 209 post_sx, /* Sx */ 210 post_delim_nb, /* Sy */ 211 post_useless, /* Tn */ 212 post_xx, /* Ux */ 213 NULL, /* Xc */ 214 NULL, /* Xo */ 215 post_fo, /* Fo */ 216 NULL, /* Fc */ 217 post_delim_nb, /* Oo */ 218 NULL, /* Oc */ 219 post_bk, /* Bk */ 220 NULL, /* Ek */ 221 post_eoln, /* Bt */ 222 post_obsolete, /* Hf */ 223 post_obsolete, /* Fr */ 224 post_eoln, /* Ud */ 225 post_lb, /* Lb */ 226 post_par, /* Lp */ 227 post_delim_nb, /* Lk */ 228 post_defaults, /* Mt */ 229 post_delim_nb, /* Brq */ 230 post_delim_nb, /* Bro */ 231 NULL, /* Brc */ 232 NULL, /* %C */ 233 post_es, /* Es */ 234 post_en, /* En */ 235 post_xx, /* Dx */ 236 NULL, /* %Q */ 237 NULL, /* %U */ 238 NULL, /* Ta */ 239 }; 240 static const v_post *const mdoc_valids = __mdoc_valids - MDOC_Dd; 241 242 #define RSORD_MAX 14 /* Number of `Rs' blocks. */ 243 244 static const enum roff_tok rsord[RSORD_MAX] = { 245 MDOC__A, 246 MDOC__T, 247 MDOC__B, 248 MDOC__I, 249 MDOC__J, 250 MDOC__R, 251 MDOC__N, 252 MDOC__V, 253 MDOC__U, 254 MDOC__P, 255 MDOC__Q, 256 MDOC__C, 257 MDOC__D, 258 MDOC__O 259 }; 260 261 static const char * const secnames[SEC__MAX] = { 262 NULL, 263 "NAME", 264 "LIBRARY", 265 "SYNOPSIS", 266 "DESCRIPTION", 267 "CONTEXT", 268 "IMPLEMENTATION NOTES", 269 "RETURN VALUES", 270 "ENVIRONMENT", 271 "FILES", 272 "EXIT STATUS", 273 "EXAMPLES", 274 "DIAGNOSTICS", 275 "COMPATIBILITY", 276 "ERRORS", 277 "SEE ALSO", 278 "STANDARDS", 279 "HISTORY", 280 "AUTHORS", 281 "CAVEATS", 282 "BUGS", 283 "SECURITY CONSIDERATIONS", 284 NULL 285 }; 286 287 288 void 289 mdoc_node_validate(struct roff_man *mdoc) 290 { 291 struct roff_node *n; 292 const v_post *p; 293 294 n = mdoc->last; 295 mdoc->last = mdoc->last->child; 296 while (mdoc->last != NULL) { 297 mdoc_node_validate(mdoc); 298 if (mdoc->last == n) 299 mdoc->last = mdoc->last->child; 300 else 301 mdoc->last = mdoc->last->next; 302 } 303 304 mdoc->last = n; 305 mdoc->next = ROFF_NEXT_SIBLING; 306 switch (n->type) { 307 case ROFFT_TEXT: 308 if (n->sec != SEC_SYNOPSIS || 309 (n->parent->tok != MDOC_Cd && n->parent->tok != MDOC_Fd)) 310 check_text(mdoc, n->line, n->pos, n->string); 311 if (n->parent->tok == MDOC_It || 312 (n->parent->type == ROFFT_BODY && 313 (n->parent->tok == MDOC_Sh || 314 n->parent->tok == MDOC_Ss))) 315 check_toptext(mdoc, n->line, n->pos, n->string); 316 break; 317 case ROFFT_EQN: 318 case ROFFT_TBL: 319 break; 320 case ROFFT_ROOT: 321 post_root(mdoc); 322 break; 323 default: 324 check_args(mdoc, mdoc->last); 325 326 /* 327 * Closing delimiters are not special at the 328 * beginning of a block, opening delimiters 329 * are not special at the end. 330 */ 331 332 if (n->child != NULL) 333 n->child->flags &= ~NODE_DELIMC; 334 if (n->last != NULL) 335 n->last->flags &= ~NODE_DELIMO; 336 337 /* Call the macro's postprocessor. */ 338 339 if (n->tok < ROFF_MAX) { 340 switch(n->tok) { 341 case ROFF_br: 342 case ROFF_sp: 343 post_par(mdoc); 344 break; 345 default: 346 roff_validate(mdoc); 347 break; 348 } 349 break; 350 } 351 352 assert(n->tok >= MDOC_Dd && n->tok < MDOC_MAX); 353 p = mdoc_valids + n->tok; 354 if (*p) 355 (*p)(mdoc); 356 if (mdoc->last == n) 357 mdoc_state(mdoc, n); 358 break; 359 } 360 } 361 362 static void 363 check_args(struct roff_man *mdoc, struct roff_node *n) 364 { 365 int i; 366 367 if (NULL == n->args) 368 return; 369 370 assert(n->args->argc); 371 for (i = 0; i < (int)n->args->argc; i++) 372 check_argv(mdoc, n, &n->args->argv[i]); 373 } 374 375 static void 376 check_argv(struct roff_man *mdoc, struct roff_node *n, struct mdoc_argv *v) 377 { 378 int i; 379 380 for (i = 0; i < (int)v->sz; i++) 381 check_text(mdoc, v->line, v->pos, v->value[i]); 382 } 383 384 static void 385 check_text(struct roff_man *mdoc, int ln, int pos, char *p) 386 { 387 char *cp; 388 389 if (MDOC_LITERAL & mdoc->flags) 390 return; 391 392 for (cp = p; NULL != (p = strchr(p, '\t')); p++) 393 mandoc_msg(MANDOCERR_FI_TAB, mdoc->parse, 394 ln, pos + (int)(p - cp), NULL); 395 } 396 397 static void 398 check_toptext(struct roff_man *mdoc, int ln, int pos, const char *p) 399 { 400 const char *cp, *cpr; 401 402 if (*p == '\0') 403 return; 404 405 if ((cp = strstr(p, "OpenBSD")) != NULL) 406 mandoc_msg(MANDOCERR_BX, mdoc->parse, 407 ln, pos + (cp - p), "Ox"); 408 if ((cp = strstr(p, "NetBSD")) != NULL) 409 mandoc_msg(MANDOCERR_BX, mdoc->parse, 410 ln, pos + (cp - p), "Nx"); 411 if ((cp = strstr(p, "FreeBSD")) != NULL) 412 mandoc_msg(MANDOCERR_BX, mdoc->parse, 413 ln, pos + (cp - p), "Fx"); 414 if ((cp = strstr(p, "DragonFly")) != NULL) 415 mandoc_msg(MANDOCERR_BX, mdoc->parse, 416 ln, pos + (cp - p), "Dx"); 417 418 cp = p; 419 while ((cp = strstr(cp + 1, "()")) != NULL) { 420 for (cpr = cp - 1; cpr >= p; cpr--) 421 if (*cpr != '_' && !isalnum((unsigned char)*cpr)) 422 break; 423 if ((cpr < p || *cpr == ' ') && cpr + 1 < cp) { 424 cpr++; 425 mandoc_vmsg(MANDOCERR_FUNC, mdoc->parse, 426 ln, pos + (cpr - p), 427 "%.*s()", (int)(cp - cpr), cpr); 428 } 429 } 430 } 431 432 static void 433 post_delim(POST_ARGS) 434 { 435 const struct roff_node *nch; 436 const char *lc; 437 enum mdelim delim; 438 enum roff_tok tok; 439 440 tok = mdoc->last->tok; 441 nch = mdoc->last->last; 442 if (nch == NULL || nch->type != ROFFT_TEXT) 443 return; 444 lc = strchr(nch->string, '\0') - 1; 445 if (lc < nch->string) 446 return; 447 delim = mdoc_isdelim(lc); 448 if (delim == DELIM_NONE || delim == DELIM_OPEN) 449 return; 450 if (*lc == ')' && (tok == MDOC_Nd || tok == MDOC_Sh || 451 tok == MDOC_Ss || tok == MDOC_Fo)) 452 return; 453 454 mandoc_vmsg(MANDOCERR_DELIM, mdoc->parse, 455 nch->line, nch->pos + (lc - nch->string), 456 "%s%s %s", roff_name[tok], 457 nch == mdoc->last->child ? "" : " ...", nch->string); 458 } 459 460 static void 461 post_delim_nb(POST_ARGS) 462 { 463 const struct roff_node *nch; 464 const char *lc, *cp; 465 int nw; 466 enum mdelim delim; 467 enum roff_tok tok; 468 469 /* 470 * Find candidates: at least two bytes, 471 * the last one a closing or middle delimiter. 472 */ 473 474 tok = mdoc->last->tok; 475 nch = mdoc->last->last; 476 if (nch == NULL || nch->type != ROFFT_TEXT) 477 return; 478 lc = strchr(nch->string, '\0') - 1; 479 if (lc <= nch->string) 480 return; 481 delim = mdoc_isdelim(lc); 482 if (delim == DELIM_NONE || delim == DELIM_OPEN) 483 return; 484 485 /* 486 * Reduce false positives by allowing various cases. 487 */ 488 489 /* Escaped delimiters. */ 490 if (lc > nch->string + 1 && lc[-2] == '\\' && 491 (lc[-1] == '&' || lc[-1] == 'e')) 492 return; 493 494 /* Specific byte sequences. */ 495 switch (*lc) { 496 case ')': 497 for (cp = lc; cp >= nch->string; cp--) 498 if (*cp == '(') 499 return; 500 break; 501 case '.': 502 if (lc > nch->string + 1 && lc[-2] == '.' && lc[-1] == '.') 503 return; 504 if (lc[-1] == '.') 505 return; 506 break; 507 case ';': 508 if (tok == MDOC_Vt) 509 return; 510 break; 511 case '?': 512 if (lc[-1] == '?') 513 return; 514 break; 515 case ']': 516 for (cp = lc; cp >= nch->string; cp--) 517 if (*cp == '[') 518 return; 519 break; 520 case '|': 521 if (lc == nch->string + 1 && lc[-1] == '|') 522 return; 523 default: 524 break; 525 } 526 527 /* Exactly two non-alphanumeric bytes. */ 528 if (lc == nch->string + 1 && !isalnum((unsigned char)lc[-1])) 529 return; 530 531 /* At least three alphabetic words with a sentence ending. */ 532 if (strchr("!.:?", *lc) != NULL && (tok == MDOC_Em || 533 tok == MDOC_Li || tok == MDOC_Po || tok == MDOC_Pq || 534 tok == MDOC_Sy)) { 535 nw = 0; 536 for (cp = lc - 1; cp >= nch->string; cp--) { 537 if (*cp == ' ') { 538 nw++; 539 if (cp > nch->string && cp[-1] == ',') 540 cp--; 541 } else if (isalpha((unsigned int)*cp)) { 542 if (nw > 1) 543 return; 544 } else 545 break; 546 } 547 } 548 549 mandoc_vmsg(MANDOCERR_DELIM_NB, mdoc->parse, 550 nch->line, nch->pos + (lc - nch->string), 551 "%s%s %s", roff_name[tok], 552 nch == mdoc->last->child ? "" : " ...", nch->string); 553 } 554 555 static void 556 post_bl_norm(POST_ARGS) 557 { 558 struct roff_node *n; 559 struct mdoc_argv *argv, *wa; 560 int i; 561 enum mdocargt mdoclt; 562 enum mdoc_list lt; 563 564 n = mdoc->last->parent; 565 n->norm->Bl.type = LIST__NONE; 566 567 /* 568 * First figure out which kind of list to use: bind ourselves to 569 * the first mentioned list type and warn about any remaining 570 * ones. If we find no list type, we default to LIST_item. 571 */ 572 573 wa = (n->args == NULL) ? NULL : n->args->argv; 574 mdoclt = MDOC_ARG_MAX; 575 for (i = 0; n->args && i < (int)n->args->argc; i++) { 576 argv = n->args->argv + i; 577 lt = LIST__NONE; 578 switch (argv->arg) { 579 /* Set list types. */ 580 case MDOC_Bullet: 581 lt = LIST_bullet; 582 break; 583 case MDOC_Dash: 584 lt = LIST_dash; 585 break; 586 case MDOC_Enum: 587 lt = LIST_enum; 588 break; 589 case MDOC_Hyphen: 590 lt = LIST_hyphen; 591 break; 592 case MDOC_Item: 593 lt = LIST_item; 594 break; 595 case MDOC_Tag: 596 lt = LIST_tag; 597 break; 598 case MDOC_Diag: 599 lt = LIST_diag; 600 break; 601 case MDOC_Hang: 602 lt = LIST_hang; 603 break; 604 case MDOC_Ohang: 605 lt = LIST_ohang; 606 break; 607 case MDOC_Inset: 608 lt = LIST_inset; 609 break; 610 case MDOC_Column: 611 lt = LIST_column; 612 break; 613 /* Set list arguments. */ 614 case MDOC_Compact: 615 if (n->norm->Bl.comp) 616 mandoc_msg(MANDOCERR_ARG_REP, 617 mdoc->parse, argv->line, 618 argv->pos, "Bl -compact"); 619 n->norm->Bl.comp = 1; 620 break; 621 case MDOC_Width: 622 wa = argv; 623 if (0 == argv->sz) { 624 mandoc_msg(MANDOCERR_ARG_EMPTY, 625 mdoc->parse, argv->line, 626 argv->pos, "Bl -width"); 627 n->norm->Bl.width = "0n"; 628 break; 629 } 630 if (NULL != n->norm->Bl.width) 631 mandoc_vmsg(MANDOCERR_ARG_REP, 632 mdoc->parse, argv->line, 633 argv->pos, "Bl -width %s", 634 argv->value[0]); 635 rewrite_macro2len(mdoc, argv->value); 636 n->norm->Bl.width = argv->value[0]; 637 break; 638 case MDOC_Offset: 639 if (0 == argv->sz) { 640 mandoc_msg(MANDOCERR_ARG_EMPTY, 641 mdoc->parse, argv->line, 642 argv->pos, "Bl -offset"); 643 break; 644 } 645 if (NULL != n->norm->Bl.offs) 646 mandoc_vmsg(MANDOCERR_ARG_REP, 647 mdoc->parse, argv->line, 648 argv->pos, "Bl -offset %s", 649 argv->value[0]); 650 rewrite_macro2len(mdoc, argv->value); 651 n->norm->Bl.offs = argv->value[0]; 652 break; 653 default: 654 continue; 655 } 656 if (LIST__NONE == lt) 657 continue; 658 mdoclt = argv->arg; 659 660 /* Check: multiple list types. */ 661 662 if (LIST__NONE != n->norm->Bl.type) { 663 mandoc_vmsg(MANDOCERR_BL_REP, 664 mdoc->parse, n->line, n->pos, 665 "Bl -%s", mdoc_argnames[argv->arg]); 666 continue; 667 } 668 669 /* The list type should come first. */ 670 671 if (n->norm->Bl.width || 672 n->norm->Bl.offs || 673 n->norm->Bl.comp) 674 mandoc_vmsg(MANDOCERR_BL_LATETYPE, 675 mdoc->parse, n->line, n->pos, "Bl -%s", 676 mdoc_argnames[n->args->argv[0].arg]); 677 678 n->norm->Bl.type = lt; 679 if (LIST_column == lt) { 680 n->norm->Bl.ncols = argv->sz; 681 n->norm->Bl.cols = (void *)argv->value; 682 } 683 } 684 685 /* Allow lists to default to LIST_item. */ 686 687 if (LIST__NONE == n->norm->Bl.type) { 688 mandoc_msg(MANDOCERR_BL_NOTYPE, mdoc->parse, 689 n->line, n->pos, "Bl"); 690 n->norm->Bl.type = LIST_item; 691 mdoclt = MDOC_Item; 692 } 693 694 /* 695 * Validate the width field. Some list types don't need width 696 * types and should be warned about them. Others should have it 697 * and must also be warned. Yet others have a default and need 698 * no warning. 699 */ 700 701 switch (n->norm->Bl.type) { 702 case LIST_tag: 703 if (n->norm->Bl.width == NULL) 704 mandoc_msg(MANDOCERR_BL_NOWIDTH, mdoc->parse, 705 n->line, n->pos, "Bl -tag"); 706 break; 707 case LIST_column: 708 case LIST_diag: 709 case LIST_ohang: 710 case LIST_inset: 711 case LIST_item: 712 if (n->norm->Bl.width != NULL) 713 mandoc_vmsg(MANDOCERR_BL_SKIPW, mdoc->parse, 714 wa->line, wa->pos, "Bl -%s", 715 mdoc_argnames[mdoclt]); 716 n->norm->Bl.width = NULL; 717 break; 718 case LIST_bullet: 719 case LIST_dash: 720 case LIST_hyphen: 721 if (n->norm->Bl.width == NULL) 722 n->norm->Bl.width = "2n"; 723 break; 724 case LIST_enum: 725 if (n->norm->Bl.width == NULL) 726 n->norm->Bl.width = "3n"; 727 break; 728 default: 729 break; 730 } 731 } 732 733 static void 734 post_bd(POST_ARGS) 735 { 736 struct roff_node *n; 737 struct mdoc_argv *argv; 738 int i; 739 enum mdoc_disp dt; 740 741 n = mdoc->last; 742 for (i = 0; n->args && i < (int)n->args->argc; i++) { 743 argv = n->args->argv + i; 744 dt = DISP__NONE; 745 746 switch (argv->arg) { 747 case MDOC_Centred: 748 dt = DISP_centered; 749 break; 750 case MDOC_Ragged: 751 dt = DISP_ragged; 752 break; 753 case MDOC_Unfilled: 754 dt = DISP_unfilled; 755 break; 756 case MDOC_Filled: 757 dt = DISP_filled; 758 break; 759 case MDOC_Literal: 760 dt = DISP_literal; 761 break; 762 case MDOC_File: 763 mandoc_msg(MANDOCERR_BD_FILE, mdoc->parse, 764 n->line, n->pos, NULL); 765 break; 766 case MDOC_Offset: 767 if (0 == argv->sz) { 768 mandoc_msg(MANDOCERR_ARG_EMPTY, 769 mdoc->parse, argv->line, 770 argv->pos, "Bd -offset"); 771 break; 772 } 773 if (NULL != n->norm->Bd.offs) 774 mandoc_vmsg(MANDOCERR_ARG_REP, 775 mdoc->parse, argv->line, 776 argv->pos, "Bd -offset %s", 777 argv->value[0]); 778 rewrite_macro2len(mdoc, argv->value); 779 n->norm->Bd.offs = argv->value[0]; 780 break; 781 case MDOC_Compact: 782 if (n->norm->Bd.comp) 783 mandoc_msg(MANDOCERR_ARG_REP, 784 mdoc->parse, argv->line, 785 argv->pos, "Bd -compact"); 786 n->norm->Bd.comp = 1; 787 break; 788 default: 789 abort(); 790 } 791 if (DISP__NONE == dt) 792 continue; 793 794 if (DISP__NONE == n->norm->Bd.type) 795 n->norm->Bd.type = dt; 796 else 797 mandoc_vmsg(MANDOCERR_BD_REP, 798 mdoc->parse, n->line, n->pos, 799 "Bd -%s", mdoc_argnames[argv->arg]); 800 } 801 802 if (DISP__NONE == n->norm->Bd.type) { 803 mandoc_msg(MANDOCERR_BD_NOTYPE, mdoc->parse, 804 n->line, n->pos, "Bd"); 805 n->norm->Bd.type = DISP_ragged; 806 } 807 } 808 809 /* 810 * Stand-alone line macros. 811 */ 812 813 static void 814 post_an_norm(POST_ARGS) 815 { 816 struct roff_node *n; 817 struct mdoc_argv *argv; 818 size_t i; 819 820 n = mdoc->last; 821 if (n->args == NULL) 822 return; 823 824 for (i = 1; i < n->args->argc; i++) { 825 argv = n->args->argv + i; 826 mandoc_vmsg(MANDOCERR_AN_REP, 827 mdoc->parse, argv->line, argv->pos, 828 "An -%s", mdoc_argnames[argv->arg]); 829 } 830 831 argv = n->args->argv; 832 if (argv->arg == MDOC_Split) 833 n->norm->An.auth = AUTH_split; 834 else if (argv->arg == MDOC_Nosplit) 835 n->norm->An.auth = AUTH_nosplit; 836 else 837 abort(); 838 } 839 840 static void 841 post_eoln(POST_ARGS) 842 { 843 struct roff_node *n; 844 845 post_useless(mdoc); 846 n = mdoc->last; 847 if (n->child != NULL) 848 mandoc_vmsg(MANDOCERR_ARG_SKIP, mdoc->parse, n->line, 849 n->pos, "%s %s", roff_name[n->tok], n->child->string); 850 851 while (n->child != NULL) 852 roff_node_delete(mdoc, n->child); 853 854 roff_word_alloc(mdoc, n->line, n->pos, n->tok == MDOC_Bt ? 855 "is currently in beta test." : "currently under development."); 856 mdoc->last->flags |= NODE_EOS | NODE_NOSRC; 857 mdoc->last = n; 858 } 859 860 static int 861 build_list(struct roff_man *mdoc, int tok) 862 { 863 struct roff_node *n; 864 int ic; 865 866 n = mdoc->last->next; 867 for (ic = 1;; ic++) { 868 roff_elem_alloc(mdoc, n->line, n->pos, tok); 869 mdoc->last->flags |= NODE_NOSRC; 870 mdoc_node_relink(mdoc, n); 871 n = mdoc->last = mdoc->last->parent; 872 mdoc->next = ROFF_NEXT_SIBLING; 873 if (n->next == NULL) 874 return ic; 875 if (ic > 1 || n->next->next != NULL) { 876 roff_word_alloc(mdoc, n->line, n->pos, ","); 877 mdoc->last->flags |= NODE_DELIMC | NODE_NOSRC; 878 } 879 n = mdoc->last->next; 880 if (n->next == NULL) { 881 roff_word_alloc(mdoc, n->line, n->pos, "and"); 882 mdoc->last->flags |= NODE_NOSRC; 883 } 884 } 885 } 886 887 static void 888 post_ex(POST_ARGS) 889 { 890 struct roff_node *n; 891 int ic; 892 893 post_std(mdoc); 894 895 n = mdoc->last; 896 mdoc->next = ROFF_NEXT_CHILD; 897 roff_word_alloc(mdoc, n->line, n->pos, "The"); 898 mdoc->last->flags |= NODE_NOSRC; 899 900 if (mdoc->last->next != NULL) 901 ic = build_list(mdoc, MDOC_Nm); 902 else if (mdoc->meta.name != NULL) { 903 roff_elem_alloc(mdoc, n->line, n->pos, MDOC_Nm); 904 mdoc->last->flags |= NODE_NOSRC; 905 roff_word_alloc(mdoc, n->line, n->pos, mdoc->meta.name); 906 mdoc->last->flags |= NODE_NOSRC; 907 mdoc->last = mdoc->last->parent; 908 mdoc->next = ROFF_NEXT_SIBLING; 909 ic = 1; 910 } else { 911 mandoc_msg(MANDOCERR_EX_NONAME, mdoc->parse, 912 n->line, n->pos, "Ex"); 913 ic = 0; 914 } 915 916 roff_word_alloc(mdoc, n->line, n->pos, 917 ic > 1 ? "utilities exit\\~0" : "utility exits\\~0"); 918 mdoc->last->flags |= NODE_NOSRC; 919 roff_word_alloc(mdoc, n->line, n->pos, 920 "on success, and\\~>0 if an error occurs."); 921 mdoc->last->flags |= NODE_EOS | NODE_NOSRC; 922 mdoc->last = n; 923 } 924 925 static void 926 post_lb(POST_ARGS) 927 { 928 struct roff_node *n; 929 const char *p; 930 931 post_delim_nb(mdoc); 932 933 n = mdoc->last; 934 assert(n->child->type == ROFFT_TEXT); 935 mdoc->next = ROFF_NEXT_CHILD; 936 937 if ((p = mdoc_a2lib(n->child->string)) != NULL) { 938 n->child->flags |= NODE_NOPRT; 939 roff_word_alloc(mdoc, n->line, n->pos, p); 940 mdoc->last->flags = NODE_NOSRC; 941 mdoc->last = n; 942 return; 943 } 944 945 mandoc_vmsg(MANDOCERR_LB_BAD, mdoc->parse, n->child->line, 946 n->child->pos, "Lb %s", n->child->string); 947 948 roff_word_alloc(mdoc, n->line, n->pos, "library"); 949 mdoc->last->flags = NODE_NOSRC; 950 roff_word_alloc(mdoc, n->line, n->pos, "\\(Lq"); 951 mdoc->last->flags = NODE_DELIMO | NODE_NOSRC; 952 mdoc->last = mdoc->last->next; 953 roff_word_alloc(mdoc, n->line, n->pos, "\\(Rq"); 954 mdoc->last->flags = NODE_DELIMC | NODE_NOSRC; 955 mdoc->last = n; 956 } 957 958 static void 959 post_rv(POST_ARGS) 960 { 961 struct roff_node *n; 962 int ic; 963 964 post_std(mdoc); 965 966 n = mdoc->last; 967 mdoc->next = ROFF_NEXT_CHILD; 968 if (n->child != NULL) { 969 roff_word_alloc(mdoc, n->line, n->pos, "The"); 970 mdoc->last->flags |= NODE_NOSRC; 971 ic = build_list(mdoc, MDOC_Fn); 972 roff_word_alloc(mdoc, n->line, n->pos, 973 ic > 1 ? "functions return" : "function returns"); 974 mdoc->last->flags |= NODE_NOSRC; 975 roff_word_alloc(mdoc, n->line, n->pos, 976 "the value\\~0 if successful;"); 977 } else 978 roff_word_alloc(mdoc, n->line, n->pos, "Upon successful " 979 "completion, the value\\~0 is returned;"); 980 mdoc->last->flags |= NODE_NOSRC; 981 982 roff_word_alloc(mdoc, n->line, n->pos, "otherwise " 983 "the value\\~\\-1 is returned and the global variable"); 984 mdoc->last->flags |= NODE_NOSRC; 985 roff_elem_alloc(mdoc, n->line, n->pos, MDOC_Va); 986 mdoc->last->flags |= NODE_NOSRC; 987 roff_word_alloc(mdoc, n->line, n->pos, "errno"); 988 mdoc->last->flags |= NODE_NOSRC; 989 mdoc->last = mdoc->last->parent; 990 mdoc->next = ROFF_NEXT_SIBLING; 991 roff_word_alloc(mdoc, n->line, n->pos, 992 "is set to indicate the error."); 993 mdoc->last->flags |= NODE_EOS | NODE_NOSRC; 994 mdoc->last = n; 995 } 996 997 static void 998 post_std(POST_ARGS) 999 { 1000 struct roff_node *n; 1001 1002 post_delim(mdoc); 1003 1004 n = mdoc->last; 1005 if (n->args && n->args->argc == 1) 1006 if (n->args->argv[0].arg == MDOC_Std) 1007 return; 1008 1009 mandoc_msg(MANDOCERR_ARG_STD, mdoc->parse, 1010 n->line, n->pos, roff_name[n->tok]); 1011 } 1012 1013 static void 1014 post_st(POST_ARGS) 1015 { 1016 struct roff_node *n, *nch; 1017 const char *p; 1018 1019 n = mdoc->last; 1020 nch = n->child; 1021 assert(nch->type == ROFFT_TEXT); 1022 1023 if ((p = mdoc_a2st(nch->string)) == NULL) { 1024 mandoc_vmsg(MANDOCERR_ST_BAD, mdoc->parse, 1025 nch->line, nch->pos, "St %s", nch->string); 1026 roff_node_delete(mdoc, n); 1027 return; 1028 } 1029 1030 nch->flags |= NODE_NOPRT; 1031 mdoc->next = ROFF_NEXT_CHILD; 1032 roff_word_alloc(mdoc, nch->line, nch->pos, p); 1033 mdoc->last->flags |= NODE_NOSRC; 1034 mdoc->last= n; 1035 } 1036 1037 static void 1038 post_obsolete(POST_ARGS) 1039 { 1040 struct roff_node *n; 1041 1042 n = mdoc->last; 1043 if (n->type == ROFFT_ELEM || n->type == ROFFT_BLOCK) 1044 mandoc_msg(MANDOCERR_MACRO_OBS, mdoc->parse, 1045 n->line, n->pos, roff_name[n->tok]); 1046 } 1047 1048 static void 1049 post_useless(POST_ARGS) 1050 { 1051 struct roff_node *n; 1052 1053 n = mdoc->last; 1054 mandoc_msg(MANDOCERR_MACRO_USELESS, mdoc->parse, 1055 n->line, n->pos, roff_name[n->tok]); 1056 } 1057 1058 /* 1059 * Block macros. 1060 */ 1061 1062 static void 1063 post_bf(POST_ARGS) 1064 { 1065 struct roff_node *np, *nch; 1066 1067 /* 1068 * Unlike other data pointers, these are "housed" by the HEAD 1069 * element, which contains the goods. 1070 */ 1071 1072 np = mdoc->last; 1073 if (np->type != ROFFT_HEAD) 1074 return; 1075 1076 assert(np->parent->type == ROFFT_BLOCK); 1077 assert(np->parent->tok == MDOC_Bf); 1078 1079 /* Check the number of arguments. */ 1080 1081 nch = np->child; 1082 if (np->parent->args == NULL) { 1083 if (nch == NULL) { 1084 mandoc_msg(MANDOCERR_BF_NOFONT, mdoc->parse, 1085 np->line, np->pos, "Bf"); 1086 return; 1087 } 1088 nch = nch->next; 1089 } 1090 if (nch != NULL) 1091 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 1092 nch->line, nch->pos, "Bf ... %s", nch->string); 1093 1094 /* Extract argument into data. */ 1095 1096 if (np->parent->args != NULL) { 1097 switch (np->parent->args->argv[0].arg) { 1098 case MDOC_Emphasis: 1099 np->norm->Bf.font = FONT_Em; 1100 break; 1101 case MDOC_Literal: 1102 np->norm->Bf.font = FONT_Li; 1103 break; 1104 case MDOC_Symbolic: 1105 np->norm->Bf.font = FONT_Sy; 1106 break; 1107 default: 1108 abort(); 1109 } 1110 return; 1111 } 1112 1113 /* Extract parameter into data. */ 1114 1115 if ( ! strcmp(np->child->string, "Em")) 1116 np->norm->Bf.font = FONT_Em; 1117 else if ( ! strcmp(np->child->string, "Li")) 1118 np->norm->Bf.font = FONT_Li; 1119 else if ( ! strcmp(np->child->string, "Sy")) 1120 np->norm->Bf.font = FONT_Sy; 1121 else 1122 mandoc_vmsg(MANDOCERR_BF_BADFONT, mdoc->parse, 1123 np->child->line, np->child->pos, 1124 "Bf %s", np->child->string); 1125 } 1126 1127 static void 1128 post_fname(POST_ARGS) 1129 { 1130 const struct roff_node *n; 1131 const char *cp; 1132 size_t pos; 1133 1134 n = mdoc->last->child; 1135 pos = strcspn(n->string, "()"); 1136 cp = n->string + pos; 1137 if ( ! (cp[0] == '\0' || (cp[0] == '(' && cp[1] == '*'))) 1138 mandoc_msg(MANDOCERR_FN_PAREN, mdoc->parse, 1139 n->line, n->pos + pos, n->string); 1140 } 1141 1142 static void 1143 post_fn(POST_ARGS) 1144 { 1145 1146 post_fname(mdoc); 1147 post_fa(mdoc); 1148 } 1149 1150 static void 1151 post_fo(POST_ARGS) 1152 { 1153 const struct roff_node *n; 1154 1155 n = mdoc->last; 1156 1157 if (n->type != ROFFT_HEAD) 1158 return; 1159 1160 if (n->child == NULL) { 1161 mandoc_msg(MANDOCERR_FO_NOHEAD, mdoc->parse, 1162 n->line, n->pos, "Fo"); 1163 return; 1164 } 1165 if (n->child != n->last) { 1166 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 1167 n->child->next->line, n->child->next->pos, 1168 "Fo ... %s", n->child->next->string); 1169 while (n->child != n->last) 1170 roff_node_delete(mdoc, n->last); 1171 } else 1172 post_delim(mdoc); 1173 1174 post_fname(mdoc); 1175 } 1176 1177 static void 1178 post_fa(POST_ARGS) 1179 { 1180 const struct roff_node *n; 1181 const char *cp; 1182 1183 for (n = mdoc->last->child; n != NULL; n = n->next) { 1184 for (cp = n->string; *cp != '\0'; cp++) { 1185 /* Ignore callbacks and alterations. */ 1186 if (*cp == '(' || *cp == '{') 1187 break; 1188 if (*cp != ',') 1189 continue; 1190 mandoc_msg(MANDOCERR_FA_COMMA, mdoc->parse, 1191 n->line, n->pos + (cp - n->string), 1192 n->string); 1193 break; 1194 } 1195 } 1196 post_delim_nb(mdoc); 1197 } 1198 1199 static void 1200 post_nm(POST_ARGS) 1201 { 1202 struct roff_node *n; 1203 1204 n = mdoc->last; 1205 1206 if (n->sec == SEC_NAME && n->child != NULL && 1207 n->child->type == ROFFT_TEXT && mdoc->meta.msec != NULL) 1208 mandoc_xr_add(mdoc->meta.msec, n->child->string, -1, -1); 1209 1210 if (n->last != NULL && 1211 (n->last->tok == MDOC_Pp || 1212 n->last->tok == MDOC_Lp)) 1213 mdoc_node_relink(mdoc, n->last); 1214 1215 if (mdoc->meta.name == NULL) 1216 deroff(&mdoc->meta.name, n); 1217 1218 if (mdoc->meta.name == NULL || 1219 (mdoc->lastsec == SEC_NAME && n->child == NULL)) 1220 mandoc_msg(MANDOCERR_NM_NONAME, mdoc->parse, 1221 n->line, n->pos, "Nm"); 1222 1223 switch (n->type) { 1224 case ROFFT_ELEM: 1225 post_delim_nb(mdoc); 1226 break; 1227 case ROFFT_HEAD: 1228 post_delim(mdoc); 1229 break; 1230 default: 1231 return; 1232 } 1233 1234 if ((n->child != NULL && n->child->type == ROFFT_TEXT) || 1235 mdoc->meta.name == NULL) 1236 return; 1237 1238 mdoc->next = ROFF_NEXT_CHILD; 1239 roff_word_alloc(mdoc, n->line, n->pos, mdoc->meta.name); 1240 mdoc->last->flags |= NODE_NOSRC; 1241 mdoc->last = n; 1242 } 1243 1244 static void 1245 post_nd(POST_ARGS) 1246 { 1247 struct roff_node *n; 1248 1249 n = mdoc->last; 1250 1251 if (n->type != ROFFT_BODY) 1252 return; 1253 1254 if (n->sec != SEC_NAME) 1255 mandoc_msg(MANDOCERR_ND_LATE, mdoc->parse, 1256 n->line, n->pos, "Nd"); 1257 1258 if (n->child == NULL) 1259 mandoc_msg(MANDOCERR_ND_EMPTY, mdoc->parse, 1260 n->line, n->pos, "Nd"); 1261 else 1262 post_delim(mdoc); 1263 1264 post_hyph(mdoc); 1265 } 1266 1267 static void 1268 post_display(POST_ARGS) 1269 { 1270 struct roff_node *n, *np; 1271 1272 n = mdoc->last; 1273 switch (n->type) { 1274 case ROFFT_BODY: 1275 if (n->end != ENDBODY_NOT) { 1276 if (n->tok == MDOC_Bd && 1277 n->body->parent->args == NULL) 1278 roff_node_delete(mdoc, n); 1279 } else if (n->child == NULL) 1280 mandoc_msg(MANDOCERR_BLK_EMPTY, mdoc->parse, 1281 n->line, n->pos, roff_name[n->tok]); 1282 else if (n->tok == MDOC_D1) 1283 post_hyph(mdoc); 1284 break; 1285 case ROFFT_BLOCK: 1286 if (n->tok == MDOC_Bd) { 1287 if (n->args == NULL) { 1288 mandoc_msg(MANDOCERR_BD_NOARG, 1289 mdoc->parse, n->line, n->pos, "Bd"); 1290 mdoc->next = ROFF_NEXT_SIBLING; 1291 while (n->body->child != NULL) 1292 mdoc_node_relink(mdoc, 1293 n->body->child); 1294 roff_node_delete(mdoc, n); 1295 break; 1296 } 1297 post_bd(mdoc); 1298 post_prevpar(mdoc); 1299 } 1300 for (np = n->parent; np != NULL; np = np->parent) { 1301 if (np->type == ROFFT_BLOCK && np->tok == MDOC_Bd) { 1302 mandoc_vmsg(MANDOCERR_BD_NEST, 1303 mdoc->parse, n->line, n->pos, 1304 "%s in Bd", roff_name[n->tok]); 1305 break; 1306 } 1307 } 1308 break; 1309 default: 1310 break; 1311 } 1312 } 1313 1314 static void 1315 post_defaults(POST_ARGS) 1316 { 1317 struct roff_node *nn; 1318 1319 if (mdoc->last->child != NULL) { 1320 post_delim_nb(mdoc); 1321 return; 1322 } 1323 1324 /* 1325 * The `Ar' defaults to "file ..." if no value is provided as an 1326 * argument; the `Mt' and `Pa' macros use "~"; the `Li' just 1327 * gets an empty string. 1328 */ 1329 1330 nn = mdoc->last; 1331 switch (nn->tok) { 1332 case MDOC_Ar: 1333 mdoc->next = ROFF_NEXT_CHILD; 1334 roff_word_alloc(mdoc, nn->line, nn->pos, "file"); 1335 mdoc->last->flags |= NODE_NOSRC; 1336 roff_word_alloc(mdoc, nn->line, nn->pos, "..."); 1337 mdoc->last->flags |= NODE_NOSRC; 1338 break; 1339 case MDOC_Pa: 1340 case MDOC_Mt: 1341 mdoc->next = ROFF_NEXT_CHILD; 1342 roff_word_alloc(mdoc, nn->line, nn->pos, "~"); 1343 mdoc->last->flags |= NODE_NOSRC; 1344 break; 1345 default: 1346 abort(); 1347 } 1348 mdoc->last = nn; 1349 } 1350 1351 static void 1352 post_at(POST_ARGS) 1353 { 1354 struct roff_node *n, *nch; 1355 const char *att; 1356 1357 n = mdoc->last; 1358 nch = n->child; 1359 1360 /* 1361 * If we have a child, look it up in the standard keys. If a 1362 * key exist, use that instead of the child; if it doesn't, 1363 * prefix "AT&T UNIX " to the existing data. 1364 */ 1365 1366 att = NULL; 1367 if (nch != NULL && ((att = mdoc_a2att(nch->string)) == NULL)) 1368 mandoc_vmsg(MANDOCERR_AT_BAD, mdoc->parse, 1369 nch->line, nch->pos, "At %s", nch->string); 1370 1371 mdoc->next = ROFF_NEXT_CHILD; 1372 if (att != NULL) { 1373 roff_word_alloc(mdoc, nch->line, nch->pos, att); 1374 nch->flags |= NODE_NOPRT; 1375 } else 1376 roff_word_alloc(mdoc, n->line, n->pos, "AT&T UNIX"); 1377 mdoc->last->flags |= NODE_NOSRC; 1378 mdoc->last = n; 1379 } 1380 1381 static void 1382 post_an(POST_ARGS) 1383 { 1384 struct roff_node *np, *nch; 1385 1386 post_an_norm(mdoc); 1387 1388 np = mdoc->last; 1389 nch = np->child; 1390 if (np->norm->An.auth == AUTH__NONE) { 1391 if (nch == NULL) 1392 mandoc_msg(MANDOCERR_MACRO_EMPTY, mdoc->parse, 1393 np->line, np->pos, "An"); 1394 else 1395 post_delim_nb(mdoc); 1396 } else if (nch != NULL) 1397 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 1398 nch->line, nch->pos, "An ... %s", nch->string); 1399 } 1400 1401 static void 1402 post_en(POST_ARGS) 1403 { 1404 1405 post_obsolete(mdoc); 1406 if (mdoc->last->type == ROFFT_BLOCK) 1407 mdoc->last->norm->Es = mdoc->last_es; 1408 } 1409 1410 static void 1411 post_es(POST_ARGS) 1412 { 1413 1414 post_obsolete(mdoc); 1415 mdoc->last_es = mdoc->last; 1416 } 1417 1418 static void 1419 post_xx(POST_ARGS) 1420 { 1421 struct roff_node *n; 1422 const char *os; 1423 char *v; 1424 1425 post_delim_nb(mdoc); 1426 1427 n = mdoc->last; 1428 switch (n->tok) { 1429 case MDOC_Bsx: 1430 os = "BSD/OS"; 1431 break; 1432 case MDOC_Dx: 1433 os = "DragonFly"; 1434 break; 1435 case MDOC_Fx: 1436 os = "FreeBSD"; 1437 break; 1438 case MDOC_Nx: 1439 os = "NetBSD"; 1440 if (n->child == NULL) 1441 break; 1442 v = n->child->string; 1443 if ((v[0] != '0' && v[0] != '1') || v[1] != '.' || 1444 v[2] < '0' || v[2] > '9' || 1445 v[3] < 'a' || v[3] > 'z' || v[4] != '\0') 1446 break; 1447 n->child->flags |= NODE_NOPRT; 1448 mdoc->next = ROFF_NEXT_CHILD; 1449 roff_word_alloc(mdoc, n->child->line, n->child->pos, v); 1450 v = mdoc->last->string; 1451 v[3] = toupper((unsigned char)v[3]); 1452 mdoc->last->flags |= NODE_NOSRC; 1453 mdoc->last = n; 1454 break; 1455 case MDOC_Ox: 1456 os = "OpenBSD"; 1457 break; 1458 case MDOC_Ux: 1459 os = "UNIX"; 1460 break; 1461 default: 1462 abort(); 1463 } 1464 mdoc->next = ROFF_NEXT_CHILD; 1465 roff_word_alloc(mdoc, n->line, n->pos, os); 1466 mdoc->last->flags |= NODE_NOSRC; 1467 mdoc->last = n; 1468 } 1469 1470 static void 1471 post_it(POST_ARGS) 1472 { 1473 struct roff_node *nbl, *nit, *nch; 1474 int i, cols; 1475 enum mdoc_list lt; 1476 1477 post_prevpar(mdoc); 1478 1479 nit = mdoc->last; 1480 if (nit->type != ROFFT_BLOCK) 1481 return; 1482 1483 nbl = nit->parent->parent; 1484 lt = nbl->norm->Bl.type; 1485 1486 switch (lt) { 1487 case LIST_tag: 1488 case LIST_hang: 1489 case LIST_ohang: 1490 case LIST_inset: 1491 case LIST_diag: 1492 if (nit->head->child == NULL) 1493 mandoc_vmsg(MANDOCERR_IT_NOHEAD, 1494 mdoc->parse, nit->line, nit->pos, 1495 "Bl -%s It", 1496 mdoc_argnames[nbl->args->argv[0].arg]); 1497 break; 1498 case LIST_bullet: 1499 case LIST_dash: 1500 case LIST_enum: 1501 case LIST_hyphen: 1502 if (nit->body == NULL || nit->body->child == NULL) 1503 mandoc_vmsg(MANDOCERR_IT_NOBODY, 1504 mdoc->parse, nit->line, nit->pos, 1505 "Bl -%s It", 1506 mdoc_argnames[nbl->args->argv[0].arg]); 1507 /* FALLTHROUGH */ 1508 case LIST_item: 1509 if ((nch = nit->head->child) != NULL) 1510 mandoc_vmsg(MANDOCERR_ARG_SKIP, mdoc->parse, 1511 nit->line, nit->pos, "It %s", 1512 nch->string == NULL ? roff_name[nch->tok] : 1513 nch->string); 1514 break; 1515 case LIST_column: 1516 cols = (int)nbl->norm->Bl.ncols; 1517 1518 assert(nit->head->child == NULL); 1519 1520 if (nit->head->next->child == NULL && 1521 nit->head->next->next == NULL) { 1522 mandoc_msg(MANDOCERR_MACRO_EMPTY, mdoc->parse, 1523 nit->line, nit->pos, "It"); 1524 roff_node_delete(mdoc, nit); 1525 break; 1526 } 1527 1528 i = 0; 1529 for (nch = nit->child; nch != NULL; nch = nch->next) { 1530 if (nch->type != ROFFT_BODY) 1531 continue; 1532 if (i++ && nch->flags & NODE_LINE) 1533 mandoc_msg(MANDOCERR_TA_LINE, mdoc->parse, 1534 nch->line, nch->pos, "Ta"); 1535 } 1536 if (i < cols || i > cols + 1) 1537 mandoc_vmsg(MANDOCERR_BL_COL, 1538 mdoc->parse, nit->line, nit->pos, 1539 "%d columns, %d cells", cols, i); 1540 else if (nit->head->next->child != NULL && 1541 nit->head->next->child->line > nit->line) 1542 mandoc_msg(MANDOCERR_IT_NOARG, mdoc->parse, 1543 nit->line, nit->pos, "Bl -column It"); 1544 break; 1545 default: 1546 abort(); 1547 } 1548 } 1549 1550 static void 1551 post_bl_block(POST_ARGS) 1552 { 1553 struct roff_node *n, *ni, *nc; 1554 1555 post_prevpar(mdoc); 1556 1557 n = mdoc->last; 1558 for (ni = n->body->child; ni != NULL; ni = ni->next) { 1559 if (ni->body == NULL) 1560 continue; 1561 nc = ni->body->last; 1562 while (nc != NULL) { 1563 switch (nc->tok) { 1564 case MDOC_Pp: 1565 case MDOC_Lp: 1566 case ROFF_br: 1567 break; 1568 default: 1569 nc = NULL; 1570 continue; 1571 } 1572 if (ni->next == NULL) { 1573 mandoc_msg(MANDOCERR_PAR_MOVE, 1574 mdoc->parse, nc->line, nc->pos, 1575 roff_name[nc->tok]); 1576 mdoc_node_relink(mdoc, nc); 1577 } else if (n->norm->Bl.comp == 0 && 1578 n->norm->Bl.type != LIST_column) { 1579 mandoc_vmsg(MANDOCERR_PAR_SKIP, 1580 mdoc->parse, nc->line, nc->pos, 1581 "%s before It", roff_name[nc->tok]); 1582 roff_node_delete(mdoc, nc); 1583 } else 1584 break; 1585 nc = ni->body->last; 1586 } 1587 } 1588 } 1589 1590 /* 1591 * If the argument of -offset or -width is a macro, 1592 * replace it with the associated default width. 1593 */ 1594 static void 1595 rewrite_macro2len(struct roff_man *mdoc, char **arg) 1596 { 1597 size_t width; 1598 enum roff_tok tok; 1599 1600 if (*arg == NULL) 1601 return; 1602 else if ( ! strcmp(*arg, "Ds")) 1603 width = 6; 1604 else if ((tok = roffhash_find(mdoc->mdocmac, *arg, 0)) == TOKEN_NONE) 1605 return; 1606 else 1607 width = macro2len(tok); 1608 1609 free(*arg); 1610 mandoc_asprintf(arg, "%zun", width); 1611 } 1612 1613 static void 1614 post_bl_head(POST_ARGS) 1615 { 1616 struct roff_node *nbl, *nh, *nch, *nnext; 1617 struct mdoc_argv *argv; 1618 int i, j; 1619 1620 post_bl_norm(mdoc); 1621 1622 nh = mdoc->last; 1623 if (nh->norm->Bl.type != LIST_column) { 1624 if ((nch = nh->child) == NULL) 1625 return; 1626 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 1627 nch->line, nch->pos, "Bl ... %s", nch->string); 1628 while (nch != NULL) { 1629 roff_node_delete(mdoc, nch); 1630 nch = nh->child; 1631 } 1632 return; 1633 } 1634 1635 /* 1636 * Append old-style lists, where the column width specifiers 1637 * trail as macro parameters, to the new-style ("normal-form") 1638 * lists where they're argument values following -column. 1639 */ 1640 1641 if (nh->child == NULL) 1642 return; 1643 1644 nbl = nh->parent; 1645 for (j = 0; j < (int)nbl->args->argc; j++) 1646 if (nbl->args->argv[j].arg == MDOC_Column) 1647 break; 1648 1649 assert(j < (int)nbl->args->argc); 1650 1651 /* 1652 * Accommodate for new-style groff column syntax. Shuffle the 1653 * child nodes, all of which must be TEXT, as arguments for the 1654 * column field. Then, delete the head children. 1655 */ 1656 1657 argv = nbl->args->argv + j; 1658 i = argv->sz; 1659 for (nch = nh->child; nch != NULL; nch = nch->next) 1660 argv->sz++; 1661 argv->value = mandoc_reallocarray(argv->value, 1662 argv->sz, sizeof(char *)); 1663 1664 nh->norm->Bl.ncols = argv->sz; 1665 nh->norm->Bl.cols = (void *)argv->value; 1666 1667 for (nch = nh->child; nch != NULL; nch = nnext) { 1668 argv->value[i++] = nch->string; 1669 nch->string = NULL; 1670 nnext = nch->next; 1671 roff_node_delete(NULL, nch); 1672 } 1673 nh->child = NULL; 1674 } 1675 1676 static void 1677 post_bl(POST_ARGS) 1678 { 1679 struct roff_node *nparent, *nprev; /* of the Bl block */ 1680 struct roff_node *nblock, *nbody; /* of the Bl */ 1681 struct roff_node *nchild, *nnext; /* of the Bl body */ 1682 const char *prev_Er; 1683 int order; 1684 1685 nbody = mdoc->last; 1686 switch (nbody->type) { 1687 case ROFFT_BLOCK: 1688 post_bl_block(mdoc); 1689 return; 1690 case ROFFT_HEAD: 1691 post_bl_head(mdoc); 1692 return; 1693 case ROFFT_BODY: 1694 break; 1695 default: 1696 return; 1697 } 1698 if (nbody->end != ENDBODY_NOT) 1699 return; 1700 1701 nchild = nbody->child; 1702 if (nchild == NULL) { 1703 mandoc_msg(MANDOCERR_BLK_EMPTY, mdoc->parse, 1704 nbody->line, nbody->pos, "Bl"); 1705 return; 1706 } 1707 while (nchild != NULL) { 1708 nnext = nchild->next; 1709 if (nchild->tok == MDOC_It || 1710 (nchild->tok == MDOC_Sm && 1711 nnext != NULL && nnext->tok == MDOC_It)) { 1712 nchild = nnext; 1713 continue; 1714 } 1715 1716 /* 1717 * In .Bl -column, the first rows may be implicit, 1718 * that is, they may not start with .It macros. 1719 * Such rows may be followed by nodes generated on the 1720 * roff level, for example .TS, which cannot be moved 1721 * out of the list. In that case, wrap such roff nodes 1722 * into an implicit row. 1723 */ 1724 1725 if (nchild->prev != NULL) { 1726 mdoc->last = nchild; 1727 mdoc->next = ROFF_NEXT_SIBLING; 1728 roff_block_alloc(mdoc, nchild->line, 1729 nchild->pos, MDOC_It); 1730 roff_head_alloc(mdoc, nchild->line, 1731 nchild->pos, MDOC_It); 1732 mdoc->next = ROFF_NEXT_SIBLING; 1733 roff_body_alloc(mdoc, nchild->line, 1734 nchild->pos, MDOC_It); 1735 while (nchild->tok != MDOC_It) { 1736 mdoc_node_relink(mdoc, nchild); 1737 if ((nchild = nnext) == NULL) 1738 break; 1739 nnext = nchild->next; 1740 mdoc->next = ROFF_NEXT_SIBLING; 1741 } 1742 mdoc->last = nbody; 1743 continue; 1744 } 1745 1746 mandoc_msg(MANDOCERR_BL_MOVE, mdoc->parse, 1747 nchild->line, nchild->pos, roff_name[nchild->tok]); 1748 1749 /* 1750 * Move the node out of the Bl block. 1751 * First, collect all required node pointers. 1752 */ 1753 1754 nblock = nbody->parent; 1755 nprev = nblock->prev; 1756 nparent = nblock->parent; 1757 1758 /* 1759 * Unlink this child. 1760 */ 1761 1762 nbody->child = nnext; 1763 if (nnext == NULL) 1764 nbody->last = NULL; 1765 else 1766 nnext->prev = NULL; 1767 1768 /* 1769 * Relink this child. 1770 */ 1771 1772 nchild->parent = nparent; 1773 nchild->prev = nprev; 1774 nchild->next = nblock; 1775 1776 nblock->prev = nchild; 1777 if (nprev == NULL) 1778 nparent->child = nchild; 1779 else 1780 nprev->next = nchild; 1781 1782 nchild = nnext; 1783 } 1784 1785 if (mdoc->meta.os_e != MANDOC_OS_NETBSD) 1786 return; 1787 1788 prev_Er = NULL; 1789 for (nchild = nbody->child; nchild != NULL; nchild = nchild->next) { 1790 if (nchild->tok != MDOC_It) 1791 continue; 1792 if ((nnext = nchild->head->child) == NULL) 1793 continue; 1794 if (nnext->type == ROFFT_BLOCK) 1795 nnext = nnext->body->child; 1796 if (nnext == NULL || nnext->tok != MDOC_Er) 1797 continue; 1798 nnext = nnext->child; 1799 if (prev_Er != NULL) { 1800 order = strcmp(prev_Er, nnext->string); 1801 if (order > 0) 1802 mandoc_vmsg(MANDOCERR_ER_ORDER, 1803 mdoc->parse, nnext->line, nnext->pos, 1804 "Er %s %s (NetBSD)", 1805 prev_Er, nnext->string); 1806 else if (order == 0) 1807 mandoc_vmsg(MANDOCERR_ER_REP, 1808 mdoc->parse, nnext->line, nnext->pos, 1809 "Er %s (NetBSD)", prev_Er); 1810 } 1811 prev_Er = nnext->string; 1812 } 1813 } 1814 1815 static void 1816 post_bk(POST_ARGS) 1817 { 1818 struct roff_node *n; 1819 1820 n = mdoc->last; 1821 1822 if (n->type == ROFFT_BLOCK && n->body->child == NULL) { 1823 mandoc_msg(MANDOCERR_BLK_EMPTY, 1824 mdoc->parse, n->line, n->pos, "Bk"); 1825 roff_node_delete(mdoc, n); 1826 } 1827 } 1828 1829 static void 1830 post_sm(POST_ARGS) 1831 { 1832 struct roff_node *nch; 1833 1834 nch = mdoc->last->child; 1835 1836 if (nch == NULL) { 1837 mdoc->flags ^= MDOC_SMOFF; 1838 return; 1839 } 1840 1841 assert(nch->type == ROFFT_TEXT); 1842 1843 if ( ! strcmp(nch->string, "on")) { 1844 mdoc->flags &= ~MDOC_SMOFF; 1845 return; 1846 } 1847 if ( ! strcmp(nch->string, "off")) { 1848 mdoc->flags |= MDOC_SMOFF; 1849 return; 1850 } 1851 1852 mandoc_vmsg(MANDOCERR_SM_BAD, 1853 mdoc->parse, nch->line, nch->pos, 1854 "%s %s", roff_name[mdoc->last->tok], nch->string); 1855 mdoc_node_relink(mdoc, nch); 1856 return; 1857 } 1858 1859 static void 1860 post_root(POST_ARGS) 1861 { 1862 const char *openbsd_arch[] = { 1863 "alpha", "amd64", "arm64", "armv7", "hppa", "i386", 1864 "landisk", "loongson", "luna88k", "macppc", "mips64", 1865 "octeon", "sgi", "socppc", "sparc64", NULL 1866 }; 1867 const char *netbsd_arch[] = { 1868 "acorn26", "acorn32", "algor", "alpha", "amiga", 1869 "arc", "atari", 1870 "bebox", "cats", "cesfic", "cobalt", "dreamcast", 1871 "emips", "evbarm", "evbmips", "evbppc", "evbsh3", "evbsh5", 1872 "hp300", "hpcarm", "hpcmips", "hpcsh", "hppa", 1873 "i386", "ibmnws", "luna68k", 1874 "mac68k", "macppc", "mipsco", "mmeye", "mvme68k", "mvmeppc", 1875 "netwinder", "news68k", "newsmips", "next68k", 1876 "pc532", "playstation2", "pmax", "pmppc", "prep", 1877 "sandpoint", "sbmips", "sgimips", "shark", 1878 "sparc", "sparc64", "sun2", "sun3", 1879 "vax", "walnut", "x68k", "x86", "x86_64", "xen", NULL 1880 }; 1881 const char **arches[] = { NULL, netbsd_arch, openbsd_arch }; 1882 1883 struct roff_node *n; 1884 const char **arch; 1885 1886 /* Add missing prologue data. */ 1887 1888 if (mdoc->meta.date == NULL) 1889 mdoc->meta.date = mdoc->quick ? mandoc_strdup("") : 1890 mandoc_normdate(mdoc, NULL, 0, 0); 1891 1892 if (mdoc->meta.title == NULL) { 1893 mandoc_msg(MANDOCERR_DT_NOTITLE, 1894 mdoc->parse, 0, 0, "EOF"); 1895 mdoc->meta.title = mandoc_strdup("UNTITLED"); 1896 } 1897 1898 if (mdoc->meta.vol == NULL) 1899 mdoc->meta.vol = mandoc_strdup("LOCAL"); 1900 1901 if (mdoc->meta.os == NULL) { 1902 mandoc_msg(MANDOCERR_OS_MISSING, 1903 mdoc->parse, 0, 0, NULL); 1904 mdoc->meta.os = mandoc_strdup(""); 1905 } else if (mdoc->meta.os_e && 1906 (mdoc->meta.rcsids & (1 << mdoc->meta.os_e)) == 0) 1907 mandoc_msg(MANDOCERR_RCS_MISSING, mdoc->parse, 0, 0, 1908 mdoc->meta.os_e == MANDOC_OS_OPENBSD ? 1909 "(OpenBSD)" : "(NetBSD)"); 1910 1911 if (mdoc->meta.arch != NULL && 1912 (arch = arches[mdoc->meta.os_e]) != NULL) { 1913 while (*arch != NULL && strcmp(*arch, mdoc->meta.arch)) 1914 arch++; 1915 if (*arch == NULL) { 1916 n = mdoc->first->child; 1917 while (n->tok != MDOC_Dt) 1918 n = n->next; 1919 n = n->child->next->next; 1920 mandoc_vmsg(MANDOCERR_ARCH_BAD, 1921 mdoc->parse, n->line, n->pos, 1922 "Dt ... %s %s", mdoc->meta.arch, 1923 mdoc->meta.os_e == MANDOC_OS_OPENBSD ? 1924 "(OpenBSD)" : "(NetBSD)"); 1925 } 1926 } 1927 1928 /* Check that we begin with a proper `Sh'. */ 1929 1930 n = mdoc->first->child; 1931 while (n != NULL && n->tok >= MDOC_Dd && 1932 mdoc_macros[n->tok].flags & MDOC_PROLOGUE) 1933 n = n->next; 1934 1935 if (n == NULL) 1936 mandoc_msg(MANDOCERR_DOC_EMPTY, mdoc->parse, 0, 0, NULL); 1937 else if (n->tok != MDOC_Sh) 1938 mandoc_msg(MANDOCERR_SEC_BEFORE, mdoc->parse, 1939 n->line, n->pos, roff_name[n->tok]); 1940 } 1941 1942 static void 1943 post_rs(POST_ARGS) 1944 { 1945 struct roff_node *np, *nch, *next, *prev; 1946 int i, j; 1947 1948 np = mdoc->last; 1949 1950 if (np->type != ROFFT_BODY) 1951 return; 1952 1953 if (np->child == NULL) { 1954 mandoc_msg(MANDOCERR_RS_EMPTY, mdoc->parse, 1955 np->line, np->pos, "Rs"); 1956 return; 1957 } 1958 1959 /* 1960 * The full `Rs' block needs special handling to order the 1961 * sub-elements according to `rsord'. Pick through each element 1962 * and correctly order it. This is an insertion sort. 1963 */ 1964 1965 next = NULL; 1966 for (nch = np->child->next; nch != NULL; nch = next) { 1967 /* Determine order number of this child. */ 1968 for (i = 0; i < RSORD_MAX; i++) 1969 if (rsord[i] == nch->tok) 1970 break; 1971 1972 if (i == RSORD_MAX) { 1973 mandoc_msg(MANDOCERR_RS_BAD, mdoc->parse, 1974 nch->line, nch->pos, roff_name[nch->tok]); 1975 i = -1; 1976 } else if (nch->tok == MDOC__J || nch->tok == MDOC__B) 1977 np->norm->Rs.quote_T++; 1978 1979 /* 1980 * Remove this child from the chain. This somewhat 1981 * repeats roff_node_unlink(), but since we're 1982 * just re-ordering, there's no need for the 1983 * full unlink process. 1984 */ 1985 1986 if ((next = nch->next) != NULL) 1987 next->prev = nch->prev; 1988 1989 if ((prev = nch->prev) != NULL) 1990 prev->next = nch->next; 1991 1992 nch->prev = nch->next = NULL; 1993 1994 /* 1995 * Scan back until we reach a node that's 1996 * to be ordered before this child. 1997 */ 1998 1999 for ( ; prev ; prev = prev->prev) { 2000 /* Determine order of `prev'. */ 2001 for (j = 0; j < RSORD_MAX; j++) 2002 if (rsord[j] == prev->tok) 2003 break; 2004 if (j == RSORD_MAX) 2005 j = -1; 2006 2007 if (j <= i) 2008 break; 2009 } 2010 2011 /* 2012 * Set this child back into its correct place 2013 * in front of the `prev' node. 2014 */ 2015 2016 nch->prev = prev; 2017 2018 if (prev == NULL) { 2019 np->child->prev = nch; 2020 nch->next = np->child; 2021 np->child = nch; 2022 } else { 2023 if (prev->next) 2024 prev->next->prev = nch; 2025 nch->next = prev->next; 2026 prev->next = nch; 2027 } 2028 } 2029 } 2030 2031 /* 2032 * For some arguments of some macros, 2033 * convert all breakable hyphens into ASCII_HYPH. 2034 */ 2035 static void 2036 post_hyph(POST_ARGS) 2037 { 2038 struct roff_node *nch; 2039 char *cp; 2040 2041 for (nch = mdoc->last->child; nch != NULL; nch = nch->next) { 2042 if (nch->type != ROFFT_TEXT) 2043 continue; 2044 cp = nch->string; 2045 if (*cp == '\0') 2046 continue; 2047 while (*(++cp) != '\0') 2048 if (*cp == '-' && 2049 isalpha((unsigned char)cp[-1]) && 2050 isalpha((unsigned char)cp[1])) 2051 *cp = ASCII_HYPH; 2052 } 2053 } 2054 2055 static void 2056 post_ns(POST_ARGS) 2057 { 2058 struct roff_node *n; 2059 2060 n = mdoc->last; 2061 if (n->flags & NODE_LINE || 2062 (n->next != NULL && n->next->flags & NODE_DELIMC)) 2063 mandoc_msg(MANDOCERR_NS_SKIP, mdoc->parse, 2064 n->line, n->pos, NULL); 2065 } 2066 2067 static void 2068 post_sx(POST_ARGS) 2069 { 2070 post_delim(mdoc); 2071 post_hyph(mdoc); 2072 } 2073 2074 static void 2075 post_sh(POST_ARGS) 2076 { 2077 2078 post_ignpar(mdoc); 2079 2080 switch (mdoc->last->type) { 2081 case ROFFT_HEAD: 2082 post_sh_head(mdoc); 2083 break; 2084 case ROFFT_BODY: 2085 switch (mdoc->lastsec) { 2086 case SEC_NAME: 2087 post_sh_name(mdoc); 2088 break; 2089 case SEC_SEE_ALSO: 2090 post_sh_see_also(mdoc); 2091 break; 2092 case SEC_AUTHORS: 2093 post_sh_authors(mdoc); 2094 break; 2095 default: 2096 break; 2097 } 2098 break; 2099 default: 2100 break; 2101 } 2102 } 2103 2104 static void 2105 post_sh_name(POST_ARGS) 2106 { 2107 struct roff_node *n; 2108 int hasnm, hasnd; 2109 2110 hasnm = hasnd = 0; 2111 2112 for (n = mdoc->last->child; n != NULL; n = n->next) { 2113 switch (n->tok) { 2114 case MDOC_Nm: 2115 if (hasnm && n->child != NULL) 2116 mandoc_vmsg(MANDOCERR_NAMESEC_PUNCT, 2117 mdoc->parse, n->line, n->pos, 2118 "Nm %s", n->child->string); 2119 hasnm = 1; 2120 continue; 2121 case MDOC_Nd: 2122 hasnd = 1; 2123 if (n->next != NULL) 2124 mandoc_msg(MANDOCERR_NAMESEC_ND, 2125 mdoc->parse, n->line, n->pos, NULL); 2126 break; 2127 case TOKEN_NONE: 2128 if (n->type == ROFFT_TEXT && 2129 n->string[0] == ',' && n->string[1] == '\0' && 2130 n->next != NULL && n->next->tok == MDOC_Nm) { 2131 n = n->next; 2132 continue; 2133 } 2134 /* FALLTHROUGH */ 2135 default: 2136 mandoc_msg(MANDOCERR_NAMESEC_BAD, mdoc->parse, 2137 n->line, n->pos, roff_name[n->tok]); 2138 continue; 2139 } 2140 break; 2141 } 2142 2143 if ( ! hasnm) 2144 mandoc_msg(MANDOCERR_NAMESEC_NONM, mdoc->parse, 2145 mdoc->last->line, mdoc->last->pos, NULL); 2146 if ( ! hasnd) 2147 mandoc_msg(MANDOCERR_NAMESEC_NOND, mdoc->parse, 2148 mdoc->last->line, mdoc->last->pos, NULL); 2149 } 2150 2151 static void 2152 post_sh_see_also(POST_ARGS) 2153 { 2154 const struct roff_node *n; 2155 const char *name, *sec; 2156 const char *lastname, *lastsec, *lastpunct; 2157 int cmp; 2158 2159 n = mdoc->last->child; 2160 lastname = lastsec = lastpunct = NULL; 2161 while (n != NULL) { 2162 if (n->tok != MDOC_Xr || 2163 n->child == NULL || 2164 n->child->next == NULL) 2165 break; 2166 2167 /* Process one .Xr node. */ 2168 2169 name = n->child->string; 2170 sec = n->child->next->string; 2171 if (lastsec != NULL) { 2172 if (lastpunct[0] != ',' || lastpunct[1] != '\0') 2173 mandoc_vmsg(MANDOCERR_XR_PUNCT, 2174 mdoc->parse, n->line, n->pos, 2175 "%s before %s(%s)", lastpunct, 2176 name, sec); 2177 cmp = strcmp(lastsec, sec); 2178 if (cmp > 0) 2179 mandoc_vmsg(MANDOCERR_XR_ORDER, 2180 mdoc->parse, n->line, n->pos, 2181 "%s(%s) after %s(%s)", name, 2182 sec, lastname, lastsec); 2183 else if (cmp == 0 && 2184 strcasecmp(lastname, name) > 0) 2185 mandoc_vmsg(MANDOCERR_XR_ORDER, 2186 mdoc->parse, n->line, n->pos, 2187 "%s after %s", name, lastname); 2188 } 2189 lastname = name; 2190 lastsec = sec; 2191 2192 /* Process the following node. */ 2193 2194 n = n->next; 2195 if (n == NULL) 2196 break; 2197 if (n->tok == MDOC_Xr) { 2198 lastpunct = "none"; 2199 continue; 2200 } 2201 if (n->type != ROFFT_TEXT) 2202 break; 2203 for (name = n->string; *name != '\0'; name++) 2204 if (isalpha((const unsigned char)*name)) 2205 return; 2206 lastpunct = n->string; 2207 if (n->next == NULL || n->next->tok == MDOC_Rs) 2208 mandoc_vmsg(MANDOCERR_XR_PUNCT, mdoc->parse, 2209 n->line, n->pos, "%s after %s(%s)", 2210 lastpunct, lastname, lastsec); 2211 n = n->next; 2212 } 2213 } 2214 2215 static int 2216 child_an(const struct roff_node *n) 2217 { 2218 2219 for (n = n->child; n != NULL; n = n->next) 2220 if ((n->tok == MDOC_An && n->child != NULL) || child_an(n)) 2221 return 1; 2222 return 0; 2223 } 2224 2225 static void 2226 post_sh_authors(POST_ARGS) 2227 { 2228 2229 if ( ! child_an(mdoc->last)) 2230 mandoc_msg(MANDOCERR_AN_MISSING, mdoc->parse, 2231 mdoc->last->line, mdoc->last->pos, NULL); 2232 } 2233 2234 /* 2235 * Return an upper bound for the string distance (allowing 2236 * transpositions). Not a full Levenshtein implementation 2237 * because Levenshtein is quadratic in the string length 2238 * and this function is called for every standard name, 2239 * so the check for each custom name would be cubic. 2240 * The following crude heuristics is linear, resulting 2241 * in quadratic behaviour for checking one custom name, 2242 * which does not cause measurable slowdown. 2243 */ 2244 static int 2245 similar(const char *s1, const char *s2) 2246 { 2247 const int maxdist = 3; 2248 int dist = 0; 2249 2250 while (s1[0] != '\0' && s2[0] != '\0') { 2251 if (s1[0] == s2[0]) { 2252 s1++; 2253 s2++; 2254 continue; 2255 } 2256 if (++dist > maxdist) 2257 return INT_MAX; 2258 if (s1[1] == s2[1]) { /* replacement */ 2259 s1++; 2260 s2++; 2261 } else if (s1[0] == s2[1] && s1[1] == s2[0]) { 2262 s1 += 2; /* transposition */ 2263 s2 += 2; 2264 } else if (s1[0] == s2[1]) /* insertion */ 2265 s2++; 2266 else if (s1[1] == s2[0]) /* deletion */ 2267 s1++; 2268 else 2269 return INT_MAX; 2270 } 2271 dist += strlen(s1) + strlen(s2); 2272 return dist > maxdist ? INT_MAX : dist; 2273 } 2274 2275 static void 2276 post_sh_head(POST_ARGS) 2277 { 2278 struct roff_node *nch; 2279 const char *goodsec; 2280 const char *const *testsec; 2281 int dist, mindist; 2282 enum roff_sec sec; 2283 2284 /* 2285 * Process a new section. Sections are either "named" or 2286 * "custom". Custom sections are user-defined, while named ones 2287 * follow a conventional order and may only appear in certain 2288 * manual sections. 2289 */ 2290 2291 sec = mdoc->last->sec; 2292 2293 /* The NAME should be first. */ 2294 2295 if (sec != SEC_NAME && mdoc->lastnamed == SEC_NONE) 2296 mandoc_vmsg(MANDOCERR_NAMESEC_FIRST, mdoc->parse, 2297 mdoc->last->line, mdoc->last->pos, "Sh %s", 2298 sec != SEC_CUSTOM ? secnames[sec] : 2299 (nch = mdoc->last->child) == NULL ? "" : 2300 nch->type == ROFFT_TEXT ? nch->string : 2301 roff_name[nch->tok]); 2302 2303 /* The SYNOPSIS gets special attention in other areas. */ 2304 2305 if (sec == SEC_SYNOPSIS) { 2306 roff_setreg(mdoc->roff, "nS", 1, '='); 2307 mdoc->flags |= MDOC_SYNOPSIS; 2308 } else { 2309 roff_setreg(mdoc->roff, "nS", 0, '='); 2310 mdoc->flags &= ~MDOC_SYNOPSIS; 2311 } 2312 2313 /* Mark our last section. */ 2314 2315 mdoc->lastsec = sec; 2316 2317 /* We don't care about custom sections after this. */ 2318 2319 if (sec == SEC_CUSTOM) { 2320 if ((nch = mdoc->last->child) == NULL || 2321 nch->type != ROFFT_TEXT || nch->next != NULL) 2322 return; 2323 goodsec = NULL; 2324 mindist = INT_MAX; 2325 for (testsec = secnames + 1; *testsec != NULL; testsec++) { 2326 dist = similar(nch->string, *testsec); 2327 if (dist < mindist) { 2328 goodsec = *testsec; 2329 mindist = dist; 2330 } 2331 } 2332 if (goodsec != NULL) 2333 mandoc_vmsg(MANDOCERR_SEC_TYPO, mdoc->parse, 2334 nch->line, nch->pos, "Sh %s instead of %s", 2335 nch->string, goodsec); 2336 return; 2337 } 2338 2339 /* 2340 * Check whether our non-custom section is being repeated or is 2341 * out of order. 2342 */ 2343 2344 if (sec == mdoc->lastnamed) 2345 mandoc_vmsg(MANDOCERR_SEC_REP, mdoc->parse, 2346 mdoc->last->line, mdoc->last->pos, 2347 "Sh %s", secnames[sec]); 2348 2349 if (sec < mdoc->lastnamed) 2350 mandoc_vmsg(MANDOCERR_SEC_ORDER, mdoc->parse, 2351 mdoc->last->line, mdoc->last->pos, 2352 "Sh %s", secnames[sec]); 2353 2354 /* Mark the last named section. */ 2355 2356 mdoc->lastnamed = sec; 2357 2358 /* Check particular section/manual conventions. */ 2359 2360 if (mdoc->meta.msec == NULL) 2361 return; 2362 2363 goodsec = NULL; 2364 switch (sec) { 2365 case SEC_ERRORS: 2366 if (*mdoc->meta.msec == '4') 2367 break; 2368 goodsec = "2, 3, 4, 9"; 2369 /* FALLTHROUGH */ 2370 case SEC_RETURN_VALUES: 2371 case SEC_LIBRARY: 2372 if (*mdoc->meta.msec == '2') 2373 break; 2374 if (*mdoc->meta.msec == '3') 2375 break; 2376 if (NULL == goodsec) 2377 goodsec = "2, 3, 9"; 2378 /* FALLTHROUGH */ 2379 case SEC_CONTEXT: 2380 if (*mdoc->meta.msec == '9') 2381 break; 2382 if (NULL == goodsec) 2383 goodsec = "9"; 2384 mandoc_vmsg(MANDOCERR_SEC_MSEC, mdoc->parse, 2385 mdoc->last->line, mdoc->last->pos, 2386 "Sh %s for %s only", secnames[sec], goodsec); 2387 break; 2388 default: 2389 break; 2390 } 2391 } 2392 2393 static void 2394 post_xr(POST_ARGS) 2395 { 2396 struct roff_node *n, *nch; 2397 2398 n = mdoc->last; 2399 nch = n->child; 2400 if (nch->next == NULL) { 2401 mandoc_vmsg(MANDOCERR_XR_NOSEC, mdoc->parse, 2402 n->line, n->pos, "Xr %s", nch->string); 2403 } else { 2404 assert(nch->next == n->last); 2405 if(mandoc_xr_add(nch->next->string, nch->string, 2406 nch->line, nch->pos)) 2407 mandoc_vmsg(MANDOCERR_XR_SELF, mdoc->parse, 2408 nch->line, nch->pos, "Xr %s %s", 2409 nch->string, nch->next->string); 2410 } 2411 post_delim_nb(mdoc); 2412 } 2413 2414 static void 2415 post_ignpar(POST_ARGS) 2416 { 2417 struct roff_node *np; 2418 2419 switch (mdoc->last->type) { 2420 case ROFFT_BLOCK: 2421 post_prevpar(mdoc); 2422 return; 2423 case ROFFT_HEAD: 2424 post_delim(mdoc); 2425 post_hyph(mdoc); 2426 return; 2427 case ROFFT_BODY: 2428 break; 2429 default: 2430 return; 2431 } 2432 2433 if ((np = mdoc->last->child) != NULL) 2434 if (np->tok == MDOC_Pp || np->tok == MDOC_Lp) { 2435 mandoc_vmsg(MANDOCERR_PAR_SKIP, 2436 mdoc->parse, np->line, np->pos, 2437 "%s after %s", roff_name[np->tok], 2438 roff_name[mdoc->last->tok]); 2439 roff_node_delete(mdoc, np); 2440 } 2441 2442 if ((np = mdoc->last->last) != NULL) 2443 if (np->tok == MDOC_Pp || np->tok == MDOC_Lp) { 2444 mandoc_vmsg(MANDOCERR_PAR_SKIP, mdoc->parse, 2445 np->line, np->pos, "%s at the end of %s", 2446 roff_name[np->tok], 2447 roff_name[mdoc->last->tok]); 2448 roff_node_delete(mdoc, np); 2449 } 2450 } 2451 2452 static void 2453 post_prevpar(POST_ARGS) 2454 { 2455 struct roff_node *n; 2456 2457 n = mdoc->last; 2458 if (NULL == n->prev) 2459 return; 2460 if (n->type != ROFFT_ELEM && n->type != ROFFT_BLOCK) 2461 return; 2462 2463 /* 2464 * Don't allow prior `Lp' or `Pp' prior to a paragraph-type 2465 * block: `Lp', `Pp', or non-compact `Bd' or `Bl'. 2466 */ 2467 2468 if (n->prev->tok != MDOC_Pp && 2469 n->prev->tok != MDOC_Lp && 2470 n->prev->tok != ROFF_br) 2471 return; 2472 if (n->tok == MDOC_Bl && n->norm->Bl.comp) 2473 return; 2474 if (n->tok == MDOC_Bd && n->norm->Bd.comp) 2475 return; 2476 if (n->tok == MDOC_It && n->parent->norm->Bl.comp) 2477 return; 2478 2479 mandoc_vmsg(MANDOCERR_PAR_SKIP, mdoc->parse, 2480 n->prev->line, n->prev->pos, "%s before %s", 2481 roff_name[n->prev->tok], roff_name[n->tok]); 2482 roff_node_delete(mdoc, n->prev); 2483 } 2484 2485 static void 2486 post_par(POST_ARGS) 2487 { 2488 struct roff_node *np; 2489 2490 np = mdoc->last; 2491 if (np->tok != ROFF_br && np->tok != ROFF_sp) 2492 post_prevpar(mdoc); 2493 2494 if (np->tok == ROFF_sp) { 2495 if (np->child != NULL && np->child->next != NULL) 2496 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 2497 np->child->next->line, np->child->next->pos, 2498 "sp ... %s", np->child->next->string); 2499 } else if (np->child != NULL) 2500 mandoc_vmsg(MANDOCERR_ARG_SKIP, 2501 mdoc->parse, np->line, np->pos, "%s %s", 2502 roff_name[np->tok], np->child->string); 2503 2504 if ((np = mdoc->last->prev) == NULL) { 2505 np = mdoc->last->parent; 2506 if (np->tok != MDOC_Sh && np->tok != MDOC_Ss) 2507 return; 2508 } else if (np->tok != MDOC_Pp && np->tok != MDOC_Lp && 2509 (mdoc->last->tok != ROFF_br || 2510 (np->tok != ROFF_sp && np->tok != ROFF_br))) 2511 return; 2512 2513 mandoc_vmsg(MANDOCERR_PAR_SKIP, mdoc->parse, 2514 mdoc->last->line, mdoc->last->pos, "%s after %s", 2515 roff_name[mdoc->last->tok], roff_name[np->tok]); 2516 roff_node_delete(mdoc, mdoc->last); 2517 } 2518 2519 static void 2520 post_dd(POST_ARGS) 2521 { 2522 struct roff_node *n; 2523 char *datestr; 2524 2525 n = mdoc->last; 2526 n->flags |= NODE_NOPRT; 2527 2528 if (mdoc->meta.date != NULL) { 2529 mandoc_msg(MANDOCERR_PROLOG_REP, mdoc->parse, 2530 n->line, n->pos, "Dd"); 2531 free(mdoc->meta.date); 2532 } else if (mdoc->flags & MDOC_PBODY) 2533 mandoc_msg(MANDOCERR_PROLOG_LATE, mdoc->parse, 2534 n->line, n->pos, "Dd"); 2535 else if (mdoc->meta.title != NULL) 2536 mandoc_msg(MANDOCERR_PROLOG_ORDER, mdoc->parse, 2537 n->line, n->pos, "Dd after Dt"); 2538 else if (mdoc->meta.os != NULL) 2539 mandoc_msg(MANDOCERR_PROLOG_ORDER, mdoc->parse, 2540 n->line, n->pos, "Dd after Os"); 2541 2542 if (n->child == NULL || n->child->string[0] == '\0') { 2543 mdoc->meta.date = mdoc->quick ? mandoc_strdup("") : 2544 mandoc_normdate(mdoc, NULL, n->line, n->pos); 2545 return; 2546 } 2547 2548 datestr = NULL; 2549 deroff(&datestr, n); 2550 if (mdoc->quick) 2551 mdoc->meta.date = datestr; 2552 else { 2553 mdoc->meta.date = mandoc_normdate(mdoc, 2554 datestr, n->line, n->pos); 2555 free(datestr); 2556 } 2557 } 2558 2559 static void 2560 post_dt(POST_ARGS) 2561 { 2562 struct roff_node *nn, *n; 2563 const char *cp; 2564 char *p; 2565 2566 n = mdoc->last; 2567 n->flags |= NODE_NOPRT; 2568 2569 if (mdoc->flags & MDOC_PBODY) { 2570 mandoc_msg(MANDOCERR_DT_LATE, mdoc->parse, 2571 n->line, n->pos, "Dt"); 2572 return; 2573 } 2574 2575 if (mdoc->meta.title != NULL) 2576 mandoc_msg(MANDOCERR_PROLOG_REP, mdoc->parse, 2577 n->line, n->pos, "Dt"); 2578 else if (mdoc->meta.os != NULL) 2579 mandoc_msg(MANDOCERR_PROLOG_ORDER, mdoc->parse, 2580 n->line, n->pos, "Dt after Os"); 2581 2582 free(mdoc->meta.title); 2583 free(mdoc->meta.msec); 2584 free(mdoc->meta.vol); 2585 free(mdoc->meta.arch); 2586 2587 mdoc->meta.title = NULL; 2588 mdoc->meta.msec = NULL; 2589 mdoc->meta.vol = NULL; 2590 mdoc->meta.arch = NULL; 2591 2592 /* Mandatory first argument: title. */ 2593 2594 nn = n->child; 2595 if (nn == NULL || *nn->string == '\0') { 2596 mandoc_msg(MANDOCERR_DT_NOTITLE, 2597 mdoc->parse, n->line, n->pos, "Dt"); 2598 mdoc->meta.title = mandoc_strdup("UNTITLED"); 2599 } else { 2600 mdoc->meta.title = mandoc_strdup(nn->string); 2601 2602 /* Check that all characters are uppercase. */ 2603 2604 for (p = nn->string; *p != '\0'; p++) 2605 if (islower((unsigned char)*p)) { 2606 mandoc_vmsg(MANDOCERR_TITLE_CASE, 2607 mdoc->parse, nn->line, 2608 nn->pos + (p - nn->string), 2609 "Dt %s", nn->string); 2610 break; 2611 } 2612 } 2613 2614 /* Mandatory second argument: section. */ 2615 2616 if (nn != NULL) 2617 nn = nn->next; 2618 2619 if (nn == NULL) { 2620 mandoc_vmsg(MANDOCERR_MSEC_MISSING, 2621 mdoc->parse, n->line, n->pos, 2622 "Dt %s", mdoc->meta.title); 2623 mdoc->meta.vol = mandoc_strdup("LOCAL"); 2624 return; /* msec and arch remain NULL. */ 2625 } 2626 2627 mdoc->meta.msec = mandoc_strdup(nn->string); 2628 2629 /* Infer volume title from section number. */ 2630 2631 cp = mandoc_a2msec(nn->string); 2632 if (cp == NULL) { 2633 mandoc_vmsg(MANDOCERR_MSEC_BAD, mdoc->parse, 2634 nn->line, nn->pos, "Dt ... %s", nn->string); 2635 mdoc->meta.vol = mandoc_strdup(nn->string); 2636 } else 2637 mdoc->meta.vol = mandoc_strdup(cp); 2638 2639 /* Optional third argument: architecture. */ 2640 2641 if ((nn = nn->next) == NULL) 2642 return; 2643 2644 for (p = nn->string; *p != '\0'; p++) 2645 *p = tolower((unsigned char)*p); 2646 mdoc->meta.arch = mandoc_strdup(nn->string); 2647 2648 /* Ignore fourth and later arguments. */ 2649 2650 if ((nn = nn->next) != NULL) 2651 mandoc_vmsg(MANDOCERR_ARG_EXCESS, mdoc->parse, 2652 nn->line, nn->pos, "Dt ... %s", nn->string); 2653 } 2654 2655 static void 2656 post_bx(POST_ARGS) 2657 { 2658 struct roff_node *n, *nch; 2659 const char *macro; 2660 2661 post_delim_nb(mdoc); 2662 2663 n = mdoc->last; 2664 nch = n->child; 2665 2666 if (nch != NULL) { 2667 macro = !strcmp(nch->string, "Open") ? "Ox" : 2668 !strcmp(nch->string, "Net") ? "Nx" : 2669 !strcmp(nch->string, "Free") ? "Fx" : 2670 !strcmp(nch->string, "DragonFly") ? "Dx" : NULL; 2671 if (macro != NULL) 2672 mandoc_msg(MANDOCERR_BX, mdoc->parse, 2673 n->line, n->pos, macro); 2674 mdoc->last = nch; 2675 nch = nch->next; 2676 mdoc->next = ROFF_NEXT_SIBLING; 2677 roff_elem_alloc(mdoc, n->line, n->pos, MDOC_Ns); 2678 mdoc->last->flags |= NODE_NOSRC; 2679 mdoc->next = ROFF_NEXT_SIBLING; 2680 } else 2681 mdoc->next = ROFF_NEXT_CHILD; 2682 roff_word_alloc(mdoc, n->line, n->pos, "BSD"); 2683 mdoc->last->flags |= NODE_NOSRC; 2684 2685 if (nch == NULL) { 2686 mdoc->last = n; 2687 return; 2688 } 2689 2690 roff_elem_alloc(mdoc, n->line, n->pos, MDOC_Ns); 2691 mdoc->last->flags |= NODE_NOSRC; 2692 mdoc->next = ROFF_NEXT_SIBLING; 2693 roff_word_alloc(mdoc, n->line, n->pos, "-"); 2694 mdoc->last->flags |= NODE_NOSRC; 2695 roff_elem_alloc(mdoc, n->line, n->pos, MDOC_Ns); 2696 mdoc->last->flags |= NODE_NOSRC; 2697 mdoc->last = n; 2698 2699 /* 2700 * Make `Bx's second argument always start with an uppercase 2701 * letter. Groff checks if it's an "accepted" term, but we just 2702 * uppercase blindly. 2703 */ 2704 2705 *nch->string = (char)toupper((unsigned char)*nch->string); 2706 } 2707 2708 static void 2709 post_os(POST_ARGS) 2710 { 2711 #ifndef OSNAME 2712 struct utsname utsname; 2713 static char *defbuf; 2714 #endif 2715 struct roff_node *n; 2716 2717 n = mdoc->last; 2718 n->flags |= NODE_NOPRT; 2719 2720 if (mdoc->meta.os != NULL) 2721 mandoc_msg(MANDOCERR_PROLOG_REP, mdoc->parse, 2722 n->line, n->pos, "Os"); 2723 else if (mdoc->flags & MDOC_PBODY) 2724 mandoc_msg(MANDOCERR_PROLOG_LATE, mdoc->parse, 2725 n->line, n->pos, "Os"); 2726 2727 post_delim(mdoc); 2728 2729 /* 2730 * Set the operating system by way of the `Os' macro. 2731 * The order of precedence is: 2732 * 1. the argument of the `Os' macro, unless empty 2733 * 2. the -Ios=foo command line argument, if provided 2734 * 3. -DOSNAME="\"foo\"", if provided during compilation 2735 * 4. "sysname release" from uname(3) 2736 */ 2737 2738 free(mdoc->meta.os); 2739 mdoc->meta.os = NULL; 2740 deroff(&mdoc->meta.os, n); 2741 if (mdoc->meta.os) 2742 goto out; 2743 2744 if (mdoc->os_s != NULL) { 2745 mdoc->meta.os = mandoc_strdup(mdoc->os_s); 2746 goto out; 2747 } 2748 2749 #ifdef OSNAME 2750 mdoc->meta.os = mandoc_strdup(OSNAME); 2751 #else /*!OSNAME */ 2752 if (defbuf == NULL) { 2753 if (uname(&utsname) == -1) { 2754 mandoc_msg(MANDOCERR_OS_UNAME, mdoc->parse, 2755 n->line, n->pos, "Os"); 2756 defbuf = mandoc_strdup("UNKNOWN"); 2757 } else 2758 mandoc_asprintf(&defbuf, "%s %s", 2759 utsname.sysname, utsname.release); 2760 } 2761 mdoc->meta.os = mandoc_strdup(defbuf); 2762 #endif /*!OSNAME*/ 2763 2764 out: 2765 if (mdoc->meta.os_e == MANDOC_OS_OTHER) { 2766 if (strstr(mdoc->meta.os, "OpenBSD") != NULL) 2767 mdoc->meta.os_e = MANDOC_OS_OPENBSD; 2768 else if (strstr(mdoc->meta.os, "NetBSD") != NULL) 2769 mdoc->meta.os_e = MANDOC_OS_NETBSD; 2770 } 2771 2772 /* 2773 * This is the earliest point where we can check 2774 * Mdocdate conventions because we don't know 2775 * the operating system earlier. 2776 */ 2777 2778 if (n->child != NULL) 2779 mandoc_vmsg(MANDOCERR_OS_ARG, mdoc->parse, 2780 n->child->line, n->child->pos, 2781 "Os %s (%s)", n->child->string, 2782 mdoc->meta.os_e == MANDOC_OS_OPENBSD ? 2783 "OpenBSD" : "NetBSD"); 2784 2785 while (n->tok != MDOC_Dd) 2786 if ((n = n->prev) == NULL) 2787 return; 2788 if ((n = n->child) == NULL) 2789 return; 2790 if (strncmp(n->string, "$" "Mdocdate", 9)) { 2791 if (mdoc->meta.os_e == MANDOC_OS_OPENBSD) 2792 mandoc_vmsg(MANDOCERR_MDOCDATE_MISSING, 2793 mdoc->parse, n->line, n->pos, 2794 "Dd %s (OpenBSD)", n->string); 2795 } else { 2796 if (mdoc->meta.os_e == MANDOC_OS_NETBSD) 2797 mandoc_vmsg(MANDOCERR_MDOCDATE, 2798 mdoc->parse, n->line, n->pos, 2799 "Dd %s (NetBSD)", n->string); 2800 } 2801 } 2802 2803 enum roff_sec 2804 mdoc_a2sec(const char *p) 2805 { 2806 int i; 2807 2808 for (i = 0; i < (int)SEC__MAX; i++) 2809 if (secnames[i] && 0 == strcmp(p, secnames[i])) 2810 return (enum roff_sec)i; 2811 2812 return SEC_CUSTOM; 2813 } 2814 2815 static size_t 2816 macro2len(enum roff_tok macro) 2817 { 2818 2819 switch (macro) { 2820 case MDOC_Ad: 2821 return 12; 2822 case MDOC_Ao: 2823 return 12; 2824 case MDOC_An: 2825 return 12; 2826 case MDOC_Aq: 2827 return 12; 2828 case MDOC_Ar: 2829 return 12; 2830 case MDOC_Bo: 2831 return 12; 2832 case MDOC_Bq: 2833 return 12; 2834 case MDOC_Cd: 2835 return 12; 2836 case MDOC_Cm: 2837 return 10; 2838 case MDOC_Do: 2839 return 10; 2840 case MDOC_Dq: 2841 return 12; 2842 case MDOC_Dv: 2843 return 12; 2844 case MDOC_Eo: 2845 return 12; 2846 case MDOC_Em: 2847 return 10; 2848 case MDOC_Er: 2849 return 17; 2850 case MDOC_Ev: 2851 return 15; 2852 case MDOC_Fa: 2853 return 12; 2854 case MDOC_Fl: 2855 return 10; 2856 case MDOC_Fo: 2857 return 16; 2858 case MDOC_Fn: 2859 return 16; 2860 case MDOC_Ic: 2861 return 10; 2862 case MDOC_Li: 2863 return 16; 2864 case MDOC_Ms: 2865 return 6; 2866 case MDOC_Nm: 2867 return 10; 2868 case MDOC_No: 2869 return 12; 2870 case MDOC_Oo: 2871 return 10; 2872 case MDOC_Op: 2873 return 14; 2874 case MDOC_Pa: 2875 return 32; 2876 case MDOC_Pf: 2877 return 12; 2878 case MDOC_Po: 2879 return 12; 2880 case MDOC_Pq: 2881 return 12; 2882 case MDOC_Ql: 2883 return 16; 2884 case MDOC_Qo: 2885 return 12; 2886 case MDOC_So: 2887 return 12; 2888 case MDOC_Sq: 2889 return 12; 2890 case MDOC_Sy: 2891 return 6; 2892 case MDOC_Sx: 2893 return 16; 2894 case MDOC_Tn: 2895 return 10; 2896 case MDOC_Va: 2897 return 12; 2898 case MDOC_Vt: 2899 return 12; 2900 case MDOC_Xr: 2901 return 10; 2902 default: 2903 break; 2904 }; 2905 return 0; 2906 }