2 * expand.c - macro expansion functions for cawf(1)
6 * Copyright (c) 1991 Purdue University Research Foundation,
7 * West Lafayette, Indiana 47907. All rights reserved.
9 * Written by Victor A. Abell <abe@mace.cc.purdue.edu>, Purdue
10 * University Computing Center. Not derived from licensed software;
11 * derived from awf(1) by Henry Spencer of the University of Toronto.
13 * Permission is granted to anyone to use this software for any
14 * purpose on any computer system, and to alter it and redistribute
15 * it freely, subject to the following restrictions:
17 * 1. The author is not responsible for any consequences of use of
18 * this software, even if they arise from flaws in it.
20 * 2. The origin of this software must not be misrepresented, either
21 * by explicit claim or by omission. Credits must appear in the
24 * 3. Altered versions must be plainly marked as such, and must not
25 * be misrepresented as being the original software. Credits must
26 * appear in the documentation.
28 * 4. This notice may not be removed or altered.
34 * Expand(line) - expand macro or if/ie/el line
37 void Expand(unsigned char *line
) {
39 unsigned char buf
[2*MAXLINE
]; /* line buffer */
40 unsigned char cmd
[4]; /* nroff command */
41 int cmdl
; /* command length */
42 int cmdx
; /* cmd index in Macrotab[] */
43 int cond
= 0; /* conditional statuses */
44 int i
, j
; /* temporary indexes */
45 int iflen
; /* if statement length */
46 int invert
; /* inversion status */
47 unsigned char *lp
; /* line pointer */
48 int mx
= -1; /* Macrotab[] index */
49 int n1
, n2
; /* temporary numbers */
50 int nargs
= 0; /* number of arguments */
51 int nleft
= 0; /* number of macro lines left */
52 char op
; /* comparison operator */
53 int prevcond
; /* previous condition (for else's) */
54 int ptr
= -1; /* Macrotxt[] index */
55 int quote
; /* quoted string status */
56 unsigned char *s1
, *s2
; /* temporary string pointers */
59 (void) sprintf((char *)buf
, ".^= %d %s", NR
, (char *)Inname
);
62 for (lp
= line
; *lp
; ) {
63 invert
= regexec(Pat
[1].pat
, lp
);
66 if (regexec(Pat
[0].pat
, lp
) == 0) {
68 * Not conditional: - ! "^[.'](i[ef]|el)"
74 else if (regexec(Pat
[2].pat
, lp
)) {
76 * Argument count comparison: -
77 * "^[.']i[ef] !?\\n\(\.\$(>|>=|=|<|<=)[0-9] "
79 iflen
= strlen(".if \\n(.$=n ") + invert
;
82 if (*s1
== '=' && (op
== '>' || op
== '<')) {
84 op
= (op
== '>') ? 'G' : 'L';
86 n1
= (int)(*s1
- '0');
89 if ((nargs
- 1) == n1
)
101 if ((nargs
- 1) >= n1
)
105 if ((nargs
- 1) <= n1
)
110 else if (regexec(Pat
[3].pat
, lp
)) {
112 * Argument string comparison: - "^[.']i[ef] !?'\\\$[0-9]'[^']*' "
114 iflen
= strlen(".if '\\$n'") + invert
;
115 n1
= (int)(*(lp
+ iflen
- 2) - '0');
116 if (n1
>= 0 && n1
< nargs
)
119 s1
= (unsigned char *)"";
120 if ((s2
= (unsigned char *)strchr((char *)lp
123 n2
= s2
- lp
- iflen
;
124 if (strncmp((char *)s1
, (char *)lp
+ iflen
, n2
)
131 else if (regexec(Pat
[4].pat
, lp
)) {
133 * Nroff or troff: - "^[.']i[ef] !?[nt] "
135 iflen
= strlen(".if n ") + invert
;
136 if (*(lp
+ iflen
- 2) == 'n')
140 else if ((*lp
== '.' || *lp
== '\'')
141 && strncmp((char *)lp
+1, "el ", 3) == 0) {
143 * Else clause: - "^[.']el "
151 * Unknown conditional:
155 (void) sprintf((char *)buf
,
156 ".tm unknown .if/.ie form: %s", (char *)lp
);
160 * Handle conditional. If case is true, locate predicate.
161 * If predicate is an .i[ef], process it.
165 if (cond
&& iflen
> 0) {
167 if (regexec(Pat
[15].pat
, lp
))
171 * Do argument substitution, as necessary.
173 if (cond
&& regexec(Pat
[5].pat
, lp
)) { /* "\$[0-9]" ??? */
175 if ((n1
= Pat
[5].pat
->startp
[0] - lp
) > 0) {
176 (void) strncpy((char *)s1
, (char *)lp
,
181 lp
= Pat
[5].pat
->endp
[0];
182 n1
= (int)(*(lp
-1) - '0');
183 if (n1
>= 0 && n1
< nargs
) {
184 (void) strcpy((char *)s1
,
186 s1
+= strlen((char *)Args
[n1
]);
190 if (regexec(Pat
[5].pat
, lp
) == 0) {
191 (void) strcpy((char *)s1
, (char *)lp
);
198 * Check for nroff command.
202 if (cond
&& (*lp
== '.' || *lp
== '\'')) {
203 if ((*cmd
= *(lp
+1)) != '\0') {
205 if ((*(cmd
+1) = *(lp
+2)) == ' ')
214 i
= i
; /* do nothing if condition is false */
215 else if (cmdl
== 0 || ((cmdx
= Findmacro(cmd
, 0)) < 0))
217 else if (Sp
>= MAXSP
) {
218 (void) sprintf((char *)buf
, " macro nesting > %d",
220 Error(WARN
, LINE
, (char *)buf
, NULL
);
229 Nleftstack
[Sp
] = nleft
;
232 Condstack
[Sp
] = cond
;
233 for (i
= 10*Sp
, j
= 0; j
< 10; i
++, j
++) {
234 Argstack
[i
] = Args
[j
];
238 * Start new stack entry.
241 ptr
= Macrotab
[mx
].bx
;
243 nleft
= Macrotab
[mx
].ct
;
244 Args
[0] = Newstr(cmd
);
248 for (s1
= lp
+ cmdl
+ 1, nargs
= 1; nargs
< 10;) {
249 while (*s1
&& (*s1
== ' ' || *s1
== '\t'))
259 if (!quote
&& (*s1
== ' ' || *s1
== '\t')) {
263 if ((*s2
= *s1
) == '\0')
266 if (quote
&& *s2
== '"') {
273 Args
[nargs
++] = Newstr(buf
);
275 for (i
= nargs
; i
< 10; i
++) {
280 * Unstack completed macros.
282 while (nleft
<= 0 && Sp
>= 0) {
283 nleft
= Nleftstack
[Sp
];
286 cond
= Condstack
[Sp
];
287 for (i
= 10*Sp
, j
= 0, nargs
= -1; j
< 10; i
++, j
++) {
289 if ((Args
[j
] = Argstack
[i
]) != NULL
)
299 lp
= Macrotxt
[ptr
++];
302 lp
= (unsigned char *)"";
304 (void) sprintf((char *)buf
, ".^# %d %s", NR
, (char *)Inname
);