PostgreSQL Source Code git master
Loading...
Searching...
No Matches
readfuncs.c File Reference
#include "postgres.h"
#include "miscadmin.h"
#include "nodes/bitmapset.h"
#include "nodes/readfuncs.h"
#include "readfuncs.funcs.c"
#include "readfuncs.switch.c"
Include dependency graph for readfuncs.c:

Go to the source code of this file.

Macros

#define READ_LOCALS_NO_FIELDS(nodeTypeName)    nodeTypeName *local_node = makeNode(nodeTypeName)
 
#define READ_TEMP_LOCALS()
 
#define READ_LOCALS(nodeTypeName)
 
#define READ_INT_FIELD(fldname)
 
#define READ_UINT_FIELD(fldname)
 
#define READ_INT64_FIELD(fldname)
 
#define READ_UINT64_FIELD(fldname)
 
#define READ_LONG_FIELD(fldname)
 
#define READ_OID_FIELD(fldname)
 
#define READ_CHAR_FIELD(fldname)
 
#define READ_ENUM_FIELD(fldname, enumtype)
 
#define READ_FLOAT_FIELD(fldname)
 
#define READ_BOOL_FIELD(fldname)
 
#define READ_STRING_FIELD(fldname)
 
#define READ_LOCATION_FIELD(fldname)
 
#define READ_NODE_FIELD(fldname)
 
#define READ_BITMAPSET_FIELD(fldname)
 
#define READ_ATTRNUMBER_ARRAY(fldname, len)
 
#define READ_OID_ARRAY(fldname, len)
 
#define READ_INT_ARRAY(fldname, len)
 
#define READ_BOOL_ARRAY(fldname, len)
 
#define READ_DONE()    return local_node
 
#define atoui(x)   ((unsigned int) strtoul((x), NULL, 10))
 
#define strtobool(x)   ((*(x) == 't') ? true : false)
 
#define MATCH(tokname, namelen)    (length == namelen && memcmp(token, tokname, namelen) == 0)
 
#define READ_SCALAR_ARRAY(fnname, datatype, convfunc)
 

Functions

static charnullable_string (const char *token, int length)
 
static Bitmapset_readBitmapset (void)
 
BitmapsetreadBitmapset (void)
 
static Const_readConst (void)
 
static BoolExpr_readBoolExpr (void)
 
static A_Const_readA_Const (void)
 
static RangeTblEntry_readRangeTblEntry (void)
 
static A_Expr_readA_Expr (void)
 
static ExtensibleNode_readExtensibleNode (void)
 
NodeparseNodeString (void)
 
Datum readDatum (bool typbyval)
 

Macro Definition Documentation

◆ atoui

#define atoui (   x)    ((unsigned int) strtoul((x), NULL, 10))

Definition at line 181 of file readfuncs.c.

◆ MATCH

#define MATCH (   tokname,
  namelen 
)     (length == namelen && memcmp(token, tokname, namelen) == 0)

◆ READ_ATTRNUMBER_ARRAY

#define READ_ATTRNUMBER_ARRAY (   fldname,
  len 
)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
int16 * readAttrNumberCols(int numCols)
const void size_t len
static int fb(int x)
const char * pg_strtok(int *length)
Definition read.c:153

Definition at line 151 of file readfuncs.c.

