lexi.c revision 1.145 1 1.145 rillig /* $NetBSD: lexi.c,v 1.145 2021/11/19 19:15:55 rillig Exp $ */
2 1.3 tls
3 1.16 kamil /*-
4 1.16 kamil * SPDX-License-Identifier: BSD-4-Clause
5 1.16 kamil *
6 1.16 kamil * Copyright (c) 1985 Sun Microsystems, Inc.
7 1.5 mrg * Copyright (c) 1980, 1993
8 1.5 mrg * The Regents of the University of California. All rights reserved.
9 1.1 cgd * All rights reserved.
10 1.1 cgd *
11 1.1 cgd * Redistribution and use in source and binary forms, with or without
12 1.1 cgd * modification, are permitted provided that the following conditions
13 1.1 cgd * are met:
14 1.1 cgd * 1. Redistributions of source code must retain the above copyright
15 1.1 cgd * notice, this list of conditions and the following disclaimer.
16 1.1 cgd * 2. Redistributions in binary form must reproduce the above copyright
17 1.1 cgd * notice, this list of conditions and the following disclaimer in the
18 1.1 cgd * documentation and/or other materials provided with the distribution.
19 1.1 cgd * 3. All advertising materials mentioning features or use of this software
20 1.1 cgd * must display the following acknowledgement:
21 1.1 cgd * This product includes software developed by the University of
22 1.1 cgd * California, Berkeley and its contributors.
23 1.1 cgd * 4. Neither the name of the University nor the names of its contributors
24 1.1 cgd * may be used to endorse or promote products derived from this software
25 1.1 cgd * without specific prior written permission.
26 1.1 cgd *
27 1.1 cgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
28 1.1 cgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
29 1.1 cgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
30 1.1 cgd * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
31 1.1 cgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
32 1.1 cgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
33 1.1 cgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
34 1.1 cgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35 1.1 cgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
36 1.1 cgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
37 1.1 cgd * SUCH DAMAGE.
38 1.1 cgd */
39 1.1 cgd
40 1.16 kamil #if 0
41 1.16 kamil static char sccsid[] = "@(#)lexi.c 8.1 (Berkeley) 6/6/93";
42 1.16 kamil #endif
43 1.16 kamil
44 1.6 lukem #include <sys/cdefs.h>
45 1.16 kamil #if defined(__NetBSD__)
46 1.145 rillig __RCSID("$NetBSD: lexi.c,v 1.145 2021/11/19 19:15:55 rillig Exp $");
47 1.16 kamil #elif defined(__FreeBSD__)
48 1.16 kamil __FBSDID("$FreeBSD: head/usr.bin/indent/lexi.c 337862 2018-08-15 18:19:45Z pstef $");
49 1.16 kamil #endif
50 1.1 cgd
51 1.142 rillig #include <assert.h>
52 1.1 cgd #include <ctype.h>
53 1.1 cgd #include <stdlib.h>
54 1.1 cgd #include <string.h>
55 1.16 kamil
56 1.16 kamil #include "indent.h"
57 1.1 cgd
58 1.127 rillig /*
59 1.127 rillig * While inside lexi_alnum, this constant just marks a type, independently of
60 1.127 rillig * the parentheses level.
61 1.127 rillig */
62 1.135 rillig #define lsym_type lsym_type_outside_parentheses
63 1.127 rillig
64 1.60 rillig /* must be sorted alphabetically, is used in binary search */
65 1.62 rillig static const struct keyword {
66 1.62 rillig const char *name;
67 1.125 rillig lexer_symbol lsym;
68 1.62 rillig } keywords[] = {
69 1.127 rillig {"_Bool", lsym_type},
70 1.127 rillig {"_Complex", lsym_type},
71 1.127 rillig {"_Imaginary", lsym_type},
72 1.127 rillig {"auto", lsym_storage_class},
73 1.127 rillig {"bool", lsym_type},
74 1.134 rillig {"break", lsym_word},
75 1.127 rillig {"case", lsym_case_label},
76 1.127 rillig {"char", lsym_type},
77 1.127 rillig {"complex", lsym_type},
78 1.127 rillig {"const", lsym_type},
79 1.134 rillig {"continue", lsym_word},
80 1.127 rillig {"default", lsym_case_label},
81 1.127 rillig {"do", lsym_do},
82 1.127 rillig {"double", lsym_type},
83 1.127 rillig {"else", lsym_else},
84 1.127 rillig {"enum", lsym_tag},
85 1.127 rillig {"extern", lsym_storage_class},
86 1.127 rillig {"float", lsym_type},
87 1.127 rillig {"for", lsym_for},
88 1.134 rillig {"goto", lsym_word},
89 1.127 rillig {"if", lsym_if},
90 1.127 rillig {"imaginary", lsym_type},
91 1.134 rillig {"inline", lsym_word},
92 1.127 rillig {"int", lsym_type},
93 1.127 rillig {"long", lsym_type},
94 1.127 rillig {"offsetof", lsym_offsetof},
95 1.127 rillig {"register", lsym_storage_class},
96 1.134 rillig {"restrict", lsym_word},
97 1.129 rillig {"return", lsym_return},
98 1.127 rillig {"short", lsym_type},
99 1.127 rillig {"signed", lsym_type},
100 1.127 rillig {"sizeof", lsym_sizeof},
101 1.127 rillig {"static", lsym_storage_class},
102 1.127 rillig {"struct", lsym_tag},
103 1.127 rillig {"switch", lsym_switch},
104 1.127 rillig {"typedef", lsym_typedef},
105 1.127 rillig {"union", lsym_tag},
106 1.127 rillig {"unsigned", lsym_type},
107 1.127 rillig {"void", lsym_type},
108 1.127 rillig {"volatile", lsym_type},
109 1.127 rillig {"while", lsym_while}
110 1.1 cgd };
111 1.1 cgd
112 1.84 rillig static struct {
113 1.64 rillig const char **items;
114 1.64 rillig unsigned int len;
115 1.64 rillig unsigned int cap;
116 1.64 rillig } typenames;
117 1.16 kamil
118 1.16 kamil /*
119 1.16 kamil * The transition table below was rewritten by hand from lx's output, given
120 1.16 kamil * the following definitions. lx is Katherine Flavel's lexer generator.
121 1.16 kamil *
122 1.16 kamil * O = /[0-7]/; D = /[0-9]/; NZ = /[1-9]/;
123 1.16 kamil * H = /[a-f0-9]/i; B = /[0-1]/; HP = /0x/i;
124 1.16 kamil * BP = /0b/i; E = /e[+\-]?/i D+; P = /p[+\-]?/i D+;
125 1.16 kamil * FS = /[fl]/i; IS = /u/i /(l|L|ll|LL)/? | /(l|L|ll|LL)/ /u/i?;
126 1.16 kamil *
127 1.16 kamil * D+ E FS? -> $float;
128 1.16 kamil * D* "." D+ E? FS? -> $float;
129 1.16 kamil * D+ "." E? FS? -> $float; HP H+ IS? -> $int;
130 1.16 kamil * HP H+ P FS? -> $float; NZ D* IS? -> $int;
131 1.16 kamil * HP H* "." H+ P FS? -> $float; "0" O* IS? -> $int;
132 1.16 kamil * HP H+ "." P FS -> $float; BP B+ IS? -> $int;
133 1.16 kamil */
134 1.71 rillig /* INDENT OFF */
135 1.82 rillig static const unsigned char lex_number_state[][26] = {
136 1.16 kamil /* examples:
137 1.16 kamil 00
138 1.16 kamil s 0xx
139 1.16 kamil t 00xaa
140 1.16 kamil a 11 101100xxa..
141 1.16 kamil r 11ee0001101lbuuxx.a.pp
142 1.16 kamil t.01.e+008bLuxll0Ll.aa.p+0
143 1.16 kamil states: ABCDEFGHIJKLMNOPQRSTUVWXYZ */
144 1.83 rillig [0] = "uuiifuufiuuiiuiiiiiuiuuuuu", /* (other) */
145 1.83 rillig [1] = "CEIDEHHHIJQ U Q VUVVZZZ", /* 0 */
146 1.83 rillig [2] = "DEIDEHHHIJQ U Q VUVVZZZ", /* 1 */
147 1.83 rillig [3] = "DEIDEHHHIJ U VUVVZZZ", /* 2 3 4 5 6 7 */
148 1.83 rillig [4] = "DEJDEHHHJJ U VUVVZZZ", /* 8 9 */
149 1.83 rillig [5] = " U VUVV ", /* A a C c D d */
150 1.83 rillig [6] = " K U VUVV ", /* B b */
151 1.83 rillig [7] = " FFF FF U VUVV ", /* E e */
152 1.83 rillig [8] = " f f U VUVV f", /* F f */
153 1.83 rillig [9] = " LLf fL PR Li L f", /* L */
154 1.83 rillig [10] = " OOf fO S P O i O f", /* l */
155 1.83 rillig [11] = " FFX ", /* P p */
156 1.83 rillig [12] = " MM M i iiM M ", /* U u */
157 1.83 rillig [13] = " N ", /* X x */
158 1.83 rillig [14] = " G Y ", /* + - */
159 1.83 rillig [15] = "B EE EE T W ", /* . */
160 1.16 kamil /* ABCDEFGHIJKLMNOPQRSTUVWXYZ */
161 1.1 cgd };
162 1.71 rillig /* INDENT ON */
163 1.1 cgd
164 1.115 rillig static const unsigned char lex_number_row[] = {
165 1.56 rillig ['0'] = 1,
166 1.56 rillig ['1'] = 2,
167 1.56 rillig ['2'] = 3, ['3'] = 3, ['4'] = 3, ['5'] = 3, ['6'] = 3, ['7'] = 3,
168 1.56 rillig ['8'] = 4, ['9'] = 4,
169 1.56 rillig ['A'] = 5, ['a'] = 5, ['C'] = 5, ['c'] = 5, ['D'] = 5, ['d'] = 5,
170 1.56 rillig ['B'] = 6, ['b'] = 6,
171 1.56 rillig ['E'] = 7, ['e'] = 7,
172 1.56 rillig ['F'] = 8, ['f'] = 8,
173 1.56 rillig ['L'] = 9,
174 1.56 rillig ['l'] = 10,
175 1.56 rillig ['P'] = 11, ['p'] = 11,
176 1.56 rillig ['U'] = 12, ['u'] = 12,
177 1.56 rillig ['X'] = 13, ['x'] = 13,
178 1.56 rillig ['+'] = 14, ['-'] = 14,
179 1.56 rillig ['.'] = 15,
180 1.56 rillig };
181 1.36 rillig
182 1.25 rillig static void
183 1.25 rillig check_size_token(size_t desired_size)
184 1.25 rillig {
185 1.58 rillig if (token.e + desired_size >= token.l)
186 1.58 rillig buf_expand(&token, desired_size);
187 1.25 rillig }
188 1.25 rillig
189 1.87 rillig static void
190 1.87 rillig token_add_char(char ch)
191 1.87 rillig {
192 1.87 rillig check_size_token(1);
193 1.87 rillig *token.e++ = ch;
194 1.87 rillig }
195 1.87 rillig
196 1.20 rillig #ifdef debug
197 1.100 rillig static const char *
198 1.100 rillig lsym_name(lexer_symbol sym)
199 1.20 rillig {
200 1.20 rillig static const char *const name[] = {
201 1.100 rillig "eof",
202 1.100 rillig "preprocessing",
203 1.100 rillig "newline",
204 1.100 rillig "form_feed",
205 1.100 rillig "comment",
206 1.100 rillig "lparen_or_lbracket",
207 1.100 rillig "rparen_or_rbracket",
208 1.100 rillig "lbrace",
209 1.100 rillig "rbrace",
210 1.100 rillig "period",
211 1.100 rillig "unary_op",
212 1.100 rillig "binary_op",
213 1.100 rillig "postfix_op",
214 1.100 rillig "question",
215 1.100 rillig "colon",
216 1.100 rillig "comma",
217 1.100 rillig "semicolon",
218 1.100 rillig "typedef",
219 1.100 rillig "storage_class",
220 1.135 rillig "type_outside_parentheses",
221 1.134 rillig "type_in_parentheses",
222 1.100 rillig "tag",
223 1.100 rillig "case_label",
224 1.100 rillig "string_prefix",
225 1.120 rillig "sizeof",
226 1.121 rillig "offsetof",
227 1.134 rillig "word",
228 1.100 rillig "funcname",
229 1.100 rillig "do",
230 1.100 rillig "else",
231 1.100 rillig "for",
232 1.100 rillig "if",
233 1.100 rillig "switch",
234 1.100 rillig "while",
235 1.129 rillig "return",
236 1.20 rillig };
237 1.20 rillig
238 1.100 rillig return name[sym];
239 1.20 rillig }
240 1.20 rillig
241 1.20 rillig static void
242 1.72 rillig debug_print_buf(const char *name, const struct buffer *buf)
243 1.20 rillig {
244 1.72 rillig if (buf->s < buf->e) {
245 1.101 rillig debug_printf("%s ", name);
246 1.101 rillig debug_vis_range("\"", buf->s, buf->e, "\"\n");
247 1.20 rillig }
248 1.20 rillig }
249 1.20 rillig
250 1.112 rillig #define debug_ps_bool(name) \
251 1.113 rillig if (ps.name != prev_ps.name) \
252 1.113 rillig debug_println("[%c] ps." #name, ps.name ? 'x' : ' ')
253 1.112 rillig #define debug_ps_int(name) \
254 1.113 rillig if (ps.name != prev_ps.name) \
255 1.113 rillig debug_println("%3d ps." #name, ps.name)
256 1.112 rillig
257 1.101 rillig static void
258 1.107 rillig debug_lexi(lexer_symbol lsym)
259 1.20 rillig {
260 1.113 rillig /*
261 1.113 rillig * Watch out for 'rolled back parser state' in the debug output; the
262 1.113 rillig * differences around these are unreliable.
263 1.113 rillig */
264 1.113 rillig static struct parser_state prev_ps;
265 1.113 rillig
266 1.104 rillig debug_println("");
267 1.134 rillig debug_printf("line %d: %s", line_no, lsym_name(lsym));
268 1.116 rillig debug_vis_range(" \"", token.s, token.e, "\"\n");
269 1.122 rillig
270 1.72 rillig debug_print_buf("label", &lab);
271 1.72 rillig debug_print_buf("code", &code);
272 1.72 rillig debug_print_buf("comment", &com);
273 1.112 rillig
274 1.122 rillig debug_println(" ps.prev_token = %s", lsym_name(ps.prev_token));
275 1.130 rillig debug_ps_bool(next_col_1);
276 1.117 rillig debug_ps_bool(curr_col_1);
277 1.112 rillig debug_ps_bool(next_unary);
278 1.131 rillig if (strcmp(ps.procname, prev_ps.procname) != 0)
279 1.145 rillig debug_println(" ps.procname = \"%s\"", ps.procname);
280 1.112 rillig debug_ps_bool(want_blank);
281 1.112 rillig debug_ps_int(paren_level);
282 1.112 rillig debug_ps_int(p_l_follow);
283 1.131 rillig if (ps.paren_level != prev_ps.paren_level) {
284 1.131 rillig debug_printf(" ps.paren_indents:");
285 1.131 rillig for (int i = 0; i < ps.paren_level; i++)
286 1.131 rillig debug_printf(" %d", ps.paren_indents[i]);
287 1.131 rillig debug_println("");
288 1.131 rillig }
289 1.112 rillig debug_ps_int(cast_mask);
290 1.112 rillig debug_ps_int(not_cast_mask);
291 1.112 rillig
292 1.112 rillig debug_ps_int(comment_delta);
293 1.112 rillig debug_ps_int(n_comment_delta);
294 1.112 rillig debug_ps_int(com_ind);
295 1.112 rillig
296 1.112 rillig debug_ps_bool(block_init);
297 1.112 rillig debug_ps_int(block_init_level);
298 1.112 rillig debug_ps_bool(init_or_struct);
299 1.112 rillig
300 1.112 rillig debug_ps_int(ind_level);
301 1.112 rillig debug_ps_int(ind_level_follow);
302 1.112 rillig
303 1.137 rillig debug_ps_int(decl_level);
304 1.112 rillig debug_ps_bool(decl_on_line);
305 1.112 rillig debug_ps_bool(in_decl);
306 1.112 rillig debug_ps_int(just_saw_decl);
307 1.112 rillig debug_ps_bool(in_parameter_declaration);
308 1.112 rillig debug_ps_bool(decl_indent_done);
309 1.112 rillig
310 1.112 rillig debug_ps_bool(in_stmt);
311 1.112 rillig debug_ps_bool(ind_stmt);
312 1.112 rillig debug_ps_bool(is_case_label);
313 1.112 rillig
314 1.112 rillig debug_ps_bool(search_stmt);
315 1.113 rillig
316 1.113 rillig prev_ps = ps;
317 1.101 rillig }
318 1.96 rillig #endif
319 1.20 rillig
320 1.104 rillig /* ARGSUSED */
321 1.101 rillig static lexer_symbol
322 1.107 rillig lexi_end(lexer_symbol lsym)
323 1.101 rillig {
324 1.101 rillig #ifdef debug
325 1.107 rillig debug_lexi(lsym);
326 1.101 rillig #endif
327 1.100 rillig return lsym;
328 1.20 rillig }
329 1.20 rillig
330 1.43 rillig static void
331 1.43 rillig lex_number(void)
332 1.43 rillig {
333 1.115 rillig for (unsigned char s = 'A'; s != 'f' && s != 'i' && s != 'u';) {
334 1.141 rillig unsigned char ch = (unsigned char)inp_peek();
335 1.94 rillig if (ch >= array_length(lex_number_row) || lex_number_row[ch] == 0)
336 1.56 rillig break;
337 1.75 rillig
338 1.115 rillig unsigned char row = lex_number_row[ch];
339 1.82 rillig if (lex_number_state[row][s - 'A'] == ' ') {
340 1.71 rillig /*-
341 1.82 rillig * lex_number_state[0][s - 'A'] now indicates the type:
342 1.74 rillig * f = floating, i = integer, u = unknown
343 1.56 rillig */
344 1.138 rillig return;
345 1.43 rillig }
346 1.75 rillig
347 1.82 rillig s = lex_number_state[row][s - 'A'];
348 1.133 rillig token_add_char(inp_next());
349 1.43 rillig }
350 1.43 rillig }
351 1.43 rillig
352 1.145 rillig static bool
353 1.145 rillig is_identifier_part(char ch)
354 1.145 rillig {
355 1.145 rillig return isalnum((unsigned char)ch) || ch == '_' || ch == '$';
356 1.145 rillig }
357 1.145 rillig
358 1.43 rillig static void
359 1.43 rillig lex_word(void)
360 1.43 rillig {
361 1.145 rillig while (is_identifier_part(inp_peek()) || inp_peek() == '\\' ) {
362 1.141 rillig if (inp_peek() == '\\') {
363 1.142 rillig if (inp_lookahead(1) == '\n') {
364 1.142 rillig inp_skip();
365 1.142 rillig inp_skip();
366 1.43 rillig } else
367 1.43 rillig break;
368 1.43 rillig }
369 1.75 rillig
370 1.133 rillig token_add_char(inp_next());
371 1.43 rillig }
372 1.43 rillig }
373 1.43 rillig
374 1.43 rillig static void
375 1.43 rillig lex_char_or_string(void)
376 1.43 rillig {
377 1.132 rillig for (char delim = token.e[-1];;) {
378 1.141 rillig if (inp_peek() == '\n') {
379 1.52 rillig diag(1, "Unterminated literal");
380 1.52 rillig return;
381 1.52 rillig }
382 1.75 rillig
383 1.133 rillig token_add_char(inp_next());
384 1.52 rillig if (token.e[-1] == delim)
385 1.52 rillig return;
386 1.75 rillig
387 1.52 rillig if (token.e[-1] == '\\') {
388 1.141 rillig if (inp_peek() == '\n')
389 1.52 rillig ++line_no;
390 1.133 rillig token_add_char(inp_next());
391 1.52 rillig }
392 1.52 rillig }
393 1.43 rillig }
394 1.43 rillig
395 1.84 rillig /* Guess whether the current token is a declared type. */
396 1.57 rillig static bool
397 1.107 rillig probably_typename(void)
398 1.57 rillig {
399 1.107 rillig if (ps.block_init || ps.in_stmt)
400 1.70 rillig return false;
401 1.142 rillig if (inp_peek() == '*' && inp_lookahead(1) != '=')
402 1.70 rillig goto maybe;
403 1.145 rillig /* XXX: is_identifier_start */
404 1.141 rillig if (isalpha((unsigned char)inp_peek()))
405 1.70 rillig goto maybe;
406 1.70 rillig return false;
407 1.70 rillig maybe:
408 1.110 rillig return ps.prev_token == lsym_semicolon ||
409 1.110 rillig ps.prev_token == lsym_lbrace ||
410 1.110 rillig ps.prev_token == lsym_rbrace;
411 1.57 rillig }
412 1.57 rillig
413 1.84 rillig static int
414 1.84 rillig bsearch_typenames(const char *key)
415 1.84 rillig {
416 1.84 rillig const char **arr = typenames.items;
417 1.84 rillig int lo = 0;
418 1.84 rillig int hi = (int)typenames.len - 1;
419 1.84 rillig
420 1.84 rillig while (lo <= hi) {
421 1.84 rillig int mid = (int)((unsigned)(lo + hi) >> 1);
422 1.84 rillig int cmp = strcmp(arr[mid], key);
423 1.84 rillig if (cmp < 0)
424 1.84 rillig lo = mid + 1;
425 1.84 rillig else if (cmp > 0)
426 1.84 rillig hi = mid - 1;
427 1.84 rillig else
428 1.84 rillig return mid;
429 1.84 rillig }
430 1.84 rillig return -(lo + 1);
431 1.84 rillig }
432 1.84 rillig
433 1.63 rillig static bool
434 1.63 rillig is_typename(void)
435 1.63 rillig {
436 1.84 rillig if (opt.auto_typedefs &&
437 1.84 rillig token.e - token.s >= 2 && memcmp(token.e - 2, "_t", 2) == 0)
438 1.84 rillig return true;
439 1.63 rillig
440 1.84 rillig return bsearch_typenames(token.s) >= 0;
441 1.63 rillig }
442 1.63 rillig
443 1.115 rillig static int
444 1.115 rillig cmp_keyword_by_name(const void *key, const void *elem)
445 1.115 rillig {
446 1.115 rillig return strcmp(key, ((const struct keyword *)elem)->name);
447 1.115 rillig }
448 1.115 rillig
449 1.138 rillig /* Read an alphanumeric token into 'token', or return lsym_eof. */
450 1.100 rillig static lexer_symbol
451 1.107 rillig lexi_alnum(void)
452 1.1 cgd {
453 1.141 rillig if (isdigit((unsigned char)inp_peek()) ||
454 1.142 rillig (inp_peek() == '.' && isdigit((unsigned char)inp_lookahead(1)))) {
455 1.89 rillig lex_number();
456 1.145 rillig } else if (is_identifier_part(inp_peek())) {
457 1.89 rillig lex_word();
458 1.102 rillig } else
459 1.102 rillig return lsym_eof; /* just as a placeholder */
460 1.102 rillig
461 1.89 rillig *token.e = '\0';
462 1.16 kamil
463 1.89 rillig if (token.s[0] == 'L' && token.s[1] == '\0' &&
464 1.141 rillig (inp_peek() == '"' || inp_peek() == '\''))
465 1.100 rillig return lsym_string_prefix;
466 1.16 kamil
467 1.133 rillig while (ch_isblank(inp_peek()))
468 1.133 rillig inp_skip();
469 1.89 rillig
470 1.110 rillig if (ps.prev_token == lsym_tag && ps.p_l_follow == 0) {
471 1.107 rillig ps.next_unary = true;
472 1.135 rillig return lsym_type_outside_parentheses;
473 1.16 kamil }
474 1.6 lukem
475 1.89 rillig /* Operator after identifier is binary unless last token was 'struct'. */
476 1.110 rillig ps.next_unary = ps.prev_token == lsym_tag;
477 1.16 kamil
478 1.89 rillig const struct keyword *kw = bsearch(token.s, keywords,
479 1.94 rillig array_length(keywords), sizeof(keywords[0]), cmp_keyword_by_name);
480 1.134 rillig bool is_type = false;
481 1.89 rillig if (kw == NULL) {
482 1.89 rillig if (is_typename()) {
483 1.134 rillig is_type = true;
484 1.107 rillig ps.next_unary = true;
485 1.89 rillig goto found_typename;
486 1.16 kamil }
487 1.89 rillig
488 1.89 rillig } else { /* we have a keyword */
489 1.134 rillig is_type = kw->lsym == lsym_type;
490 1.107 rillig ps.next_unary = true;
491 1.127 rillig if (kw->lsym != lsym_tag && kw->lsym != lsym_type)
492 1.125 rillig return kw->lsym;
493 1.118 rillig
494 1.118 rillig found_typename:
495 1.118 rillig if (ps.p_l_follow > 0) {
496 1.118 rillig /* inside parentheses: cast, param list, offsetof or sizeof */
497 1.118 rillig ps.cast_mask |= (1 << ps.p_l_follow) & ~ps.not_cast_mask;
498 1.118 rillig }
499 1.118 rillig if (ps.prev_token != lsym_period && ps.prev_token != lsym_unary_op) {
500 1.126 rillig if (kw != NULL && kw->lsym == lsym_tag)
501 1.100 rillig return lsym_tag;
502 1.118 rillig if (ps.p_l_follow == 0)
503 1.135 rillig return lsym_type_outside_parentheses;
504 1.90 rillig }
505 1.90 rillig }
506 1.89 rillig
507 1.141 rillig if (inp_peek() == '(' && ps.tos <= 1 && ps.ind_level == 0 &&
508 1.107 rillig !ps.in_parameter_declaration && !ps.block_init) {
509 1.89 rillig
510 1.143 rillig for (const char *p = inp_p(), *e = inp_line_end(); p < e;)
511 1.89 rillig if (*p++ == ')' && (*p == ';' || *p == ','))
512 1.118 rillig goto no_function_definition;
513 1.89 rillig
514 1.107 rillig strncpy(ps.procname, token.s, sizeof ps.procname - 1);
515 1.107 rillig if (ps.in_decl)
516 1.107 rillig ps.in_parameter_declaration = true;
517 1.100 rillig return lsym_funcname;
518 1.118 rillig no_function_definition:;
519 1.89 rillig
520 1.136 rillig } else if (ps.p_l_follow == 0 && probably_typename()) {
521 1.107 rillig ps.next_unary = true;
522 1.135 rillig return lsym_type_outside_parentheses;
523 1.89 rillig }
524 1.89 rillig
525 1.134 rillig return is_type ? lsym_type_in_parentheses : lsym_word;
526 1.89 rillig }
527 1.75 rillig
528 1.89 rillig /* Reads the next token, placing it in the global variable "token". */
529 1.100 rillig lexer_symbol
530 1.106 rillig lexi(void)
531 1.89 rillig {
532 1.90 rillig token.e = token.s;
533 1.130 rillig ps.curr_col_1 = ps.next_col_1;
534 1.130 rillig ps.next_col_1 = false;
535 1.75 rillig
536 1.141 rillig while (ch_isblank(inp_peek())) {
537 1.117 rillig ps.curr_col_1 = false;
538 1.133 rillig inp_skip();
539 1.89 rillig }
540 1.75 rillig
541 1.107 rillig lexer_symbol alnum_lsym = lexi_alnum();
542 1.100 rillig if (alnum_lsym != lsym_eof)
543 1.107 rillig return lexi_end(alnum_lsym);
544 1.16 kamil
545 1.16 kamil /* Scan a non-alphanumeric token */
546 1.16 kamil
547 1.90 rillig check_size_token(3); /* for things like "<<=" */
548 1.133 rillig *token.e++ = inp_next();
549 1.50 rillig *token.e = '\0';
550 1.16 kamil
551 1.100 rillig lexer_symbol lsym;
552 1.89 rillig bool unary_delim = false; /* whether the current token forces a
553 1.89 rillig * following operator to be unary */
554 1.89 rillig
555 1.132 rillig switch (token.e[-1]) {
556 1.16 kamil case '\n':
557 1.107 rillig unary_delim = ps.next_unary;
558 1.130 rillig ps.next_col_1 = true;
559 1.47 rillig /* if data has been exhausted, the newline is a dummy. */
560 1.100 rillig lsym = had_eof ? lsym_eof : lsym_newline;
561 1.16 kamil break;
562 1.16 kamil
563 1.43 rillig case '\'':
564 1.43 rillig case '"':
565 1.44 rillig lex_char_or_string();
566 1.134 rillig lsym = lsym_word;
567 1.16 kamil break;
568 1.6 lukem
569 1.40 rillig case '(':
570 1.40 rillig case '[':
571 1.16 kamil unary_delim = true;
572 1.100 rillig lsym = lsym_lparen_or_lbracket;
573 1.16 kamil break;
574 1.16 kamil
575 1.40 rillig case ')':
576 1.40 rillig case ']':
577 1.100 rillig lsym = lsym_rparen_or_rbracket;
578 1.16 kamil break;
579 1.16 kamil
580 1.16 kamil case '#':
581 1.107 rillig unary_delim = ps.next_unary;
582 1.100 rillig lsym = lsym_preprocessing;
583 1.16 kamil break;
584 1.16 kamil
585 1.16 kamil case '?':
586 1.16 kamil unary_delim = true;
587 1.100 rillig lsym = lsym_question;
588 1.16 kamil break;
589 1.16 kamil
590 1.40 rillig case ':':
591 1.100 rillig lsym = lsym_colon;
592 1.16 kamil unary_delim = true;
593 1.16 kamil break;
594 1.16 kamil
595 1.40 rillig case ';':
596 1.16 kamil unary_delim = true;
597 1.100 rillig lsym = lsym_semicolon;
598 1.16 kamil break;
599 1.16 kamil
600 1.40 rillig case '{':
601 1.16 kamil unary_delim = true;
602 1.100 rillig lsym = lsym_lbrace;
603 1.16 kamil break;
604 1.16 kamil
605 1.40 rillig case '}':
606 1.16 kamil unary_delim = true;
607 1.100 rillig lsym = lsym_rbrace;
608 1.16 kamil break;
609 1.16 kamil
610 1.69 rillig case '\f':
611 1.107 rillig unary_delim = ps.next_unary;
612 1.130 rillig ps.next_col_1 = true;
613 1.100 rillig lsym = lsym_form_feed;
614 1.16 kamil break;
615 1.16 kamil
616 1.40 rillig case ',':
617 1.16 kamil unary_delim = true;
618 1.100 rillig lsym = lsym_comma;
619 1.16 kamil break;
620 1.16 kamil
621 1.16 kamil case '.':
622 1.16 kamil unary_delim = false;
623 1.100 rillig lsym = lsym_period;
624 1.16 kamil break;
625 1.1 cgd
626 1.16 kamil case '-':
627 1.90 rillig case '+':
628 1.107 rillig lsym = ps.next_unary ? lsym_unary_op : lsym_binary_op;
629 1.16 kamil unary_delim = true;
630 1.16 kamil
631 1.141 rillig if (inp_peek() == token.e[-1]) { /* ++, -- */
632 1.141 rillig *token.e++ = inp_next();
633 1.134 rillig if (ps.prev_token == lsym_word ||
634 1.110 rillig ps.prev_token == lsym_rparen_or_rbracket) {
635 1.107 rillig lsym = ps.next_unary ? lsym_unary_op : lsym_postfix_op;
636 1.1 cgd unary_delim = false;
637 1.16 kamil }
638 1.75 rillig
639 1.141 rillig } else if (inp_peek() == '=') { /* += */
640 1.141 rillig *token.e++ = inp_next();
641 1.75 rillig
642 1.141 rillig } else if (inp_peek() == '>') { /* -> */
643 1.141 rillig *token.e++ = inp_next();
644 1.16 kamil unary_delim = false;
645 1.100 rillig lsym = lsym_unary_op;
646 1.107 rillig ps.want_blank = false;
647 1.16 kamil }
648 1.90 rillig break;
649 1.16 kamil
650 1.16 kamil case '=':
651 1.107 rillig if (ps.init_or_struct)
652 1.107 rillig ps.block_init = true;
653 1.141 rillig if (inp_peek() == '=') { /* == */
654 1.141 rillig *token.e++ = inp_next();
655 1.67 rillig *token.e = '\0';
656 1.16 kamil }
657 1.100 rillig lsym = lsym_binary_op;
658 1.16 kamil unary_delim = true;
659 1.16 kamil break;
660 1.16 kamil
661 1.16 kamil case '>':
662 1.16 kamil case '<':
663 1.16 kamil case '!': /* ops like <, <<, <=, !=, etc */
664 1.141 rillig if (inp_peek() == '>' || inp_peek() == '<' || inp_peek() == '=')
665 1.141 rillig *token.e++ = inp_next();
666 1.141 rillig if (inp_peek() == '=')
667 1.133 rillig *token.e++ = inp_next();
668 1.107 rillig lsym = ps.next_unary ? lsym_unary_op : lsym_binary_op;
669 1.16 kamil unary_delim = true;
670 1.16 kamil break;
671 1.16 kamil
672 1.16 kamil case '*':
673 1.16 kamil unary_delim = true;
674 1.107 rillig if (!ps.next_unary) {
675 1.141 rillig if (inp_peek() == '=')
676 1.141 rillig *token.e++ = inp_next();
677 1.100 rillig lsym = lsym_binary_op;
678 1.16 kamil break;
679 1.16 kamil }
680 1.75 rillig
681 1.141 rillig while (inp_peek() == '*' || isspace((unsigned char)inp_peek())) {
682 1.141 rillig if (inp_peek() == '*')
683 1.87 rillig token_add_char('*');
684 1.133 rillig inp_skip();
685 1.16 kamil }
686 1.75 rillig
687 1.16 kamil if (ps.in_decl) {
688 1.144 rillig const char *tp = inp_p(), *e = inp_line_end();
689 1.6 lukem
690 1.145 rillig /* XXX: is_identifier_start */
691 1.144 rillig while (tp < e && (isalpha((unsigned char)*tp) ||
692 1.144 rillig isspace((unsigned char)*tp)))
693 1.144 rillig tp++;
694 1.144 rillig
695 1.144 rillig if (tp < e && *tp == '(')
696 1.144 rillig ps.procname[0] = ' '; /* XXX: why not '\0'? */
697 1.16 kamil }
698 1.75 rillig
699 1.100 rillig lsym = lsym_unary_op;
700 1.16 kamil break;
701 1.1 cgd
702 1.16 kamil default:
703 1.141 rillig if (token.e[-1] == '/' && (inp_peek() == '*' || inp_peek() == '/')) {
704 1.133 rillig *token.e++ = inp_next();
705 1.100 rillig lsym = lsym_comment;
706 1.107 rillig unary_delim = ps.next_unary;
707 1.16 kamil break;
708 1.1 cgd }
709 1.75 rillig
710 1.132 rillig /* handle '||', '&&', etc., and also things as in 'int *****i' */
711 1.141 rillig while (token.e[-1] == inp_peek() || inp_peek() == '=')
712 1.133 rillig token_add_char(inp_next());
713 1.75 rillig
714 1.107 rillig lsym = ps.next_unary ? lsym_unary_op : lsym_binary_op;
715 1.16 kamil unary_delim = true;
716 1.47 rillig }
717 1.16 kamil
718 1.107 rillig ps.next_unary = unary_delim;
719 1.75 rillig
720 1.25 rillig check_size_token(1);
721 1.50 rillig *token.e = '\0';
722 1.75 rillig
723 1.107 rillig return lexi_end(lsym);
724 1.1 cgd }
725 1.16 kamil
726 1.6 lukem void
727 1.128 rillig register_typename(const char *name)
728 1.1 cgd {
729 1.64 rillig if (typenames.len >= typenames.cap) {
730 1.64 rillig typenames.cap = 16 + 2 * typenames.cap;
731 1.64 rillig typenames.items = xrealloc(typenames.items,
732 1.64 rillig sizeof(typenames.items[0]) * typenames.cap);
733 1.64 rillig }
734 1.16 kamil
735 1.84 rillig int pos = bsearch_typenames(name);
736 1.64 rillig if (pos >= 0)
737 1.64 rillig return; /* already in the list */
738 1.75 rillig
739 1.64 rillig pos = -(pos + 1);
740 1.64 rillig memmove(typenames.items + pos + 1, typenames.items + pos,
741 1.73 rillig sizeof(typenames.items[0]) * (typenames.len++ - (unsigned)pos));
742 1.64 rillig typenames.items[pos] = xstrdup(name);
743 1.1 cgd }
744