/*-------------------------------------------------------------------------
 *
 * readfuncs.c--
 *    Reader functions for Postgres tree nodes.
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
 *    /usr/local/devel/pglite/cvs/src/backend/nodes/readfuncs.c,v 1.20 1995/03/17 20:26:03 andrew Exp
 *
 * NOTES
 *    Most of the read functions for plan nodes are tested. (In fact, they
 *    pass the regression test as of 11/8/94.) The rest (for path selection)
 *    are probably never used. No effort has been made to get them to work.
 *    The simplest way to test these functions is by doing the following in
 *    ProcessQuery (before executing the plan):
 *    		plan = stringToNode(nodeToString(plan));
 *    Then, run the regression test. Let's just say you'll notice if either
 *    of the above function are not properly done.
 *							- ay 11/94
 *    
 *-------------------------------------------------------------------------
 */
#include <stdio.h>
#include <math.h>
#include <string.h>

#include "postgres.h"

#include "access/heapam.h"
#include "access/htup.h"
#include "fmgr.h"
#include "utils/builtins.h"
#include "utils/elog.h"
#include "utils/palloc.h"

#include "utils/lsyscache.h"
#include "utils/syscache.h"
#include "catalog/pg_type.h"

#include "nodes/primnodes.h"
#include "nodes/plannodes.h"
#include "nodes/parsenodes.h"
#include "nodes/execnodes.h"
#include "nodes/relation.h"
#include "nodes/readfuncs.h"

/* ----------------
 *	node creator declarations
 * ----------------
 */

static Datum readDatum(Oid type);

static List *toIntList(List *list)
{
    List *l;
    foreach(l, list) {
	/* ugly manipulation, should probably free the Value node too */
	lfirst(l) = (void*)intVal(lfirst(l));
    }
    return list;
}

/* ----------------
 *	_readQuery
 * ----------------
 */
static Query *
_readQuery()
{
    Query *local_node;
    char *token;
    int length;
    
    local_node = makeNode(Query);

    token = lsptok(NULL, &length);	/* skip the :command */
    token = lsptok(NULL, &length);	/* get the commandType */
    local_node->commandType = atoi(token);

    token = lsptok(NULL, &length);	/* skip the :resrel */
    token = lsptok(NULL, &length);	/* get the resultRelation */
    local_node->resultRelation = atoi(token);
    
    token = lsptok(NULL, &length);    	/* skip :rtable */
    local_node->rtable = nodeRead(true);

    token = lsptok(NULL, &length);	/* skip the :unique */
    token = lsptok(NULL, &length);	/* get the uniqueFlag */
/*    local_node->uniqueFlag = (bool)atoi(token); */
    if (token[0]=='"' && token[1] == '"') /* non-unique */
      local_node->uniqueFlag = NULL;
    else {
      local_node->uniqueFlag = palloc(length + 1);
      strncpy(local_node->uniqueFlag,token,length);
      local_node->uniqueFlag[length] = '\0';
    }

    token = lsptok(NULL, &length);    	/* skip :targetlist */
    local_node->targetList = nodeRead(true);
    
    token = lsptok(NULL, &length);    	/* skip :qual */
    local_node->qual = nodeRead(true);
    
    return (local_node);
}

/* ----------------
 *	_getPlan
 * ----------------
 */
static void
_getPlan(Plan *node)
{
    char *token;
    int length;
    
    token = lsptok(NULL, &length);    	/* first token is :cost */
    token = lsptok(NULL, &length);    	/* next is the actual cost */
    node->cost = (Cost) atof(token);
    
    token = lsptok(NULL, &length);		/* skip the :size */
    token = lsptok(NULL, &length);		/* get the plan_size */
    node->plan_size = atoi(token);
    
    token = lsptok(NULL, &length);		/* skip the :width */
    token = lsptok(NULL, &length);		/* get the plan_width */
    node->plan_width = atoi(token);
    
    token = lsptok(NULL, &length);    	/* eat the :state stuff */
    token = lsptok(NULL, &length);    	/* now get the state */ 
    
    if (!strncmp(token, "nil", 3)) {
	node->state = (EState*) NULL;
    }else { /* Disgusting hack until I figure out what to do here */
	node->state = (EState*) ! NULL;
    }
    
    token = lsptok(NULL, &length);    	/* eat :qptargetlist */
    node->targetlist = nodeRead(true);
    
    token = lsptok(NULL, &length);    	/* eat :qpqual */
    node->qual = nodeRead(true);
    
    token = lsptok(NULL, &length);    	/* eat :lefttree */
    node->lefttree = (Plan*) nodeRead(true);
    
    token = lsptok(NULL, &length);    	/* eat :righttree */
    node->righttree = (Plan*) nodeRead(true);

    return;
}

/*
 *  Stuff from plannodes.h
 */

/* ----------------
 *	_readPlan
 * ----------------
 */
static Plan *
_readPlan()
{
    Plan *local_node;
    
    local_node = makeNode(Plan);
    
    _getPlan(local_node);
    
    return (local_node);
}

/* ----------------
 *	_readResult
 *
 * 	Does some obscene, possibly unportable, magic with
 *	sizes of things.
 * ----------------
 */
static Result *
_readResult()
{
    Result	*local_node;
    char *token;
    int length;
    
    local_node = makeNode(Result);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    	/* eat :resconstantqual */
    local_node->resconstantqual = nodeRead(true);	/* now read it */
    
    return( local_node );
}

/* ----------------
 *	_readExistential
 *
 *	Existential nodes are only used by the planner.
 * ----------------
 */
static Existential *
_readExistential()
{
    Existential	*local_node;
    
    local_node = makeNode(Existential);
    
    _getPlan((Plan*)local_node);
    
    return( local_node );
}

/* ----------------
 *	_readAppend
 *
 *  Append is a subclass of Plan.
 * ----------------
 */

