]>
git.cameronkatri.com Git - mandoc.git/blob - eqn.c
1 /* $Id: eqn.c,v 1.84 2020/01/08 12:16:24 schwarze Exp $ */
3 * Copyright (c) 2011, 2014 Kristaps Dzonsons <kristaps@bsd.lv>
4 * Copyright (c) 2014,2015,2017,2018,2020 Ingo Schwarze <schwarze@openbsd.org>
6 * Permission to use, copy, modify, and distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
20 #include <sys/types.h>
30 #include "mandoc_aux.h"
34 #include "libmandoc.h"
35 #include "eqn_parse.h"
37 #define EQN_NEST_MAX 128 /* maximum nesting of defines */
38 #define STRNEQ(p1, sz1, p2, sz2) \
39 ((sz1) == (sz2) && 0 == strncmp((p1), (p2), (sz1)))
94 static const char *eqn_toks
[EQN_TOK__MAX
] = {
95 "dyad", /* EQN_TOK_DYAD */
96 "vec", /* EQN_TOK_VEC */
97 "under", /* EQN_TOK_UNDER */
98 "bar", /* EQN_TOK_BAR */
99 "tilde", /* EQN_TOK_TILDE */
100 "hat", /* EQN_TOK_HAT */
101 "dot", /* EQN_TOK_DOT */
102 "dotdot", /* EQN_TOK_DOTDOT */
103 "fwd", /* EQN_TOK_FWD * */
104 "back", /* EQN_TOK_BACK */
105 "down", /* EQN_TOK_DOWN */
106 "up", /* EQN_TOK_UP */
107 "fat", /* EQN_TOK_FAT */
108 "roman", /* EQN_TOK_ROMAN */
109 "italic", /* EQN_TOK_ITALIC */
110 "bold", /* EQN_TOK_BOLD */
111 "size", /* EQN_TOK_SIZE */
112 "sub", /* EQN_TOK_SUB */
113 "sup", /* EQN_TOK_SUP */
114 "sqrt", /* EQN_TOK_SQRT */
115 "over", /* EQN_TOK_OVER */
116 "from", /* EQN_TOK_FROM */
117 "to", /* EQN_TOK_TO */
118 "{", /* EQN_TOK_BRACE_OPEN */
119 "}", /* EQN_TOK_BRACE_CLOSE */
120 "gsize", /* EQN_TOK_GSIZE */
121 "gfont", /* EQN_TOK_GFONT */
122 "mark", /* EQN_TOK_MARK */
123 "lineup", /* EQN_TOK_LINEUP */
124 "left", /* EQN_TOK_LEFT */
125 "right", /* EQN_TOK_RIGHT */
126 "pile", /* EQN_TOK_PILE */
127 "lpile", /* EQN_TOK_LPILE */
128 "rpile", /* EQN_TOK_RPILE */
129 "cpile", /* EQN_TOK_CPILE */
130 "matrix", /* EQN_TOK_MATRIX */
131 "ccol", /* EQN_TOK_CCOL */
132 "lcol", /* EQN_TOK_LCOL */
133 "rcol", /* EQN_TOK_RCOL */
134 "delim", /* EQN_TOK_DELIM */
135 "define", /* EQN_TOK_DEFINE */
136 "tdefine", /* EQN_TOK_TDEFINE */
137 "ndefine", /* EQN_TOK_NDEFINE */
138 "undef", /* EQN_TOK_UNDEF */
139 "above", /* EQN_TOK_ABOVE */
142 static const char *const eqn_func
[] = {
143 "acos", "acsc", "and", "arc", "asec", "asin", "atan",
144 "cos", "cosh", "coth", "csc", "det", "exp", "for",
145 "if", "lim", "ln", "log", "max", "min",
146 "sec", "sin", "sinh", "tan", "tanh", "Im", "Re",
218 static const struct eqnsym eqnsyms
[EQNSYM__MAX
] = {
219 { "alpha", "*a" }, /* EQNSYM_alpha */
220 { "beta", "*b" }, /* EQNSYM_beta */
221 { "chi", "*x" }, /* EQNSYM_chi */
222 { "delta", "*d" }, /* EQNSYM_delta */
223 { "epsilon", "*e" }, /* EQNSYM_epsilon */
224 { "eta", "*y" }, /* EQNSYM_eta */
225 { "gamma", "*g" }, /* EQNSYM_gamma */
226 { "iota", "*i" }, /* EQNSYM_iota */
227 { "kappa", "*k" }, /* EQNSYM_kappa */
228 { "lambda", "*l" }, /* EQNSYM_lambda */
229 { "mu", "*m" }, /* EQNSYM_mu */
230 { "nu", "*n" }, /* EQNSYM_nu */
231 { "omega", "*w" }, /* EQNSYM_omega */
232 { "omicron", "*o" }, /* EQNSYM_omicron */
233 { "phi", "*f" }, /* EQNSYM_phi */
234 { "pi", "*p" }, /* EQNSYM_pi */
235 { "psi", "*q" }, /* EQNSYM_psi */
236 { "rho", "*r" }, /* EQNSYM_rho */
237 { "sigma", "*s" }, /* EQNSYM_sigma */
238 { "tau", "*t" }, /* EQNSYM_tau */
239 { "theta", "*h" }, /* EQNSYM_theta */
240 { "upsilon", "*u" }, /* EQNSYM_upsilon */
241 { "xi", "*c" }, /* EQNSYM_xi */
242 { "zeta", "*z" }, /* EQNSYM_zeta */
243 { "DELTA", "*D" }, /* EQNSYM_DELTA */
244 { "GAMMA", "*G" }, /* EQNSYM_GAMMA */
245 { "LAMBDA", "*L" }, /* EQNSYM_LAMBDA */
246 { "OMEGA", "*W" }, /* EQNSYM_OMEGA */
247 { "PHI", "*F" }, /* EQNSYM_PHI */
248 { "PI", "*P" }, /* EQNSYM_PI */
249 { "PSI", "*Q" }, /* EQNSYM_PSI */
250 { "SIGMA", "*S" }, /* EQNSYM_SIGMA */
251 { "THETA", "*H" }, /* EQNSYM_THETA */
252 { "UPSILON", "*U" }, /* EQNSYM_UPSILON */
253 { "XI", "*C" }, /* EQNSYM_XI */
254 { "inter", "ca" }, /* EQNSYM_inter */
255 { "union", "cu" }, /* EQNSYM_union */
256 { "prod", "product" }, /* EQNSYM_prod */
257 { "int", "integral" }, /* EQNSYM_int */
258 { "sum", "sum" }, /* EQNSYM_sum */
259 { "grad", "gr" }, /* EQNSYM_grad */
260 { "del", "gr" }, /* EQNSYM_del */
261 { "times", "mu" }, /* EQNSYM_times */
262 { "cdot", "pc" }, /* EQNSYM_cdot */
263 { "nothing", "&" }, /* EQNSYM_nothing */
264 { "approx", "~~" }, /* EQNSYM_approx */
265 { "prime", "fm" }, /* EQNSYM_prime */
266 { "half", "12" }, /* EQNSYM_half */
267 { "partial", "pd" }, /* EQNSYM_partial */
268 { "inf", "if" }, /* EQNSYM_inf */
269 { ">>", ">>" }, /* EQNSYM_muchgreat */
270 { "<<", "<<" }, /* EQNSYM_muchless */
271 { "<-", "<-" }, /* EQNSYM_larrow */
272 { "->", "->" }, /* EQNSYM_rarrow */
273 { "+-", "+-" }, /* EQNSYM_pm */
274 { "!=", "!=" }, /* EQNSYM_nequal */
275 { "==", "==" }, /* EQNSYM_equiv */
276 { "<=", "<=" }, /* EQNSYM_lessequal */
277 { ">=", ">=" }, /* EQNSYM_moreequal */
278 { "-", "mi" }, /* EQNSYM_minus */
295 static struct eqn_box
*eqn_box_alloc(struct eqn_node
*, struct eqn_box
*);
296 static struct eqn_box
*eqn_box_makebinary(struct eqn_node
*,
298 static void eqn_def(struct eqn_node
*);
299 static struct eqn_def
*eqn_def_find(struct eqn_node
*);
300 static void eqn_delim(struct eqn_node
*);
301 static enum eqn_tok
eqn_next(struct eqn_node
*, enum parse_mode
);
302 static void eqn_undef(struct eqn_node
*);
310 ep
= mandoc_calloc(1, sizeof(*ep
));
311 ep
->gsize
= EQN_DEFSIZE
;
316 eqn_reset(struct eqn_node
*ep
)
319 ep
->data
= ep
->start
= ep
->end
= NULL
;
320 ep
->sz
= ep
->toksz
= 0;
324 eqn_read(struct eqn_node
*ep
, const char *p
)
328 if (ep
->data
== NULL
) {
330 ep
->data
= mandoc_strdup(p
);
332 ep
->sz
= mandoc_asprintf(&cp
, "%s %s", ep
->data
, p
);
340 * Find the key "key" of the give size within our eqn-defined values.
342 static struct eqn_def
*
343 eqn_def_find(struct eqn_node
*ep
)
347 for (i
= 0; i
< (int)ep
->defsz
; i
++)
348 if (ep
->defs
[i
].keysz
&& STRNEQ(ep
->defs
[i
].key
,
349 ep
->defs
[i
].keysz
, ep
->start
, ep
->toksz
))
356 * Parse a token from the input text. The modes are:
357 * MODE_QUOTED: Use *ep->start as the delimiter; the token ends
358 * before its next occurence. Do not interpret the token in any
359 * way and return EQN_TOK_QUOTED. All other modes behave like
360 * MODE_QUOTED when *ep->start is '"'.
361 * MODE_NOSUB: If *ep->start is a curly brace, the token ends after it;
362 * otherwise, it ends before the next whitespace or brace.
363 * Do not interpret the token and return EQN_TOK__MAX.
364 * MODE_SUB: Like MODE_NOSUB, but try to interpret the token as an
365 * alias created with define. If it is an alias, replace it with
366 * its string value and reparse.
367 * MODE_TOK: Like MODE_SUB, but also check the token against the list
368 * of tokens, and if there is a match, return that token. Otherwise,
369 * if the token matches a symbol, return EQN_TOK_SYM; if it matches
370 * a function name, EQN_TOK_FUNC, or else EQN_TOK__MAX. Except for
371 * a token match, *ep->start is set to an allocated string that the
372 * caller is expected to free.
373 * All modes skip whitespace following the end of the token.
376 eqn_next(struct eqn_node
*ep
, enum parse_mode mode
)
378 static int last_len
, lim
;
386 * Reset the recursion counter after advancing
387 * beyond the end of the previous substitution.
389 if (ep
->end
- ep
->data
>= last_len
)
393 quoted
= mode
== MODE_QUOTED
;
395 switch (*ep
->start
) {
414 ep
->end
= strchr(ep
->start
+ 1, *ep
->start
);
415 ep
->start
++; /* Skip opening quote. */
416 if (ep
->end
== NULL
) {
417 mandoc_msg(MANDOCERR_ARG_QUOTE
,
418 ep
->node
->line
, ep
->node
->pos
, NULL
);
419 ep
->end
= strchr(ep
->start
, '\0');
422 ep
->end
= ep
->start
+ 1;
423 if (*ep
->start
!= '{' && *ep
->start
!= '}')
424 ep
->end
+= strcspn(ep
->end
, " ^~\"{}\t");
426 ep
->toksz
= ep
->end
- ep
->start
;
427 if (quoted
&& *ep
->end
!= '\0')
428 ep
->end
++; /* Skip closing quote. */
429 while (*ep
->end
!= '\0' && strchr(" \t^~", *ep
->end
) != NULL
)
431 if (quoted
) /* Cannot return, may have to strndup. */
433 if (mode
== MODE_NOSUB
)
435 if ((def
= eqn_def_find(ep
)) == NULL
)
437 if (++lim
> EQN_NEST_MAX
) {
438 mandoc_msg(MANDOCERR_ROFFLOOP
,
439 ep
->node
->line
, ep
->node
->pos
, NULL
);
443 /* Replace a defined name with its string value. */
444 if ((diff
= def
->valsz
- ep
->toksz
) > 0) {
445 start
= ep
->start
- ep
->data
;
447 ep
->data
= mandoc_realloc(ep
->data
, ep
->sz
+ 1);
448 ep
->start
= ep
->data
+ start
;
451 memmove(ep
->start
+ def
->valsz
, ep
->start
+ ep
->toksz
,
452 strlen(ep
->start
+ ep
->toksz
) + 1);
453 memcpy(ep
->start
, def
->val
, def
->valsz
);
454 last_len
= ep
->start
- ep
->data
+ def
->valsz
;
456 if (mode
!= MODE_TOK
)
457 return quoted
? EQN_TOK_QUOTED
: EQN_TOK__MAX
;
459 ep
->start
= mandoc_strndup(ep
->start
, ep
->toksz
);
460 return EQN_TOK_QUOTED
;
462 for (tok
= 0; tok
< EQN_TOK__MAX
; tok
++)
463 if (STRNEQ(ep
->start
, ep
->toksz
,
464 eqn_toks
[tok
], strlen(eqn_toks
[tok
])))
467 for (i
= 0; i
< EQNSYM__MAX
; i
++) {
468 if (STRNEQ(ep
->start
, ep
->toksz
,
469 eqnsyms
[i
].str
, strlen(eqnsyms
[i
].str
))) {
470 mandoc_asprintf(&ep
->start
,
471 "\\[%s]", eqnsyms
[i
].sym
);
475 ep
->start
= mandoc_strndup(ep
->start
, ep
->toksz
);
476 for (i
= 0; i
< (int)(sizeof(eqn_func
)/sizeof(*eqn_func
)); i
++)
477 if (STRNEQ(ep
->start
, ep
->toksz
,
478 eqn_func
[i
], strlen(eqn_func
[i
])))
484 eqn_box_free(struct eqn_box
*bp
)
490 eqn_box_free(bp
->first
);
492 eqn_box_free(bp
->next
);
507 bp
= mandoc_calloc(1, sizeof(*bp
));
508 bp
->expectargs
= UINT_MAX
;
513 * Allocate a box as the last child of the parent node.
515 static struct eqn_box
*
516 eqn_box_alloc(struct eqn_node
*ep
, struct eqn_box
*parent
)
523 bp
->font
= bp
->parent
->font
;
524 bp
->size
= ep
->gsize
;
526 if (NULL
!= parent
->first
) {
527 parent
->last
->next
= bp
;
528 bp
->prev
= parent
->last
;
537 * Reparent the current last node (of the current parent) under a new
538 * EQN_SUBEXPR as the first element.
539 * Then return the new parent.
540 * The new EQN_SUBEXPR will have a two-child limit.
542 static struct eqn_box
*
543 eqn_box_makebinary(struct eqn_node
*ep
, struct eqn_box
*parent
)
545 struct eqn_box
*b
, *newb
;
547 assert(NULL
!= parent
->last
);
549 if (parent
->last
== parent
->first
)
550 parent
->first
= NULL
;
552 parent
->last
= b
->prev
;
554 newb
= eqn_box_alloc(ep
, parent
);
555 newb
->type
= EQN_SUBEXPR
;
556 newb
->expectargs
= 2;
558 newb
->first
= newb
->last
= b
;
559 newb
->first
->next
= NULL
;
565 * Parse the "delim" control statement.
568 eqn_delim(struct eqn_node
*ep
)
570 if (ep
->end
[0] == '\0' || ep
->end
[1] == '\0') {
571 mandoc_msg(MANDOCERR_REQ_EMPTY
,
572 ep
->node
->line
, ep
->node
->pos
, "delim");
573 if (ep
->end
[0] != '\0')
575 } else if (strncmp(ep
->end
, "off", 3) == 0) {
578 } else if (strncmp(ep
->end
, "on", 2) == 0) {
579 if (ep
->odelim
&& ep
->cdelim
)
583 ep
->odelim
= *ep
->end
++;
584 ep
->cdelim
= *ep
->end
++;
590 * Undefine a previously-defined string.
593 eqn_undef(struct eqn_node
*ep
)
597 if (eqn_next(ep
, MODE_NOSUB
) == EQN_TOK_EOF
) {
598 mandoc_msg(MANDOCERR_REQ_EMPTY
,
599 ep
->node
->line
, ep
->node
->pos
, "undef");
602 if ((def
= eqn_def_find(ep
)) == NULL
)
606 def
->key
= def
->val
= NULL
;
607 def
->keysz
= def
->valsz
= 0;
611 eqn_def(struct eqn_node
*ep
)
616 if (eqn_next(ep
, MODE_NOSUB
) == EQN_TOK_EOF
) {
617 mandoc_msg(MANDOCERR_REQ_EMPTY
,
618 ep
->node
->line
, ep
->node
->pos
, "define");
623 * Search for a key that already exists.
624 * Create a new key if none is found.
626 if ((def
= eqn_def_find(ep
)) == NULL
) {
627 /* Find holes in string array. */
628 for (i
= 0; i
< (int)ep
->defsz
; i
++)
629 if (0 == ep
->defs
[i
].keysz
)
632 if (i
== (int)ep
->defsz
) {
634 ep
->defs
= mandoc_reallocarray(ep
->defs
,
635 ep
->defsz
, sizeof(struct eqn_def
));
636 ep
->defs
[i
].key
= ep
->defs
[i
].val
= NULL
;
641 def
->key
= mandoc_strndup(ep
->start
, ep
->toksz
);
642 def
->keysz
= ep
->toksz
;
645 if (eqn_next(ep
, MODE_QUOTED
) == EQN_TOK_EOF
) {
646 mandoc_msg(MANDOCERR_REQ_EMPTY
,
647 ep
->node
->line
, ep
->node
->pos
, "define %s", def
->key
);
650 def
->key
= def
->val
= NULL
;
651 def
->keysz
= def
->valsz
= 0;
655 def
->val
= mandoc_strndup(ep
->start
, ep
->toksz
);
656 def
->valsz
= ep
->toksz
;
660 eqn_parse(struct eqn_node
*ep
)
662 struct eqn_box
*cur
, *nbox
, *parent
, *split
;
663 const char *cp
, *cpn
;
666 enum { CCL_LET
, CCL_DIG
, CCL_PUN
} ccl
, ccln
;
669 parent
= ep
->node
->eqn
;
670 assert(parent
!= NULL
);
674 * Do not add it to the high-level syntax tree.
677 if (ep
->data
== NULL
)
680 ep
->start
= ep
->end
= ep
->data
;
683 tok
= eqn_next(ep
, MODE_TOK
);
688 case EQN_TOK_NDEFINE
:
692 case EQN_TOK_TDEFINE
:
693 if (eqn_next(ep
, MODE_NOSUB
) == EQN_TOK_EOF
||
694 eqn_next(ep
, MODE_QUOTED
) == EQN_TOK_EOF
)
695 mandoc_msg(MANDOCERR_REQ_EMPTY
,
696 ep
->node
->line
, ep
->node
->pos
, "tdefine");
702 if (eqn_next(ep
, MODE_SUB
) == EQN_TOK_EOF
)
703 mandoc_msg(MANDOCERR_REQ_EMPTY
, ep
->node
->line
,
704 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
718 if (parent
->last
== NULL
) {
719 mandoc_msg(MANDOCERR_EQN_NOBOX
, ep
->node
->line
,
720 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
721 cur
= eqn_box_alloc(ep
, parent
);
722 cur
->type
= EQN_TEXT
;
723 cur
->text
= mandoc_strdup("");
725 parent
= eqn_box_makebinary(ep
, parent
);
726 parent
->type
= EQN_LIST
;
727 parent
->expectargs
= 1;
728 parent
->font
= EQNFONT_ROMAN
;
731 parent
->top
= mandoc_strdup("\\[ad]");
734 parent
->top
= mandoc_strdup("\\[->]");
737 parent
->top
= mandoc_strdup("\\[<>]");
740 parent
->top
= mandoc_strdup("\\[a~]");
743 parent
->bottom
= mandoc_strdup("\\[ul]");
746 parent
->top
= mandoc_strdup("\\[rn]");
749 parent
->top
= mandoc_strdup("\\[a.]");
752 parent
->top
= mandoc_strdup("\\[ha]");
757 parent
= parent
->parent
;
763 if (eqn_next(ep
, MODE_SUB
) == EQN_TOK_EOF
)
764 mandoc_msg(MANDOCERR_REQ_EMPTY
, ep
->node
->line
,
765 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
771 while (parent
->args
== parent
->expectargs
)
772 parent
= parent
->parent
;
774 * These values apply to the next word or sequence of
775 * words; thus, we mark that we'll have a child with
776 * exactly one of those.
778 parent
= eqn_box_alloc(ep
, parent
);
779 parent
->type
= EQN_LIST
;
780 parent
->expectargs
= 1;
783 parent
->font
= EQNFONT_FAT
;
786 parent
->font
= EQNFONT_ROMAN
;
789 parent
->font
= EQNFONT_ITALIC
;
792 parent
->font
= EQNFONT_BOLD
;
800 /* Accept two values: integral size and a single. */
801 if (eqn_next(ep
, MODE_SUB
) == EQN_TOK_EOF
) {
802 mandoc_msg(MANDOCERR_REQ_EMPTY
, ep
->node
->line
,
803 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
806 size
= mandoc_strntoi(ep
->start
, ep
->toksz
, 10);
808 mandoc_msg(MANDOCERR_IT_NONUM
, ep
->node
->line
,
809 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
812 if (EQN_TOK_GSIZE
== tok
) {
816 while (parent
->args
== parent
->expectargs
)
817 parent
= parent
->parent
;
818 parent
= eqn_box_alloc(ep
, parent
);
819 parent
->type
= EQN_LIST
;
820 parent
->expectargs
= 1;
828 * We have a left-right-associative expression.
829 * Repivot under a positional node, open a child scope
830 * and keep on reading.
832 if (parent
->last
== NULL
) {
833 mandoc_msg(MANDOCERR_EQN_NOBOX
, ep
->node
->line
,
834 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
835 cur
= eqn_box_alloc(ep
, parent
);
836 cur
->type
= EQN_TEXT
;
837 cur
->text
= mandoc_strdup("");
839 while (parent
->expectargs
== 1 && parent
->args
== 1)
840 parent
= parent
->parent
;
841 if (tok
== EQN_TOK_FROM
|| tok
== EQN_TOK_TO
) {
842 for (cur
= parent
; cur
!= NULL
; cur
= cur
->parent
)
843 if (cur
->pos
== EQNPOS_SUB
||
844 cur
->pos
== EQNPOS_SUP
||
845 cur
->pos
== EQNPOS_SUBSUP
||
846 cur
->pos
== EQNPOS_SQRT
||
847 cur
->pos
== EQNPOS_OVER
)
850 parent
= cur
->parent
;
852 if (tok
== EQN_TOK_SUP
&& parent
->pos
== EQNPOS_SUB
) {
853 parent
->expectargs
= 3;
854 parent
->pos
= EQNPOS_SUBSUP
;
857 if (tok
== EQN_TOK_TO
&& parent
->pos
== EQNPOS_FROM
) {
858 parent
->expectargs
= 3;
859 parent
->pos
= EQNPOS_FROMTO
;
862 parent
= eqn_box_makebinary(ep
, parent
);
865 parent
->pos
= EQNPOS_FROM
;
868 parent
->pos
= EQNPOS_TO
;
871 parent
->pos
= EQNPOS_SUP
;
874 parent
->pos
= EQNPOS_SUB
;
881 while (parent
->args
== parent
->expectargs
)
882 parent
= parent
->parent
;
884 * Accept a left-right-associative set of arguments just
885 * like sub and sup and friends but without rebalancing
888 parent
= eqn_box_alloc(ep
, parent
);
889 parent
->type
= EQN_SUBEXPR
;
890 parent
->pos
= EQNPOS_SQRT
;
891 parent
->expectargs
= 1;
895 * We have a right-left-associative fraction.
896 * Close out anything that's currently open, then
897 * rebalance and continue reading.
899 if (parent
->last
== NULL
) {
900 mandoc_msg(MANDOCERR_EQN_NOBOX
, ep
->node
->line
,
901 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
902 cur
= eqn_box_alloc(ep
, parent
);
903 cur
->type
= EQN_TEXT
;
904 cur
->text
= mandoc_strdup("");
906 while (parent
->args
== parent
->expectargs
)
907 parent
= parent
->parent
;
908 while (EQN_SUBEXPR
== parent
->type
)
909 parent
= parent
->parent
;
910 parent
= eqn_box_makebinary(ep
, parent
);
911 parent
->pos
= EQNPOS_OVER
;
914 case EQN_TOK_BRACE_CLOSE
:
916 * Close out the existing brace.
917 * FIXME: this is a shitty sentinel: we should really
918 * have a native EQN_BRACE type or whatnot.
920 for (cur
= parent
; cur
!= NULL
; cur
= cur
->parent
)
921 if (cur
->type
== EQN_LIST
&&
922 cur
->expectargs
> 1 &&
923 (tok
== EQN_TOK_BRACE_CLOSE
||
927 mandoc_msg(MANDOCERR_BLK_NOTOPEN
, ep
->node
->line
,
928 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
932 if (EQN_TOK_RIGHT
== tok
) {
933 if (eqn_next(ep
, MODE_SUB
) == EQN_TOK_EOF
) {
934 mandoc_msg(MANDOCERR_REQ_EMPTY
,
935 ep
->node
->line
, ep
->node
->pos
,
936 "%s", eqn_toks
[tok
]);
939 /* Handling depends on right/left. */
940 if (STRNEQ(ep
->start
, ep
->toksz
, "ceiling", 7))
941 parent
->right
= mandoc_strdup("\\[rc]");
942 else if (STRNEQ(ep
->start
, ep
->toksz
, "floor", 5))
943 parent
->right
= mandoc_strdup("\\[rf]");
946 mandoc_strndup(ep
->start
, ep
->toksz
);
948 parent
= parent
->parent
;
949 if (tok
== EQN_TOK_BRACE_CLOSE
&&
950 (parent
->type
== EQN_PILE
||
951 parent
->type
== EQN_MATRIX
))
952 parent
= parent
->parent
;
953 /* Close out any "singleton" lists. */
954 while (parent
->type
== EQN_LIST
&&
955 parent
->expectargs
== 1 &&
957 parent
= parent
->parent
;
959 case EQN_TOK_BRACE_OPEN
:
962 * If we already have something in the stack and we're
963 * in an expression, then rewind til we're not any more
964 * (just like with the text node).
966 while (parent
->args
== parent
->expectargs
)
967 parent
= parent
->parent
;
968 if (EQN_TOK_LEFT
== tok
&&
969 eqn_next(ep
, MODE_SUB
) == EQN_TOK_EOF
) {
970 mandoc_msg(MANDOCERR_REQ_EMPTY
, ep
->node
->line
,
971 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
974 parent
= eqn_box_alloc(ep
, parent
);
975 parent
->type
= EQN_LIST
;
976 if (EQN_TOK_LEFT
== tok
) {
977 if (STRNEQ(ep
->start
, ep
->toksz
, "ceiling", 7))
978 parent
->left
= mandoc_strdup("\\[lc]");
979 else if (STRNEQ(ep
->start
, ep
->toksz
, "floor", 5))
980 parent
->left
= mandoc_strdup("\\[lf]");
983 mandoc_strndup(ep
->start
, ep
->toksz
);
993 while (parent
->args
== parent
->expectargs
)
994 parent
= parent
->parent
;
995 parent
= eqn_box_alloc(ep
, parent
);
996 parent
->type
= EQN_PILE
;
997 parent
->expectargs
= 1;
1000 for (cur
= parent
; cur
!= NULL
; cur
= cur
->parent
)
1001 if (cur
->type
== EQN_PILE
)
1004 mandoc_msg(MANDOCERR_IT_STRAY
, ep
->node
->line
,
1005 ep
->node
->pos
, "%s", eqn_toks
[tok
]);
1008 parent
= eqn_box_alloc(ep
, cur
);
1009 parent
->type
= EQN_LIST
;
1011 case EQN_TOK_MATRIX
:
1012 while (parent
->args
== parent
->expectargs
)
1013 parent
= parent
->parent
;
1014 parent
= eqn_box_alloc(ep
, parent
);
1015 parent
->type
= EQN_MATRIX
;
1016 parent
->expectargs
= 1;
1022 case EQN_TOK_QUOTED
:
1027 * If we already have something in the stack and we're
1028 * in an expression, then rewind til we're not any more.
1030 while (parent
->args
== parent
->expectargs
)
1031 parent
= parent
->parent
;
1032 cur
= eqn_box_alloc(ep
, parent
);
1033 cur
->type
= EQN_TEXT
;
1037 cur
->font
= EQNFONT_ROMAN
;
1039 case EQN_TOK_QUOTED
:
1040 if (cur
->font
== EQNFONT_NONE
)
1041 cur
->font
= EQNFONT_ITALIC
;
1046 if (cur
->font
!= EQNFONT_NONE
|| *p
== '\0')
1052 /* Advance to next character. */
1055 ccln
= isalpha((unsigned char)*cpn
) ? CCL_LET
:
1056 isdigit((unsigned char)*cpn
) ||
1057 (*cpn
== '.' && (ccl
== CCL_DIG
||
1058 isdigit((unsigned char)cpn
[1]))) ?
1060 /* No boundary before first character. */
1063 cur
->font
= ccl
== CCL_LET
?
1064 EQNFONT_ITALIC
: EQNFONT_ROMAN
;
1066 mandoc_escape(&cpn
, NULL
, NULL
);
1067 /* No boundary after last character. */
1070 if (ccln
== ccl
&& *cp
!= ',' && *cpn
!= ',')
1072 /* Boundary found, split the text. */
1073 if (parent
->args
== parent
->expectargs
) {
1074 /* Remove the text from the tree. */
1075 if (cur
->prev
== NULL
)
1076 parent
->first
= cur
->next
;
1078 cur
->prev
->next
= NULL
;
1079 parent
->last
= cur
->prev
;
1081 /* Set up a list instead. */
1082 split
= eqn_box_alloc(ep
, parent
);
1083 split
->type
= EQN_LIST
;
1084 /* Insert the word into the list. */
1085 split
->first
= split
->last
= cur
;
1086 cur
->parent
= split
;
1090 /* Append a new text box. */
1091 nbox
= eqn_box_alloc(ep
, parent
);
1092 nbox
->type
= EQN_TEXT
;
1093 nbox
->text
= mandoc_strdup(cpn
);
1094 /* Truncate the old box. */
1095 p
= mandoc_strndup(cur
->text
,
1099 /* Setup to process the new box. */
1106 parent
= split
->parent
;
1117 eqn_free(struct eqn_node
*p
)
1124 for (i
= 0; i
< (int)p
->defsz
; i
++) {
1125 free(p
->defs
[i
].key
);
1126 free(p
->defs
[i
].val
);