1 /* $Id: roff_validate.c,v 1.11 2018/12/04 02:53:51 schwarze Exp $ */
3 * Copyright (c) 2010, 2017, 2018 Ingo Schwarze <schwarze@openbsd.org>
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 #include <sys/types.h>
25 #include "libmandoc.h"
28 #define ROFF_VALID_ARGS struct roff_man *man, struct roff_node *n
30 typedef void (*roff_valid_fp
)(ROFF_VALID_ARGS
);
32 static void roff_valid_br(ROFF_VALID_ARGS
);
33 static void roff_valid_ft(ROFF_VALID_ARGS
);
34 static void roff_valid_sp(ROFF_VALID_ARGS
);
36 static const roff_valid_fp roff_valids
[ROFF_MAX
] = {
37 roff_valid_br
, /* br */
39 roff_valid_ft
, /* ft */
44 roff_valid_sp
, /* sp */
51 roff_validate(struct roff_man
*man
)
56 assert(n
->tok
< ROFF_MAX
);
57 if (roff_valids
[n
->tok
] != NULL
)
58 (*roff_valids
[n
->tok
])(man
, n
);
62 roff_valid_br(ROFF_VALID_ARGS
)
67 mandoc_vmsg(MANDOCERR_ARG_SKIP
, man
->parse
,
68 n
->line
, n
->pos
, "br %s", n
->child
->string
);
70 if ((np
= n
->prev
) == NULL
)
77 mandoc_vmsg(MANDOCERR_PAR_SKIP
, man
->parse
,
78 n
->line
, n
->pos
, "br after %s", roff_name
[np
->tok
]);
79 roff_node_delete(man
, n
);
87 roff_valid_ft(ROFF_VALID_ARGS
)
91 if (n
->child
== NULL
) {
92 man
->next
= ROFF_NEXT_CHILD
;
93 roff_word_alloc(man
, n
->line
, n
->pos
, "P");
98 cp
= n
->child
->string
;
111 if (cp
[1] == '\0' || (cp
[1] == 'I' && cp
[2] == '\0'))
115 if (cp
[1] != '\0' && cp
[2] == '\0' &&
116 strchr("BIRW", cp
[1]) != NULL
)
123 mandoc_vmsg(MANDOCERR_FT_BAD
, man
->parse
,
124 n
->line
, n
->pos
, "ft %s", cp
);
125 roff_node_delete(man
, n
);
129 roff_valid_sp(ROFF_VALID_ARGS
)
131 struct roff_node
*np
;
133 if (n
->child
!= NULL
&& n
->child
->next
!= NULL
)
134 mandoc_vmsg(MANDOCERR_ARG_EXCESS
, man
->parse
,
135 n
->child
->next
->line
, n
->child
->next
->pos
,
136 "sp ... %s", n
->child
->next
->string
);
138 if ((np
= n
->prev
) == NULL
)
143 mandoc_msg(MANDOCERR_PAR_SKIP
, man
->parse
,
144 np
->line
, np
->pos
, "br before sp");
145 roff_node_delete(man
, np
);
148 mandoc_msg(MANDOCERR_PAR_SKIP
, man
->parse
,
149 n
->line
, n
->pos
, "sp after Pp");
150 roff_node_delete(man
, n
);