static Append *
_readAppend()
{
    Append *local_node;
    char *token;
    int length;
    
    local_node = makeNode(Append);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :unionplans */
    local_node->unionplans = nodeRead(true); 	/* now read it */
    
    token = lsptok(NULL, &length);    		/* eat :unionrelid */
    token = lsptok(NULL, &length);    		/* get unionrelid */
    local_node->unionrelid = atoi(token);
    
    token = lsptok(NULL, &length);    	/* eat :unionrtentries */
    local_node->unionrtentries = nodeRead(true);	/* now read it */
    
    return(local_node);
}

/* ----------------
 *	_getJoin
 *
 * In case Join is not the same structure as Plan someday.
 * ----------------
 */
static void
_getJoin(Join *node)
{
    _getPlan((Plan*)node);
}


/* ----------------
 *	_readJoin
 *
 *  Join is a subclass of Plan
 * ----------------
 */
static Join *
_readJoin()
{
    Join	*local_node;
    
    local_node = makeNode(Join);
    
    _getJoin(local_node);
    
    return( local_node );
}

/* ----------------
 *	_readNestLoop
 *	
 *  NestLoop is a subclass of Join
 * ----------------
 */

static NestLoop *
_readNestLoop()
{
    NestLoop	*local_node;
    
    local_node = makeNode(NestLoop);
    
    _getJoin((Join*)local_node);
    
    return( local_node );
}

/* ----------------
 *	_readMergeJoin
 *	
 *  MergeJoin is a subclass of Join
 * ----------------
 */
static MergeJoin *
_readMergeJoin()
{
    MergeJoin	*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(MergeJoin);
    
    _getJoin((Join*)local_node);
    token = lsptok(NULL, &length);    		/* eat :mergeclauses */
    local_node->mergeclauses = nodeRead(true);	/* now read it */
    
    token = lsptok(NULL, &length);    		/* eat :mergesortop */
    token = lsptok(NULL, &length);    		/* get mergesortop */
    local_node->mergesortop = atol(token);
    
    return( local_node );
}

/* ----------------
 *	_readHashJoin
 *	
 *  HashJoin is a subclass of Join.
 * ----------------
 */
static HashJoin *
_readHashJoin()
{
    HashJoin	*local_node;
    char 		*token;
    int length;
    
    local_node = makeNode(HashJoin);
    
    _getJoin((Join*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :hashclauses */
    local_node->hashclauses = nodeRead(true);	/* now read it */
    
    token = lsptok(NULL, &length);    		/* eat :hashjoinop */
    token = lsptok(NULL, &length);    		/* get hashjoinop */
    local_node->hashjoinop = atoi(token);
    
    token = lsptok(NULL, &length);		/* eat :hashjointable */
    token = lsptok(NULL, &length);		/* eat hashjointable */
    local_node->hashjointable = NULL;
    
    token = lsptok(NULL, &length);		/* eat :hashjointablekey */
    token = lsptok(NULL, &length);		/* eat hashjointablekey */
    local_node->hashjointablekey = 0;
    
    token = lsptok(NULL, &length);		/* eat :hashjointablesize */
    token = lsptok(NULL, &length);		/* eat hashjointablesize */
    local_node->hashjointablesize = 0;
    
    token = lsptok(NULL, &length);		/* eat :hashdone */
    token = lsptok(NULL, &length);		/* eat hashdone */
    local_node->hashdone = false;
    
    return( local_node );
}

/* ----------------
 *	_getScan
 *
 *  Scan is a subclass of Node
 *  (Actually, according to the plannodes.h include file, it is a
 *  subclass of Plan.  This is why _getPlan is used here.)
 *
 *  Scan gets its own get function since stuff inherits it.
 * ----------------
 */
static void 
_getScan(Scan *node)
{
    char *token;
    int length;
    
    _getPlan((Plan*)node);
    
    token = lsptok(NULL, &length);    		/* eat :scanrelid */
    token = lsptok(NULL, &length);    		/* get scanrelid */
    node->scanrelid = atoi(token);
}

/* ----------------
 *	_readScan
 *	
 * Scan is a subclass of Plan (Not Node, see above).
 * ----------------
 */
static Scan *
_readScan()
{
    Scan 	*local_node;
    
    local_node = makeNode(Scan);
    
    _getScan(local_node);
    
    return(local_node);
}

/* ----------------
 *	_readSeqScan
 *	
 *  SeqScan is a subclass of Scan
 * ----------------
 */
static SeqScan *
_readSeqScan()
{
    SeqScan 	*local_node;
    
    local_node = makeNode(SeqScan);
    
    _getScan((Scan*)local_node);
    
    return(local_node);
}

/* ----------------
 *	_readIndexScan
 *	
 *  IndexScan is a subclass of Scan
 * ----------------
 */
static IndexScan *
_readIndexScan()
{
    IndexScan	*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(IndexScan);
    
    _getScan((Scan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :indxid */
    local_node->indxid =
	toIntList(nodeRead(true));		/* now read it */
    
    token = lsptok(NULL, &length);    		/* eat :indxqual */
    local_node->indxqual = nodeRead(true); 		/* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readTemp
 *	
 *  Temp is a subclass of Plan
 * ----------------
 */
static Temp *
_readTemp()
{
    Temp		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Temp);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :tempid */
    token = lsptok(NULL, &length);    		/* get tempid */
    local_node->tempid = atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :keycount */
    token = lsptok(NULL, &length);    		/* get keycount */
    local_node->keycount = atoi(token);
    
    return(local_node);
}

/* ----------------
 *	_readSort
 *	
 *  Sort is a subclass of Temp
 * ----------------
 */
static Sort *
_readSort()
{
    Sort		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Sort);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :tempid */
    token = lsptok(NULL, &length);    		/* get tempid */
    local_node->tempid = atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :keycount */
    token = lsptok(NULL, &length);    		/* get keycount */
    local_node->keycount = atoi(token);
    
    return(local_node);
}

static Agg *
_readAgg()
{
    Agg             *local_node;
    char            *token;
    int length;
    
    local_node = makeNode(Agg);
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);                  /* eat :numagg */
    token = lsptok(NULL, &length);                  /* get numagg */
    local_node->numAgg = atoi(token);

    return(local_node);
}

/* ----------------
 *	_readUnique
 *
 * For some reason, unique is a subclass of Temp.
 */
static Unique *
_readUnique()
{
    Unique		*local_node;
    char		*token;
    int		length;
    
    local_node = makeNode(Unique);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :tempid */
    token = lsptok(NULL, &length);    		/* get :tempid */
    local_node->tempid = atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :keycount */
    token = lsptok(NULL, &length);    		/* get :keycount */
    local_node->keycount = atoi(token);
    
    return(local_node);
}

/* ----------------
 *	_readHash
 *	
 *  Hash is a subclass of Temp
 * ----------------
 */
static Hash *
_readHash()
{
    Hash		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Hash);
    
    _getPlan((Plan*)local_node);
    
    token = lsptok(NULL, &length);    		/* eat :hashkey */
    local_node->hashkey = (Var*) nodeRead(true);
    
    token = lsptok(NULL, &length);    	/* eat :hashtable */
    token = lsptok(NULL, &length);    	/* eat hashtable address*/
    local_node->hashtable = NULL;
    
    token = lsptok(NULL, &length);    	/* eat :hashtablekey*/
    token = lsptok(NULL, &length);    	/* get hashtablekey */
    local_node->hashtablekey = 0;
    
    token = lsptok(NULL, &length);    	/* eat :hashtablesize*/
    token = lsptok(NULL, &length);    	/* get hashtablesize */
    local_node->hashtablesize = 0;
    
    return(local_node);
}

/*
 *  Stuff from primnodes.h.
 */

/* ----------------
 *	_readResdom
 *	
 *  Resdom is a subclass of Node
 * ----------------
 */
static Resdom *
_readResdom()
{
    Resdom		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Resdom);
    
    token = lsptok(NULL, &length);    		/* eat :resno */
    token = lsptok(NULL, &length);    		/* get resno */
    local_node->resno = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :restype */
    token = lsptok(NULL, &length);    		/* get restype */
    local_node->restype = atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :reslen */
    token = lsptok(NULL, &length);    		/* get reslen */
    local_node->reslen = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :resname */
    token = lsptok(NULL, &length);    		/* get the name */
    
    if (!strncmp(token, "\"null\"", 5)) {
	local_node->resname = NULL;
    }else {
	/*
	 * Peel off ""'s, then make a true copy.
	 */
	
	token++;
	token[length - 2] = '\0';
	
	local_node->resname = (Name) palloc(NAMEDATALEN);
	namestrcpy(local_node->resname, token);
	token[length - 2] = '\"';
    }
    
    token = lsptok(NULL, &length);    		/* eat :reskey */
    token = lsptok(NULL, &length);    		/* get reskey */
    local_node->reskey = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :reskeyop */
    token = lsptok(NULL, &length);    		/* get reskeyop */
    local_node->reskeyop = (Oid) atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :resjunk */
    token = lsptok(NULL, &length);    		/* get resjunk */
    local_node->resjunk = atoi(token);
    
    return(local_node);
}