182 : false)
183
184static char *
185nullable_string(const char *token, int length)
186{
187 /* outToken emits <> for NULL, and pg_strtok makes that an empty string */
188 if (length == 0)
189 return NULL;
190 /* outToken emits "" for empty string */
191 if (length == 2 && token[0] == '"' && token[1] == '"')
192 return pstrdup("");
193 /* otherwise, we must remove protective backslashes added by outToken */
194 return debackslash(token, length);
195}
196
197
198/*
199 * _readBitmapset
200 *
201 * Note: this code is used in contexts where we know that a Bitmapset
202 * is expected. There is equivalent code in nodeRead() that can read a
203 * Bitmapset when we come across one in other contexts.
204 */
205static Bitmapset *
206_readBitmapset(void)
207{
208 Bitmapset *result = NULL;
209
211
212 token = pg_strtok(&length);
213 if (token == NULL)
214 elog(ERROR, "incomplete Bitmapset structure");
215 if (length != 1 || token[0] != '(')
216 elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
217
218 token = pg_strtok(&length);
219 if (token == NULL)
220 elog(ERROR, "incomplete Bitmapset structure");
221 if (length != 1 || token[0] != 'b')
222 elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
223
224 for (;;)
225 {
226 int val;
227 char *endptr;
228
229 token = pg_strtok(&length);
230 if (token == NULL)
231 elog(ERROR, "unterminated Bitmapset structure");
232 if (length == 1 && token[0] == ')')
233 break;
234 val = (int) strtol(token, &endptr, 10);
235 if (endptr != token + length)
236 elog(ERROR, "unrecognized integer: \"%.*s\"", length, token);
237 result = bms_add_member(result, val);
238 }
239
240 return result;
241}
242
243/*
244 * We export this function for use by extensions that define extensible nodes.
245 * That's somewhat historical, though, because calling nodeRead() will work.
246 */
247Bitmapset *
248readBitmapset(void)
249{
250 return _readBitmapset();
251}
252
253#include "readfuncs.funcs.c"
254
255
256/*
257 * Support functions for nodes with custom_read_write attribute or
258 * special_read_write attribute
259 */
260
261static Const *
262_readConst(void)
263{
265
266 READ_OID_FIELD(consttype);
272 READ_LOCATION_FIELD(location);
273
274 token = pg_strtok(&length); /* skip :constvalue */
275 if (local_node->constisnull)
276 token = pg_strtok(&length); /* skip "<>" */
277 else
278 local_node->constvalue = readDatum(local_node->constbyval);
279
280 READ_DONE();
281}
282
283static BoolExpr *
284_readBoolExpr(void)
285{
287
288 /* do-it-yourself enum representation */
289 token = pg_strtok(&length); /* skip :boolop */
290 token = pg_strtok(&length); /* get field value */
291 if (length == 3 && strncmp(token, "and", 3) == 0)
292 local_node->boolop = AND_EXPR;
293 else if (length == 2 && strncmp(token, "or", 2) == 0)
294 local_node->boolop = OR_EXPR;
295 else if (length == 3 && strncmp(token, "not", 3) == 0)
296 local_node->boolop = NOT_EXPR;
297 else
298 elog(ERROR, "unrecognized boolop \"%.*s\"", length, token);
299
300 READ_NODE_FIELD(args);
301 READ_LOCATION_FIELD(location);
302
303 READ_DONE();
304}
305
306static A_Const *
307_readA_Const(void)
308{
310
311 /* We expect either NULL or :val here */
312 token = pg_strtok(&length);
313 if (length == 4 && strncmp(token, "NULL", 4) == 0)
314 local_node->isnull = true;
315 else
316 {
317 union ValUnion *tmp = nodeRead(NULL, 0);
318
319 /* To forestall valgrind complaints, copy only the valid data */
320 switch (nodeTag(tmp))
321 {
322 case T_Integer:
323 memcpy(&local_node->val, tmp, sizeof(Integer));
324 break;
325 case T_Float:
326 memcpy(&local_node->val, tmp, sizeof(Float));
327 break;
328 case T_Boolean:
329 memcpy(&local_node->val, tmp, sizeof(Boolean));
330 break;
331 case T_String:
332 memcpy(&local_node->val, tmp, sizeof(String));
333 break;
334 case T_BitString:
335 memcpy(&local_node->val, tmp, sizeof(BitString));
336 break;
337 default:
338 elog(ERROR, "unrecognized node type: %d",
339 (int) nodeTag(tmp));
340 break;
341 }
342 }
343
344 READ_LOCATION_FIELD(location);
345
346 READ_DONE();
347}
348
349static RangeTblEntry *
351{
353
354 READ_NODE_FIELD(alias);
356 READ_ENUM_FIELD(rtekind, RTEKind);
357
358 switch (local_node->rtekind)
359 {
360 case RTE_RELATION:
361 READ_OID_FIELD(relid);
362 READ_BOOL_FIELD(inh);
363 READ_CHAR_FIELD(relkind);
366 READ_NODE_FIELD(tablesample);
367 break;
368 case RTE_SUBQUERY:
369 READ_NODE_FIELD(subquery);
370 READ_BOOL_FIELD(security_barrier);
371 /* we re-use these RELATION fields, too: */
372 READ_OID_FIELD(relid);
373 READ_BOOL_FIELD(inh);
374 READ_CHAR_FIELD(relkind);
377 break;
378 case RTE_JOIN:
379 READ_ENUM_FIELD(jointype, JoinType);
385 break;
386 case RTE_FUNCTION:
388 READ_BOOL_FIELD(funcordinality);
389 break;
390 case RTE_TABLEFUNC:
391 READ_NODE_FIELD(tablefunc);
392 /* The RTE must have a copy of the column type info, if any */
393 if (local_node->tablefunc)
394 {
395 TableFunc *tf = local_node->tablefunc;
396
397 local_node->coltypes = tf->coltypes;
398 local_node->coltypmods = tf->coltypmods;
399 local_node->colcollations = tf->colcollations;
400 }
401 break;
402 case RTE_VALUES:
403 READ_NODE_FIELD(values_lists);
407 break;
408 case RTE_CTE:
409 READ_STRING_FIELD(ctename);
410 READ_UINT_FIELD(ctelevelsup);
415 break;
417 READ_STRING_FIELD(enrname);
418 READ_FLOAT_FIELD(enrtuples);
422 /* we re-use these RELATION fields, too: */
423 READ_OID_FIELD(relid);
424 break;
425 case RTE_RESULT:
426 /* no extra fields */
427 break;
428 case RTE_GROUP:
429 READ_NODE_FIELD(groupexprs);
430 break;
431 default:
432 elog(ERROR, "unrecognized RTE kind: %d",
433 (int) local_node->rtekind);
434 break;
435 }
436
437 READ_BOOL_FIELD(lateral);
440
441 READ_DONE();
442}
443
444static A_Expr *
445_readA_Expr(void)
446{
448
449 token = pg_strtok(&length);
450
451 if (length == 3 && strncmp(token, "ANY", 3) == 0)
452 {
453 local_node->kind = AEXPR_OP_ANY;
455 }
456 else if (length == 3 && strncmp(token, "ALL", 3) == 0)
457 {
458 local_node->kind = AEXPR_OP_ALL;
460 }
461 else if (length == 8 && strncmp(token, "DISTINCT", 8) == 0)
462 {
465 }
466 else if (length == 12 && strncmp(token, "NOT_DISTINCT", 12) == 0)
467 {
470 }
471 else if (length == 6 && strncmp(token, "NULLIF", 6) == 0)
472 {
473 local_node->kind = AEXPR_NULLIF;
475 }
476 else if (length == 2 && strncmp(token, "IN", 2) == 0)
477 {
478 local_node->kind = AEXPR_IN;
480 }
481 else if (length == 4 && strncmp(token, "LIKE", 4) == 0)
482 {
483 local_node->kind = AEXPR_LIKE;
485 }
486 else if (length == 5 && strncmp(token, "ILIKE", 5) == 0)
487 {
488 local_node->kind = AEXPR_ILIKE;
490 }
491 else if (length == 7 && strncmp(token, "SIMILAR", 7) == 0)
492 {
495 }
496 else if (length == 7 && strncmp(token, "BETWEEN", 7) == 0)
497 {
500 }
501 else if (length == 11 && strncmp(token, "NOT_BETWEEN", 11) == 0)
502 {
505 }
506 else if (length == 11 && strncmp(token, "BETWEEN_SYM", 11) == 0)
507 {
510 }
511 else if (length == 15 && strncmp(token, "NOT_BETWEEN_SYM", 15) == 0)
512 {
515 }
516 else if (length == 5 && strncmp(token, ":name", 5) == 0)
517 {
518 local_node->kind = AEXPR_OP;
519 local_node->name = nodeRead(NULL, 0);
520 }
521 else
522 elog(ERROR, "unrecognized A_Expr kind: \"%.*s\"", length, token);
523
524 READ_NODE_FIELD(lexpr);
525 READ_NODE_FIELD(rexpr);
526 READ_LOCATION_FIELD(rexpr_list_start);
527 READ_LOCATION_FIELD(rexpr_list_end);
528 READ_LOCATION_FIELD(location);
529
530 READ_DONE();
531}
532
533static ExtensibleNode *
535{
536 const ExtensibleNodeMethods *methods;
538 const char *extnodename;
539
541
542 token = pg_strtok(&length); /* skip :extnodename */
543 token = pg_strtok(&length); /* get extnodename */
544
545 extnodename = nullable_string(token, length);
546 if (!extnodename)
547 elog(ERROR, "extnodename has to be supplied");
548 methods = GetExtensibleNodeMethods(extnodename, false);
549
552 local_node->extnodename = extnodename;
553
554 /* deserialize the private fields */
555 methods->nodeRead(local_node);
556
557 READ_DONE();
558}
559
560
561/*
562 * parseNodeString
563 *
564 * Given a character string representing a node tree, parseNodeString creates
565 * the internal node structure.
566 *
567 * The string to be read must already have been loaded into pg_strtok().
568 */
569Node *
570parseNodeString(void)
571{
573
574 /* Guard against stack overflow due to overly complex expressions */
576
577 token = pg_strtok(&length);
578
579#define MATCH(tokname, namelen) \
580 (length == namelen && memcmp(token, tokname, namelen) == 0)
581
582#include "readfuncs.switch.c"
583
584 elog(ERROR, "badly formatted node string \"%.32s\"...", token);
585 return NULL; /* keep compiler quiet */
586}
587
588
589/*
590 * readDatum
591 *
592 * Given a string representation of a constant, recreate the appropriate
593 * Datum. The string representation embeds length info, but not byValue,
594 * so we must be told that.
595 */
596Datum
597readDatum(bool typbyval)
598{
599 Size length;
600 int tokenLength;
601 const char *token;
602 Datum res;
603 char *s;
604
605 /*
606 * read the actual length of the value
607 */
609 length = atoui(token);
610
611 token = pg_strtok(&tokenLength); /* read the '[' */
612 if (token == NULL || token[0] != '[')
613 elog(ERROR, "expected \"[\" to start datum, but got \"%s\"; length = %zu",
614 token ? token : "[NULL]", length);
615
616 if (typbyval)
617 {
618 if (length > (Size) sizeof(Datum))
619 elog(ERROR, "byval datum but length = %zu", length);
620 res = (Datum) 0;
621 s = (char *) (&res);
622 for (Size i = 0; i < (Size) sizeof(Datum); i++)
623 {
625 s[i] = (char) atoi(token);
626 }
627 }
628 else if (length <= 0)
629 res = (Datum) 0;
630 else
631 {
632 s = (char *) palloc(length);
633 for (Size i = 0; i < length; i++)
634 {
636 s[i] = (char) atoi(token);
637 }
638 res = PointerGetDatum(s);
639 }
640
641 token = pg_strtok(&tokenLength); /* read the ']' */
642 if (token == NULL || token[0] != ']')
643 elog(ERROR, "expected \"]\" to end datum, but got \"%s\"; length = %zu",
644 token ? token : "[NULL]", length);
645
646 return res;
647}
648
649/*
650 * common implementation for scalar-array-reading functions
651 *
652 * The data format is either "<>" for a NULL pointer (in which case numCols
653 * is ignored) or "(item item item)" where the number of items must equal
654 * numCols. The convfunc must be okay with stopping at whitespace or a
655 * right parenthesis, since pg_strtok won't null-terminate the token.
656 */
657#define READ_SCALAR_ARRAY(fnname, datatype, convfunc) \
658datatype * \
659fnname(int numCols) \
660{ \
661 datatype *vals; \
662 READ_TEMP_LOCALS(); \
663 token = pg_strtok(&length); \
664 if (token == NULL) \
665 elog(ERROR, "incomplete scalar array"); \
666 if (length == 0) \
667 return NULL; /* it was "<>", so return NULL pointer */ \
668 if (length != 1 || token[0] != '(') \
669 elog(ERROR, "unrecognized token: \"%.*s\"", length, token); \
670 vals = (datatype *) palloc(numCols * sizeof(datatype)); \
671 for (int i = 0; i < numCols; i++) \
672 { \
673 token = pg_strtok(&length); \
674 if (token == NULL || token[0] == ')') \
675 elog(ERROR, "incomplete scalar array"); \
676 vals[i] = convfunc(token); \
677 } \
678 token = pg_strtok(&length); \
679 if (token == NULL || length != 1 || token[0] != ')') \
680 elog(ERROR, "incomplete scalar array"); \
681 return vals; \
682}
683
684/*
685 * Note: these functions are exported in nodes.h for possible use by
686 * extensions, so don't mess too much with their names or API.
687 */
690/* outfuncs.c has writeIndexCols, but we don't yet need that here */
691/* READ_SCALAR_ARRAY(readIndexCols, Index, atoui) */
Bitmapset * bms_add_member(Bitmapset *a, int x)
Definition bitmapset.c:814
int16_t int16
Definition c.h:541
size_t Size
Definition c.h:619
#define ERROR
Definition elog.h:39
#define elog(elevel,...)
Definition elog.h:226
const ExtensibleNodeMethods * GetExtensibleNodeMethods(const char *extnodename, bool missing_ok)
Definition extensible.c:125
#define token
long val
Definition informix.c:689
int i
Definition isn.c:77
char * pstrdup(const char *in)
Definition mcxt.c:1781
void * palloc(Size size)
Definition mcxt.c:1387
#define nodeTag(nodeptr)
Definition nodes.h:139
int * readIntCols(int numCols)
static Node * newNode(size_t size, NodeTag tag)
Definition nodes.h:150
Oid * readOidCols(int numCols)
JoinType
Definition nodes.h:298
bool * readBoolCols(int numCols)
@ AEXPR_BETWEEN
Definition parsenodes.h:340
@ AEXPR_NULLIF
Definition parsenodes.h:335
@ AEXPR_NOT_DISTINCT
Definition parsenodes.h:334
@ AEXPR_BETWEEN_SYM
Definition parsenodes.h:342
@ AEXPR_NOT_BETWEEN_SYM
Definition parsenodes.h:343
@ AEXPR_ILIKE
Definition parsenodes.h:338
@ AEXPR_IN
Definition parsenodes.h:336
@ AEXPR_NOT_BETWEEN
Definition parsenodes.h:341
@ AEXPR_DISTINCT
Definition parsenodes.h:333
@ AEXPR_SIMILAR
Definition parsenodes.h:339
@ AEXPR_LIKE
Definition parsenodes.h:337
@ AEXPR_OP
Definition parsenodes.h:330
@ AEXPR_OP_ANY
Definition parsenodes.h:331
@ AEXPR_OP_ALL
Definition parsenodes.h:332
RTEKind
@ RTE_JOIN
@ RTE_CTE
@ RTE_NAMEDTUPLESTORE
@ RTE_VALUES
@ RTE_SUBQUERY
@ RTE_RESULT
@ RTE_FUNCTION
@ RTE_TABLEFUNC
@ RTE_GROUP
@ RTE_RELATION
static Datum PointerGetDatum(const void *X)
Definition postgres.h:352
uint64_t Datum
Definition postgres.h:70
unsigned int Oid
#define atooid(x)
@ AND_EXPR
Definition primnodes.h:963
@ OR_EXPR
Definition primnodes.h:963
@ NOT_EXPR
Definition primnodes.h:963
char * debackslash(const char *token, int length)
Definition read.c:214
void * nodeRead(const char *token, int tok_len)
Definition read.c:320
#define READ_INT_FIELD(fldname)
Definition readfuncs.c:58
#define READ_UINT_FIELD(fldname)
Definition readfuncs.c:64
#define READ_NODE_FIELD(fldname)
Definition readfuncs.c:139
static ExtensibleNode * _readExtensibleNode(void)
Definition readfuncs.c:535
#define READ_CHAR_FIELD(fldname)
Definition readfuncs.c:94
static char * nullable_string(const char *token, int length)
Definition readfuncs.c:186
#define READ_SCALAR_ARRAY(fnname, datatype, convfunc)
Definition readfuncs.c:658
Node * parseNodeString(void)
Definition readfuncs.c:571
#define READ_OID_FIELD(fldname)
Definition readfuncs.c:88
#define READ_LOCATION_FIELD(fldname)
Definition readfuncs.c:131
static Bitmapset * _readBitmapset(void)
Definition readfuncs.c:207
static A_Const * _readA_Const(void)
Definition readfuncs.c:308
static Const * _readConst(void)
Definition readfuncs.c:263
#define READ_STRING_FIELD(fldname)
Definition readfuncs.c:119
#define READ_FLOAT_FIELD(fldname)
Definition readfuncs.c:107
Datum readDatum(bool typbyval)
Definition readfuncs.c:598
#define READ_BOOL_FIELD(fldname)
Definition readfuncs.c:113
#define atoui(x)
Definition readfuncs.c:181
#define strtobool(x)
Definition readfuncs.c:183
static BoolExpr * _readBoolExpr(void)
Definition readfuncs.c:285
#define READ_ENUM_FIELD(fldname, enumtype)
Definition readfuncs.c:101
#define READ_TEMP_LOCALS()
Definition readfuncs.c:48
static A_Expr * _readA_Expr(void)
Definition readfuncs.c:446
Bitmapset * readBitmapset(void)
Definition readfuncs.c:249
#define READ_LOCALS(nodeTypeName)
Definition readfuncs.c:53
#define READ_DONE()
Definition readfuncs.c:171
static RangeTblEntry * _readRangeTblEntry(void)
Definition readfuncs.c:351
static const struct fns functions
Definition regcomp.c:358
void check_stack_depth(void)
Definition stack_depth.c:95
void(* nodeRead)(struct ExtensibleNode *node)
Definition extensible.h:72
Definition value.h:48
Definition nodes.h:135
Definition value.h:64
const char * name

