Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * readfuncs.c
4 : : * Reader functions for Postgres tree nodes.
5 : : *
6 : : * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
7 : : * Portions Copyright (c) 1994, Regents of the University of California
8 : : *
9 : : *
10 : : * IDENTIFICATION
11 : : * src/backend/nodes/readfuncs.c
12 : : *
13 : : * NOTES
14 : : * Parse location fields are written out by outfuncs.c, but only for
15 : : * debugging use. When reading a location field, we normally discard
16 : : * the stored value and set the location field to -1 (ie, "unknown").
17 : : * This is because nodes coming from a stored rule should not be thought
18 : : * to have a known location in the current query's text.
19 : : *
20 : : * However, if restore_location_fields is true, we do restore location
21 : : * fields from the string. This is currently intended only for use by the
22 : : * debug_write_read_parse_plan_trees test code, which doesn't want to cause
23 : : * any change in the node contents.
24 : : *
25 : : *-------------------------------------------------------------------------
26 : : */
27 : : #include "postgres.h"
28 : :
29 : : #include "miscadmin.h"
30 : : #include "nodes/bitmapset.h"
31 : : #include "nodes/readfuncs.h"
32 : :
33 : :
34 : : /*
35 : : * Macros to simplify reading of different kinds of fields. Use these
36 : : * wherever possible to reduce the chance for silly typos. Note that these
37 : : * hard-wire conventions about the names of the local variables in a Read
38 : : * routine.
39 : : */
40 : :
41 : : /* Macros for declaring appropriate local variables */
42 : :
43 : : /* A few guys need only local_node */
44 : : #define READ_LOCALS_NO_FIELDS(nodeTypeName) \
45 : : nodeTypeName *local_node = makeNode(nodeTypeName)
46 : :
47 : : /* And a few guys need only the pg_strtok support fields */
48 : : #define READ_TEMP_LOCALS() \
49 : : const char *token; \
50 : : int length
51 : :
52 : : /* ... but most need both */
53 : : #define READ_LOCALS(nodeTypeName) \
54 : : READ_LOCALS_NO_FIELDS(nodeTypeName); \
55 : : READ_TEMP_LOCALS()
56 : :
57 : : /* Read an integer field (anything written as ":fldname %d") */
58 : : #define READ_INT_FIELD(fldname) \
59 : : token = pg_strtok(&length); /* skip :fldname */ \
60 : : token = pg_strtok(&length); /* get field value */ \
61 : : local_node->fldname = atoi(token)
62 : :
63 : : /* Read an unsigned integer field (anything written as ":fldname %u") */
64 : : #define READ_UINT_FIELD(fldname) \
65 : : token = pg_strtok(&length); /* skip :fldname */ \
66 : : token = pg_strtok(&length); /* get field value */ \
67 : : local_node->fldname = atoui(token)
68 : :
69 : : /* Read a signed integer field (anything written using INT64_FORMAT) */
70 : : #define READ_INT64_FIELD(fldname) \
71 : : token = pg_strtok(&length); /* skip :fldname */ \
72 : : token = pg_strtok(&length); /* get field value */ \
73 : : local_node->fldname = strtoi64(token, NULL, 10)
74 : :
75 : : /* Read an unsigned integer field (anything written using UINT64_FORMAT) */
76 : : #define READ_UINT64_FIELD(fldname) \
77 : : token = pg_strtok(&length); /* skip :fldname */ \
78 : : token = pg_strtok(&length); /* get field value */ \
79 : : local_node->fldname = strtou64(token, NULL, 10)
80 : :
81 : : /* Read a long integer field (anything written as ":fldname %ld") */
82 : : #define READ_LONG_FIELD(fldname) \
83 : : token = pg_strtok(&length); /* skip :fldname */ \
84 : : token = pg_strtok(&length); /* get field value */ \
85 : : local_node->fldname = atol(token)
86 : :
87 : : /* Read an OID field (don't hard-wire assumption that OID is same as uint) */
88 : : #define READ_OID_FIELD(fldname) \
89 : : token = pg_strtok(&length); /* skip :fldname */ \
90 : : token = pg_strtok(&length); /* get field value */ \
91 : : local_node->fldname = atooid(token)
92 : :
93 : : /* Read a char field (ie, one ascii character) */
94 : : #define READ_CHAR_FIELD(fldname) \
95 : : token = pg_strtok(&length); /* skip :fldname */ \
96 : : token = pg_strtok(&length); /* get field value */ \
97 : : /* avoid overhead of calling debackslash() for one char */ \
98 : : local_node->fldname = (length == 0) ? '\0' : (token[0] == '\\' ? token[1] : token[0])
99 : :
100 : : /* Read an enumerated-type field that was written as an integer code */
101 : : #define READ_ENUM_FIELD(fldname, enumtype) \
102 : : token = pg_strtok(&length); /* skip :fldname */ \
103 : : token = pg_strtok(&length); /* get field value */ \
104 : : local_node->fldname = (enumtype) atoi(token)
105 : :
106 : : /* Read a float field */
107 : : #define READ_FLOAT_FIELD(fldname) \
108 : : token = pg_strtok(&length); /* skip :fldname */ \
109 : : token = pg_strtok(&length); /* get field value */ \
110 : : local_node->fldname = atof(token)
111 : :
112 : : /* Read a boolean field */
113 : : #define READ_BOOL_FIELD(fldname) \
114 : : token = pg_strtok(&length); /* skip :fldname */ \
115 : : token = pg_strtok(&length); /* get field value */ \
116 : : local_node->fldname = strtobool(token)
117 : :
118 : : /* Read a character-string field */
119 : : #define READ_STRING_FIELD(fldname) \
120 : : token = pg_strtok(&length); /* skip :fldname */ \
121 : : token = pg_strtok(&length); /* get field value */ \
122 : : local_node->fldname = nullable_string(token, length)
123 : :
124 : : /* Read a parse location field (and possibly throw away the value) */
125 : : #ifdef DEBUG_NODE_TESTS_ENABLED
126 : : #define READ_LOCATION_FIELD(fldname) \
127 : : token = pg_strtok(&length); /* skip :fldname */ \
128 : : token = pg_strtok(&length); /* get field value */ \
129 : : local_node->fldname = restore_location_fields ? atoi(token) : -1
130 : : #else
131 : : #define READ_LOCATION_FIELD(fldname) \
132 : : token = pg_strtok(&length); /* skip :fldname */ \
133 : : token = pg_strtok(&length); /* get field value */ \
134 : : (void) token; /* in case not used elsewhere */ \
135 : : local_node->fldname = -1 /* set field to "unknown" */
136 : : #endif
137 : :
138 : : /* Read a Node field */
139 : : #define READ_NODE_FIELD(fldname) \
140 : : token = pg_strtok(&length); /* skip :fldname */ \
141 : : (void) token; /* in case not used elsewhere */ \
142 : : local_node->fldname = nodeRead(NULL, 0)
143 : :
144 : : /* Read a bitmapset field */
145 : : #define READ_BITMAPSET_FIELD(fldname) \
146 : : token = pg_strtok(&length); /* skip :fldname */ \
147 : : (void) token; /* in case not used elsewhere */ \
148 : : local_node->fldname = _readBitmapset()
149 : :
150 : : /* Read an attribute number array */
151 : : #define READ_ATTRNUMBER_ARRAY(fldname, len) \
152 : : token = pg_strtok(&length); /* skip :fldname */ \
153 : : local_node->fldname = readAttrNumberCols(len)
154 : :
155 : : /* Read an oid array */
156 : : #define READ_OID_ARRAY(fldname, len) \
157 : : token = pg_strtok(&length); /* skip :fldname */ \
158 : : local_node->fldname = readOidCols(len)
159 : :
160 : : /* Read an int array */
161 : : #define READ_INT_ARRAY(fldname, len) \
162 : : token = pg_strtok(&length); /* skip :fldname */ \
163 : : local_node->fldname = readIntCols(len)
164 : :
165 : : /* Read a bool array */
166 : : #define READ_BOOL_ARRAY(fldname, len) \
167 : : token = pg_strtok(&length); /* skip :fldname */ \
168 : : local_node->fldname = readBoolCols(len)
169 : :
170 : : /* Routine exit */
171 : : #define READ_DONE() \
172 : : return local_node
173 : :
174 : :
175 : : /*
176 : : * NOTE: use atoi() to read values written with %d, or atoui() to read
177 : : * values written with %u in outfuncs.c. An exception is OID values,
178 : : * for which use atooid(). (As of 7.1, outfuncs.c writes OIDs as %u,
179 : : * but this will probably change in the future.)
180 : : */
181 : : #define atoui(x) ((unsigned int) strtoul((x), NULL, 10))
182 : :
183 : : #define strtobool(x) ((*(x) == 't') ? true : false)
184 : :
185 : : static char *
1317 peter@eisentraut.org 186 :CBC 646991 : nullable_string(const char *token, int length)
187 : : {
188 : : /* outToken emits <> for NULL, and pg_strtok makes that an empty string */
189 [ + + ]: 646991 : if (length == 0)
190 : 133173 : return NULL;
191 : : /* outToken emits "" for empty string */
192 [ + + - + : 513818 : if (length == 2 && token[0] == '"' && token[1] == '"')
- - ]
1317 peter@eisentraut.org 193 :UBC 0 : return pstrdup("");
194 : : /* otherwise, we must remove protective backslashes added by outToken */
1317 peter@eisentraut.org 195 :CBC 513818 : return debackslash(token, length);
196 : : }
197 : :
198 : :
199 : : /*
200 : : * _readBitmapset
201 : : *
202 : : * Note: this code is used in contexts where we know that a Bitmapset
203 : : * is expected. There is equivalent code in nodeRead() that can read a
204 : : * Bitmapset when we come across one in other contexts.
205 : : */
206 : : static Bitmapset *
6312 tgl@sss.pgh.pa.us 207 : 1844240 : _readBitmapset(void)
208 : : {
6172 bruce@momjian.us 209 : 1844240 : Bitmapset *result = NULL;
210 : :
211 : : READ_TEMP_LOCALS();
212 : :
6312 tgl@sss.pgh.pa.us 213 : 1844240 : token = pg_strtok(&length);
214 [ - + ]: 1844240 : if (token == NULL)
6312 tgl@sss.pgh.pa.us 215 [ # # ]:UBC 0 : elog(ERROR, "incomplete Bitmapset structure");
6312 tgl@sss.pgh.pa.us 216 [ + - - + ]:CBC 1844240 : if (length != 1 || token[0] != '(')
6312 tgl@sss.pgh.pa.us 217 [ # # ]:UBC 0 : elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
218 : :
6312 tgl@sss.pgh.pa.us 219 :CBC 1844240 : token = pg_strtok(&length);
220 [ - + ]: 1844240 : if (token == NULL)
6312 tgl@sss.pgh.pa.us 221 [ # # ]:UBC 0 : elog(ERROR, "incomplete Bitmapset structure");
6312 tgl@sss.pgh.pa.us 222 [ + - + - ]:CBC 1844240 : if (length != 1 || token[0] != 'b')
6312 tgl@sss.pgh.pa.us 223 [ # # ]:UBC 0 : elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
224 : :
225 : : for (;;)
6312 tgl@sss.pgh.pa.us 226 :CBC 456642 : {
227 : : int val;
228 : : char *endptr;
229 : :
230 : 2300882 : token = pg_strtok(&length);
231 [ - + ]: 2300882 : if (token == NULL)
6312 tgl@sss.pgh.pa.us 232 [ # # ]:UBC 0 : elog(ERROR, "unterminated Bitmapset structure");
6312 tgl@sss.pgh.pa.us 233 [ + + + + ]:CBC 2300882 : if (length == 1 && token[0] == ')')
234 : 1844240 : break;
235 : 456642 : val = (int) strtol(token, &endptr, 10);
236 [ - + ]: 456642 : if (endptr != token + length)
6312 tgl@sss.pgh.pa.us 237 [ # # ]:UBC 0 : elog(ERROR, "unrecognized integer: \"%.*s\"", length, token);
6312 tgl@sss.pgh.pa.us 238 :CBC 456642 : result = bms_add_member(result, val);
239 : : }
240 : :
241 : 1844240 : return result;
242 : : }
243 : :
244 : : /*
245 : : * We export this function for use by extensions that define extensible nodes.
246 : : * That's somewhat historical, though, because calling nodeRead() will work.
247 : : */
248 : : Bitmapset *
3735 rhaas@postgresql.org 249 :UBC 0 : readBitmapset(void)
250 : : {
251 : 0 : return _readBitmapset();
252 : : }
253 : :
254 : : #include "readfuncs.funcs.c"
255 : :
256 : :
257 : : /*
258 : : * Support functions for nodes with custom_read_write attribute or
259 : : * special_read_write attribute
260 : : */
261 : :
262 : : static Const *
1396 peter@eisentraut.org 263 :CBC 442121 : _readConst(void)
264 : : {
265 : 442121 : READ_LOCALS(Const);
266 : :
267 : 442121 : READ_OID_FIELD(consttype);
268 : 442121 : READ_INT_FIELD(consttypmod);
269 : 442121 : READ_OID_FIELD(constcollid);
270 : 442121 : READ_INT_FIELD(constlen);
271 : 442121 : READ_BOOL_FIELD(constbyval);
272 : 442121 : READ_BOOL_FIELD(constisnull);
4007 andres@anarazel.de 273 [ - + ]: 442121 : READ_LOCATION_FIELD(location);
274 : :
1396 peter@eisentraut.org 275 : 442121 : token = pg_strtok(&length); /* skip :constvalue */
276 [ + + ]: 442121 : if (local_node->constisnull)
277 : 46768 : token = pg_strtok(&length); /* skip "<>" */
278 : : else
279 : 395353 : local_node->constvalue = readDatum(local_node->constbyval);
280 : :
1919 281 : 442121 : READ_DONE();
282 : : }
283 : :
284 : : static BoolExpr *
1396 285 : 32468 : _readBoolExpr(void)
286 : : {
287 : 32468 : READ_LOCALS(BoolExpr);
288 : :
289 : : /* do-it-yourself enum representation */
290 : 32468 : token = pg_strtok(&length); /* skip :boolop */
291 : 32468 : token = pg_strtok(&length); /* get field value */
1319 292 [ + + + + ]: 32468 : if (length == 3 && strncmp(token, "and", 3) == 0)
1396 293 : 22419 : local_node->boolop = AND_EXPR;
1319 294 [ + + + - ]: 10049 : else if (length == 2 && strncmp(token, "or", 2) == 0)
1396 295 : 6756 : local_node->boolop = OR_EXPR;
1319 296 [ + - + - ]: 3293 : else if (length == 3 && strncmp(token, "not", 3) == 0)
1396 297 : 3293 : local_node->boolop = NOT_EXPR;
298 : : else
1396 peter@eisentraut.org 299 [ # # ]:UBC 0 : elog(ERROR, "unrecognized boolop \"%.*s\"", length, token);
300 : :
1396 peter@eisentraut.org 301 :CBC 32468 : READ_NODE_FIELD(args);
6422 tgl@sss.pgh.pa.us 302 [ - + ]: 32468 : READ_LOCATION_FIELD(location);
303 : :
8562 304 : 32468 : READ_DONE();
305 : : }
306 : :
307 : : static A_Const *
1319 peter@eisentraut.org 308 :UBC 0 : _readA_Const(void)
309 : : {
310 : 0 : READ_LOCALS(A_Const);
311 : :
312 : : /* We expect either NULL or :val here */
313 : 0 : token = pg_strtok(&length);
314 [ # # # # ]: 0 : if (length == 4 && strncmp(token, "NULL", 4) == 0)
315 : 0 : local_node->isnull = true;
316 : : else
317 : : {
318 : 0 : union ValUnion *tmp = nodeRead(NULL, 0);
319 : :
320 : : /* To forestall valgrind complaints, copy only the valid data */
1143 tgl@sss.pgh.pa.us 321 [ # # # # : 0 : switch (nodeTag(tmp))
# # ]
322 : : {
323 : 0 : case T_Integer:
324 : 0 : memcpy(&local_node->val, tmp, sizeof(Integer));
325 : 0 : break;
326 : 0 : case T_Float:
327 : 0 : memcpy(&local_node->val, tmp, sizeof(Float));
328 : 0 : break;
329 : 0 : case T_Boolean:
330 : 0 : memcpy(&local_node->val, tmp, sizeof(Boolean));
331 : 0 : break;
332 : 0 : case T_String:
333 : 0 : memcpy(&local_node->val, tmp, sizeof(String));
334 : 0 : break;
335 : 0 : case T_BitString:
336 : 0 : memcpy(&local_node->val, tmp, sizeof(BitString));
337 : 0 : break;
338 : 0 : default:
339 [ # # ]: 0 : elog(ERROR, "unrecognized node type: %d",
340 : : (int) nodeTag(tmp));
341 : : break;
342 : : }
343 : : }
344 : :
1319 peter@eisentraut.org 345 [ # # ]: 0 : READ_LOCATION_FIELD(location);
346 : :
347 : 0 : READ_DONE();
348 : : }
349 : :
350 : : static RangeTblEntry *
1396 peter@eisentraut.org 351 :CBC 128578 : _readRangeTblEntry(void)
352 : : {
353 : 128578 : READ_LOCALS(RangeTblEntry);
354 : :
355 : 128578 : READ_NODE_FIELD(alias);
356 : 128578 : READ_NODE_FIELD(eref);
357 : 128578 : READ_ENUM_FIELD(rtekind, RTEKind);
358 : :
359 [ + + + + : 128578 : switch (local_node->rtekind)
+ + + - +
+ + - ]
360 : : {
361 : 78282 : case RTE_RELATION:
362 : 78282 : READ_OID_FIELD(relid);
789 363 : 78282 : READ_BOOL_FIELD(inh);
1396 364 [ + - - + ]: 78282 : READ_CHAR_FIELD(relkind);
365 : 78282 : READ_INT_FIELD(rellockmode);
1246 alvherre@alvh.no-ip. 366 : 78282 : READ_UINT_FIELD(perminfoindex);
791 peter@eisentraut.org 367 : 78282 : READ_NODE_FIELD(tablesample);
1396 368 : 78282 : break;
369 : 15237 : case RTE_SUBQUERY:
370 : 15237 : READ_NODE_FIELD(subquery);
371 : 15237 : READ_BOOL_FIELD(security_barrier);
372 : : /* we re-use these RELATION fields, too: */
1203 tgl@sss.pgh.pa.us 373 : 15237 : READ_OID_FIELD(relid);
789 peter@eisentraut.org 374 : 15237 : READ_BOOL_FIELD(inh);
1057 amitlan@postgresql.o 375 [ + + - + ]: 15237 : READ_CHAR_FIELD(relkind);
1203 tgl@sss.pgh.pa.us 376 : 15237 : READ_INT_FIELD(rellockmode);
377 : 15237 : READ_UINT_FIELD(perminfoindex);
1396 peter@eisentraut.org 378 : 15237 : break;
379 : 23997 : case RTE_JOIN:
380 : 23997 : READ_ENUM_FIELD(jointype, JoinType);
381 : 23997 : READ_INT_FIELD(joinmergedcols);
382 : 23997 : READ_NODE_FIELD(joinaliasvars);
383 : 23997 : READ_NODE_FIELD(joinleftcols);
384 : 23997 : READ_NODE_FIELD(joinrightcols);
385 : 23997 : READ_NODE_FIELD(join_using_alias);
386 : 23997 : break;
387 : 8181 : case RTE_FUNCTION:
388 : 8181 : READ_NODE_FIELD(functions);
389 : 8181 : READ_BOOL_FIELD(funcordinality);
390 : 8181 : break;
391 : 178 : case RTE_TABLEFUNC:
392 : 178 : READ_NODE_FIELD(tablefunc);
393 : : /* The RTE must have a copy of the column type info, if any */
394 [ + - ]: 178 : if (local_node->tablefunc)
395 : : {
396 : 178 : TableFunc *tf = local_node->tablefunc;
397 : :
398 : 178 : local_node->coltypes = tf->coltypes;
399 : 178 : local_node->coltypmods = tf->coltypmods;
400 : 178 : local_node->colcollations = tf->colcollations;
401 : : }
402 : 178 : break;
403 : 975 : case RTE_VALUES:
404 : 975 : READ_NODE_FIELD(values_lists);
405 : 975 : READ_NODE_FIELD(coltypes);
406 : 975 : READ_NODE_FIELD(coltypmods);
407 : 975 : READ_NODE_FIELD(colcollations);
408 : 975 : break;
409 : 435 : case RTE_CTE:
410 : 435 : READ_STRING_FIELD(ctename);
411 : 435 : READ_UINT_FIELD(ctelevelsup);
412 : 435 : READ_BOOL_FIELD(self_reference);
413 : 435 : READ_NODE_FIELD(coltypes);
414 : 435 : READ_NODE_FIELD(coltypmods);
415 : 435 : READ_NODE_FIELD(colcollations);
416 : 435 : break;
1396 peter@eisentraut.org 417 :UBC 0 : case RTE_NAMEDTUPLESTORE:
418 : 0 : READ_STRING_FIELD(enrname);
419 : 0 : READ_FLOAT_FIELD(enrtuples);
420 : 0 : READ_NODE_FIELD(coltypes);
421 : 0 : READ_NODE_FIELD(coltypmods);
422 : 0 : READ_NODE_FIELD(colcollations);
423 : : /* we re-use these RELATION fields, too: */
1203 tgl@sss.pgh.pa.us 424 : 0 : READ_OID_FIELD(relid);
1396 peter@eisentraut.org 425 : 0 : break;
50 peter@eisentraut.org 426 :GNC 82 : case RTE_GRAPH_TABLE:
427 : 82 : READ_NODE_FIELD(graph_pattern);
428 : 82 : READ_NODE_FIELD(graph_table_columns);
429 : : /* we re-use these RELATION fields, too: */
430 : 82 : READ_OID_FIELD(relid);
431 [ + - - + ]: 82 : READ_CHAR_FIELD(relkind);
432 : 82 : READ_INT_FIELD(rellockmode);
433 : 82 : READ_UINT_FIELD(perminfoindex);
434 : 82 : break;
1396 peter@eisentraut.org 435 :CBC 76 : case RTE_RESULT:
436 : : /* no extra fields */
437 : 76 : break;
602 rguo@postgresql.org 438 : 1135 : case RTE_GROUP:
439 : 1135 : READ_NODE_FIELD(groupexprs);
440 : 1135 : break;
1396 peter@eisentraut.org 441 :UBC 0 : default:
442 [ # # ]: 0 : elog(ERROR, "unrecognized RTE kind: %d",
443 : : (int) local_node->rtekind);
444 : : break;
445 : : }
446 : :
1396 peter@eisentraut.org 447 :CBC 128578 : READ_BOOL_FIELD(lateral);
448 : 128578 : READ_BOOL_FIELD(inFromCl);
449 : 128578 : READ_NODE_FIELD(securityQuals);
450 : :
8562 tgl@sss.pgh.pa.us 451 : 128578 : READ_DONE();
452 : : }
453 : :
454 : : static A_Expr *
1319 peter@eisentraut.org 455 :UBC 0 : _readA_Expr(void)
456 : : {
457 : 0 : READ_LOCALS(A_Expr);
458 : :
459 : 0 : token = pg_strtok(&length);
460 : :
461 [ # # # # ]: 0 : if (length == 3 && strncmp(token, "ANY", 3) == 0)
462 : : {
463 : 0 : local_node->kind = AEXPR_OP_ANY;
464 : 0 : READ_NODE_FIELD(name);
465 : : }
466 [ # # # # ]: 0 : else if (length == 3 && strncmp(token, "ALL", 3) == 0)
467 : : {
468 : 0 : local_node->kind = AEXPR_OP_ALL;
469 : 0 : READ_NODE_FIELD(name);
470 : : }
471 [ # # # # ]: 0 : else if (length == 8 && strncmp(token, "DISTINCT", 8) == 0)
472 : : {
473 : 0 : local_node->kind = AEXPR_DISTINCT;
474 : 0 : READ_NODE_FIELD(name);
475 : : }
476 [ # # # # ]: 0 : else if (length == 12 && strncmp(token, "NOT_DISTINCT", 12) == 0)
477 : : {
478 : 0 : local_node->kind = AEXPR_NOT_DISTINCT;
479 : 0 : READ_NODE_FIELD(name);
480 : : }
481 [ # # # # ]: 0 : else if (length == 6 && strncmp(token, "NULLIF", 6) == 0)
482 : : {
483 : 0 : local_node->kind = AEXPR_NULLIF;
484 : 0 : READ_NODE_FIELD(name);
485 : : }
486 [ # # # # ]: 0 : else if (length == 2 && strncmp(token, "IN", 2) == 0)
487 : : {
488 : 0 : local_node->kind = AEXPR_IN;
489 : 0 : READ_NODE_FIELD(name);
490 : : }
491 [ # # # # ]: 0 : else if (length == 4 && strncmp(token, "LIKE", 4) == 0)
492 : : {
493 : 0 : local_node->kind = AEXPR_LIKE;
494 : 0 : READ_NODE_FIELD(name);
495 : : }
496 [ # # # # ]: 0 : else if (length == 5 && strncmp(token, "ILIKE", 5) == 0)
497 : : {
498 : 0 : local_node->kind = AEXPR_ILIKE;
499 : 0 : READ_NODE_FIELD(name);
500 : : }
501 [ # # # # ]: 0 : else if (length == 7 && strncmp(token, "SIMILAR", 7) == 0)
502 : : {
503 : 0 : local_node->kind = AEXPR_SIMILAR;
504 : 0 : READ_NODE_FIELD(name);
505 : : }
506 [ # # # # ]: 0 : else if (length == 7 && strncmp(token, "BETWEEN", 7) == 0)
507 : : {
508 : 0 : local_node->kind = AEXPR_BETWEEN;
509 : 0 : READ_NODE_FIELD(name);
510 : : }
511 [ # # # # ]: 0 : else if (length == 11 && strncmp(token, "NOT_BETWEEN", 11) == 0)
512 : : {
513 : 0 : local_node->kind = AEXPR_NOT_BETWEEN;
514 : 0 : READ_NODE_FIELD(name);
515 : : }
516 [ # # # # ]: 0 : else if (length == 11 && strncmp(token, "BETWEEN_SYM", 11) == 0)
517 : : {
518 : 0 : local_node->kind = AEXPR_BETWEEN_SYM;
519 : 0 : READ_NODE_FIELD(name);
520 : : }
521 [ # # # # ]: 0 : else if (length == 15 && strncmp(token, "NOT_BETWEEN_SYM", 15) == 0)
522 : : {
523 : 0 : local_node->kind = AEXPR_NOT_BETWEEN_SYM;
524 : 0 : READ_NODE_FIELD(name);
525 : : }
526 [ # # # # ]: 0 : else if (length == 5 && strncmp(token, ":name", 5) == 0)
527 : : {
528 : 0 : local_node->kind = AEXPR_OP;
529 : 0 : local_node->name = nodeRead(NULL, 0);
530 : : }
531 : : else
532 [ # # ]: 0 : elog(ERROR, "unrecognized A_Expr kind: \"%.*s\"", length, token);
533 : :
534 : 0 : READ_NODE_FIELD(lexpr);
535 : 0 : READ_NODE_FIELD(rexpr);
327 alvherre@kurilemu.de 536 [ # # ]: 0 : READ_LOCATION_FIELD(rexpr_list_start);
537 [ # # ]: 0 : READ_LOCATION_FIELD(rexpr_list_end);
1319 peter@eisentraut.org 538 [ # # ]: 0 : READ_LOCATION_FIELD(location);
539 : :
540 : 0 : READ_DONE();
541 : : }
542 : :
543 : : static ExtensibleNode *
1396 544 : 0 : _readExtensibleNode(void)
545 : : {
546 : : const ExtensibleNodeMethods *methods;
547 : : ExtensibleNode *local_node;
548 : : const char *extnodename;
549 : :
550 : : READ_TEMP_LOCALS();
551 : :
552 : 0 : token = pg_strtok(&length); /* skip :extnodename */
553 : 0 : token = pg_strtok(&length); /* get extnodename */
554 : :
555 : 0 : extnodename = nullable_string(token, length);
556 [ # # ]: 0 : if (!extnodename)
557 [ # # ]: 0 : elog(ERROR, "extnodename has to be supplied");
558 : 0 : methods = GetExtensibleNodeMethods(extnodename, false);
559 : :
560 : 0 : local_node = (ExtensibleNode *) newNode(methods->node_size,
561 : : T_ExtensibleNode);
562 : 0 : local_node->extnodename = extnodename;
563 : :
564 : : /* deserialize the private fields */
565 : 0 : methods->nodeRead(local_node);
566 : :
8562 tgl@sss.pgh.pa.us 567 : 0 : READ_DONE();
568 : : }
569 : :
570 : :
571 : : /*
572 : : * parseNodeString
573 : : *
574 : : * Given a character string representing a node tree, parseNodeString creates
575 : : * the internal node structure.
576 : : *
577 : : * The string to be read must already have been loaded into pg_strtok().
578 : : */
579 : : Node *
8562 tgl@sss.pgh.pa.us 580 :CBC 3791619 : parseNodeString(void)
581 : : {
582 : : READ_TEMP_LOCALS();
583 : :
584 : : /* Guard against stack overflow due to overly complex expressions */
2703 585 : 3791619 : check_stack_depth();
586 : :
9249 587 : 3791619 : token = pg_strtok(&length);
588 : :
589 : : #define MATCH(tokname, namelen) \
590 : : (length == namelen && memcmp(token, tokname, namelen) == 0)
591 : :
592 : : #include "readfuncs.switch.c"
593 : :
1056 noah@leadboat.com 594 [ # # ]:UBC 0 : elog(ERROR, "badly formatted node string \"%.32s\"...", token);
595 : : return NULL; /* keep compiler quiet */
596 : : }
597 : :
598 : :
599 : : /*
600 : : * readDatum
601 : : *
602 : : * Given a string representation of a constant, recreate the appropriate
603 : : * Datum. The string representation embeds length info, but not byValue,
604 : : * so we must be told that.
605 : : */
606 : : Datum
9249 tgl@sss.pgh.pa.us 607 :CBC 395353 : readDatum(bool typbyval)
608 : : {
609 : : Size length;
610 : : int tokenLength;
611 : : const char *token;
612 : : Datum res;
613 : : char *s;
614 : :
615 : : /*
616 : : * read the actual length of the value
617 : : */
618 : 395353 : token = pg_strtok(&tokenLength);
9248 619 : 395353 : length = atoui(token);
620 : :
9072 621 : 395353 : token = pg_strtok(&tokenLength); /* read the '[' */
622 [ + - - + ]: 395353 : if (token == NULL || token[0] != '[')
4485 tgl@sss.pgh.pa.us 623 [ # # # # ]:UBC 0 : elog(ERROR, "expected \"[\" to start datum, but got \"%s\"; length = %zu",
624 : : token ? token : "[NULL]", length);
625 : :
9249 tgl@sss.pgh.pa.us 626 [ + + ]:CBC 395353 : if (typbyval)
627 : : {
9248 628 [ - + ]: 306276 : if (length > (Size) sizeof(Datum))
4485 tgl@sss.pgh.pa.us 629 [ # # ]:UBC 0 : elog(ERROR, "byval datum but length = %zu", length);
9608 tgl@sss.pgh.pa.us 630 :CBC 306276 : res = (Datum) 0;
10467 bruce@momjian.us 631 : 306276 : s = (char *) (&res);
161 peter@eisentraut.org 632 [ + + ]:GNC 2756484 : for (Size i = 0; i < (Size) sizeof(Datum); i++)
633 : : {
9249 tgl@sss.pgh.pa.us 634 :CBC 2450208 : token = pg_strtok(&tokenLength);
10467 bruce@momjian.us 635 : 2450208 : s[i] = (char) atoi(token);
636 : : }
637 : : }
638 [ - + ]: 89077 : else if (length <= 0)
270 tgl@sss.pgh.pa.us 639 :UNC 0 : res = (Datum) 0;
640 : : else
641 : : {
10467 bruce@momjian.us 642 :CBC 89077 : s = (char *) palloc(length);
161 peter@eisentraut.org 643 [ + + ]:GNC 2808835 : for (Size i = 0; i < length; i++)
644 : : {
9249 tgl@sss.pgh.pa.us 645 :CBC 2719758 : token = pg_strtok(&tokenLength);
10467 bruce@momjian.us 646 : 2719758 : s[i] = (char) atoi(token);
647 : : }
648 : 89077 : res = PointerGetDatum(s);
649 : : }
650 : :
9072 tgl@sss.pgh.pa.us 651 : 395353 : token = pg_strtok(&tokenLength); /* read the ']' */
9249 652 [ + - - + ]: 395353 : if (token == NULL || token[0] != ']')
4485 tgl@sss.pgh.pa.us 653 [ # # # # ]:UBC 0 : elog(ERROR, "expected \"]\" to end datum, but got \"%s\"; length = %zu",
654 : : token ? token : "[NULL]", length);
655 : :
10108 bruce@momjian.us 656 :CBC 395353 : return res;
657 : : }
658 : :
659 : : /*
660 : : * common implementation for scalar-array-reading functions
661 : : *
662 : : * The data format is either "<>" for a NULL pointer (in which case numCols
663 : : * is ignored) or "(item item item)" where the number of items must equal
664 : : * numCols. The convfunc must be okay with stopping at whitespace or a
665 : : * right parenthesis, since pg_strtok won't null-terminate the token.
666 : : */
667 : : #define READ_SCALAR_ARRAY(fnname, datatype, convfunc) \
668 : : datatype * \
669 : : fnname(int numCols) \
670 : : { \
671 : : datatype *vals; \
672 : : READ_TEMP_LOCALS(); \
673 : : token = pg_strtok(&length); \
674 : : if (token == NULL) \
675 : : elog(ERROR, "incomplete scalar array"); \
676 : : if (length == 0) \
677 : : return NULL; /* it was "<>", so return NULL pointer */ \
678 : : if (length != 1 || token[0] != '(') \
679 : : elog(ERROR, "unrecognized token: \"%.*s\"", length, token); \
680 : : vals = (datatype *) palloc(numCols * sizeof(datatype)); \
681 : : for (int i = 0; i < numCols; i++) \
682 : : { \
683 : : token = pg_strtok(&length); \
684 : : if (token == NULL || token[0] == ')') \
685 : : elog(ERROR, "incomplete scalar array"); \
686 : : vals[i] = convfunc(token); \
687 : : } \
688 : : token = pg_strtok(&length); \
689 : : if (token == NULL || length != 1 || token[0] != ')') \
690 : : elog(ERROR, "incomplete scalar array"); \
691 : : return vals; \
692 : : }
693 : :
694 : : /*
695 : : * Note: these functions are exported in nodes.h for possible use by
696 : : * extensions, so don't mess too much with their names or API.
697 : : */
1385 tgl@sss.pgh.pa.us 698 [ - + - - : 2875 : READ_SCALAR_ARRAY(readAttrNumberCols, int16, atoi)
+ + + - -
+ - - + -
- + - - +
+ + - + -
- + - - ]
699 [ - + - - : 7358 : READ_SCALAR_ARRAY(readOidCols, Oid, atooid)
+ + + - -
+ - - + -
- + - - +
+ + - + -
- + - - ]
700 : : /* outfuncs.c has writeIndexCols, but we don't yet need that here */
701 : : /* READ_SCALAR_ARRAY(readIndexCols, Index, atoui) */
702 [ - + - - : 3096 : READ_SCALAR_ARRAY(readIntCols, int, atoi)
- + + - -
+ - - + -
- + - - +
+ + - + -
- + - - ]
703 [ - + - - : 1766 : READ_SCALAR_ARRAY(readBoolCols, bool, strtobool)
- + + - -
+ - - + -
- + - - +
+ + - + -
- + - - ]
|