/* ----------------
 *	_readExpr
 *	
 *  Expr is a subclass of Node
 * ----------------
 */
static Expr *
_readExpr()
{
    Expr *local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Expr);
    
    token = lsptok(NULL, &length);    		/* eat :typeOid */
    token = lsptok(NULL, &length);    		/* get typeOid */
    local_node->typeOid = (Oid)atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :opType */
    token = lsptok(NULL, &length);    		/* get opType */
    if (!strncmp(token, "op", 2)) {
	local_node->opType = OP_EXPR;
    } else if (!strncmp(token, "func", 4)) {
	local_node->opType = FUNC_EXPR;
    } else if (!strncmp(token, "or", 2)) {
	local_node->opType = OR_EXPR;
    } else if (!strncmp(token, "and", 3)) {
	local_node->opType = AND_EXPR;
    } else if (!strncmp(token, "not", 3)) {
	local_node->opType = NOT_EXPR;
    }
    
    token = lsptok(NULL, &length);    		/* eat :oper */
    local_node->oper = nodeRead(true);
    
    token = lsptok(NULL, &length);    		/* eat :args */
    local_node->args = nodeRead(true);		/* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readVar
 *	
 *  Var is a subclass of Expr
 * ----------------
 */
static Var *
_readVar()
{
    Var		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Var);
    
    token = lsptok(NULL, &length);    		/* eat :varno */
    token = lsptok(NULL, &length);    		/* get varno */
    local_node->varno = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :varattno */
    token = lsptok(NULL, &length);    		/* get varattno */
    local_node->varattno = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :vartype */
    token = lsptok(NULL, &length);    		/* get vartype */
    local_node->vartype = (Oid) atol(token);

    token = lsptok(NULL, &length);    		/* eat :varnoold */
    token = lsptok(NULL, &length);    		/* get varnoold */
    local_node->varnoold = (Oid) atol(token);
    
    token = lsptok(NULL, &length);    		/* eat :varoattno */
    token = lsptok(NULL, &length);    		/* eat :varoattno */
    local_node->varoattno = (int) atol(token);
    
    return(local_node);
}

/* ----------------
 * _readArray
 *
 * Array is a subclass of Expr
 * ----------------
 */
static Array *
_readArray()
{
    Array		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(Array);
    
    token = lsptok(NULL, &length);    		/* eat :arrayelemtype */
    token = lsptok(NULL, &length);    		/* get arrayelemtype */
    local_node->arrayelemtype = (Oid) atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :arrayelemlength */
    token = lsptok(NULL, &length);    		/* get arrayelemlength */
    local_node->arrayelemlength = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :arrayelembyval */
    token = lsptok(NULL, &length);    		/* get arrayelembyval */
    local_node->arrayelembyval = (token[0] == 't') ? true : false;
    
    token = lsptok(NULL, &length);    		/* eat :arraylow */
    token = lsptok(NULL, &length);    		/* get arraylow */
    local_node->arraylow.indx[0] = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :arrayhigh */
    token = lsptok(NULL, &length);    		/* get arrayhigh */
    local_node->arrayhigh.indx[0] = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :arraylen */
    token = lsptok(NULL, &length);    		/* get arraylen */
    local_node->arraylen = atoi(token);
    
    return(local_node);
}