◆ READ_BITMAPSET_FIELD

#define READ_BITMAPSET_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
(void) token; /* in case not used elsewhere */ \

Definition at line 145 of file readfuncs.c.

◆ READ_BOOL_ARRAY

#define READ_BOOL_ARRAY (   fldname,
  len 
)
Value:
token = pg_strtok(&length); /* skip :fldname */ \

Definition at line 166 of file readfuncs.c.

◆ READ_BOOL_FIELD

#define READ_BOOL_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \

Definition at line 113 of file readfuncs.c.

◆ READ_CHAR_FIELD

#define READ_CHAR_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
/* avoid overhead of calling debackslash() for one char */ \
local_node->fldname = (length == 0) ? '\0' : (token[0] == '\\' ? token[1] : token[0])

Definition at line 94 of file readfuncs.c.

98 : (token[0] == '\\' ? token[1] : token[0])

◆ READ_DONE

#define READ_DONE ( )     return local_node

Definition at line 171 of file readfuncs.c.

◆ READ_ENUM_FIELD

#define READ_ENUM_FIELD (   fldname,
  enumtype 
)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = (enumtype) atoi(token)

Definition at line 101 of file readfuncs.c.

◆ READ_FLOAT_FIELD

#define READ_FLOAT_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = atof(token)

