950 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			950 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*	$Vendor-Id: eqn.c,v 1.38 2011/07/25 15:37:00 kristaps Exp $ */
 | 
						|
/*
 | 
						|
 * Copyright (c) 2011 Kristaps Dzonsons <kristaps@bsd.lv>
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, and distribute this software for any
 | 
						|
 * purpose with or without fee is hereby granted, provided that the above
 | 
						|
 * copyright notice and this permission notice appear in all copies.
 | 
						|
 *
 | 
						|
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 | 
						|
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 | 
						|
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 | 
						|
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | 
						|
 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 | 
						|
 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 | 
						|
 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | 
						|
 */
 | 
						|
#ifdef HAVE_CONFIG_H
 | 
						|
#include "config.h"
 | 
						|
#endif
 | 
						|
 | 
						|
#include <assert.h>
 | 
						|
#include <limits.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <time.h>
 | 
						|
 | 
						|
#include "mandoc.h"
 | 
						|
#include "libmandoc.h"
 | 
						|
#include "libroff.h"
 | 
						|
 | 
						|
#define	EQN_NEST_MAX	 128 /* maximum nesting of defines */
 | 
						|
#define	EQN_MSG(t, x)	 mandoc_msg((t), (x)->parse, (x)->eqn.ln, (x)->eqn.pos, NULL)
 | 
						|
 | 
						|
enum	eqn_rest {
 | 
						|
	EQN_DESCOPE,
 | 
						|
	EQN_ERR,
 | 
						|
	EQN_OK,
 | 
						|
	EQN_EOF
 | 
						|
};
 | 
						|
 | 
						|
enum	eqn_symt {
 | 
						|
	EQNSYM_alpha,
 | 
						|
	EQNSYM_beta,
 | 
						|
	EQNSYM_chi,
 | 
						|
	EQNSYM_delta,
 | 
						|
	EQNSYM_epsilon,
 | 
						|
	EQNSYM_eta,
 | 
						|
	EQNSYM_gamma,
 | 
						|
	EQNSYM_iota,
 | 
						|
	EQNSYM_kappa,
 | 
						|
	EQNSYM_lambda,
 | 
						|
	EQNSYM_mu,
 | 
						|
	EQNSYM_nu,
 | 
						|
	EQNSYM_omega,
 | 
						|
	EQNSYM_omicron,
 | 
						|
	EQNSYM_phi,
 | 
						|
	EQNSYM_pi,
 | 
						|
	EQNSYM_ps,
 | 
						|
	EQNSYM_rho,
 | 
						|
	EQNSYM_sigma,
 | 
						|
	EQNSYM_tau,
 | 
						|
	EQNSYM_theta,
 | 
						|
	EQNSYM_upsilon,
 | 
						|
	EQNSYM_xi,
 | 
						|
	EQNSYM_zeta,
 | 
						|
	EQNSYM_DELTA,
 | 
						|
	EQNSYM_GAMMA,
 | 
						|
	EQNSYM_LAMBDA,
 | 
						|
	EQNSYM_OMEGA,
 | 
						|
	EQNSYM_PHI,
 | 
						|
	EQNSYM_PI,
 | 
						|
	EQNSYM_PSI,
 | 
						|
	EQNSYM_SIGMA,
 | 
						|
	EQNSYM_THETA,
 | 
						|
	EQNSYM_UPSILON,
 | 
						|
	EQNSYM_XI,
 | 
						|
	EQNSYM_inter,
 | 
						|
	EQNSYM_union,
 | 
						|
	EQNSYM_prod,
 | 
						|
	EQNSYM_int,
 | 
						|
	EQNSYM_sum,
 | 
						|
	EQNSYM_grad,
 | 
						|
	EQNSYM_del,
 | 
						|
	EQNSYM_times,
 | 
						|
	EQNSYM_cdot,
 | 
						|
	EQNSYM_nothing,
 | 
						|
	EQNSYM_approx,
 | 
						|
	EQNSYM_prime,
 | 
						|
	EQNSYM_half,
 | 
						|
	EQNSYM_partial,
 | 
						|
	EQNSYM_inf,
 | 
						|
	EQNSYM_muchgreat,
 | 
						|
	EQNSYM_muchless,
 | 
						|
	EQNSYM_larrow,
 | 
						|
	EQNSYM_rarrow,
 | 
						|
	EQNSYM_pm,
 | 
						|
	EQNSYM_nequal,
 | 
						|
	EQNSYM_equiv,
 | 
						|
	EQNSYM_lessequal,
 | 
						|
	EQNSYM_moreequal,
 | 
						|
	EQNSYM__MAX
 | 
						|
};
 | 
						|
 | 
						|
enum	eqnpartt {
 | 
						|
	EQN_DEFINE = 0,
 | 
						|
	EQN_NDEFINE,
 | 
						|
	EQN_TDEFINE,
 | 
						|
	EQN_SET,
 | 
						|
	EQN_UNDEF,
 | 
						|
	EQN_GFONT,
 | 
						|
	EQN_GSIZE,
 | 
						|
	EQN_BACK,
 | 
						|
	EQN_FWD,
 | 
						|
	EQN_UP,
 | 
						|
	EQN_DOWN,
 | 
						|
	EQN__MAX
 | 
						|
};
 | 
						|
 | 
						|
struct	eqnstr {
 | 
						|
	const char	*name;
 | 
						|
	size_t		 sz;
 | 
						|
};
 | 
						|
 | 
						|
#define	STRNEQ(p1, sz1, p2, sz2) \
 | 
						|
	((sz1) == (sz2) && 0 == strncmp((p1), (p2), (sz1)))
 | 
						|