/* ----------------
 * _readArrayRef
 *
 * ArrayRef is a subclass of Expr
 * ----------------
 */
static ArrayRef *
_readArrayRef()
{
    ArrayRef	*local_node;
    char	*token;
    int		length;
    
    local_node = makeNode(ArrayRef);
    
    token = lsptok(NULL, &length);    		/* eat :refelemtype */
    token = lsptok(NULL, &length);    		/* get refelemtype */
    local_node->refelemtype = (Oid) atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :refattrlength */
    token = lsptok(NULL, &length);    		/* get refattrlength */
    local_node->refattrlength = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :refelemlength */
    token = lsptok(NULL, &length);    		/* get refelemlength */
    local_node->refelemlength = atoi(token);
    
    token = lsptok(NULL, &length);    		/* eat :refelembyval */
    token = lsptok(NULL, &length);    		/* get refelembyval */
    local_node->refelembyval = (token[0] == 't') ? true : false;
    
    token = lsptok(NULL, &length);    		/* eat :refupperindex */
    local_node->refupperindexpr = nodeRead(true);
    
    token = lsptok(NULL, &length);    		/* eat :reflowerindex */
    local_node->reflowerindexpr = nodeRead(true);
    
    token = lsptok(NULL, &length);    		/* eat :refexpr */
    local_node->refexpr = nodeRead(true);
    
    token = lsptok(NULL, &length);    		/* eat :refassgnexpr */
    local_node->refassgnexpr = nodeRead(true);
    
    return(local_node);
}

/* ----------------
 *	_readConst
 *	
 *  Const is a subclass of Expr
 * ----------------
 */
static Const *
_readConst()
{
    Const	*local_node;
    char *token;
    int length;
    
    local_node = makeNode(Const);
    
    token = lsptok(NULL, &length);      /* get :consttype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->consttype = atol(token);
    
    
    token = lsptok(NULL, &length);      /* get :constlen */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->constlen = atoi(token);
    
    token = lsptok(NULL, &length);      /* get :constisnull */
    token = lsptok(NULL, &length);      /* now read it */
    
    if (!strncmp(token, "true", 4)) {
	local_node->constisnull = true;
    }else {
	local_node->constisnull = false;
    }
    
    
    token = lsptok(NULL, &length);      /* get :constvalue */
    
    if (local_node->constisnull) {
	token = lsptok(NULL, &length);      /* skip "NIL" */
    }else {
	/*
	 * read the value
	 */
	local_node->constvalue = readDatum(local_node->consttype);
    }
    
    token = lsptok(NULL, &length);      /* get :constbyval */
    token = lsptok(NULL, &length);      /* now read it */
    
    if (!strncmp(token, "true", 4)) {
	local_node->constbyval = true;
    }else {
	local_node->constbyval = false;
    }
    
    return(local_node);
}

/* ----------------
 *	_readFunc
 *	
 *  Func is a subclass of Expr
 * ----------------
 */
static Func *
_readFunc()
{
    Func	*local_node;
    char *token;
    int length;
    
    local_node = makeNode(Func);
    
    token = lsptok(NULL, &length);      /* get :funcid */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->funcid = atol(token);
    
    token = lsptok(NULL, &length);      /* get :functype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->functype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :funcisindex */
    token = lsptok(NULL, &length);      /* now read it */
    
    if (!strncmp(token, "true", 4)) {
	local_node->funcisindex = true;
    }else {
	local_node->funcisindex = false;
    }
    
    token = lsptok(NULL, &length);      /* get :funcsize */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->funcsize = atol(token);
    
    token = lsptok(NULL, &length); 	    /* get :func_fcache */
    token = lsptok(NULL, &length);	    /* get @ */
    token = lsptok(NULL, &length);	    /* now read it */
    
    local_node->func_fcache = (FunctionCache *) NULL;
    
    token = lsptok(NULL, &length);            /* get :func_tlist */
    local_node->func_tlist = nodeRead(true);  /* now read it */
    
    token = lsptok(NULL, &length);              /* get :func_planlist */
    local_node->func_planlist = nodeRead(true); /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readOper
 *	
 *  Oper is a subclass of Expr
 * ----------------
 */
static Oper *
_readOper()
{
    Oper	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(Oper);
    
    token = lsptok(NULL, &length);      /* get :opno */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->opno = atol(token);
    
    token = lsptok(NULL, &length);      /* get :opid */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->opid = atol(token);
    
    token = lsptok(NULL, &length);      /* get :opresulttype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->opresulttype = atol(token);
    
    /*
     * NOTE: Alternatively we can call 'replace_opid' 
     * which initializes both 'opid' and 'op_fcache'.
     */
    local_node->op_fcache = (FunctionCache *) NULL;
    
    return(local_node);
}

/* ----------------
 *	_readParam
 *	
 *  Param is a subclass of Expr
 * ----------------
 */
static Param *
_readParam()
{
    Param	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(Param);
    
    token = lsptok(NULL, &length);      /* get :paramkind */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->paramkind = atoi(token);
    
    token = lsptok(NULL, &length);      /* get :paramid */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->paramid = atol(token);
    
    token = lsptok(NULL, &length);      /* get :paramname */
    token = lsptok(NULL, &length);      /* now read it */
    token++;			    /* skip the first `"' */
    token[length - 2] = '\0';	    /* this is the 2nd `"' */
    
    local_node->paramname = (Name) palloc(NAMEDATALEN);
    namestrcpy(local_node->paramname, token);
    token[length - 2] = '\"';	/* restore the 2nd `"' */
    
    token = lsptok(NULL, &length);      /* get :paramtype */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->paramtype = atol(token);
    token = lsptok(NULL, &length);             /* get :param_tlist */
    local_node->param_tlist = nodeRead(true);  /* now read it */
    
    return(local_node);
}

/*
 *  Stuff from execnodes.h
 */

/* ----------------
 *	_readEState
 *	
 *  EState is a subclass of Node.
 * ----------------
 */
static EState *
_readEState()
{
    EState	*local_node;
    char *token;
    int length;
    
    local_node = makeNode(EState);
    
    token = lsptok(NULL, &length);      /* get :direction */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->es_direction = atoi(token);

    token = lsptok(NULL, &length);      /* get :range_table */
    
    local_node->es_range_table = nodeRead(true); /* now read it */
    
#if 0
    token = lsptok(NULL, &length);      /* get :time */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->es_time = (unsigned long) atoi(token);
    
    token = lsptok(NULL, &length);      /* get :owner */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->es_owner = atol(token);
    

    token = lsptok(NULL, &length);      /* get :locks */
    
    local_node->es_locks = nodeRead(true);   /* now read it */
    
    token = lsptok(NULL, &length);      /* get :subplan_info */
    
    local_node->es_subplan_info = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :error_message */
    token = lsptok(NULL, &length);      /* now read it */
    token++;
    token[length-2] = '\0';
    
    local_node->es_error_message = (Name) palloc(NAMEDATALEN);
    namestrcpy(local_node->es_error_message, token);
    
    token[length-2] = '\"';
    
    token = lsptok(NULL, &length);      /* get :qualification_tuple */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->es_qualification_tuple);
    
    token = lsptok(NULL, &length);      /* get :qualification_tuple_id */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->es_qualification_tuple_id);
    
    token = lsptok(NULL, &length); /* get :relation_relation_descriptor */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->es_relation_relation_descriptor);
    