Definition at line 107 of file readfuncs.c.

◆ READ_INT64_FIELD

#define READ_INT64_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = strtoi64(token, NULL, 10)

Definition at line 70 of file readfuncs.c.

◆ READ_INT_ARRAY

#define READ_INT_ARRAY (   fldname,
  len 
)
Value:
token = pg_strtok(&length); /* skip :fldname */ \

Definition at line 161 of file readfuncs.c.

◆ READ_INT_FIELD

#define READ_INT_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = atoi(token)

Definition at line 58 of file readfuncs.c.

◆ READ_LOCALS

#define READ_LOCALS (   nodeTypeName)
Value:
#define READ_LOCALS_NO_FIELDS(nodeTypeName)
Definition readfuncs.c:44

Definition at line 53 of file readfuncs.c.

◆ READ_LOCALS_NO_FIELDS

#define READ_LOCALS_NO_FIELDS (   nodeTypeName)     nodeTypeName *local_node = makeNode(nodeTypeName)

Definition at line 44 of file readfuncs.c.

◆ READ_LOCATION_FIELD

#define READ_LOCATION_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
(void) token; /* in case not used elsewhere */ \
local_node->fldname = -1 /* set field to "unknown" */

Definition at line 131 of file readfuncs.c.

◆ READ_LONG_FIELD