#define	EQNSTREQ(x, p, sz) \
 | 
						|
	STRNEQ((x)->name, (x)->sz, (p), (sz))
 | 
						|
 | 
						|
struct	eqnpart {
 | 
						|
	struct eqnstr	 str;
 | 
						|
	int		(*fp)(struct eqn_node *);
 | 
						|
};
 | 
						|
 | 
						|
struct	eqnsym {
 | 
						|
	struct eqnstr	 str;
 | 
						|
	const char	*sym;
 | 
						|
};
 | 
						|
 | 
						|
 | 
						|
static	enum eqn_rest	 eqn_box(struct eqn_node *, struct eqn_box *);
 | 
						|
static	struct eqn_box	*eqn_box_alloc(struct eqn_node *, 
 | 
						|
				struct eqn_box *);
 | 
						|
static	void		 eqn_box_free(struct eqn_box *);
 | 
						|
static	struct eqn_def	*eqn_def_find(struct eqn_node *, 
 | 
						|
				const char *, size_t);
 | 
						|
static	int		 eqn_do_gfont(struct eqn_node *);
 | 
						|
static	int		 eqn_do_gsize(struct eqn_node *);
 | 
						|
static	int		 eqn_do_define(struct eqn_node *);
 | 
						|
static	int		 eqn_do_ign1(struct eqn_node *);
 | 
						|
static	int		 eqn_do_ign2(struct eqn_node *);
 | 
						|
static	int		 eqn_do_tdefine(struct eqn_node *);
 | 
						|
static	int		 eqn_do_undef(struct eqn_node *);
 | 
						|
static	enum eqn_rest	 eqn_eqn(struct eqn_node *, struct eqn_box *);
 | 
						|
static	enum eqn_rest	 eqn_list(struct eqn_node *, struct eqn_box *);
 | 
						|
static	enum eqn_rest	 eqn_matrix(struct eqn_node *, struct eqn_box *);
 | 
						|
static	const char	*eqn_nexttok(struct eqn_node *, size_t *);
 | 
						|
static	const char	*eqn_nextrawtok(struct eqn_node *, size_t *);
 | 
						|
static	const char	*eqn_next(struct eqn_node *, 
 | 
						|
				char, size_t *, int);
 | 
						|
static	void		 eqn_rewind(struct eqn_node *);
 | 
						|
 | 
						|
static	const struct eqnpart eqnparts[EQN__MAX] = {
 | 
						|
	{ { "define", 6 }, eqn_do_define }, /* EQN_DEFINE */
 | 
						|
	{ { "ndefine", 7 }, eqn_do_define }, /* EQN_NDEFINE */
 | 
						|
	{ { "tdefine", 7 }, eqn_do_tdefine }, /* EQN_TDEFINE */
 | 
						|
	{ { "set", 3 }, eqn_do_ign2 }, /* EQN_SET */
 | 
						|
	{ { "undef", 5 }, eqn_do_undef }, /* EQN_UNDEF */
 | 
						|
	{ { "gfont", 5 }, eqn_do_gfont }, /* EQN_GFONT */
 | 
						|
	{ { "gsize", 5 }, eqn_do_gsize }, /* EQN_GSIZE */
 | 
						|
	{ { "back", 4 }, eqn_do_ign1 }, /* EQN_BACK */
 | 
						|
	{ { "fwd", 3 }, eqn_do_ign1 }, /* EQN_FWD */
 | 
						|
	{ { "up", 2 }, eqn_do_ign1 }, /* EQN_UP */
 | 
						|
	{ { "down", 4 }, eqn_do_ign1 }, /* EQN_DOWN */
 | 
						|
};
 | 
						|
 | 
						|
static	const struct eqnstr eqnmarks[EQNMARK__MAX] = {
 | 
						|
	{ "", 0 }, /* EQNMARK_NONE */
 | 
						|
	{ "dot", 3 }, /* EQNMARK_DOT */
 | 
						|
	{ "dotdot", 6 }, /* EQNMARK_DOTDOT */
 | 
						|
	{ "hat", 3 }, /* EQNMARK_HAT */
 | 
						|
	{ "tilde", 5 }, /* EQNMARK_TILDE */
 | 
						|
	{ "vec", 3 }, /* EQNMARK_VEC */
 | 
						|
	{ "dyad", 4 }, /* EQNMARK_DYAD */
 | 
						|
	{ "bar", 3 }, /* EQNMARK_BAR */
 | 
						|
	{ "under", 5 }, /* EQNMARK_UNDER */
 | 
						|
};
 | 
						|
 | 
						|
static	const struct eqnstr eqnfonts[EQNFONT__MAX] = {
 | 
						|
	{ "", 0 }, /* EQNFONT_NONE */
 | 
						|
	{ "roman", 5 }, /* EQNFONT_ROMAN */
 | 
						|
	{ "bold", 4 }, /* EQNFONT_BOLD */
 | 
						|
	{ "fat", 3 }, /* EQNFONT_FAT */
 | 
						|
	{ "italic", 6 }, /* EQNFONT_ITALIC */
 | 
						|
};
 | 
						|
 | 
						|
static	const struct eqnstr eqnposs[EQNPOS__MAX] = {
 | 
						|
	{ "", 0 }, /* EQNPOS_NONE */
 | 
						|
	{ "over", 4 }, /* EQNPOS_OVER */
 | 
						|
	{ "sup", 3 }, /* EQNPOS_SUP */
 | 
						|
	{ "sub", 3 }, /* EQNPOS_SUB */
 | 
						|
	{ "to", 2 }, /* EQNPOS_TO */
 | 
						|
	{ "from", 4 }, /* EQNPOS_FROM */
 | 
						|
};
 | 
						|
 | 
						|
