PostgreSQL Source Code  git master
readfuncs.c
Go to the documentation of this file.
1 /*-------------------------------------------------------------------------
2  *
3  * readfuncs.c
4  * Reader functions for Postgres tree nodes.
5  *
6  * Portions Copyright (c) 1996-2023, 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  * 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 <math.h>
30 
31 #include "miscadmin.h"
32 #include "nodes/bitmapset.h"
33 #include "nodes/readfuncs.h"
34 
35 
36 /*
37  * Macros to simplify reading of different kinds of fields. Use these
38  * wherever possible to reduce the chance for silly typos. Note that these
39  * hard-wire conventions about the names of the local variables in a Read
40  * routine.
41  */
42 
43 /* Macros for declaring appropriate local variables */
44 
45 /* A few guys need only local_node */
46 #define READ_LOCALS_NO_FIELDS(nodeTypeName) \
47  nodeTypeName *local_node = makeNode(nodeTypeName)
48 
49 /* And a few guys need only the pg_strtok support fields */
50 #define READ_TEMP_LOCALS() \
51  const char *token; \
52  int length
53 
54 /* ... but most need both */
55 #define READ_LOCALS(nodeTypeName) \
56  READ_LOCALS_NO_FIELDS(nodeTypeName); \
57  READ_TEMP_LOCALS()
58 
59 /* Read an integer field (anything written as ":fldname %d") */
60 #define READ_INT_FIELD(fldname) \
61  token = pg_strtok(&length); /* skip :fldname */ \
62  token = pg_strtok(&length); /* get field value */ \
63  local_node->fldname = atoi(token)
64 
65 /* Read an unsigned integer field (anything written as ":fldname %u") */
66 #define READ_UINT_FIELD(fldname) \
67  token = pg_strtok(&length); /* skip :fldname */ \
68  token = pg_strtok(&length); /* get field value */ \
69  local_node->fldname = atoui(token)
70 
71 /* Read an unsigned integer field (anything written using UINT64_FORMAT) */
72 #define READ_UINT64_FIELD(fldname) \
73  token = pg_strtok(&length); /* skip :fldname */ \
74  token = pg_strtok(&length); /* get field value */ \
75  local_node->fldname = strtou64(token, NULL, 10)
76 
77 /* Read a long integer field (anything written as ":fldname %ld") */
78 #define READ_LONG_FIELD(fldname) \
79  token = pg_strtok(&length); /* skip :fldname */ \
80  token = pg_strtok(&length); /* get field value */ \
81  local_node->fldname = atol(token)
82 
83 /* Read an OID field (don't hard-wire assumption that OID is same as uint) */
84 #define READ_OID_FIELD(fldname) \
85  token = pg_strtok(&length); /* skip :fldname */ \
86  token = pg_strtok(&length); /* get field value */ \
87  local_node->fldname = atooid(token)
88 
89 /* Read a char field (ie, one ascii character) */
90 #define READ_CHAR_FIELD(fldname) \
91  token = pg_strtok(&length); /* skip :fldname */ \
92  token = pg_strtok(&length); /* get field value */ \
93  /* avoid overhead of calling debackslash() for one char */ \
94  local_node->fldname = (length == 0) ? '\0' : (token[0] == '\\' ? token[1] : token[0])
95 
96 /* Read an enumerated-type field that was written as an integer code */
97 #define READ_ENUM_FIELD(fldname, enumtype) \
98  token = pg_strtok(&length); /* skip :fldname */ \
99  token = pg_strtok(&length); /* get field value */ \
100  local_node->fldname = (enumtype) atoi(token)
101 
102 /* Read a float field */
103 #define READ_FLOAT_FIELD(fldname) \
104  token = pg_strtok(&length); /* skip :fldname */ \
105  token = pg_strtok(&length); /* get field value */ \
106  local_node->fldname = atof(token)
107 
108 /* Read a boolean field */
109 #define READ_BOOL_FIELD(fldname) \
110  token = pg_strtok(&length); /* skip :fldname */ \
111  token = pg_strtok(&length); /* get field value */ \
112  local_node->fldname = strtobool(token)
113 
114 /* Read a character-string field */
115 #define READ_STRING_FIELD(fldname) \
116  token = pg_strtok(&length); /* skip :fldname */ \
117  token = pg_strtok(&length); /* get field value */ \
118  local_node->fldname = nullable_string(token, length)
119 
120 /* Read a parse location field (and possibly throw away the value) */
121 #ifdef WRITE_READ_PARSE_PLAN_TREES
122 #define READ_LOCATION_FIELD(fldname) \
123  token = pg_strtok(&length); /* skip :fldname */ \
124  token = pg_strtok(&length); /* get field value */ \
125  local_node->fldname = restore_location_fields ? atoi(token) : -1
126 #else
127 #define READ_LOCATION_FIELD(fldname) \
128  token = pg_strtok(&length); /* skip :fldname */ \
129  token = pg_strtok(&length); /* get field value */ \
130  (void) token; /* in case not used elsewhere */ \
131  local_node->fldname = -1 /* set field to "unknown" */
132 #endif
133 
134 /* Read a Node field */
135 #define READ_NODE_FIELD(fldname) \
136  token = pg_strtok(&length); /* skip :fldname */ \
137  (void) token; /* in case not used elsewhere */ \
138  local_node->fldname = nodeRead(NULL, 0)
139 
140 /* Read a bitmapset field */
141 #define READ_BITMAPSET_FIELD(fldname) \
142  token = pg_strtok(&length); /* skip :fldname */ \
143  (void) token; /* in case not used elsewhere */ \
144  local_node->fldname = _readBitmapset()
145 
146 /* Read an attribute number array */
147 #define READ_ATTRNUMBER_ARRAY(fldname, len) \
148  token = pg_strtok(&length); /* skip :fldname */ \
149  local_node->fldname = readAttrNumberCols(len)
150 
151 /* Read an oid array */
152 #define READ_OID_ARRAY(fldname, len) \
153  token = pg_strtok(&length); /* skip :fldname */ \
154  local_node->fldname = readOidCols(len)
155 
156 /* Read an int array */
157 #define READ_INT_ARRAY(fldname, len) \
158  token = pg_strtok(&length); /* skip :fldname */ \
159  local_node->fldname = readIntCols(len)
160 
161 /* Read a bool array */
162 #define READ_BOOL_ARRAY(fldname, len) \
163  token = pg_strtok(&length); /* skip :fldname */ \
164  local_node->fldname = readBoolCols(len)
165 
166 /* Routine exit */
167 #define READ_DONE() \
168  return local_node
169 
170 
171 /*
172  * NOTE: use atoi() to read values written with %d, or atoui() to read
173  * values written with %u in outfuncs.c. An exception is OID values,
174  * for which use atooid(). (As of 7.1, outfuncs.c writes OIDs as %u,
175  * but this will probably change in the future.)
176  */
177 #define atoui(x) ((unsigned int) strtoul((x), NULL, 10))
178 
179 #define strtobool(x) ((*(x) == 't') ? true : false)
180 
181 static char *
182 nullable_string(const char *token, int length)
183 {
184  /* outToken emits <> for NULL, and pg_strtok makes that an empty string */
185  if (length == 0)
186  return NULL;
187  /* outToken emits "" for empty string */
188  if (length == 2 && token[0] == '"' && token[1] == '"')
189  return pstrdup("");
190  /* otherwise, we must remove protective backslashes added by outToken */
191  return debackslash(token, length);
192 }
193 
194 
195 /*
196  * _readBitmapset
197  *
198  * Note: this code is used in contexts where we know that a Bitmapset
199  * is expected. There is equivalent code in nodeRead() that can read a
200  * Bitmapset when we come across one in other contexts.
201  */
202 static Bitmapset *
204 {
205  Bitmapset *result = NULL;
206 
208 
209  token = pg_strtok(&length);
210  if (token == NULL)
211  elog(ERROR, "incomplete Bitmapset structure");
212  if (length != 1 || token[0] != '(')
213  elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
214 
215  token = pg_strtok(&length);
216  if (token == NULL)
217  elog(ERROR, "incomplete Bitmapset structure");
218  if (length != 1 || token[0] != 'b')
219  elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
220 
221  for (;;)
222  {
223  int val;
224  char *endptr;
225 
226  token = pg_strtok(&length);
227  if (token == NULL)
228  elog(ERROR, "unterminated Bitmapset structure");
229  if (length == 1 && token[0] == ')')
230  break;
231  val = (int) strtol(token, &endptr, 10);
232  if (endptr != token + length)
233  elog(ERROR, "unrecognized integer: \"%.*s\"", length, token);
234  result = bms_add_member(result, val);
235  }
236 
237  return result;
238 }
239 
240 /*
241  * We export this function for use by extensions that define extensible nodes.
242  * That's somewhat historical, though, because calling nodeRead() will work.
243  */
244 Bitmapset *
246 {
247  return _readBitmapset();
248 }
249 
250 #include "readfuncs.funcs.c"
251 
252 
253 /*
254  * Support functions for nodes with custom_read_write attribute or
255  * special_read_write attribute
256  */
257 
258 static Const *
260 {
262 
263  READ_OID_FIELD(consttype);
264  READ_INT_FIELD(consttypmod);
265  READ_OID_FIELD(constcollid);
266  READ_INT_FIELD(constlen);
267  READ_BOOL_FIELD(constbyval);
268  READ_BOOL_FIELD(constisnull);
269  READ_LOCATION_FIELD(location);
270 
271  token = pg_strtok(&length); /* skip :constvalue */
272  if (local_node->constisnull)
273  token = pg_strtok(&length); /* skip "<>" */
274  else
275  local_node->constvalue = readDatum(local_node->constbyval);
276 
277  READ_DONE();
278 }
279 
280 static BoolExpr *
282 {
284 
285  /* do-it-yourself enum representation */
286  token = pg_strtok(&length); /* skip :boolop */
287  token = pg_strtok(&length); /* get field value */
288  if (length == 3 && strncmp(token, "and", 3) == 0)
289  local_node->boolop = AND_EXPR;
290  else if (length == 2 && strncmp(token, "or", 2) == 0)
291  local_node->boolop = OR_EXPR;
292  else if (length == 3 && strncmp(token, "not", 3) == 0)
293  local_node->boolop = NOT_EXPR;
294  else
295  elog(ERROR, "unrecognized boolop \"%.*s\"", length, token);
296 
298  READ_LOCATION_FIELD(location);
299 
300  READ_DONE();
301 }
302 
303 static A_Const *
305 {
307 
308  /* We expect either NULL or :val here */
309  token = pg_strtok(&length);
310  if (length == 4 && strncmp(token, "NULL", 4) == 0)
311  local_node->isnull = true;
312  else
313  {
314  union ValUnion *tmp = nodeRead(NULL, 0);
315 
316  /* To forestall valgrind complaints, copy only the valid data */
317  switch (nodeTag(tmp))
318  {
319  case T_Integer:
320  memcpy(&local_node->val, tmp, sizeof(Integer));
321  break;
322  case T_Float:
323  memcpy(&local_node->val, tmp, sizeof(Float));
324  break;
325  case T_Boolean:
326  memcpy(&local_node->val, tmp, sizeof(Boolean));
327  break;
328  case T_String:
329  memcpy(&local_node->val, tmp, sizeof(String));
330  break;
331  case T_BitString:
332  memcpy(&local_node->val, tmp, sizeof(BitString));
333  break;
334  default:
335  elog(ERROR, "unrecognized node type: %d",
336  (int) nodeTag(tmp));
337  break;
338  }
339  }
340 
341  READ_LOCATION_FIELD(location);
342 
343  READ_DONE();
344 }
345 
346 /*
347  * _readConstraint
348  */
349 static Constraint *
351 {
353 
354  READ_STRING_FIELD(conname);
355  READ_BOOL_FIELD(deferrable);
356  READ_BOOL_FIELD(initdeferred);
357  READ_LOCATION_FIELD(location);
358 
359  token = pg_strtok(&length); /* skip :contype */
360  token = pg_strtok(&length); /* get field value */
361  if (length == 4 && strncmp(token, "NULL", 4) == 0)
362  local_node->contype = CONSTR_NULL;
363  else if (length == 8 && strncmp(token, "NOT_NULL", 8) == 0)
364  local_node->contype = CONSTR_NOTNULL;
365  else if (length == 7 && strncmp(token, "DEFAULT", 7) == 0)
366  local_node->contype = CONSTR_DEFAULT;
367  else if (length == 8 && strncmp(token, "IDENTITY", 8) == 0)
368  local_node->contype = CONSTR_IDENTITY;
369  else if (length == 9 && strncmp(token, "GENERATED", 9) == 0)
370  local_node->contype = CONSTR_GENERATED;
371  else if (length == 5 && strncmp(token, "CHECK", 5) == 0)
372  local_node->contype = CONSTR_CHECK;
373  else if (length == 11 && strncmp(token, "PRIMARY_KEY", 11) == 0)
374  local_node->contype = CONSTR_PRIMARY;
375  else if (length == 6 && strncmp(token, "UNIQUE", 6) == 0)
376  local_node->contype = CONSTR_UNIQUE;
377  else if (length == 9 && strncmp(token, "EXCLUSION", 9) == 0)
378  local_node->contype = CONSTR_EXCLUSION;
379  else if (length == 11 && strncmp(token, "FOREIGN_KEY", 11) == 0)
380  local_node->contype = CONSTR_FOREIGN;
381  else if (length == 15 && strncmp(token, "ATTR_DEFERRABLE", 15) == 0)
382  local_node->contype = CONSTR_ATTR_DEFERRABLE;
383  else if (length == 19 && strncmp(token, "ATTR_NOT_DEFERRABLE", 19) == 0)
384  local_node->contype = CONSTR_ATTR_NOT_DEFERRABLE;
385  else if (length == 13 && strncmp(token, "ATTR_DEFERRED", 13) == 0)
386  local_node->contype = CONSTR_ATTR_DEFERRED;
387  else if (length == 14 && strncmp(token, "ATTR_IMMEDIATE", 14) == 0)
388  local_node->contype = CONSTR_ATTR_IMMEDIATE;
389 
390  switch (local_node->contype)
391  {
392  case CONSTR_NULL:
393  case CONSTR_NOTNULL:
394  /* no extra fields */
395  break;
396 
397  case CONSTR_DEFAULT:
398  READ_NODE_FIELD(raw_expr);
399  READ_STRING_FIELD(cooked_expr);
400  break;
401 
402  case CONSTR_IDENTITY:
404  READ_CHAR_FIELD(generated_when);
405  break;
406 
407  case CONSTR_GENERATED:
408  READ_NODE_FIELD(raw_expr);
409  READ_STRING_FIELD(cooked_expr);
410  READ_CHAR_FIELD(generated_when);
411  break;
412 
413  case CONSTR_CHECK:
414  READ_BOOL_FIELD(is_no_inherit);
415  READ_NODE_FIELD(raw_expr);
416  READ_STRING_FIELD(cooked_expr);
417  READ_BOOL_FIELD(skip_validation);
418  READ_BOOL_FIELD(initially_valid);
419  break;
420 
421  case CONSTR_PRIMARY:
422  READ_NODE_FIELD(keys);
423  READ_NODE_FIELD(including);
425  READ_STRING_FIELD(indexname);
426  READ_STRING_FIELD(indexspace);
427  READ_BOOL_FIELD(reset_default_tblspc);
428  /* access_method and where_clause not currently used */
429  break;
430 
431  case CONSTR_UNIQUE:
432  READ_BOOL_FIELD(nulls_not_distinct);
433  READ_NODE_FIELD(keys);
434  READ_NODE_FIELD(including);
436  READ_STRING_FIELD(indexname);
437  READ_STRING_FIELD(indexspace);
438  READ_BOOL_FIELD(reset_default_tblspc);
439  /* access_method and where_clause not currently used */
440  break;
441 
442  case CONSTR_EXCLUSION:
443  READ_NODE_FIELD(exclusions);
444  READ_NODE_FIELD(including);
446  READ_STRING_FIELD(indexname);
447  READ_STRING_FIELD(indexspace);
448  READ_BOOL_FIELD(reset_default_tblspc);
449  READ_STRING_FIELD(access_method);
450  READ_NODE_FIELD(where_clause);
451  break;
452 
453  case CONSTR_FOREIGN:
454  READ_NODE_FIELD(pktable);
455  READ_NODE_FIELD(fk_attrs);
456  READ_NODE_FIELD(pk_attrs);
457  READ_CHAR_FIELD(fk_matchtype);
458  READ_CHAR_FIELD(fk_upd_action);
459  READ_CHAR_FIELD(fk_del_action);
460  READ_NODE_FIELD(fk_del_set_cols);
461  READ_NODE_FIELD(old_conpfeqop);
462  READ_OID_FIELD(old_pktable_oid);
463  READ_BOOL_FIELD(skip_validation);
464  READ_BOOL_FIELD(initially_valid);
465  break;
466 
471  /* no extra fields */
472  break;
473 
474  default:
475  elog(ERROR, "unrecognized ConstrType: %d", (int) local_node->contype);
476  break;
477  }
478 
479  READ_DONE();
480 }
481 
482 static RangeTblEntry *
484 {
486 
487  /* put alias + eref first to make dump more legible */
488  READ_NODE_FIELD(alias);
489  READ_NODE_FIELD(eref);
490  READ_ENUM_FIELD(rtekind, RTEKind);
491 
492  switch (local_node->rtekind)
493  {
494  case RTE_RELATION:
495  READ_OID_FIELD(relid);
496  READ_CHAR_FIELD(relkind);
497  READ_INT_FIELD(rellockmode);
498  READ_NODE_FIELD(tablesample);
499  READ_UINT_FIELD(perminfoindex);
500  break;
501  case RTE_SUBQUERY:
502  READ_NODE_FIELD(subquery);
503  READ_BOOL_FIELD(security_barrier);
504  /* we re-use these RELATION fields, too: */
505  READ_OID_FIELD(relid);
506  READ_INT_FIELD(rellockmode);
507  READ_UINT_FIELD(perminfoindex);
508  break;
509  case RTE_JOIN:
510  READ_ENUM_FIELD(jointype, JoinType);
511  READ_INT_FIELD(joinmergedcols);
512  READ_NODE_FIELD(joinaliasvars);
513  READ_NODE_FIELD(joinleftcols);
514  READ_NODE_FIELD(joinrightcols);
515  READ_NODE_FIELD(join_using_alias);
516  break;
517  case RTE_FUNCTION:
519  READ_BOOL_FIELD(funcordinality);
520  break;
521  case RTE_TABLEFUNC:
522  READ_NODE_FIELD(tablefunc);
523  /* The RTE must have a copy of the column type info, if any */
524  if (local_node->tablefunc)
525  {
526  TableFunc *tf = local_node->tablefunc;
527 
528  local_node->coltypes = tf->coltypes;
529  local_node->coltypmods = tf->coltypmods;
530  local_node->colcollations = tf->colcollations;
531  }
532  break;
533  case RTE_VALUES:
534  READ_NODE_FIELD(values_lists);
535  READ_NODE_FIELD(coltypes);
536  READ_NODE_FIELD(coltypmods);
537  READ_NODE_FIELD(colcollations);
538  break;
539  case RTE_CTE:
540  READ_STRING_FIELD(ctename);
541  READ_UINT_FIELD(ctelevelsup);
542  READ_BOOL_FIELD(self_reference);
543  READ_NODE_FIELD(coltypes);
544  READ_NODE_FIELD(coltypmods);
545  READ_NODE_FIELD(colcollations);
546  break;
547  case RTE_NAMEDTUPLESTORE:
548  READ_STRING_FIELD(enrname);
549  READ_FLOAT_FIELD(enrtuples);
550  READ_NODE_FIELD(coltypes);
551  READ_NODE_FIELD(coltypmods);
552  READ_NODE_FIELD(colcollations);
553  /* we re-use these RELATION fields, too: */
554  READ_OID_FIELD(relid);
555  break;
556  case RTE_RESULT:
557  /* no extra fields */
558  break;
559  default:
560  elog(ERROR, "unrecognized RTE kind: %d",
561  (int) local_node->rtekind);
562  break;
563  }
564 
565  READ_BOOL_FIELD(lateral);
566  READ_BOOL_FIELD(inh);
567  READ_BOOL_FIELD(inFromCl);
568  READ_NODE_FIELD(securityQuals);
569 
570  READ_DONE();
571 }
572 
573 static A_Expr *
575 {
577 
578  token = pg_strtok(&length);
579 
580  if (length == 3 && strncmp(token, "ANY", 3) == 0)
581  {
582  local_node->kind = AEXPR_OP_ANY;
584  }
585  else if (length == 3 && strncmp(token, "ALL", 3) == 0)
586  {
587  local_node->kind = AEXPR_OP_ALL;
589  }
590  else if (length == 8 && strncmp(token, "DISTINCT", 8) == 0)
591  {
592  local_node->kind = AEXPR_DISTINCT;
594  }
595  else if (length == 12 && strncmp(token, "NOT_DISTINCT", 12) == 0)
596  {
597  local_node->kind = AEXPR_NOT_DISTINCT;
599  }
600  else if (length == 6 && strncmp(token, "NULLIF", 6) == 0)
601  {
602  local_node->kind = AEXPR_NULLIF;
604  }
605  else if (length == 2 && strncmp(token, "IN", 2) == 0)
606  {
607  local_node->kind = AEXPR_IN;
609  }
610  else if (length == 4 && strncmp(token, "LIKE", 4) == 0)
611  {
612  local_node->kind = AEXPR_LIKE;
614  }
615  else if (length == 5 && strncmp(token, "ILIKE", 5) == 0)
616  {
617  local_node->kind = AEXPR_ILIKE;
619  }
620  else if (length == 7 && strncmp(token, "SIMILAR", 7) == 0)
621  {
622  local_node->kind = AEXPR_SIMILAR;
624  }
625  else if (length == 7 && strncmp(token, "BETWEEN", 7) == 0)
626  {
627  local_node->kind = AEXPR_BETWEEN;
629  }
630  else if (length == 11 && strncmp(token, "NOT_BETWEEN", 11) == 0)
631  {
632  local_node->kind = AEXPR_NOT_BETWEEN;
634  }
635  else if (length == 11 && strncmp(token, "BETWEEN_SYM", 11) == 0)
636  {
637  local_node->kind = AEXPR_BETWEEN_SYM;
639  }
640  else if (length == 15 && strncmp(token, "NOT_BETWEEN_SYM", 15) == 0)
641  {
642  local_node->kind = AEXPR_NOT_BETWEEN_SYM;
644  }
645  else if (length == 5 && strncmp(token, ":name", 5) == 0)
646  {
647  local_node->kind = AEXPR_OP;
648  local_node->name = nodeRead(NULL, 0);
649  }
650  else
651  elog(ERROR, "unrecognized A_Expr kind: \"%.*s\"", length, token);
652 
653  READ_NODE_FIELD(lexpr);
654  READ_NODE_FIELD(rexpr);
655  READ_LOCATION_FIELD(location);
656 
657  READ_DONE();
658 }
659 
660 static ExtensibleNode *
662 {
663  const ExtensibleNodeMethods *methods;
664  ExtensibleNode *local_node;
665  const char *extnodename;
666 
668 
669  token = pg_strtok(&length); /* skip :extnodename */
670  token = pg_strtok(&length); /* get extnodename */
671 
672  extnodename = nullable_string(token, length);
673  if (!extnodename)
674  elog(ERROR, "extnodename has to be supplied");
675  methods = GetExtensibleNodeMethods(extnodename, false);
676 
677  local_node = (ExtensibleNode *) newNode(methods->node_size,
678  T_ExtensibleNode);
679  local_node->extnodename = extnodename;
680 
681  /* deserialize the private fields */
682  methods->nodeRead(local_node);
683 
684  READ_DONE();
685 }
686 
687 
688 /*
689  * parseNodeString
690  *
691  * Given a character string representing a node tree, parseNodeString creates
692  * the internal node structure.
693  *
694  * The string to be read must already have been loaded into pg_strtok().
695  */
696 Node *
698 {
699  void *return_value;
700 
702 
703  /* Guard against stack overflow due to overly complex expressions */
705 
706  token = pg_strtok(&length);
707 
708 #define MATCH(tokname, namelen) \
709  (length == namelen && memcmp(token, tokname, namelen) == 0)
710 
711  if (false)
712  ;
713 #include "readfuncs.switch.c"
714  else
715  {
716  elog(ERROR, "badly formatted node string \"%.32s\"...", token);
717  return_value = NULL; /* keep compiler quiet */
718  }
719 
720  return (Node *) return_value;
721 }
722 
723 
724 /*
725  * readDatum
726  *
727  * Given a string representation of a constant, recreate the appropriate
728  * Datum. The string representation embeds length info, but not byValue,
729  * so we must be told that.
730  */
731 Datum
732 readDatum(bool typbyval)
733 {
734  Size length,
735  i;
736  int tokenLength;
737  const char *token;
738  Datum res;
739  char *s;
740 
741  /*
742  * read the actual length of the value
743  */
744  token = pg_strtok(&tokenLength);
745  length = atoui(token);
746 
747  token = pg_strtok(&tokenLength); /* read the '[' */
748  if (token == NULL || token[0] != '[')
749  elog(ERROR, "expected \"[\" to start datum, but got \"%s\"; length = %zu",
750  token ? token : "[NULL]", length);
751 
752  if (typbyval)
753  {
754  if (length > (Size) sizeof(Datum))
755  elog(ERROR, "byval datum but length = %zu", length);
756  res = (Datum) 0;
757  s = (char *) (&res);
758  for (i = 0; i < (Size) sizeof(Datum); i++)
759  {
760  token = pg_strtok(&tokenLength);
761  s[i] = (char) atoi(token);
762  }
763  }
764  else if (length <= 0)
765  res = (Datum) NULL;
766  else
767  {
768  s = (char *) palloc(length);
769  for (i = 0; i < length; i++)
770  {
771  token = pg_strtok(&tokenLength);
772  s[i] = (char) atoi(token);
773  }
774  res = PointerGetDatum(s);
775  }
776 
777  token = pg_strtok(&tokenLength); /* read the ']' */
778  if (token == NULL || token[0] != ']')
779  elog(ERROR, "expected \"]\" to end datum, but got \"%s\"; length = %zu",
780  token ? token : "[NULL]", length);
781 
782  return res;
783 }
784 
785 /*
786  * common implementation for scalar-array-reading functions
787  *
788  * The data format is either "<>" for a NULL pointer (in which case numCols
789  * is ignored) or "(item item item)" where the number of items must equal
790  * numCols. The convfunc must be okay with stopping at whitespace or a
791  * right parenthesis, since pg_strtok won't null-terminate the token.
792  */
793 #define READ_SCALAR_ARRAY(fnname, datatype, convfunc) \
794 datatype * \
795 fnname(int numCols) \
796 { \
797  datatype *vals; \
798  READ_TEMP_LOCALS(); \
799  token = pg_strtok(&length); \
800  if (token == NULL) \
801  elog(ERROR, "incomplete scalar array"); \
802  if (length == 0) \
803  return NULL; /* it was "<>", so return NULL pointer */ \
804  if (length != 1 || token[0] != '(') \
805  elog(ERROR, "unrecognized token: \"%.*s\"", length, token); \
806  vals = (datatype *) palloc(numCols * sizeof(datatype)); \
807  for (int i = 0; i < numCols; i++) \
808  { \
809  token = pg_strtok(&length); \
810  if (token == NULL || token[0] == ')') \
811  elog(ERROR, "incomplete scalar array"); \
812  vals[i] = convfunc(token); \
813  } \
814  token = pg_strtok(&length); \
815  if (token == NULL || length != 1 || token[0] != ')') \
816  elog(ERROR, "incomplete scalar array"); \
817  return vals; \
818 }
819 
820 /*
821  * Note: these functions are exported in nodes.h for possible use by
822  * extensions, so don't mess too much with their names or API.
823  */
826 /* outfuncs.c has writeIndexCols, but we don't yet need that here */
827 /* READ_SCALAR_ARRAY(readIndexCols, Index, atoui) */
828 READ_SCALAR_ARRAY(readIntCols, int, atoi)
Bitmapset * bms_add_member(Bitmapset *a, int x)
Definition: bitmapset.c:755
signed short int16
Definition: c.h:477
size_t Size
Definition: c.h:589
#define ERROR
Definition: elog.h:39
const char * name
Definition: encode.c:571
const ExtensibleNodeMethods * GetExtensibleNodeMethods(const char *extnodename, bool missing_ok)
Definition: extensible.c:125
#define token
Definition: indent_globs.h:126
long val
Definition: informix.c:664
int i
Definition: isn.c:73
char * pstrdup(const char *in)
Definition: mcxt.c:1624
void * palloc(Size size)
Definition: mcxt.c:1210
#define newNode(size, tag)
Definition: nodes.h:166
#define nodeTag(nodeptr)
Definition: nodes.h:133
int * readIntCols(int numCols)
bool * readBoolCols(int numCols)
Oid * readOidCols(int numCols)
int16 * readAttrNumberCols(int numCols)
JoinType
Definition: nodes.h:299
@ AEXPR_BETWEEN
Definition: parsenodes.h:321
@ AEXPR_NULLIF
Definition: parsenodes.h:316
@ AEXPR_NOT_DISTINCT
Definition: parsenodes.h:315
@ AEXPR_BETWEEN_SYM
Definition: parsenodes.h:323
@ AEXPR_NOT_BETWEEN_SYM
Definition: parsenodes.h:324
@ AEXPR_ILIKE
Definition: parsenodes.h:319
@ AEXPR_IN
Definition: parsenodes.h:317
@ AEXPR_NOT_BETWEEN
Definition: parsenodes.h:322
@ AEXPR_DISTINCT
Definition: parsenodes.h:314
@ AEXPR_SIMILAR
Definition: parsenodes.h:320
@ AEXPR_LIKE
Definition: parsenodes.h:318
@ AEXPR_OP
Definition: parsenodes.h:311
@ AEXPR_OP_ANY
Definition: parsenodes.h:312
@ AEXPR_OP_ALL
Definition: parsenodes.h:313
RTEKind
Definition: parsenodes.h:1013
@ RTE_JOIN
Definition: parsenodes.h:1016
@ RTE_CTE
Definition: parsenodes.h:1020
@ RTE_NAMEDTUPLESTORE
Definition: parsenodes.h:1021
@ RTE_VALUES
Definition: parsenodes.h:1019
@ RTE_SUBQUERY
Definition: parsenodes.h:1015
@ RTE_RESULT
Definition: parsenodes.h:1022
@ RTE_FUNCTION
Definition: parsenodes.h:1017
@ RTE_TABLEFUNC
Definition: parsenodes.h:1018
@ RTE_RELATION
Definition: parsenodes.h:1014
@ CONSTR_FOREIGN
Definition: parsenodes.h:2426
@ CONSTR_ATTR_DEFERRED
Definition: parsenodes.h:2429
@ CONSTR_IDENTITY
Definition: parsenodes.h:2420
@ CONSTR_UNIQUE
Definition: parsenodes.h:2424
@ CONSTR_ATTR_NOT_DEFERRABLE
Definition: parsenodes.h:2428
@ CONSTR_DEFAULT
Definition: parsenodes.h:2419
@ CONSTR_NOTNULL
Definition: parsenodes.h:2418
@ CONSTR_ATTR_IMMEDIATE
Definition: parsenodes.h:2430
@ CONSTR_CHECK
Definition: parsenodes.h:2422
@ CONSTR_NULL
Definition: parsenodes.h:2416
@ CONSTR_GENERATED
Definition: parsenodes.h:2421
@ CONSTR_EXCLUSION
Definition: parsenodes.h:2425
@ CONSTR_ATTR_DEFERRABLE
Definition: parsenodes.h:2427
@ CONSTR_PRIMARY
Definition: parsenodes.h:2423
void check_stack_depth(void)
Definition: postgres.c:3461
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:322
uintptr_t Datum
Definition: postgres.h:64
unsigned int Oid
Definition: postgres_ext.h:31
#define atooid(x)
Definition: postgres_ext.h:42
@ AND_EXPR
Definition: primnodes.h:858
@ OR_EXPR
Definition: primnodes.h:858
@ NOT_EXPR
Definition: primnodes.h:858
const char * pg_strtok(int *length)
Definition: read.c:153
char * debackslash(const char *token, int length)
Definition: read.c:214
void * nodeRead(const char *token, int tok_len)
Definition: read.c:320
static A_Const * _readA_Const(void)
Definition: readfuncs.c:304
#define READ_INT_FIELD(fldname)
Definition: readfuncs.c:60
static char * nullable_string(const char *token, int length)
Definition: readfuncs.c:182
#define READ_UINT_FIELD(fldname)
Definition: readfuncs.c:66
#define READ_NODE_FIELD(fldname)
Definition: readfuncs.c:135
static RangeTblEntry * _readRangeTblEntry(void)
Definition: readfuncs.c:483
#define READ_CHAR_FIELD(fldname)
Definition: readfuncs.c:90
static Constraint * _readConstraint(void)
Definition: readfuncs.c:350
#define READ_SCALAR_ARRAY(fnname, datatype, convfunc)
Definition: readfuncs.c:793
Node * parseNodeString(void)
Definition: readfuncs.c:697
static Const * _readConst(void)
Definition: readfuncs.c:259
#define READ_OID_FIELD(fldname)
Definition: readfuncs.c:84
#define READ_LOCATION_FIELD(fldname)
Definition: readfuncs.c:127
#define READ_STRING_FIELD(fldname)
Definition: readfuncs.c:115
static BoolExpr * _readBoolExpr(void)
Definition: readfuncs.c:281
#define READ_FLOAT_FIELD(fldname)
Definition: readfuncs.c:103
Datum readDatum(bool typbyval)
Definition: readfuncs.c:732
static ExtensibleNode * _readExtensibleNode(void)
Definition: readfuncs.c:661
#define READ_BOOL_FIELD(fldname)
Definition: readfuncs.c:109
Bitmapset * readBitmapset(void)
Definition: readfuncs.c:245
#define atoui(x)
Definition: readfuncs.c:177
#define strtobool(x)
Definition: readfuncs.c:179
static A_Expr * _readA_Expr(void)
Definition: readfuncs.c:574
static Bitmapset * _readBitmapset(void)
Definition: readfuncs.c:203
#define READ_ENUM_FIELD(fldname, enumtype)
Definition: readfuncs.c:97
#define READ_TEMP_LOCALS()
Definition: readfuncs.c:50
#define READ_LOCALS(nodeTypeName)
Definition: readfuncs.c:55
#define READ_DONE()
Definition: readfuncs.c:167
static const struct fns functions
Definition: regcomp.c:357
Definition: value.h:56
void(* nodeRead)(struct ExtensibleNode *node)
Definition: extensible.h:72
const char * extnodename
Definition: extensible.h:37
Definition: value.h:48
Definition: value.h:29
Definition: nodes.h:129
Definition: value.h:64