#define READ_LONG_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = atol(token)

Definition at line 82 of file readfuncs.c.

◆ READ_NODE_FIELD

#define READ_NODE_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
(void) token; /* in case not used elsewhere */ \
local_node->fldname = nodeRead(NULL, 0)

Definition at line 139 of file readfuncs.c.

◆ READ_OID_ARRAY

#define READ_OID_ARRAY (   fldname,
  len 
)
Value:
token = pg_strtok(&length); /* skip :fldname */ \

Definition at line 156 of file readfuncs.c.

◆ READ_OID_FIELD

#define READ_OID_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = atooid(token)

Definition at line 88 of file readfuncs.c.

◆ READ_SCALAR_ARRAY

#define READ_SCALAR_ARRAY (   fnname,
  datatype,
  convfunc 
)
Value:
datatype * \
fnname(int numCols) \
{ \
datatype *vals; \
token = pg_strtok(&length); \
elog(ERROR, "incomplete scalar array"); \
if (length == 0) \
return NULL; /* it was "<>", so return NULL pointer */ \
if (length != 1 || token[0] != '(') \
elog(ERROR, "unrecognized token: \"%.*s\"", length, token); \
vals = (datatype *) palloc(numCols * sizeof(datatype)); \
for (int i = 0; i < numCols; i++) \
{ \
token = pg_strtok(&length); \
if (token == NULL || token[0] == ')') \
elog(ERROR, "incomplete scalar array"); \
vals[i] = convfunc(token); \
} \
token = pg_strtok(&length); \
if (token == NULL || length != 1 || token[0] != ')') \
elog(ERROR, "incomplete scalar array"); \
return vals; \
}

Definition at line 658 of file readfuncs.c.

661{ \
662 datatype *vals; \
664 token = pg_strtok(&length); \
665 if (token == NULL) \
666 elog(ERROR, "incomplete scalar array"); \
667 if (length == 0) \
668 return NULL; /* it was "<>", so return NULL pointer */ \
669 if (length != 1 || token[0] != '(') \
670 elog(ERROR, "unrecognized token: \"%.*s\"", length, token); \
671 vals = (datatype *) palloc(numCols * sizeof(datatype)); \
672 for (int i = 0; i < numCols; i++) \
673 { \
674 token = pg_strtok(&length); \
675 if (token == NULL || token[0] == ')') \
676 elog(ERROR, "incomplete scalar array"); \
677 vals[i] = convfunc(token); \
678 } \
679 token = pg_strtok(&length); \
680 if (token == NULL || length != 1 || token[0] != ')') \
681 elog(ERROR, "incomplete scalar array"); \
682 return vals; \
683}