static	const struct eqnstr eqnpiles[EQNPILE__MAX] = {
 | 
						|
	{ "", 0 }, /* EQNPILE_NONE */
 | 
						|
	{ "pile", 4 }, /* EQNPILE_PILE */
 | 
						|
	{ "cpile", 5 }, /* EQNPILE_CPILE */
 | 
						|
	{ "rpile", 5 }, /* EQNPILE_RPILE */
 | 
						|
	{ "lpile", 5 }, /* EQNPILE_LPILE */
 | 
						|
	{ "col", 3 }, /* EQNPILE_COL */
 | 
						|
	{ "ccol", 4 }, /* EQNPILE_CCOL */
 | 
						|
	{ "rcol", 4 }, /* EQNPILE_RCOL */
 | 
						|
	{ "lcol", 4 }, /* EQNPILE_LCOL */
 | 
						|
};
 | 
						|
 | 
						|
static	const struct eqnsym eqnsyms[EQNSYM__MAX] = {
 | 
						|
	{ { "alpha", 5 }, "*a" }, /* EQNSYM_alpha */
 | 
						|
	{ { "beta", 4 }, "*b" }, /* EQNSYM_beta */
 | 
						|
	{ { "chi", 3 }, "*x" }, /* EQNSYM_chi */
 | 
						|
	{ { "delta", 5 }, "*d" }, /* EQNSYM_delta */
 | 
						|
	{ { "epsilon", 7 }, "*e" }, /* EQNSYM_epsilon */
 | 
						|
	{ { "eta", 3 }, "*y" }, /* EQNSYM_eta */
 | 
						|
	{ { "gamma", 5 }, "*g" }, /* EQNSYM_gamma */
 | 
						|
	{ { "iota", 4 }, "*i" }, /* EQNSYM_iota */
 | 
						|
	{ { "kappa", 5 }, "*k" }, /* EQNSYM_kappa */
 | 
						|
	{ { "lambda", 6 }, "*l" }, /* EQNSYM_lambda */
 | 
						|
	{ { "mu", 2 }, "*m" }, /* EQNSYM_mu */
 | 
						|
	{ { "nu", 2 }, "*n" }, /* EQNSYM_nu */
 | 
						|
	{ { "omega", 5 }, "*w" }, /* EQNSYM_omega */
 | 
						|
	{ { "omicron", 7 }, "*o" }, /* EQNSYM_omicron */
 | 
						|
	{ { "phi", 3 }, "*f" }, /* EQNSYM_phi */
 | 
						|
	{ { "pi", 2 }, "*p" }, /* EQNSYM_pi */
 | 
						|
	{ { "psi", 2 }, "*q" }, /* EQNSYM_psi */
 | 
						|
	{ { "rho", 3 }, "*r" }, /* EQNSYM_rho */
 | 
						|
	{ { "sigma", 5 }, "*s" }, /* EQNSYM_sigma */
 | 
						|
	{ { "tau", 3 }, "*t" }, /* EQNSYM_tau */
 | 
						|
	{ { "theta", 5 }, "*h" }, /* EQNSYM_theta */
 | 
						|
	{ { "upsilon", 7 }, "*u" }, /* EQNSYM_upsilon */
 | 
						|
	{ { "xi", 2 }, "*c" }, /* EQNSYM_xi */
 | 
						|
	{ { "zeta", 4 }, "*z" }, /* EQNSYM_zeta */
 | 
						|
	{ { "DELTA", 5 }, "*D" }, /* EQNSYM_DELTA */
 | 
						|
	{ { "GAMMA", 5 }, "*G" }, /* EQNSYM_GAMMA */
 | 
						|
	{ { "LAMBDA", 6 }, "*L" }, /* EQNSYM_LAMBDA */
 | 
						|
	{ { "OMEGA", 5 }, "*W" }, /* EQNSYM_OMEGA */
 | 
						|
	{ { "PHI", 3 }, "*F" }, /* EQNSYM_PHI */
 | 
						|
	{ { "PI", 2 }, "*P" }, /* EQNSYM_PI */
 | 
						|
	{ { "PSI", 3 }, "*Q" }, /* EQNSYM_PSI */
 | 
						|
	{ { "SIGMA", 5 }, "*S" }, /* EQNSYM_SIGMA */
 | 
						|
	{ { "THETA", 5 }, "*H" }, /* EQNSYM_THETA */
 | 
						|
	{ { "UPSILON", 7 }, "*U" }, /* EQNSYM_UPSILON */
 | 
						|
	{ { "XI", 2 }, "*C" }, /* EQNSYM_XI */
 | 
						|
	{ { "inter", 5 }, "ca" }, /* EQNSYM_inter */
 | 
						|
	{ { "union", 5 }, "cu" }, /* EQNSYM_union */
 | 
						|
	{ { "prod", 4 }, "product" }, /* EQNSYM_prod */
 | 
						|
	{ { "int", 3 }, "integral" }, /* EQNSYM_int */
 | 
						|
	{ { "sum", 3 }, "sum" }, /* EQNSYM_sum */
 | 
						|
	{ { "grad", 4 }, "gr" }, /* EQNSYM_grad */
 | 
						|
	{ { "del", 3 }, "gr" }, /* EQNSYM_del */
 | 
						|
	{ { "times", 5 }, "mu" }, /* EQNSYM_times */
 | 
						|
	{ { "cdot", 4 }, "pc" }, /* EQNSYM_cdot */
 | 
						|
	{ { "nothing", 7 }, "&" }, /* EQNSYM_nothing */
 | 
						|
	{ { "approx", 6 }, "~~" }, /* EQNSYM_approx */
 | 
						|
	{ { "prime", 5 }, "aq" }, /* EQNSYM_prime */
 | 
						|
	{ { "half", 4 }, "12" }, /* EQNSYM_half */
 | 
						|
	{ { "partial", 7 }, "pd" }, /* EQNSYM_partial */
 | 
						|
	{ { "inf", 3 }, "if" }, /* EQNSYM_inf */
 | 
						|
	{ { ">>", 2 }, ">>" }, /* EQNSYM_muchgreat */
 | 
						|
	{ { "<<", 2 }, "<<" }, /* EQNSYM_muchless */
 | 
						|
	{ { "<-", 2 }, "<-" }, /* EQNSYM_larrow */
 | 
						|
	{ { "->", 2 }, "->" }, /* EQNSYM_rarrow */
 | 
						|
	{ { "+-", 2 }, "+-" }, /* EQNSYM_pm */
 | 
						|
	{ { "!=", 2 }, "!=" }, /* EQNSYM_nequal */
 | 
						|
	{ { "==", 2 }, "==" }, /* EQNSYM_equiv */
 | 
						|
	{ { "<=", 2 }, "<=" }, /* EQNSYM_lessequal */
 | 
						|
	{ { ">=", 2 }, ">=" }, /* EQNSYM_moreequal */
 | 
						|
};
 | 
						|
 | 
						|
