371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov/* $Id: mdoc_man.c,v 1.96 2016/01/08 17:48:09 schwarze Exp $ */
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov * Copyright (c) 2011-2016 Ingo Schwarze <schwarze@openbsd.org>
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Permission to use, copy, modify, and distribute this software for any
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * purpose with or without fee is hereby granted, provided that the above
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * copyright notice and this permission notice appear in all copies.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov#define DECL_ARGS const struct roff_meta *meta, struct roff_node *n
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore int (*cond)(DECL_ARGS); /* DON'T run actions */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore int (*pre)(DECL_ARGS); /* pre-node action */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore void (*post)(DECL_ARGS); /* post-node action */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore const char *prefix; /* pre-node string constant */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore const char *suffix; /* post-node string constant */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void font_push(char);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void font_pop(void);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void mid_it(void);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void print_word(const char *);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void print_line(const char *, int);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void print_block(const char *, int);
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankovstatic void print_offs(const char *, int);
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov const struct roff_node *);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic void print_count(int *);
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amorestatic const struct manact manacts[MDOC_MAX + 1] = {
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ap, NULL, NULL, NULL }, /* Ap */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_sect, post_sect, ".SH", NULL }, /* Sh */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_sect, post_sect, ".SS", NULL }, /* Ss */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_pp, NULL, NULL, NULL }, /* Pp */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_dl, post_dl, NULL, NULL }, /* D1 */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_dl, post_dl, NULL, NULL }, /* Dl */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_bd, post_bd, NULL, NULL }, /* Bd */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { cond_body, pre_bl, post_bl, NULL, NULL }, /* Bl */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_it, post_it, NULL, NULL }, /* It */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Ad */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_an, NULL, NULL, NULL }, /* An */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Ar */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sy, post_font, NULL, NULL }, /* Cd */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sy, post_font, NULL, NULL }, /* Cm */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_li, post_font, NULL, NULL }, /* Dv */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_li, post_font, NULL, NULL }, /* Er */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_li, post_font, NULL, NULL }, /* Ev */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_fa, post_fa, NULL, NULL }, /* Fa */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_fd, post_fd, NULL, NULL }, /* Fd */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_fl, post_fl, NULL, NULL }, /* Fl */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_fn, post_fn, NULL, NULL }, /* Fn */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_ft, post_font, NULL, NULL }, /* Ft */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sy, post_font, NULL, NULL }, /* Ic */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_in, post_in, NULL, NULL }, /* In */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_li, post_font, NULL, NULL }, /* Li */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_head, pre_enc, NULL, "\\- ", NULL }, /* Nd */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_nm, post_nm, NULL, NULL }, /* Nm */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "[", "]" }, /* Op */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { NULL, pre_ft, post_font, NULL, NULL }, /* Ot */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Pa */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Va */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_vt, post_vt, NULL, NULL }, /* Vt */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_xr, NULL, NULL, NULL }, /* Xr */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %A */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_percent, NULL, NULL }, /* %B */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %D */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_percent, NULL, NULL }, /* %I */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_percent, NULL, NULL }, /* %J */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %N */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %O */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %P */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %R */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre__t, post__t, NULL, NULL }, /* %T */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %V */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_aq, post_aq, NULL, NULL }, /* Ao */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_aq, post_aq, NULL, NULL }, /* Aq */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_bf, post_bf, NULL, NULL }, /* Bf */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "[", "]" }, /* Bo */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "[", "]" }, /* Bq */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "BSD/OS", NULL }, /* Bsx */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_bx, NULL, NULL, NULL }, /* Bx */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_enc, post_enc, "\\(Lq", "\\(Rq" }, /* Do */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_enc, post_enc, "\\(Lq", "\\(Rq" }, /* Dq */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Em */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_eo, post_eo, NULL, NULL }, /* Eo */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "FreeBSD", NULL }, /* Fx */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sy, post_font, NULL, NULL }, /* Ms */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_no, NULL, NULL, NULL }, /* No */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ns, NULL, NULL, NULL }, /* Ns */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "NetBSD", NULL }, /* Nx */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "OpenBSD", NULL }, /* Ox */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, NULL, post_pf, NULL, NULL }, /* Pf */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "(", ")" }, /* Po */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "(", ")" }, /* Pq */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { cond_body, pre_enc, post_enc, "\\(oq", "\\(cq" }, /* Ql */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "\"", "\"" }, /* Qo */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "\"", "\"" }, /* Qq */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { cond_body, pre_rs, NULL, NULL, NULL }, /* Rs */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { cond_body, pre_enc, post_enc, "\\(oq", "\\(cq" }, /* So */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { cond_body, pre_enc, post_enc, "\\(oq", "\\(cq" }, /* Sq */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sm, NULL, NULL, NULL }, /* Sm */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Sx */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_sy, post_font, NULL, NULL }, /* Sy */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_li, post_font, NULL, NULL }, /* Tn */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "UNIX", NULL }, /* Ux */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_fo, post_fo, NULL, NULL }, /* Fo */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "[", "]" }, /* Oo */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_bk, post_bk, NULL, NULL }, /* Bk */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "is currently in beta test.", NULL }, /* Bt */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { NULL, pre_em, post_font, NULL, NULL }, /* Fr */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "currently under development.", NULL }, /* Ud */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_lb, NULL, NULL }, /* Lb */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_pp, NULL, NULL, NULL }, /* Lp */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_lk, NULL, NULL, NULL }, /* Lk */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, pre_em, post_font, NULL, NULL }, /* Mt */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "{", "}" }, /* Brq */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { cond_body, pre_enc, post_enc, "{", "}" }, /* Bro */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %C */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov { cond_body, pre_en, post_en, NULL, NULL }, /* En */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_ux, NULL, "DragonFly", NULL }, /* Dx */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %Q */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_br, NULL, NULL, NULL }, /* br */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, pre_sp, post_sp, NULL, NULL }, /* sp */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore { NULL, NULL, post_percent, NULL, NULL }, /* %U */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore { NULL, NULL, NULL, NULL, NULL }, /* ROOT */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_spc (1 << 0) /* blank character before next word */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_spc_force (1 << 1) /* even before trailing punctuation */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_nl (1 << 2) /* break man(7) code line */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_br (1 << 3) /* break output line */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_sp (1 << 4) /* insert a blank output line */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_PP (1 << 5) /* reset indentation etc. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_Sm (1 << 6) /* horizontal spacing mode */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_Bk (1 << 7) /* word keep mode */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_Bk_susp (1 << 8) /* suspend this (after a macro) */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_An_split (1 << 9) /* author mode is "split" */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_An_nosplit (1 << 10) /* author mode is "nosplit" */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_PD (1 << 11) /* inter-paragraph spacing disabled */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore#define MMAN_nbrword (1 << 12) /* do not break the next word */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankovstatic int Bl_stack[BL_STACK_MAX]; /* offsets [chars] */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic int Bl_stack_post[BL_STACK_MAX]; /* add final .RE */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic int Bl_stack_len; /* number of nested Bl blocks */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic int TPremain; /* characters before tag is full */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amorestatic struct {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (fontqueue.head + fontqueue.size <= ++fontqueue.tail) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore fontqueue.head = mandoc_realloc(fontqueue.head,
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if ((MMAN_PP | MMAN_sp | MMAN_br | MMAN_nl) & outflags) {
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * If we need a newline, print it now and start afresh.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore outflags &= ~(MMAN_PP|MMAN_sp|MMAN_br|MMAN_nl|MMAN_spc);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * If we need a space, only print it if
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * (1) it is forced by `No' or
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * (2) what follows is not terminating punctuation or
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * (3) what follows is longer than one character.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (MMAN_spc_force & outflags || '\0' == s[0] ||
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore NULL == strchr(".,:;)]?!", s[0]) || '\0' != s[1]) {
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Reassign needing space if we're not following opening
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * punctuation.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore (('(' != s[0] && '[' != s[0]) || '\0' != s[1])))
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore outflags &= ~(MMAN_spc_force | MMAN_Bk_susp);
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore for ( ; *s; s++) {
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore switch (*s) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* FALLTHROUGH */
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore putchar((unsigned char)*s);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Convert v into a number (of characters). */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov if (NULL == v || '\0' == *v || (keywords && !strcmp(v, "left")))
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * If we are inside an enclosing list,
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * there is no easy way to add the two
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * indentations because they are provided
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * in terms of different units.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * We are inside an enclosing list.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * Add the two indentations.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * Set up the indentation for a list item; used from pre_it().
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankovprint_width(const struct mdoc_bl *bl, const struct roff_node *child)
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov /* Convert the width into a number (of characters). */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov else if (a2roffsu(bl->width, &su, SCALE_MAX) > 1) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* XXX Rough estimation, might have multiple parts. */
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov else if (child != NULL && child->type == ROFFT_TEXT)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Maybe we are inside an enclosing list? */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * Save our own indentation,
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * such that child lists can use it.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Set up the current list. */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov (void)snprintf(buf, sizeof(buf), "%dn", sz + 2);
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov (void)snprintf(buf, sizeof(buf), "%d.\\&", ++*count);
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Dump the keep buffer.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * We're guaranteed by now that this exists (is non-NULL).
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Flush stdout afterward, just in case.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore fputs(mparse_getkeep(man_mparse(man)), stdout);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore printf(".TH \"%s\" \"%s\" \"%s\" \"%s\" \"%s\"\n",
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov (mdoc->meta.msec == NULL ? "" : mdoc->meta.msec),
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov mdoc->meta.date, mdoc->meta.os, mdoc->meta.vol);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Disable hyphenation and if nroff, disable justification. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore fontqueue.head = fontqueue.tail = mandoc_malloc(8);
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov for (n = mdoc->first->child; n != NULL; n = n->next)
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Break the line if we were parsed subsequent the current node.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * This makes the page structure be more consistent.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (MMAN_spc & outflags && MDOC_LINE & n->flags)
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Make sure that we don't happen to start with a
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * control character at the start of a line.
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov if (outflags & MMAN_Sm && ! (n->flags & MDOC_DELIMC))
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov if (outflags & MMAN_Sm && ! (n->flags & MDOC_DELIMO))
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Conditionally run the pre-node action handler for a
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov cond = act->cond == NULL || (*act->cond)(meta, n);
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Conditionally run all child nodes.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Note that this iterates over children instead of using
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * recursion. This prevents unnecessary depth in the stack.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Lastly, conditionally run the post-node handler.
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov for (nch = n->child; nch != NULL; nch = nch->next) {
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov if (nch->prev != NULL || nch->next->next != NULL) {
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov print_word("on success, and\\~>0 if an error occurs.");
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Print before a section header.
95c635efb7c3b86efc493e0447eaec7aecca3f0fGarrett D'Amore * Print subsequent a section header.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (MDOC_Sh == n->tok && SEC_AUTHORS == n->sec)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore outflags &= ~(MMAN_An_split | MMAN_An_nosplit);
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore/* See mdoc_term.c, synopsis_pre() for comments. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (NULL == n->prev || ! (MDOC_SYNPRETTY & n->flags))
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (MDOC_Fn != n->tok && MDOC_Fo != n->tok) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* FALLTHROUGH */
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov print_word(n->child != NULL && n->child->next == NULL &&
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov print_word(n->child != NULL && n->child->next == NULL &&
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (0 == n->norm->Bd.comp && NULL != n->parent->prev)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Close out this display. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Maybe we are inside an enclosing list? */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * print_offs() will increase the -offset to account for
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * a possible enclosing .It, but any enclosed .It blocks
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * just nest and do not add up their indentation.
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov for (icol = 0; icol < n->norm->Bl.ncols; icol++)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Maybe we are inside an enclosing list? */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Maybe we are inside an enclosing list? */
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov else if (n->end != ENDBODY_NOT ? n->child != NULL :
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov n->parent->head->child != NULL && (n->child != NULL ||
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov (n->parent->tail != NULL && n->parent->tail->child != NULL)))
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov body = n->child != NULL || n->parent->head->child != NULL;
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov tail = n->parent->tail != NULL && n->parent->tail->child != NULL;
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov else if ( ! tail)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore while (NULL != n) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (NULL != n->next && MDOC_Fa == n->next->tok)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * This function is called after closing out an indented block.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * If we are inside an enclosing list, restore its indentation.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Nothing to do outside a list. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore if (0 == Bl_stack_len || 0 == Bl_stack[Bl_stack_len - 1])
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* The indentation has already been set up. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Restore the indentation of the enclosing list. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore /* Remeber to close out this .RS block later. */
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * Our indentation had to be restored
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * after a child display or child list.
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore * Close out that indentation block now.
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov if (n->type != ROFFT_ELEM && n->type != ROFFT_HEAD)
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore name = n->child ? n->child->string : meta->name;
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov if ( ! (n->next == NULL || n->next->flags & MDOC_LINE))
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov for (nch = n->child; nch != NULL; nch = nch->next) {
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov if (nch->prev != NULL || nch->next->next != NULL) {
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov "the value\\~0 is returned;");
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov print_word("otherwise the value\\~\\-1 is returned"
260e9a87725c090ba5835b1f9f0b62fa2f96036fYuri Pankov " and the global variable");
698f87a48e2e945bfe5493ce168e0d0ae1cedd5cGarrett D'Amore switch (n->type) {
371584c2eae4cf827fd406ba26c14f021adaaa70Yuri Pankov if (n->flags & MDOC_SYNPRETTY && n->type != ROFFT_BODY)