◆ READ_STRING_FIELD

#define READ_STRING_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = nullable_string(token, length)

Definition at line 119 of file readfuncs.c.

◆ READ_TEMP_LOCALS

#define READ_TEMP_LOCALS ( )
Value:
const char *token; \
int length

Definition at line 48 of file readfuncs.c.

◆ READ_UINT64_FIELD

#define READ_UINT64_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = strtou64(token, NULL, 10)

Definition at line 76 of file readfuncs.c.

◆ READ_UINT_FIELD

#define READ_UINT_FIELD (   fldname)
Value:
token = pg_strtok(&length); /* skip :fldname */ \
token = pg_strtok(&length); /* get field value */ \
local_node->fldname = atoui(token)

Definition at line 64 of file readfuncs.c.

◆ strtobool

#define strtobool (   x)    ((*(x) == 't') ? true : false)

Definition at line 183 of file readfuncs.c.

Function Documentation

◆ _readA_Const()

static A_Const * _readA_Const ( void  )
static

Definition at line 308 of file readfuncs.c.

309{
311
312 /* We expect either NULL or :val here */
313 token = pg_strtok(&length);
314 if (length == 4 && strncmp(token, "NULL", 4) == 0)
315 local_node->isnull = true;
316 else
317 {
318 union ValUnion *tmp = nodeRead(NULL, 0);
319
320 /* To forestall valgrind complaints, copy only the valid data */
321 switch (nodeTag(tmp))
322 {
323 case T_Integer:
324 memcpy(&local_node->val, tmp, sizeof(Integer));
325 break;
326 case T_Float:
327 memcpy(&local_node->val, tmp, sizeof(Float));
328 break;
329 case T_Boolean:
330 memcpy(&local_node->val, tmp, sizeof(Boolean));
331 break;
332 case T_String:
333 memcpy(&local_node->val, tmp, sizeof(String));
334 break;
335 case T_BitString:
336 memcpy(&local_node->val, tmp, sizeof(BitString));
337 break;
338 default:
339 elog(ERROR, "unrecognized node type: %d",
340 (int) nodeTag(tmp));
341 break;
342 }
343 }
344
345 READ_LOCATION_FIELD(location);
346
347 READ_DONE();
348}

References elog, ERROR, fb(), nodeRead(), nodeTag, pg_strtok(), READ_DONE, READ_LOCALS, and READ_LOCATION_FIELD.

◆ _readA_Expr()

static A_Expr * _readA_Expr ( void  )
static

Definition at line 446 of file readfuncs.c.

447{
449
450 token = pg_strtok(&length);
451
452 if (length == 3 && strncmp(token, "ANY", 3) == 0)
453 {
454 local_node->kind = AEXPR_OP_ANY;
456 }
457 else if (length == 3 && strncmp(token, "ALL", 3) == 0)
458 {
459 local_node->kind = AEXPR_OP_ALL;
461 }
462 else if (length == 8 && strncmp(token, "DISTINCT", 8) == 0)
463 {
466 }
467 else if (length == 12 && strncmp(token, "NOT_DISTINCT", 12) == 0)
468 {
471 }
472 else if (length == 6 && strncmp(token, "NULLIF", 6) == 0)
473 {
474 local_node->kind = AEXPR_NULLIF;
476 }
477 else if (length == 2 && strncmp(token, "IN", 2) == 0)
478 {
479 local_node->kind = AEXPR_IN;
481 }
482 else if (length == 4 && strncmp(token, "LIKE", 4) == 0)
483 {
484 local_node->kind = AEXPR_LIKE;
486 }
487 else if (length == 5 && strncmp(token, "ILIKE", 5) == 0)
488 {
489 local_node->kind = AEXPR_ILIKE;
491 }
492 else if (length == 7 && strncmp(token, "SIMILAR", 7) == 0)
493 {
496 }
497 else if (length == 7 && strncmp(token, "BETWEEN", 7) == 0)
498 {
501 }
502 else if (length == 11 && strncmp(token, "NOT_BETWEEN", 11) == 0)
503 {
506 }
507 else if (length == 11 && strncmp(token, "BETWEEN_SYM", 11) == 0)
508 {
511 }
512 else if (length == 15 && strncmp(token, "NOT_BETWEEN_SYM", 15) == 0)
513 {
516 }
517 else if (length == 5 && strncmp(token, ":name", 5) == 0)
518 {
519 local_node->kind = AEXPR_OP;
520 local_node->name = nodeRead(NULL, 0);
521 }
522 else
523 elog(ERROR, "unrecognized A_Expr kind: \"%.*s\"", length, token);
524
525 READ_NODE_FIELD(lexpr);
526 READ_NODE_FIELD(rexpr);
527 READ_LOCATION_FIELD(rexpr_list_start);
528 READ_LOCATION_FIELD(rexpr_list_end);
529 READ_LOCATION_FIELD(location);
530
531 READ_DONE();
532}