/* ARGSUSED */
 | 
						|
enum rofferr
 | 
						|
eqn_read(struct eqn_node **epp, int ln, 
 | 
						|
		const char *p, int pos, int *offs)
 | 
						|
{
 | 
						|
	size_t		 sz;
 | 
						|
	struct eqn_node	*ep;
 | 
						|
	enum rofferr	 er;
 | 
						|
 | 
						|
	ep = *epp;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If we're the terminating mark, unset our equation status and
 | 
						|
	 * validate the full equation.
 | 
						|
	 */
 | 
						|
 | 
						|
	if (0 == strncmp(p, ".EN", 3)) {
 | 
						|
		er = eqn_end(epp);
 | 
						|
		p += 3;
 | 
						|
		while (' ' == *p || '\t' == *p)
 | 
						|
			p++;
 | 
						|
		if ('\0' == *p) 
 | 
						|
			return(er);
 | 
						|
		mandoc_msg(MANDOCERR_ARGSLOST, ep->parse, ln, pos, NULL);
 | 
						|
		return(er);
 | 
						|
	}
 | 
						|
 | 
						|
	/*
 | 
						|
	 * Build up the full string, replacing all newlines with regular
 | 
						|
	 * whitespace.
 | 
						|
	 */
 | 
						|
 | 
						|
	sz = strlen(p + pos) + 1;
 | 
						|
	ep->data = mandoc_realloc(ep->data, ep->sz + sz + 1);
 | 
						|
 | 
						|
	/* First invocation: nil terminate the string. */
 | 
						|
 | 
						|
	if (0 == ep->sz)
 | 
						|
		*ep->data = '\0';
 | 
						|
 | 
						|
	ep->sz += sz;
 | 
						|
	strlcat(ep->data, p + pos, ep->sz + 1);
 | 
						|
	strlcat(ep->data, " ", ep->sz + 1);
 | 
						|
	return(ROFF_IGN);
 | 
						|
}
 | 
						|
 | 
						|
struct eqn_node *
 | 
						|
eqn_alloc(const char *name, int pos, int line, struct mparse *parse)
 | 
						|
{
 | 
						|
	struct eqn_node	*p;
 | 
						|
	size_t		 sz;
 | 
						|
	const char	*end;
 | 
						|
 | 
						|
	p = mandoc_calloc(1, sizeof(struct eqn_node));
 | 
						|
 | 
						|
	if (name && '\0' != *name) {
 | 
						|
		sz = strlen(name);
 | 
						|
		assert(sz);
 | 
						|
		do {
 | 
						|
			sz--;
 | 
						|
			end = name + (int)sz;
 | 
						|
		} while (' ' == *end || '\t' == *end);
 | 
						|
		p->eqn.name = mandoc_strndup(name, sz + 1);
 | 
						|
	}
 | 
						|
 | 
						|
	p->parse = parse;
 | 
						|
	p->eqn.ln = line;
 | 
						|
	p->eqn.pos = pos;
 | 
						|
	p->gsize = EQN_DEFSIZE;
 | 
						|
 | 
						|
	return(p);
 | 
						|
}
 | 
						|
 | 
						|
enum rofferr
 | 
						|
eqn_end(struct eqn_node **epp)
 | 
						|
{
 | 
						|
	struct eqn_node	*ep;
 | 
						|
	struct eqn_box	*root;
 | 
						|
	enum eqn_rest	 c;
 | 
						|
 | 
						|
	ep = *epp;
 | 
						|
	*epp = NULL;
 | 
						|
 | 
						|
	ep->eqn.root = mandoc_calloc(1, sizeof(struct eqn_box));
 | 
						|
 | 
						|
	root = ep->eqn.root;
 | 
						|
	root->type = EQN_ROOT;
 | 
						|
 | 
						|
	if (0 == ep->sz)
 | 
						|
		return(ROFF_IGN);
 | 
						|
 | 
						|
	if (EQN_DESCOPE == (c = eqn_eqn(ep, root))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNNSCOPE, ep);
 | 
						|
		c = EQN_ERR;
 | 
						|
	}
 | 
						|
 | 
						|
	return(EQN_EOF == c ? ROFF_EQN : ROFF_IGN);
 | 
						|
}
 | 
						|
 | 
						|
