Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * parser.c
4 : * Main entry point/driver for PostgreSQL grammar
5 : *
6 : * Note that the grammar is not allowed to perform any table access
7 : * (since we need to be able to do basic parsing even while inside an
8 : * aborted transaction). Therefore, the data structures returned by
9 : * the grammar are "raw" parsetrees that still need to be analyzed by
10 : * analyze.c and related files.
11 : *
12 : *
13 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
14 : * Portions Copyright (c) 1994, Regents of the University of California
15 : *
16 : * IDENTIFICATION
17 : * src/backend/parser/parser.c
18 : *
19 : *-------------------------------------------------------------------------
20 : */
21 :
22 : #include "postgres.h"
23 :
24 : #include "gramparse.h"
25 : #include "mb/pg_wchar.h"
26 : #include "parser/parser.h"
27 : #include "parser/scansup.h"
28 :
29 : static bool check_uescapechar(unsigned char escape);
30 : static char *str_udeescape(const char *str, char escape,
31 : int position, core_yyscan_t yyscanner);
32 :
33 :
34 : /*
35 : * raw_parser
36 : * Given a query in string form, do lexical and grammatical analysis.
37 : *
38 : * Returns a list of raw (un-analyzed) parse trees. The contents of the
39 : * list have the form required by the specified RawParseMode.
40 : */
41 : List *
42 759914 : raw_parser(const char *str, RawParseMode mode)
43 : {
44 : core_yyscan_t yyscanner;
45 : base_yy_extra_type yyextra;
46 : int yyresult;
47 :
48 : /* initialize the flex scanner */
49 759914 : yyscanner = scanner_init(str, &yyextra.core_yy_extra,
50 : &ScanKeywords, ScanKeywordTokens);
51 :
52 : /* base_yylex() only needs us to initialize the lookahead token, if any */
53 759914 : if (mode == RAW_PARSE_DEFAULT)
54 711054 : yyextra.have_lookahead = false;
55 : else
56 : {
57 : /* this array is indexed by RawParseMode enum */
58 : static const int mode_token[] = {
59 : [RAW_PARSE_DEFAULT] = 0,
60 : [RAW_PARSE_TYPE_NAME] = MODE_TYPE_NAME,
61 : [RAW_PARSE_PLPGSQL_EXPR] = MODE_PLPGSQL_EXPR,
62 : [RAW_PARSE_PLPGSQL_ASSIGN1] = MODE_PLPGSQL_ASSIGN1,
63 : [RAW_PARSE_PLPGSQL_ASSIGN2] = MODE_PLPGSQL_ASSIGN2,
64 : [RAW_PARSE_PLPGSQL_ASSIGN3] = MODE_PLPGSQL_ASSIGN3,
65 : };
66 :
67 48860 : yyextra.have_lookahead = true;
68 48860 : yyextra.lookahead_token = mode_token[mode];
69 48860 : yyextra.lookahead_yylloc = 0;
70 48860 : yyextra.lookahead_end = NULL;
71 : }
72 :
73 : /* initialize the bison parser */
74 759914 : parser_init(&yyextra);
75 :
76 : /* Parse! */
77 759914 : yyresult = base_yyparse(yyscanner);
78 :
79 : /* Clean up (release memory) */
80 758708 : scanner_finish(yyscanner);
81 :
82 758708 : if (yyresult) /* error */
83 0 : return NIL;
84 :
85 758708 : return yyextra.parsetree;
86 : }
87 :
88 :
89 : /*
90 : * Intermediate filter between parser and core lexer (core_yylex in scan.l).
91 : *
92 : * This filter is needed because in some cases the standard SQL grammar
93 : * requires more than one token lookahead. We reduce these cases to one-token
94 : * lookahead by replacing tokens here, in order to keep the grammar LALR(1).
95 : *
96 : * Using a filter is simpler than trying to recognize multiword tokens
97 : * directly in scan.l, because we'd have to allow for comments between the
98 : * words. Furthermore it's not clear how to do that without re-introducing
99 : * scanner backtrack, which would cost more performance than this filter
100 : * layer does.
101 : *
102 : * We also use this filter to convert UIDENT and USCONST sequences into
103 : * plain IDENT and SCONST tokens. While that could be handled by additional
104 : * productions in the main grammar, it's more efficient to do it like this.
105 : *
106 : * The filter also provides a convenient place to translate between
107 : * the core_YYSTYPE and YYSTYPE representations (which are really the
108 : * same thing anyway, but notationally they're different).
109 : */
110 : int
111 18217512 : base_yylex(YYSTYPE *lvalp, YYLTYPE *llocp, core_yyscan_t yyscanner)
112 : {
113 18217512 : base_yy_extra_type *yyextra = pg_yyget_extra(yyscanner);
114 : int cur_token;
115 : int next_token;
116 : int cur_token_length;
117 : YYLTYPE cur_yylloc;
118 :
119 : /* Get next token --- we might already have it */
120 18217512 : if (yyextra->have_lookahead)
121 : {
122 116158 : cur_token = yyextra->lookahead_token;
123 116158 : lvalp->core_yystype = yyextra->lookahead_yylval;
124 116158 : *llocp = yyextra->lookahead_yylloc;
125 116158 : if (yyextra->lookahead_end)
126 67298 : *(yyextra->lookahead_end) = yyextra->lookahead_hold_char;
127 116158 : yyextra->have_lookahead = false;
128 : }
129 : else
130 18101354 : cur_token = core_yylex(&(lvalp->core_yystype), llocp, yyscanner);
131 :
132 : /*
133 : * If this token isn't one that requires lookahead, just return it. If it
134 : * does, determine the token length. (We could get that via strlen(), but
135 : * since we have such a small set of possibilities, hardwiring seems
136 : * feasible and more efficient --- at least for the fixed-length cases.)
137 : */
138 18217266 : switch (cur_token)
139 : {
140 3202 : case FORMAT:
141 3202 : cur_token_length = 6;
142 3202 : break;
143 40924 : case NOT:
144 40924 : cur_token_length = 3;
145 40924 : break;
146 2238 : case NULLS_P:
147 2238 : cur_token_length = 5;
148 2238 : break;
149 19092 : case WITH:
150 19092 : cur_token_length = 4;
151 19092 : break;
152 500 : case UIDENT:
153 : case USCONST:
154 500 : cur_token_length = strlen(yyextra->core_yy_extra.scanbuf + *llocp);
155 500 : break;
156 1464 : case WITHOUT:
157 1464 : cur_token_length = 7;
158 1464 : break;
159 18149846 : default:
160 18149846 : return cur_token;
161 : }
162 :
163 : /*
164 : * Identify end+1 of current token. core_yylex() has temporarily stored a
165 : * '\0' here, and will undo that when we call it again. We need to redo
166 : * it to fully revert the lookahead call for error reporting purposes.
167 : */
168 67420 : yyextra->lookahead_end = yyextra->core_yy_extra.scanbuf +
169 67420 : *llocp + cur_token_length;
170 : Assert(*(yyextra->lookahead_end) == '\0');
171 :
172 : /*
173 : * Save and restore *llocp around the call. It might look like we could
174 : * avoid this by just passing &lookahead_yylloc to core_yylex(), but that
175 : * does not work because flex actually holds onto the last-passed pointer
176 : * internally, and will use that for error reporting. We need any error
177 : * reports to point to the current token, not the next one.
178 : */
179 67420 : cur_yylloc = *llocp;
180 :
181 : /* Get next token, saving outputs into lookahead variables */
182 67420 : next_token = core_yylex(&(yyextra->lookahead_yylval), llocp, yyscanner);
183 67420 : yyextra->lookahead_token = next_token;
184 67420 : yyextra->lookahead_yylloc = *llocp;
185 :
186 67420 : *llocp = cur_yylloc;
187 :
188 : /* Now revert the un-truncation of the current token */
189 67420 : yyextra->lookahead_hold_char = *(yyextra->lookahead_end);
190 67420 : *(yyextra->lookahead_end) = '\0';
191 :
192 67420 : yyextra->have_lookahead = true;
193 :
194 : /* Replace cur_token if needed, based on lookahead */
195 67420 : switch (cur_token)
196 : {
197 3202 : case FORMAT:
198 : /* Replace FORMAT by FORMAT_LA if it's followed by JSON */
199 : switch (next_token)
200 : {
201 660 : case JSON:
202 660 : cur_token = FORMAT_LA;
203 660 : break;
204 : }
205 3202 : break;
206 :
207 40924 : case NOT:
208 : /* Replace NOT by NOT_LA if it's followed by BETWEEN, IN, etc */
209 : switch (next_token)
210 : {
211 5332 : case BETWEEN:
212 : case IN_P:
213 : case LIKE:
214 : case ILIKE:
215 : case SIMILAR:
216 5332 : cur_token = NOT_LA;
217 5332 : break;
218 : }
219 40924 : break;
220 :
221 2238 : case NULLS_P:
222 : /* Replace NULLS_P by NULLS_LA if it's followed by FIRST or LAST */
223 : switch (next_token)
224 : {
225 2044 : case FIRST_P:
226 : case LAST_P:
227 2044 : cur_token = NULLS_LA;
228 2044 : break;
229 : }
230 2238 : break;
231 :
232 19092 : case WITH:
233 : /* Replace WITH by WITH_LA if it's followed by TIME or ORDINALITY */
234 : switch (next_token)
235 : {
236 2600 : case TIME:
237 : case ORDINALITY:
238 2600 : cur_token = WITH_LA;
239 2600 : break;
240 : }
241 19092 : break;
242 :
243 1464 : case WITHOUT:
244 : /* Replace WITHOUT by WITHOUT_LA if it's followed by TIME */
245 : switch (next_token)
246 : {
247 592 : case TIME:
248 592 : cur_token = WITHOUT_LA;
249 592 : break;
250 : }
251 1464 : break;
252 :
253 500 : case UIDENT:
254 : case USCONST:
255 : /* Look ahead for UESCAPE */
256 500 : if (next_token == UESCAPE)
257 : {
258 : /* Yup, so get third token, which had better be SCONST */
259 : const char *escstr;
260 :
261 : /* Again save and restore *llocp */
262 46 : cur_yylloc = *llocp;
263 :
264 : /* Un-truncate current token so errors point to third token */
265 46 : *(yyextra->lookahead_end) = yyextra->lookahead_hold_char;
266 :
267 : /* Get third token */
268 46 : next_token = core_yylex(&(yyextra->lookahead_yylval),
269 : llocp, yyscanner);
270 :
271 : /* If we throw error here, it will point to third token */
272 46 : if (next_token != SCONST)
273 6 : scanner_yyerror("UESCAPE must be followed by a simple string literal",
274 : yyscanner);
275 :
276 40 : escstr = yyextra->lookahead_yylval.str;
277 40 : if (strlen(escstr) != 1 || !check_uescapechar(escstr[0]))
278 6 : scanner_yyerror("invalid Unicode escape character",
279 : yyscanner);
280 :
281 : /* Now restore *llocp; errors will point to first token */
282 34 : *llocp = cur_yylloc;
283 :
284 : /* Apply Unicode conversion */
285 34 : lvalp->core_yystype.str =
286 34 : str_udeescape(lvalp->core_yystype.str,
287 34 : escstr[0],
288 : *llocp,
289 : yyscanner);
290 :
291 : /*
292 : * We don't need to revert the un-truncation of UESCAPE. What
293 : * we do want to do is clear have_lookahead, thereby consuming
294 : * all three tokens.
295 : */
296 34 : yyextra->have_lookahead = false;
297 : }
298 : else
299 : {
300 : /* No UESCAPE, so convert using default escape character */
301 406 : lvalp->core_yystype.str =
302 454 : str_udeescape(lvalp->core_yystype.str,
303 : '\\',
304 : *llocp,
305 : yyscanner);
306 : }
307 :
308 440 : if (cur_token == UIDENT)
309 : {
310 : /* It's an identifier, so truncate as appropriate */
311 28 : truncate_identifier(lvalp->core_yystype.str,
312 28 : strlen(lvalp->core_yystype.str),
313 : true);
314 28 : cur_token = IDENT;
315 : }
316 412 : else if (cur_token == USCONST)
317 : {
318 412 : cur_token = SCONST;
319 : }
320 440 : break;
321 : }
322 :
323 67360 : return cur_token;
324 : }
325 :
326 : /* convert hex digit (caller should have verified that) to value */
327 : static unsigned int
328 2816 : hexval(unsigned char c)
329 : {
330 2816 : if (c >= '0' && c <= '9')
331 2446 : return c - '0';
332 370 : if (c >= 'a' && c <= 'f')
333 66 : return c - 'a' + 0xA;
334 304 : if (c >= 'A' && c <= 'F')
335 304 : return c - 'A' + 0xA;
336 0 : elog(ERROR, "invalid hexadecimal digit");
337 : return 0; /* not reached */
338 : }
339 :
340 : /* is Unicode code point acceptable? */
341 : static void
342 682 : check_unicode_value(pg_wchar c)
343 : {
344 682 : if (!is_valid_unicode_codepoint(c))
345 6 : ereport(ERROR,
346 : (errcode(ERRCODE_SYNTAX_ERROR),
347 : errmsg("invalid Unicode escape value")));
348 676 : }
349 :
350 : /* is 'escape' acceptable as Unicode escape character (UESCAPE syntax) ? */
351 : static bool
352 40 : check_uescapechar(unsigned char escape)
353 : {
354 40 : if (isxdigit(escape)
355 40 : || escape == '+'
356 34 : || escape == '\''
357 34 : || escape == '"'
358 34 : || scanner_isspace(escape))
359 6 : return false;
360 : else
361 34 : return true;
362 : }
363 :
364 : /*
365 : * Process Unicode escapes in "str", producing a palloc'd plain string
366 : *
367 : * escape: the escape character to use
368 : * position: start position of U&'' or U&"" string token
369 : * yyscanner: context information needed for error reports
370 : */
371 : static char *
372 488 : str_udeescape(const char *str, char escape,
373 : int position, core_yyscan_t yyscanner)
374 : {
375 : const char *in;
376 : char *new,
377 : *out;
378 : size_t new_len;
379 488 : pg_wchar pair_first = 0;
380 : ScannerCallbackState scbstate;
381 :
382 : /*
383 : * Guesstimate that result will be no longer than input, but allow enough
384 : * padding for Unicode conversion.
385 : */
386 488 : new_len = strlen(str) + MAX_UNICODE_EQUIVALENT_STRING + 1;
387 488 : new = palloc(new_len);
388 :
389 488 : in = str;
390 488 : out = new;
391 2442 : while (*in)
392 : {
393 : /* Enlarge string if needed */
394 1996 : size_t out_dist = out - new;
395 :
396 1996 : if (out_dist > new_len - (MAX_UNICODE_EQUIVALENT_STRING + 1))
397 : {
398 0 : new_len *= 2;
399 0 : new = repalloc(new, new_len);
400 0 : out = new + out_dist;
401 : }
402 :
403 1996 : if (in[0] == escape)
404 : {
405 : /*
406 : * Any errors reported while processing this escape sequence will
407 : * have an error cursor pointing at the escape.
408 : */
409 706 : setup_scanner_errposition_callback(&scbstate, yyscanner,
410 706 : in - str + position + 3); /* 3 for U&" */
411 706 : if (in[1] == escape)
412 : {
413 12 : if (pair_first)
414 6 : goto invalid_pair;
415 6 : *out++ = escape;
416 6 : in += 2;
417 : }
418 694 : else if (isxdigit((unsigned char) in[1]) &&
419 644 : isxdigit((unsigned char) in[2]) &&
420 644 : isxdigit((unsigned char) in[3]) &&
421 644 : isxdigit((unsigned char) in[4]))
422 632 : {
423 : pg_wchar unicode;
424 :
425 638 : unicode = (hexval(in[1]) << 12) +
426 638 : (hexval(in[2]) << 8) +
427 638 : (hexval(in[3]) << 4) +
428 638 : hexval(in[4]);
429 638 : check_unicode_value(unicode);
430 638 : if (pair_first)
431 : {
432 6 : if (is_utf16_surrogate_second(unicode))
433 : {
434 0 : unicode = surrogate_pair_to_codepoint(pair_first, unicode);
435 0 : pair_first = 0;
436 : }
437 : else
438 6 : goto invalid_pair;
439 : }
440 632 : else if (is_utf16_surrogate_second(unicode))
441 0 : goto invalid_pair;
442 :
443 632 : if (is_utf16_surrogate_first(unicode))
444 24 : pair_first = unicode;
445 : else
446 : {
447 608 : pg_unicode_to_server(unicode, (unsigned char *) out);
448 608 : out += strlen(out);
449 : }
450 632 : in += 5;
451 : }
452 56 : else if (in[1] == '+' &&
453 50 : isxdigit((unsigned char) in[2]) &&
454 50 : isxdigit((unsigned char) in[3]) &&
455 50 : isxdigit((unsigned char) in[4]) &&
456 50 : isxdigit((unsigned char) in[5]) &&
457 50 : isxdigit((unsigned char) in[6]) &&
458 44 : isxdigit((unsigned char) in[7]))
459 32 : {
460 : pg_wchar unicode;
461 :
462 44 : unicode = (hexval(in[2]) << 20) +
463 44 : (hexval(in[3]) << 16) +
464 44 : (hexval(in[4]) << 12) +
465 44 : (hexval(in[5]) << 8) +
466 44 : (hexval(in[6]) << 4) +
467 44 : hexval(in[7]);
468 44 : check_unicode_value(unicode);
469 38 : if (pair_first)
470 : {
471 6 : if (is_utf16_surrogate_second(unicode))
472 : {
473 0 : unicode = surrogate_pair_to_codepoint(pair_first, unicode);
474 0 : pair_first = 0;
475 : }
476 : else
477 6 : goto invalid_pair;
478 : }
479 32 : else if (is_utf16_surrogate_second(unicode))
480 0 : goto invalid_pair;
481 :
482 32 : if (is_utf16_surrogate_first(unicode))
483 6 : pair_first = unicode;
484 : else
485 : {
486 26 : pg_unicode_to_server(unicode, (unsigned char *) out);
487 26 : out += strlen(out);
488 : }
489 32 : in += 8;
490 : }
491 : else
492 12 : ereport(ERROR,
493 : (errcode(ERRCODE_SYNTAX_ERROR),
494 : errmsg("invalid Unicode escape"),
495 : errhint("Unicode escapes must be \\XXXX or \\+XXXXXX.")));
496 :
497 670 : cancel_scanner_errposition_callback(&scbstate);
498 : }
499 : else
500 : {
501 1290 : if (pair_first)
502 6 : goto invalid_pair;
503 :
504 1284 : *out++ = *in++;
505 : }
506 : }
507 :
508 : /* unfinished surrogate pair? */
509 446 : if (pair_first)
510 6 : goto invalid_pair;
511 :
512 440 : *out = '\0';
513 440 : return new;
514 :
515 : /*
516 : * We might get here with the error callback active, or not. Call
517 : * scanner_errposition to make sure an error cursor appears; if the
518 : * callback is active, this is duplicative but harmless.
519 : */
520 30 : invalid_pair:
521 30 : ereport(ERROR,
522 : (errcode(ERRCODE_SYNTAX_ERROR),
523 : errmsg("invalid Unicode surrogate pair"),
524 : scanner_errposition(in - str + position + 3, /* 3 for U&" */
525 : yyscanner)));
526 : return NULL; /* keep compiler quiet */
527 : }
|