References AEXPR_BETWEEN, AEXPR_BETWEEN_SYM, AEXPR_DISTINCT, AEXPR_ILIKE, AEXPR_IN, AEXPR_LIKE, AEXPR_NOT_BETWEEN, AEXPR_NOT_BETWEEN_SYM, AEXPR_NOT_DISTINCT, AEXPR_NULLIF, AEXPR_OP, AEXPR_OP_ALL, AEXPR_OP_ANY, AEXPR_SIMILAR, elog, ERROR, fb(), name, nodeRead(), pg_strtok(), READ_DONE, READ_LOCALS, READ_LOCATION_FIELD, and READ_NODE_FIELD.

◆ _readBitmapset()

static Bitmapset * _readBitmapset ( void  )
static

Definition at line 207 of file readfuncs.c.

208{
209 Bitmapset *result = NULL;
210
212
213 token = pg_strtok(&length);
214 if (token == NULL)
215 elog(ERROR, "incomplete Bitmapset structure");
216 if (length != 1 || token[0] != '(')
217 elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
218
219 token = pg_strtok(&length);
220 if (token == NULL)
221 elog(ERROR, "incomplete Bitmapset structure");
222 if (length != 1 || token[0] != 'b')
223 elog(ERROR, "unrecognized token: \"%.*s\"", length, token);
224
225 for (;;)
226 {
227 int val;
228 char *endptr;
229
230 token = pg_strtok(&length);
231 if (token == NULL)
232 elog(ERROR, "unterminated Bitmapset structure");
233 if (length == 1 && token[0] == ')')
234 break;
235 val = (int) strtol(token, &endptr, 10);
236 if (endptr != token + length)
237 elog(ERROR, "unrecognized integer: \"%.*s\"", length, token);
238 result = bms_add_member(result, val);
239 }
240
241 return result;
242}

References bms_add_member(), elog, ERROR, fb(), pg_strtok(), READ_TEMP_LOCALS, and val.

Referenced by readBitmapset().

◆ _readBoolExpr()

static BoolExpr * _readBoolExpr ( void  )
static

Definition at line 285 of file readfuncs.c.

286{
288
289 /* do-it-yourself enum representation */
290 token = pg_strtok(&length); /* skip :boolop */
291 token = pg_strtok(&length); /* get field value */
292 if (length == 3 && strncmp(token, "and", 3) == 0)
293 local_node->boolop = AND_EXPR;
294 else if (length == 2 && strncmp(token, "or", 2) == 0)
295 local_node->boolop = OR_EXPR;
296 else if (length == 3 && strncmp(token, "not", 3) == 0)
297 local_node->boolop = NOT_EXPR;
298 else
299 elog(ERROR, "unrecognized boolop \"%.*s\"", length, token);
300
301 READ_NODE_FIELD(args);
302 READ_LOCATION_FIELD(location);
303
304 READ_DONE();
305}

References AND_EXPR, elog, ERROR, fb(), NOT_EXPR, OR_EXPR, pg_strtok(), READ_DONE, READ_LOCALS, READ_LOCATION_FIELD, and READ_NODE_FIELD.

◆ _readConst()

static Const * _readConst ( void  )
static

Definition at line 263 of file readfuncs.c.

264{
266
267 READ_OID_FIELD(consttype);
273 READ_LOCATION_FIELD(location);
274
275 token = pg_strtok(&length); /* skip :constvalue */
276 if (local_node->constisnull)
277 token = pg_strtok(&length); /* skip "<>" */
278 else
279 local_node->constvalue = readDatum(local_node->constbyval);
280
281 READ_DONE();
282}

References fb(), pg_strtok(), READ_BOOL_FIELD, READ_DONE, READ_INT_FIELD, READ_LOCALS, READ_LOCATION_FIELD, READ_OID_FIELD, and readDatum().

◆ _readExtensibleNode()

static ExtensibleNode * _readExtensibleNode ( void  )
static

Definition at line 535 of file readfuncs.c.

536{
537 const ExtensibleNodeMethods *methods;
539 const char *extnodename;
540
542
543 token = pg_strtok(&length); /* skip :extnodename */
544 token = pg_strtok(&length); /* get extnodename */
545
546 extnodename = nullable_string(token, length);
547 if (!extnodename)
548 elog(ERROR, "extnodename has to be supplied");
549 methods = GetExtensibleNodeMethods(extnodename, false);
550
553 local_node->extnodename = extnodename;
554
555 /* deserialize the private fields */
556 methods->nodeRead(local_node);
557
558 READ_DONE();
559}

References elog, ERROR, fb(), GetExtensibleNodeMethods(), newNode(), ExtensibleNodeMethods::node_size, ExtensibleNodeMethods::nodeRead, nullable_string(), pg_strtok(), READ_DONE, and READ_TEMP_LOCALS.

◆ _readRangeTblEntry()

static RangeTblEntry * _readRangeTblEntry ( void  )
static

Definition at line 351 of file readfuncs.c.