static enum eqn_rest
 | 
						|
eqn_eqn(struct eqn_node *ep, struct eqn_box *last)
 | 
						|
{
 | 
						|
	struct eqn_box	*bp;
 | 
						|
	enum eqn_rest	 c;
 | 
						|
 | 
						|
	bp = eqn_box_alloc(ep, last);
 | 
						|
	bp->type = EQN_SUBEXPR;
 | 
						|
 | 
						|
	while (EQN_OK == (c = eqn_box(ep, bp)))
 | 
						|
		/* Spin! */ ;
 | 
						|
 | 
						|
	return(c);
 | 
						|
}
 | 
						|
 | 
						|
static enum eqn_rest
 | 
						|
eqn_matrix(struct eqn_node *ep, struct eqn_box *last)
 | 
						|
{
 | 
						|
	struct eqn_box	*bp;
 | 
						|
	const char	*start;
 | 
						|
	size_t		 sz;
 | 
						|
	enum eqn_rest	 c;
 | 
						|
 | 
						|
	bp = eqn_box_alloc(ep, last);
 | 
						|
	bp->type = EQN_MATRIX;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nexttok(ep, &sz))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
	if ( ! STRNEQ(start, sz, "{", 1)) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNSYNT, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
 | 
						|
	while (EQN_OK == (c = eqn_box(ep, bp)))
 | 
						|
		switch (bp->last->pile) {
 | 
						|
		case (EQNPILE_LCOL):
 | 
						|
			/* FALLTHROUGH */
 | 
						|
		case (EQNPILE_CCOL):
 | 
						|
			/* FALLTHROUGH */
 | 
						|
		case (EQNPILE_RCOL):
 | 
						|
			continue;
 | 
						|
		default:
 | 
						|
			EQN_MSG(MANDOCERR_EQNSYNT, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		};
 | 
						|
 | 
						|
	if (EQN_DESCOPE != c) {
 | 
						|
		if (EQN_EOF == c)
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
 | 
						|
	eqn_rewind(ep);
 | 
						|
	start = eqn_nexttok(ep, &sz);
 | 
						|
	assert(start);
 | 
						|
	if (STRNEQ(start, sz, "}", 1))
 | 
						|
		return(EQN_OK);
 | 
						|
 | 
						|
	EQN_MSG(MANDOCERR_EQNBADSCOPE, ep);
 | 
						|
	return(EQN_ERR);
 | 
						|
}
 | 
						|
 | 
						|
static enum eqn_rest
 | 
						|
eqn_list(struct eqn_node *ep, struct eqn_box *last)
 | 
						|
{
 | 
						|
	struct eqn_box	*bp;
 | 
						|
	const char	*start;
 | 
						|
	size_t		 sz;
 | 
						|
	enum eqn_rest	 c;
 | 
						|
 | 
						|
	bp = eqn_box_alloc(ep, last);
 | 
						|
	bp->type = EQN_LIST;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nexttok(ep, &sz))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
	if ( ! STRNEQ(start, sz, "{", 1)) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNSYNT, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
 | 
						|
	while (EQN_DESCOPE == (c = eqn_eqn(ep, bp))) {
 | 
						|
		eqn_rewind(ep);
 | 
						|
		start = eqn_nexttok(ep, &sz);
 | 
						|
		assert(start);
 | 
						|
		if ( ! STRNEQ(start, sz, "above", 5))
 | 
						|
			break;
 | 
						|
	}
 | 
						|
 | 
						|
	if (EQN_DESCOPE != c) {
 | 
						|
		if (EQN_ERR != c)
 | 
						|
			EQN_MSG(MANDOCERR_EQNSCOPE, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	}
 | 
						|
 | 
						|
	eqn_rewind(ep);
 | 
						|
	start = eqn_nexttok(ep, &sz);
 | 
						|
	assert(start);
 | 
						|
	if (STRNEQ(start, sz, "}", 1))
 | 
						|
		return(EQN_OK);
 | 
						|
 | 
						|
	EQN_MSG(MANDOCERR_EQNBADSCOPE, ep);
 | 
						|
	return(EQN_ERR);
 | 
						|
}
 | 
						|
 | 
						|
static enum eqn_rest
 | 
						|
eqn_box(struct eqn_node *ep, struct eqn_box *last)
 | 
						|
{
 | 
						|
	size_t		 sz;
 | 
						|
	const char	*start;
 | 
						|
	char		*left;
 | 
						|
	char		 sym[64];
 | 
						|
	enum eqn_rest	 c;
 | 
						|
	int		 i, size;
 | 
						|
	struct eqn_box	*bp;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nexttok(ep, &sz)))
 | 
						|
		return(EQN_EOF);
 | 
						|
 | 
						|
	if (STRNEQ(start, sz, "}", 1))
 | 
						|
		return(EQN_DESCOPE);
 | 
						|
	else if (STRNEQ(start, sz, "right", 5))
 | 
						|
		return(EQN_DESCOPE);
 | 
						|
	else if (STRNEQ(start, sz, "above", 5))
 | 
						|
		return(EQN_DESCOPE);
 | 
						|
	else if (STRNEQ(start, sz, "mark", 4))
 | 
						|
		return(EQN_OK);
 | 
						|
	else if (STRNEQ(start, sz, "lineup", 6))
 | 
						|
		return(EQN_OK);
 | 
						|
 | 
						|
	for (i = 0; i < (int)EQN__MAX; i++) {
 | 
						|
		if ( ! EQNSTREQ(&eqnparts[i].str, start, sz))
 | 
						|
			continue;
 | 
						|
		return((*eqnparts[i].fp)(ep) ? 
 | 
						|
				EQN_OK : EQN_ERR);
 | 
						|
	} 
 | 
						|
 | 
						|
	if (STRNEQ(start, sz, "{", 1)) {
 | 
						|
		if (EQN_DESCOPE != (c = eqn_eqn(ep, last))) {
 | 
						|
			if (EQN_ERR != c)
 | 
						|
				EQN_MSG(MANDOCERR_EQNSCOPE, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		eqn_rewind(ep);
 | 
						|
		start = eqn_nexttok(ep, &sz);
 | 
						|
		assert(start);
 | 
						|
		if (STRNEQ(start, sz, "}", 1))
 | 
						|
			return(EQN_OK);
 | 
						|
		EQN_MSG(MANDOCERR_EQNBADSCOPE, ep);
 | 
						|
		return(EQN_ERR);
 | 
						|
	} 
 | 
						|
 | 
						|
	for (i = 0; i < (int)EQNPILE__MAX; i++) {
 | 
						|
		if ( ! EQNSTREQ(&eqnpiles[i], start, sz))
 | 
						|
			continue;
 | 
						|
		if (EQN_OK == (c = eqn_list(ep, last)))
 | 
						|
			last->last->pile = (enum eqn_pilet)i;
 | 
						|
		return(c);
 | 
						|
	}
 | 
						|
 | 
						|
	if (STRNEQ(start, sz, "matrix", 6))
 | 
						|
		return(eqn_matrix(ep, last));
 | 
						|
 | 
						|
	if (STRNEQ(start, sz, "left", 4)) {
 | 
						|
		if (NULL == (start = eqn_nexttok(ep, &sz))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		left = mandoc_strndup(start, sz);
 | 
						|
		c = eqn_eqn(ep, last);
 | 
						|
		if (last->last)
 | 
						|
			last->last->left = left;
 | 
						|
		else
 | 
						|
			free(left);
 | 
						|
		if (EQN_DESCOPE != c)
 | 
						|
			return(c);
 | 
						|
		assert(last->last);
 | 
						|
		eqn_rewind(ep);
 | 
						|
		start = eqn_nexttok(ep, &sz);
 | 
						|
		assert(start);
 | 
						|
		if ( ! STRNEQ(start, sz, "right", 5))
 | 
						|
			return(EQN_DESCOPE);
 | 
						|
		if (NULL == (start = eqn_nexttok(ep, &sz))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		last->last->right = mandoc_strndup(start, sz);
 | 
						|
		return(EQN_OK);
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < (int)EQNPOS__MAX; i++) {
 | 
						|
		if ( ! EQNSTREQ(&eqnposs[i], start, sz))
 | 
						|
			continue;
 | 
						|
		if (NULL == last->last) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNSYNT, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		} 
 | 
						|
		last->last->pos = (enum eqn_post)i;
 | 
						|
		if (EQN_EOF == (c = eqn_box(ep, last))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		return(c);
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < (int)EQNMARK__MAX; i++) {
 | 
						|
		if ( ! EQNSTREQ(&eqnmarks[i], start, sz))
 | 
						|
			continue;
 | 
						|
		if (NULL == last->last) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNSYNT, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		} 
 | 
						|
		last->last->mark = (enum eqn_markt)i;
 | 
						|
		if (EQN_EOF == (c = eqn_box(ep, last))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		return(c);
 | 
						|
	}
 | 
						|
 | 
						|
	for (i = 0; i < (int)EQNFONT__MAX; i++) {
 | 
						|
		if ( ! EQNSTREQ(&eqnfonts[i], start, sz))
 | 
						|
			continue;
 | 
						|
		if (EQN_EOF == (c = eqn_box(ep, last))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		} else if (EQN_OK == c)
 | 
						|
			last->last->font = (enum eqn_fontt)i;
 | 
						|
		return(c);
 | 
						|
	}
 | 
						|
 | 
						|
	if (STRNEQ(start, sz, "size", 4)) {
 | 
						|
		if (NULL == (start = eqn_nexttok(ep, &sz))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		}
 | 
						|
		size = mandoc_strntoi(start, sz, 10);
 | 
						|
		if (EQN_EOF == (c = eqn_box(ep, last))) {
 | 
						|
			EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
			return(EQN_ERR);
 | 
						|
		} else if (EQN_OK != c)
 | 
						|
			return(c);
 | 
						|
		last->last->size = size;
 | 
						|
	}
 | 
						|
 | 
						|
	bp = eqn_box_alloc(ep, last);
 | 
						|
	bp->type = EQN_TEXT;
 | 
						|
	for (i = 0; i < (int)EQNSYM__MAX; i++)
 | 
						|
		if (EQNSTREQ(&eqnsyms[i].str, start, sz)) {
 | 
						|
			sym[63] = '\0';
 | 
						|
			snprintf(sym, 62, "\\[%s]", eqnsyms[i].sym);
 | 
						|
			bp->text = mandoc_strdup(sym);
 | 
						|
			return(EQN_OK);
 | 
						|
		}
 | 
						|
 | 
						|
	bp->text = mandoc_strndup(start, sz);
 | 
						|
	return(EQN_OK);
 | 
						|
}
 | 
						|
 | 
						|
void
 | 
						|
eqn_free(struct eqn_node *p)
 | 
						|
{
 | 
						|
	int		 i;
 | 
						|
 | 
						|
	eqn_box_free(p->eqn.root);
 | 
						|
 | 
						|
	for (i = 0; i < (int)p->defsz; i++) {
 | 
						|
		free(p->defs[i].key);
 | 
						|
		free(p->defs[i].val);
 | 
						|
	}
 | 
						|
 | 
						|
	free(p->eqn.name);
 | 
						|
	free(p->data);
 | 
						|
	free(p->defs);
 | 
						|
	free(p);
 | 
						|
}
 | 
						|
 | 
						|
static struct eqn_box *
 | 
						|
eqn_box_alloc(struct eqn_node *ep, struct eqn_box *parent)
 | 
						|
{
 | 
						|
	struct eqn_box	*bp;
 | 
						|
 | 
						|
	bp = mandoc_calloc(1, sizeof(struct eqn_box));
 | 
						|
	bp->parent = parent;
 | 
						|
	bp->size = ep->gsize;
 | 
						|
 | 
						|
	if (NULL == parent->first)
 | 
						|
		parent->first = bp;
 | 
						|
	else
 | 
						|
		parent->last->next = bp;
 | 
						|
 | 
						|
	parent->last = bp;
 | 
						|
	return(bp);
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eqn_box_free(struct eqn_box *bp)
 | 
						|
{
 | 
						|
 | 
						|
	if (bp->first)
 | 
						|
		eqn_box_free(bp->first);
 | 
						|
	if (bp->next)
 | 
						|
		eqn_box_free(bp->next);
 | 
						|
 | 
						|
	free(bp->text);
 | 
						|
	free(bp->left);
 | 
						|
	free(bp->right);
 | 
						|
	free(bp);
 | 
						|
}
 | 
						|
 | 
						|
static const char *
 | 
						|
eqn_nextrawtok(struct eqn_node *ep, size_t *sz)
 | 
						|
{
 | 
						|
 | 
						|
	return(eqn_next(ep, '"', sz, 0));
 | 
						|
}
 | 
						|
 | 
						|
static const char *
 | 
						|
eqn_nexttok(struct eqn_node *ep, size_t *sz)
 | 
						|
{
 | 
						|
 | 
						|
	return(eqn_next(ep, '"', sz, 1));
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
eqn_rewind(struct eqn_node *ep)
 | 
						|
{
 | 
						|
 | 
						|
	ep->cur = ep->rew;
 | 
						|
}
 | 
						|
 | 
						|
static const char *
 | 
						|
eqn_next(struct eqn_node *ep, char quote, size_t *sz, int repl)
 | 
						|
{
 | 
						|
	char		*start, *next;
 | 
						|
	int		 q, diff, lim;
 | 
						|
	size_t		 ssz, dummy;
 | 
						|
	struct eqn_def	*def;
 | 
						|
 | 
						|
	if (NULL == sz)
 | 
						|
		sz = &dummy;
 | 
						|
 | 
						|
	lim = 0;
 | 
						|
	ep->rew = ep->cur;
 | 
						|
again:
 | 
						|
	/* Prevent self-definitions. */
 | 
						|
 | 
						|
	if (lim >= EQN_NEST_MAX) {
 | 
						|
		EQN_MSG(MANDOCERR_ROFFLOOP, ep);
 | 
						|
		return(NULL);
 | 
						|
	}
 | 
						|
 | 
						|
	ep->cur = ep->rew;
 | 
						|
	start = &ep->data[(int)ep->cur];
 | 
						|
	q = 0;
 | 
						|
 | 
						|
	if ('\0' == *start)
 | 
						|
		return(NULL);
 | 
						|
 | 
						|
	if (quote == *start) {
 | 
						|
		ep->cur++;
 | 
						|
		q = 1;
 | 
						|
	}
 | 
						|
 | 
						|
	start = &ep->data[(int)ep->cur];
 | 
						|
 | 
						|
	if ( ! q) {
 | 
						|
		if ('{' == *start || '}' == *start)
 | 
						|
			ssz = 1;
 | 
						|
		else
 | 
						|
			ssz = strcspn(start + 1, " ^~\"{}\t") + 1;
 | 
						|
		next = start + (int)ssz;
 | 
						|
		if ('\0' == *next)
 | 
						|
			next = NULL;
 | 
						|
	} else
 | 
						|
		next = strchr(start, quote);
 | 
						|
 | 
						|
	if (NULL != next) {
 | 
						|
		*sz = (size_t)(next - start);
 | 
						|
		ep->cur += *sz;
 | 
						|
		if (q)
 | 
						|
			ep->cur++;
 | 
						|
		while (' ' == ep->data[(int)ep->cur] ||
 | 
						|
				'\t' == ep->data[(int)ep->cur] ||
 | 
						|
				'^' == ep->data[(int)ep->cur] ||
 | 
						|
				'~' == ep->data[(int)ep->cur])
 | 
						|
			ep->cur++;
 | 
						|
	} else {
 | 
						|
		if (q)
 | 
						|
			EQN_MSG(MANDOCERR_BADQUOTE, ep);
 | 
						|
		next = strchr(start, '\0');
 | 
						|
		*sz = (size_t)(next - start);
 | 
						|
		ep->cur += *sz;
 | 
						|
	}
 | 
						|
 | 
						|
	/* Quotes aren't expanded for values. */
 | 
						|
 | 
						|
	if (q || ! repl)
 | 
						|
		return(start);
 | 
						|
 | 
						|
	if (NULL != (def = eqn_def_find(ep, start, *sz))) {
 | 
						|
		diff = def->valsz - *sz;
 | 
						|
 | 
						|
		if (def->valsz > *sz) {
 | 
						|
			ep->sz += diff;
 | 
						|
			ep->data = mandoc_realloc(ep->data, ep->sz + 1);
 | 
						|
			ep->data[ep->sz] = '\0';
 | 
						|
			start = &ep->data[(int)ep->rew];
 | 
						|
		}
 | 
						|
 | 
						|
		diff = def->valsz - *sz;
 | 
						|
		memmove(start + *sz + diff, start + *sz, 
 | 
						|
				(strlen(start) - *sz) + 1);
 | 
						|
		memcpy(start, def->val, def->valsz);
 | 
						|
		goto again;
 | 
						|
	}
 | 
						|
 | 
						|
	return(start);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_ign1(struct eqn_node *ep)
 | 
						|
{
 | 
						|
 | 
						|
	if (NULL == eqn_nextrawtok(ep, NULL))
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
	else
 | 
						|
		return(1);
 | 
						|
 | 
						|
	return(0);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_ign2(struct eqn_node *ep)
 | 
						|
{
 | 
						|
 | 
						|
	if (NULL == eqn_nextrawtok(ep, NULL))
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
	else if (NULL == eqn_nextrawtok(ep, NULL))
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
	else
 | 
						|
		return(1);
 | 
						|
 | 
						|
	return(0);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_tdefine(struct eqn_node *ep)
 | 
						|
{
 | 
						|
 | 
						|
	if (NULL == eqn_nextrawtok(ep, NULL))
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
	else if (NULL == eqn_next(ep, ep->data[(int)ep->cur], NULL, 0))
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
	else
 | 
						|
		return(1);
 | 
						|
 | 
						|
	return(0);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_define(struct eqn_node *ep)
 | 
						|
{
 | 
						|
	const char	*start;
 | 
						|
	size_t		 sz;
 | 
						|
	struct eqn_def	*def;
 | 
						|
	int		 i;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nextrawtok(ep, &sz))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(0);
 | 
						|
	}
 | 
						|
 | 
						|
	/* 
 | 
						|
	 * Search for a key that already exists. 
 | 
						|
	 * Create a new key if none is found.
 | 
						|
	 */
 | 
						|
 | 
						|
	if (NULL == (def = eqn_def_find(ep, start, sz))) {
 | 
						|
		/* Find holes in string array. */
 | 
						|
		for (i = 0; i < (int)ep->defsz; i++)
 | 
						|
			if (0 == ep->defs[i].keysz)
 | 
						|
				break;
 | 
						|
 | 
						|
		if (i == (int)ep->defsz) {
 | 
						|
			ep->defsz++;
 | 
						|
			ep->defs = mandoc_realloc
 | 
						|
				(ep->defs, ep->defsz * 
 | 
						|
				 sizeof(struct eqn_def));
 | 
						|
			ep->defs[i].key = ep->defs[i].val = NULL;
 | 
						|
		}
 | 
						|
 | 
						|
		ep->defs[i].keysz = sz;
 | 
						|
		ep->defs[i].key = mandoc_realloc
 | 
						|
			(ep->defs[i].key, sz + 1);
 | 
						|
 | 
						|
		memcpy(ep->defs[i].key, start, sz);
 | 
						|
		ep->defs[i].key[(int)sz] = '\0';
 | 
						|
		def = &ep->defs[i];
 | 
						|
	}
 | 
						|
 | 
						|
	start = eqn_next(ep, ep->data[(int)ep->cur], &sz, 0);
 | 
						|
 | 
						|
	if (NULL == start) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(0);
 | 
						|
	}
 | 
						|
 | 
						|
	def->valsz = sz;
 | 
						|
	def->val = mandoc_realloc(def->val, sz + 1);
 | 
						|
	memcpy(def->val, start, sz);
 | 
						|
	def->val[(int)sz] = '\0';
 | 
						|
	return(1);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_gfont(struct eqn_node *ep)
 | 
						|
{
 | 
						|
 | 
						|
	if (NULL == eqn_nextrawtok(ep, NULL)) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(0);
 | 
						|
	} 
 | 
						|
	return(1);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_gsize(struct eqn_node *ep)
 | 
						|
{
 | 
						|
	const char	*start;
 | 
						|
	size_t		 sz;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nextrawtok(ep, &sz))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(0);
 | 
						|
	} 
 | 
						|
	ep->gsize = mandoc_strntoi(start, sz, 10);
 | 
						|
	return(1);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
eqn_do_undef(struct eqn_node *ep)
 | 
						|
{
 | 
						|
	const char	*start;
 | 
						|
	struct eqn_def	*def;
 | 
						|
	size_t		 sz;
 | 
						|
 | 
						|
	if (NULL == (start = eqn_nextrawtok(ep, &sz))) {
 | 
						|
		EQN_MSG(MANDOCERR_EQNEOF, ep);
 | 
						|
		return(0);
 | 
						|
	} else if (NULL != (def = eqn_def_find(ep, start, sz)))
 | 
						|
		def->keysz = 0;
 | 
						|
 | 
						|
	return(1);
 | 
						|
}
 | 
						|
 | 
						|
static struct eqn_def *
 | 
						|
eqn_def_find(struct eqn_node *ep, const char *key, size_t sz)
 | 
						|
{
 | 
						|
	int		 i;
 | 
						|
 | 
						|
	for (i = 0; i < (int)ep->defsz; i++) 
 | 
						|
		if (ep->defs[i].keysz && STRNEQ(ep->defs[i].key, 
 | 
						|
					ep->defs[i].keysz, key, sz))
 | 
						|
			return(&ep->defs[i]);
 | 
						|
 | 
						|
	return(NULL);
 | 
						|
}
 |