#endif /*0*/
    token = lsptok(NULL, &length);      /* get :result_relation_info */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->es_result_relation_info);
    
    return(local_node);
}

/*
 *  Stuff from relation.h
 */

/* ----------------
 *	_readRel
 * ----------------
 */
static Rel *
_readRel()
{
    Rel	*local_node;
    char *token;
    int length;
    
    local_node = makeNode(Rel);
    
    token = lsptok(NULL, &length);      /* get :relids */
    local_node->relids =
	toIntList(nodeRead(true)); 	/* now read it */
    
    token = lsptok(NULL, &length);      /* get :indexed */
    token = lsptok(NULL, &length);      /* now read it */
    
    if (!strncmp(token, "true", 4))
	{
	    local_node->indexed = true;
	}
    else
	{
	    local_node->indexed = false;
	}
    
    token = lsptok(NULL, &length);      /* get :pages */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->pages = (unsigned int) atoi(token);
    
    token = lsptok(NULL, &length);      /* get :tuples */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->tuples = (unsigned int) atoi(token);
    
    token = lsptok(NULL, &length);      /* get :size */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->size = (unsigned int) atoi(token);
    
    token = lsptok(NULL, &length);      /* get :width */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->width = (unsigned int) atoi(token);
    
    token = lsptok(NULL, &length);      /* get :targetlist */
    local_node->targetlist = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :pathlist */
    local_node->pathlist = nodeRead(true); /* now read it */
    
    /*
     *  Not sure if these are nodes or not.  They're declared as
     *  struct Path *.  Since i don't know, i'll just print the
     *  addresses for now.  This can be changed later, if necessary.
     */
    
    token = lsptok(NULL, &length);      /* get :unorderpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->unorderedpath);
    
    token = lsptok(NULL, &length);      /* get :cheapestpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    sscanf(token, "%x", &local_node->cheapestpath);
    
#if 0
    token = lsptok(NULL, &length);      /* get :classlist */
    local_node->classlist = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :indexkeys */
    local_node->indexkeys = nodeRead(true);/* now read it */
    
    token = lsptok(NULL, &length);      /* get :ordering */
    local_node->ordering = nodeRead(true); /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :clauseinfo */
    local_node->clauseinfo = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :joininfo */
    local_node->joininfo = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :innerjoin */
    local_node->innerjoin = nodeRead(true); /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readTargetEntry
 * ----------------
 */
static TargetEntry *
_readTargetEntry()
{
    TargetEntry *local_node;
    char *token;
    int length;

    local_node = makeNode(TargetEntry);

    token = lsptok(NULL, &length);	/* get :resdom */
    local_node->resdom = nodeRead(true);	/* now read it */

    token = lsptok(NULL, &length);	/* get :expr */
    local_node->expr = nodeRead(true);	/* now read it */

    return (local_node);
}

/* ----------------
 *	_readTargetEntry
 * ----------------
 */
static RangeTblEntry *
_readRangeTblEntry()
{
    RangeTblEntry *local_node;
    char *token;
    int length;

    local_node = makeNode(RangeTblEntry);

    token = lsptok(NULL, &length);	/* eat :relname */
    token = lsptok(NULL, &length);	/* get :relname */
    if (!strncmp(token, "\"null\"", 5)) {
	local_node->relname = NULL;
    }else {
	/*
	 * Peel off ""'s, then make a true copy.
	 */
	
	token++;
	token[length - 2] = '\0';
	
	local_node->relname = (Name) palloc(NAMEDATALEN);
	namestrcpy(local_node->relname, token);
	token[length - 2] = '\"';
    }

    token = lsptok(NULL, &length);	/* eat :inh */
    token = lsptok(NULL, &length);	/* get :inh */
    local_node->inh = atoi(token);

    token = lsptok(NULL, &length);	/* eat :refname */
    token = lsptok(NULL, &length);	/* get :refname */
    if (!strncmp(token, "\"null\"", 5)) {
	local_node->refname = NULL;
    }else {
	/*
	 * Peel off ""'s, then make a true copy.
	 */
	
	token++;
	token[length - 2] = '\0';

	local_node->refname = (char*)strdup(token);
	token[length - 2] = '\"';
    }
    
    token = lsptok(NULL, &length);	/* eat :relid */
    token = lsptok(NULL, &length);	/* get :relid */
    local_node->relid = atoi(token);

    return (local_node);
}

/* ----------------
 *	_readPath
 *	
 *  Path is a subclass of Node.
 * ----------------
 */
static Path *
_readPath()
{
    Path	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(Path);
    
    token = lsptok(NULL, &length);      /* get :pathtype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->pathtype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :cost */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->path_cost = (Cost) atof(token);
    
#if 0
    token = lsptok(NULL, &length);      /* get :p_ordering */
    local_node->p_ordering =
	nodeRead(true);		        /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :keys */
    local_node->keys = nodeRead(true);       /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readIndexPath
 *	
 *  IndexPath is a subclass of Path.
 * ----------------
 */
static IndexPath *
_readIndexPath()
{
    IndexPath	*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(IndexPath);
    
    token = lsptok(NULL, &length);      /* get :pathtype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->path.pathtype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :cost */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->path.path_cost = (Cost) atof(token);
    
#if 0
    token = lsptok(NULL, &length);      /* get :p_ordering */
    local_node->path.p_ordering = nodeRead(true);      /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :keys */
    local_node->path.keys = nodeRead(true);       /* now read it */
    
    token = lsptok(NULL, &length);      /* get :indexid */
    local_node->indexid =
	toIntList(nodeRead(true));
    
    token = lsptok(NULL, &length);      /* get :indexqual */
    local_node->indexqual = nodeRead(true);      /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readJoinPath
 *	
 *  JoinPath is a subclass of Path
 * ----------------
 */
static JoinPath *
_readJoinPath()
{
    JoinPath	*local_node;
    char		*token;
    int length;
    
    
    local_node = makeNode(JoinPath);
    
    token = lsptok(NULL, &length);      /* get :pathtype */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->path.pathtype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :cost */
    token = lsptok(NULL, &length);      /* now read it */
    local_node->path.path_cost = (Cost) atof(token);
    
#if 0
    token = lsptok(NULL, &length);      /* get :p_ordering */
    local_node->path.p_ordering = nodeRead(true);           /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :keys */
    local_node->path.keys = nodeRead(true);            /* now read it */
    
    token = lsptok(NULL, &length);      /* get :pathclauseinfo */
    local_node->pathclauseinfo = nodeRead(true);         /* now read it */
    
    /*
     *  Not sure if these are nodes; they're declared as "struct path *".
     *  For now, i'll just print the addresses.
     *
     * GJK:  Since I am parsing this stuff, I'll just ignore the addresses,
     * and initialize these pointers to NULL.
     */
    
    token = lsptok(NULL, &length);      /* get :outerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */

    local_node->outerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :innerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */

    local_node->innerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :outerjoincost */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->path.outerjoincost = (Cost) atof(token);
    
    token = lsptok(NULL, &length);      /* get :joinid */
    local_node->path.joinid =
	toIntList(nodeRead(true));          /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readMergePath
 *	
 *  MergePath is a subclass of JoinPath.
 * ----------------
 */
static MergePath *
_readMergePath()
{
    MergePath	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(MergePath);
    
    token = lsptok(NULL, &length);      /* get :pathtype */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.pathtype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :cost */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.path_cost = (Cost) atof(token);
    
#if 0
    token = lsptok(NULL, &length);      /* get :p_ordering */
    local_node->jpath.path.p_ordering = nodeRead(true);           /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :keys */
    local_node->jpath.path.keys = nodeRead(true);            /* now read it */
    
    token = lsptok(NULL, &length);      /* get :pathclauseinfo */
    local_node->jpath.pathclauseinfo = nodeRead(true);        /* now read it */
    
    /*
     *  Not sure if these are nodes; they're declared as "struct path *".
     *  For now, i'll just print the addresses.
     *
     * GJK:  Since I am parsing this stuff, I'll just ignore the addresses,
     * and initialize these pointers to NULL.
     */
    
    token = lsptok(NULL, &length);      /* get :outerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.outerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :innerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.innerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :outerjoincost */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.outerjoincost = (Cost) atof(token);
    
    token = lsptok(NULL, &length);      /* get :joinid */
    local_node->jpath.path.joinid =
	toIntList(nodeRead(true));          /* now read it */
    
    token = lsptok(NULL, &length);      /* get :path_mergeclauses */
    local_node->path_mergeclauses = nodeRead(true);      /* now read it */
    
    token = lsptok(NULL, &length);      /* get :outersortkeys */
    local_node->outersortkeys = nodeRead(true);           /* now read it */
    
    token = lsptok(NULL, &length);      /* get :innersortkeys */
    local_node->innersortkeys = nodeRead(true);           /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readHashPath
 *	
 *  HashPath is a subclass of JoinPath.
 * ----------------
 */
static HashPath *
_readHashPath()
{
    HashPath	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(HashPath);
    
    token = lsptok(NULL, &length);      /* get :pathtype */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.pathtype = atol(token);
    
    token = lsptok(NULL, &length);      /* get :cost */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.path_cost = (Cost) atof(token);
    
#if 0
    token = lsptok(NULL, &length);      /* get :p_ordering */
    local_node->jpath.path.p_ordering = nodeRead(true); /* now read it */
#endif
    
    token = lsptok(NULL, &length);      /* get :keys */
    local_node->jpath.path.keys = nodeRead(true);       /* now read it */
    
    token = lsptok(NULL, &length);      /* get :pathclauseinfo */
    local_node->jpath.pathclauseinfo = nodeRead(true); /* now read it */
    
    /*
     *  Not sure if these are nodes; they're declared as "struct path *".
     *  For now, i'll just print the addresses.
     *
     * GJK:  Since I am parsing this stuff, I'll just ignore the addresses,
     * and initialize these pointers to NULL.
     */
    
    token = lsptok(NULL, &length);      /* get :outerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.outerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :innerjoinpath */
    token = lsptok(NULL, &length);      /* get @ */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.innerjoinpath = NULL;
    
    token = lsptok(NULL, &length);      /* get :outerjoincost */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->jpath.path.outerjoincost = (Cost) atof(token);
    
    token = lsptok(NULL, &length);      /* get :joinid */
    local_node->jpath.path.joinid =
	toIntList(nodeRead(true));     /* now read it */
    
    token = lsptok(NULL, &length);      /* get :path_hashclauses */
    local_node->path_hashclauses = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :outerhashkeys */
    local_node->outerhashkeys = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :innerhashkeys */
    local_node->innerhashkeys = nodeRead(true); /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readOrderKey
 *	
 *  OrderKey is a subclass of Node.
 * ----------------
 */
static OrderKey *
_readOrderKey()
{
    OrderKey	*local_node;
    char	*token;
    int length;
    
    local_node = makeNode(OrderKey);
    
    token = lsptok(NULL, &length);      /* get :attribute_number */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->attribute_number = atoi(token);
    
    token = lsptok(NULL, &length);      /* get :array_index */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->array_index = atoi(token);
    
    return(local_node);
}

/* ----------------
 *	_readJoinKey
 *	
 *  JoinKey is a subclass of Node.
 * ----------------
 */
static JoinKey *
_readJoinKey()
{
    JoinKey		*local_node;
    char		*token;
    int length;
    
    local_node = makeNode(JoinKey);
    
    token = lsptok(NULL, &length);      /* get :outer */
    local_node->outer = nodeRead(true);      /* now read it */
    
    token = lsptok(NULL, &length);      /* get :inner */
    local_node->inner = nodeRead(true);      /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readMergeOrder
 *	
 *  MergeOrder is a subclass of Node.
 * ----------------
 */
static MergeOrder *
_readMergeOrder()
{
    MergeOrder	*local_node;
    char	*token;
    int length;
    
    local_node = makeNode(MergeOrder);
    token = lsptok(NULL, &length);      /* get :join_operator */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->join_operator = atol(token);
    
    token = lsptok(NULL, &length);      /* get :left_operator */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->left_operator = atol(token);
    
    token = lsptok(NULL, &length);      /* get :right_operator */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->right_operator = atol(token);
    
    token = lsptok(NULL, &length);      /* get :left_type */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->left_type = atol(token);
    
    token = lsptok(NULL, &length);      /* get :right_type */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->right_type = atol(token);
    
    return(local_node);
}

/* ----------------
 *	_readCInfo
 *	
 *  CInfo is a subclass of Node.
 * ----------------
 */
static CInfo *
_readCInfo()
{
    CInfo	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(CInfo);
    
    token = lsptok(NULL, &length);      /* get :clause */
    local_node->clause =  nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :selectivity */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->selectivity = atof(token);
    
    token = lsptok(NULL, &length);      /* get :notclause */
    token = lsptok(NULL, &length);      /* now read it */
    
    if (!strncmp(token, "true", 4))
	{
	    local_node->notclause = true;
	}
    else
	{
	    local_node->notclause = false;
	}
    
    token = lsptok(NULL, &length);      /* get :indexids */
    local_node->indexids = nodeRead(true);   /* now read it */
    
    token = lsptok(NULL, &length);      /* get :mergesortorder */
    local_node->mergesortorder = (MergeOrder*) nodeRead(true);
    
    token = lsptok(NULL, &length);      /* get :hashjoinoperator */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->hashjoinoperator = atol(token);
    
    return(local_node);
}

/* ----------------
 *	_readJoinMethod
 *	
 *  JoinMethod is a subclass of Node.
 * ----------------
 */
static JoinMethod *
_readJoinMethod()
{
    JoinMethod 	*local_node;
    char	*token;
    int length;
    
    local_node = makeNode(JoinMethod);
    
    token = lsptok(NULL, &length);      /* get :jmkeys */
    local_node->jmkeys = nodeRead(true);/* now read it */
    
    token = lsptok(NULL, &length);      /* get :clauses */
    local_node->clauses = nodeRead(true); /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readHInfo
 *	
 * HInfo is a subclass of JoinMethod.
 * ----------------
 */
static HInfo *
_readHInfo()
{
    HInfo 	*local_node;
    char 	*token;
    int length;
    
    local_node = makeNode(HInfo);
    
    token = lsptok(NULL, &length);      /* get :hashop */
    token = lsptok(NULL, &length);      /* now read it */
    
    local_node->hashop = atoi(token);
    
    token = lsptok(NULL, &length);      /* get :jmkeys */
    local_node->jmethod.jmkeys = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :clauses */
    local_node->jmethod.clauses = nodeRead(true);           /* now read it */
    
    return(local_node);
}

/* ----------------
 *	_readJInfo()
 *	
 *  JInfo is a subclass of Node.
 * ----------------
 */
static JInfo *
_readJInfo()
{
    JInfo	*local_node;
    char	*token;
    int length;
    
    local_node = makeNode(JInfo);
    
    token = lsptok(NULL, &length);      /* get :otherrels */
    local_node->otherrels =
	toIntList(nodeRead(true)); 	/* now read it */
    
    token = lsptok(NULL, &length);      /* get :jinfoclauseinfo */
    local_node->jinfoclauseinfo = nodeRead(true); /* now read it */
    
    token = lsptok(NULL, &length);      /* get :mergesortable */
    
    if (!strncmp(token, "true", 4))
	{
	    local_node->mergesortable = true;
	}
    else
	{
	    local_node->mergesortable = false;
	}
    
    token = lsptok(NULL, &length);      /* get :hashjoinable */
    
    if (!strncmp(token, "true", 4))
	{
	    local_node->hashjoinable = true;
	}
    else
	{
	    local_node->hashjoinable = false;
	}
    
    return(local_node);
}

/* ----------------
 *	_readIter()
 *
 * ----------------
 */
static Iter *
_readIter()
{
    Iter	*local_node;
    char	*token;
    int length;
    
    local_node = makeNode(Iter);
    
    token = lsptok(NULL, &length);      /* eat :iterexpr */
    local_node->iterexpr = nodeRead(true); 	/* now read it */
    
    return(local_node);
}


/* ----------------
 *	parsePlanString
 *
 * Given a character string containing a plan, parsePlanString sets up the
 * plan structure representing that plan.
 *
 * The string passed to parsePlanString must be null-terminated.
 * ----------------
 */
Node *
parsePlanString()
{
    char *token;
    int length;
    void *return_value;
    
    token = lsptok(NULL, &length);
    
    if (!strncmp(token, "PLAN", 4)) {
	return_value = _readPlan();
    }else if (!strncmp(token, "RESULT", 6)) {
	return_value = _readResult();
    }else if (!strncmp(token, "EXISTENTIAL", 11)) {
	return_value = _readExistential();
    }else if (!strncmp(token, "APPEND", 6)) {
	return_value = _readAppend();
    }else if (!strncmp(token, "JOIN", 4)) {
	return_value = _readJoin();
    }else if (!strncmp(token, "NESTLOOP", 8)) {
	return_value = _readNestLoop();
    }else if (!strncmp(token, "MERGEJOIN", 9)) {
	return_value = _readMergeJoin();
    }else if (!strncmp(token, "HASHJOIN", 8)) {
	return_value = _readHashJoin();
    }else if (!strncmp(token, "SCAN", 4)) {
	return_value = _readScan();
    }else if (!strncmp(token, "SEQSCAN", 7)) {
	return_value = _readSeqScan();
    }else if (!strncmp(token, "INDEXSCAN", 9)) {
	return_value = _readIndexScan();
    }else if (!strncmp(token, "TEMP", 4)) {
	return_value = _readTemp();
    }else if (!strncmp(token, "SORT", 4)) {
	return_value = _readSort();
    }else if (!strncmp(token, "AGG", 3)) {
	return_value = _readAgg();
    }else if (!strncmp(token, "UNIQUE", 4)) {
	return_value = _readUnique();
    }else if (!strncmp(token, "HASH", 4)) {
	return_value = _readHash();
    }else if (!strncmp(token, "RESDOM", 6)) {
	return_value = _readResdom();
    }else if (!strncmp(token, "EXPR", 4)) {
	return_value = _readExpr();
    }else if (!strncmp(token, "ARRAYREF", 7)) {
	/* make sure this strncmp is done before that of ARRAY */
	return_value = _readArrayRef();
    }else if (!strncmp(token, "ARRAY", 5)) {
	return_value = _readArray();
    }else if (!strncmp(token, "VAR", 3)) {
	return_value = _readVar();
    }else if (!strncmp(token, "CONST", 5)) {
	return_value = _readConst();
    }else if (!strncmp(token, "FUNC", 4)) {
	return_value = _readFunc();
    }else if (!strncmp(token, "OPER", 4)) {
	return_value = _readOper();
    }else if (!strncmp(token, "PARAM", 5)) {
	return_value = _readParam();
    }else if (!strncmp(token, "ESTATE", 6)) {
	return_value = _readEState();
    }else if (!strncmp(token, "REL", 3)) {
	return_value = _readRel();
    }else if (!strncmp(token, "TLE", 3)) {
	return_value = _readTargetEntry();
    }else if (!strncmp(token, "RTE", 3)) {
	return_value = _readRangeTblEntry();
    }else if (!strncmp(token, "PATH", 4)) {
	return_value = _readPath();
    }else if (!strncmp(token, "INDEXPATH", 9)) {
	return_value = _readIndexPath();
    }else if (!strncmp(token, "JOINPATH", 8)) {
	return_value = _readJoinPath();
    }else if (!strncmp(token, "MERGEPATH", 9)) {
	return_value = _readMergePath();
    }else if (!strncmp(token, "HASHPATH", 8)) {
	return_value = _readHashPath();
    }else if (!strncmp(token, "ORDERKEY", 8)) {
	return_value = _readOrderKey();
    }else if (!strncmp(token, "JOINKEY", 7)) {
	return_value = _readJoinKey();
    }else if (!strncmp(token, "MERGEORDER", 10)) {
	return_value = _readMergeOrder();
    }else if (!strncmp(token, "CINFO", 5)) {
	return_value = _readCInfo();
    }else if (!strncmp(token, "JOINMETHOD", 10)) {
	return_value = _readJoinMethod();
    }else if (!strncmp(token, "JINFO", 5)) {
	return_value = _readJInfo();
    }else if (!strncmp(token, "HINFO", 5)) {
	return_value = _readHInfo();
    }else if (!strncmp(token, "ITER", 4)) {
	return_value = _readIter();
    }else if (!strncmp(token, "QUERY", 5)) {
	return_value = _readQuery();
    }else {
	elog(WARN, "badly formatted planstring \"%.10s\"...\n", token);
    }

    return ((Node*)return_value);
}
/*------------------------------------------------------------*/

/* ----------------
 *	readDatum
 *
 * given a string representation of the value of the given type,
 * create the appropriate Datum
 * ----------------
 */
static Datum
readDatum(Oid type)
{
    int 	length;
    int 	tokenLength;
    char 	*token;
    bool	byValue;
    Datum 	res;
    char 	*s;
    int 	i;
    
    byValue	= get_typbyval(type);
    
    /*
     * read the actual length of the value
     */
    token = lsptok(NULL, &tokenLength);
    length = atoi(token);
    token = lsptok(NULL, &tokenLength);	/* skip the '[' */
    
    if (byValue) {
	if (length > sizeof(Datum)) {
	    elog(WARN, "readValue: byval & length = %d", length);
	}
	s = (char *) (&res);
	for (i=0; i<sizeof(Datum); i++) {
	    token = lsptok(NULL, &tokenLength);
	    s[i] = (char) atoi(token);
	}
    } else if (length <= 0) {
	s = NULL;
    } else if (length >= 1) {
	s = (char*)palloc(length);
	Assert( s!=NULL );
	for (i=0; i<length; i++) {
	    token = lsptok(NULL, &tokenLength);
	    s[i] = (char) atoi(token);
	}
	res = PointerGetDatum(s);
    }
    
    token = lsptok(NULL, &tokenLength);	/* skip the ']' */
    if (token[0] != ']') {
	elog(WARN, "readValue: ']' expected, length =%d", length);
    }
    
    return(res);
}