352{
354
355 READ_NODE_FIELD(alias);
357 READ_ENUM_FIELD(rtekind, RTEKind);
358
359 switch (local_node->rtekind)
360 {
361 case RTE_RELATION:
362 READ_OID_FIELD(relid);
363 READ_BOOL_FIELD(inh);
364 READ_CHAR_FIELD(relkind);
367 READ_NODE_FIELD(tablesample);
368 break;
369 case RTE_SUBQUERY:
370 READ_NODE_FIELD(subquery);
371 READ_BOOL_FIELD(security_barrier);
372 /* we re-use these RELATION fields, too: */
373 READ_OID_FIELD(relid);
374 READ_BOOL_FIELD(inh);
375 READ_CHAR_FIELD(relkind);
378 break;
379 case RTE_JOIN:
380 READ_ENUM_FIELD(jointype, JoinType);
386 break;
387 case RTE_FUNCTION:
389 READ_BOOL_FIELD(funcordinality);
390 break;
391 case RTE_TABLEFUNC:
392 READ_NODE_FIELD(tablefunc);
393 /* The RTE must have a copy of the column type info, if any */
394 if (local_node->tablefunc)
395 {
396 TableFunc *tf = local_node->tablefunc;
397
398 local_node->coltypes = tf->coltypes;
399 local_node->coltypmods = tf->coltypmods;
400 local_node->colcollations = tf->colcollations;
401 }
402 break;
403 case RTE_VALUES:
404 READ_NODE_FIELD(values_lists);
408 break;
409 case RTE_CTE:
410 READ_STRING_FIELD(ctename);
411 READ_UINT_FIELD(ctelevelsup);
416 break;
418 READ_STRING_FIELD(enrname);
419 READ_FLOAT_FIELD(enrtuples);
423 /* we re-use these RELATION fields, too: */
424 READ_OID_FIELD(relid);
425 break;
426 case RTE_RESULT:
427 /* no extra fields */
428 break;
429 case RTE_GROUP:
430 READ_NODE_FIELD(groupexprs);
431 break;
432 default:
433 elog(ERROR, "unrecognized RTE kind: %d",
434 (int) local_node->rtekind);
435 break;
436 }
437
438 READ_BOOL_FIELD(lateral);
441
442 READ_DONE();
443}

References elog, ERROR, fb(), functions, READ_BOOL_FIELD, READ_CHAR_FIELD, READ_DONE, READ_ENUM_FIELD, READ_FLOAT_FIELD, READ_INT_FIELD, READ_LOCALS, READ_NODE_FIELD, READ_OID_FIELD, READ_STRING_FIELD, READ_UINT_FIELD, RTE_CTE, RTE_FUNCTION, RTE_GROUP, RTE_JOIN, RTE_NAMEDTUPLESTORE, RTE_RELATION, RTE_RESULT, RTE_SUBQUERY, RTE_TABLEFUNC, and RTE_VALUES.

◆ nullable_string()

static char * nullable_string ( const char token,
int  length 
)
static

Definition at line 186 of file readfuncs.c.

187{
188 /* outToken emits <> for NULL, and pg_strtok makes that an empty string */
189 if (length == 0)
190 return NULL;
191 /* outToken emits "" for empty string */
192 if (length == 2 && token[0] == '"' && token[1] == '"')
193 return pstrdup("");
194 /* otherwise, we must remove protective backslashes added by outToken */
195 return debackslash(token, length);
196}

References debackslash(), fb(), and pstrdup().

Referenced by _readExtensibleNode().

◆ parseNodeString()

Node * parseNodeString ( void  )

Definition at line 571 of file readfuncs.c.

572{
574
575 /* Guard against stack overflow due to overly complex expressions */
577
578 token = pg_strtok(&length);
579
580#define MATCH(tokname, namelen) \
581 (length == namelen && memcmp(token, tokname, namelen) == 0)
582
583#include "readfuncs.switch.c"
584
585 elog(ERROR, "badly formatted node string \"%.32s\"...", token);
586 return NULL; /* keep compiler quiet */
587}

References check_stack_depth(), elog, ERROR, fb(), pg_strtok(), and READ_TEMP_LOCALS.

Referenced by nodeRead().

◆ readBitmapset()

Bitmapset * readBitmapset ( void  )

Definition at line 249 of file readfuncs.c.

250{
251 return _readBitmapset();
252}

References _readBitmapset().

◆ readDatum()

Datum readDatum ( bool  typbyval)

Definition at line 598 of file readfuncs.c.

599{
600 Size length;
601 int tokenLength;
602 const char *token;
603 Datum res;
604 char *s;
605
606 /*
607 * read the actual length of the value
608 */
610 length = atoui(token);
611
612 token = pg_strtok(&tokenLength); /* read the '[' */
613 if (token == NULL || token[0] != '[')
614 elog(ERROR, "expected \"[\" to start datum, but got \"%s\"; length = %zu",
615 token ? token : "[NULL]", length);
616
617 if (typbyval)
618 {
619 if (length > (Size) sizeof(Datum))
620 elog(ERROR, "byval datum but length = %zu", length);
621 res = (Datum) 0;
622 s = (char *) (&res);
623 for (Size i = 0; i < (Size) sizeof(Datum); i++)
624 {
626 s[i] = (char) atoi(token);
627 }
628 }
629 else if (length <= 0)
630 res = (Datum) 0;
631 else
632 {
633 s = (char *) palloc(length);
634 for (Size i = 0; i < length; i++)
635 {
637 s[i] = (char) atoi(token);
638 }
639 res = PointerGetDatum(s);
640 }
641
642 token = pg_strtok(&tokenLength); /* read the ']' */
643 if (token == NULL || token[0] != ']')
644 elog(ERROR, "expected \"]\" to end datum, but got \"%s\"; length = %zu",
645 token ? token : "[NULL]", length);
646
647 return res;
648}

References atoui, elog, ERROR, fb(), i, palloc(), pg_strtok(), PointerGetDatum(), and token.

Referenced by _readConst().