outfuncs.c
上传用户:blenddy
上传日期:2007-01-07
资源大小:6495k
文件大小:35k
源码类别:

数据库系统

开发平台:

Unix_Linux

  1. /*
  2.  *
  3.  * outfuncs.c
  4.  *   routines to convert a node to ascii representation
  5.  *
  6.  * Copyright (c) 1994, Regents of the University of California
  7.  *
  8.  * $Id: outfuncs.c,v 1.86 1999/05/25 22:41:14 momjian Exp $
  9.  *
  10.  * NOTES
  11.  *   Every (plan) node in POSTGRES has an associated "out" routine which
  12.  *   knows how to create its ascii representation. These functions are
  13.  *   useful for debugging as well as for storing plans in the system
  14.  *   catalogs (eg. indexes). This is also the plan string sent out in
  15.  *   Mariposa.
  16.  *
  17.  *   These functions update the in/out argument of type StringInfo
  18.  *   passed to them. This argument contains the string holding the ASCII
  19.  *   representation plus some other information (string length, etc.)
  20.  *
  21.  */
  22. #include <stdio.h>
  23. #include "postgres.h"
  24. #include "access/heapam.h"
  25. #include "access/htup.h"
  26. #include "utils/syscache.h"
  27. #include "utils/lsyscache.h"
  28. #include "fmgr.h"
  29. #include "utils/elog.h"
  30. #include "utils/datum.h"
  31. #include "utils/palloc.h"
  32. #include "nodes/nodes.h"
  33. #include "nodes/execnodes.h"
  34. #include "nodes/pg_list.h"
  35. #include "nodes/plannodes.h"
  36. #include "nodes/parsenodes.h"
  37. #include "nodes/primnodes.h"
  38. #include "nodes/relation.h"
  39. #include "catalog/pg_type.h"
  40. #include "lib/stringinfo.h"
  41. #include "../parse.h"
  42. static void _outDatum(StringInfo str, Datum value, Oid type);
  43. static void _outNode(StringInfo str, void *obj);
  44. /*
  45.  * _outIntList -
  46.  *    converts a List of integers
  47.  */
  48. static void
  49. _outIntList(StringInfo str, List *list)
  50. {
  51. List    *l;
  52. appendStringInfo(str, "(");
  53. foreach(l, list)
  54. appendStringInfo(str, " %d ", lfirsti(l));
  55. appendStringInfo(str, ")");
  56. }
  57. static void
  58. _outCreateStmt(StringInfo str, CreateStmt *node)
  59. {
  60. appendStringInfo(str, " CREATE :relname %s ",
  61.  stringStringInfo(node->relname));
  62. appendStringInfo(str, " :istemp %s ",
  63.  node->istemp ? "true" : "false");
  64. appendStringInfo(str, " :columns ");
  65. _outNode(str, node->tableElts);
  66. appendStringInfo(str, " :inhRelnames ");
  67. _outNode(str, node->inhRelnames);
  68. appendStringInfo(str, " :constraints ");
  69. _outNode(str, node->constraints);
  70. }
  71. static void
  72. _outIndexStmt(StringInfo str, IndexStmt *node)
  73. {
  74. appendStringInfo(str,
  75.  " INDEX :idxname %s :relname %s :accessMethod %s :indexParams ",
  76.  stringStringInfo(node->idxname),
  77.  stringStringInfo(node->relname),
  78.  stringStringInfo(node->accessMethod));
  79. _outNode(str, node->indexParams);
  80. appendStringInfo(str, " :withClause ");
  81. _outNode(str, node->withClause);
  82. appendStringInfo(str, " :whereClause ");
  83. _outNode(str, node->whereClause);
  84. appendStringInfo(str, " :rangetable ");
  85. _outNode(str, node->rangetable);
  86. appendStringInfo(str, " :lossy %s :unique %s ",
  87.  node->lossy ? "true" : "false",
  88.  node->unique ? "true" : "false");
  89. }
  90. static void
  91. _outSelectStmt(StringInfo str, SelectStmt *node)
  92. {
  93. appendStringInfo(str, "SELECT :where ");
  94. _outNode(str, node->whereClause);
  95. }
  96. static void
  97. _outFuncCall(StringInfo str, FuncCall *node)
  98. {
  99. appendStringInfo(str, "FUNCTION %s :args ", stringStringInfo(node->funcname));
  100. _outNode(str, node->args);
  101. }
  102. static void
  103. _outColumnDef(StringInfo str, ColumnDef *node)
  104. {
  105. appendStringInfo(str, " COLUMNDEF :colname %s :typename ",
  106.  stringStringInfo(node->colname));
  107. _outNode(str, node->typename);
  108. appendStringInfo(str, " :is_not_null %s :defval %s :constraints ",
  109.  node->is_not_null ? "true" : "false",
  110.  stringStringInfo(node->defval));
  111. _outNode(str, node->constraints);
  112. }
  113. static void
  114. _outTypeName(StringInfo str, TypeName *node)
  115. {
  116. appendStringInfo(str,
  117.  " TYPENAME :name %s :timezone %s :setof %s typmod %d :arrayBounds ",
  118.  stringStringInfo(node->name),
  119.  node->timezone ? "true" : "false",
  120.  node->setof ? "true" : "false",
  121.  node->typmod);
  122. appendStringInfo(str, " :arrayBounds ");
  123. _outNode(str, node->arrayBounds);
  124. }
  125. static void
  126. _outIndexElem(StringInfo str, IndexElem *node)
  127. {
  128. appendStringInfo(str, " INDEXELEM :name %s :args ",
  129.  stringStringInfo(node->name));
  130. _outNode(str, node->args);
  131. appendStringInfo(str, " :class %s :typename ", stringStringInfo(node->class));
  132. _outNode(str, node->typename);
  133. }
  134. static void
  135. _outQuery(StringInfo str, Query *node)
  136. {
  137. appendStringInfo(str, " QUERY :command %d ", node->commandType);
  138. if (node->utilityStmt)
  139. {
  140. switch (nodeTag(node->utilityStmt))
  141. {
  142. case T_CreateStmt:
  143. appendStringInfo(str, " :create %s ",
  144.  stringStringInfo(((CreateStmt *) (node->utilityStmt))->relname));
  145. _outNode(str, node->utilityStmt);
  146. break;
  147. case T_IndexStmt:
  148. appendStringInfo(str, " :index %s on %s ",
  149.  stringStringInfo(((IndexStmt *) (node->utilityStmt))->idxname),
  150.  stringStringInfo(((IndexStmt *) (node->utilityStmt))->relname));
  151. _outNode(str, node->utilityStmt);
  152. break;
  153. case T_NotifyStmt:
  154. appendStringInfo(str, " :utility %s ",
  155.  stringStringInfo(((NotifyStmt *) (node->utilityStmt))->relname));
  156. break;
  157. default:
  158. appendStringInfo(str, " :utility ? ");
  159. }
  160. }
  161. else
  162. appendStringInfo(str, " :utility <>");
  163. appendStringInfo(str,
  164.  " :resultRelation %u :into %s :isPortal %s :isBinary %s :isTemp %s :unionall %s ",
  165.  node->resultRelation,
  166.  stringStringInfo(node->into),
  167.  node->isPortal ? "true" : "false",
  168.  node->isBinary ? "true" : "false",
  169.  node->isTemp ? "true" : "false",
  170.  node->unionall ? "true" : "false");
  171. appendStringInfo(str, " :unique %s :sortClause ",
  172.  stringStringInfo(node->uniqueFlag));
  173. _outNode(str, node->sortClause);
  174. appendStringInfo(str, " :rtable ");
  175. _outNode(str, node->rtable);
  176. appendStringInfo(str, " :targetlist ");
  177. _outNode(str, node->targetList);
  178. appendStringInfo(str, " :qual ");
  179. _outNode(str, node->qual);
  180. appendStringInfo(str, " :groupClause ");
  181. _outNode(str, node->groupClause);
  182. appendStringInfo(str, " :havingQual ");
  183. _outNode(str, node->havingQual);
  184. appendStringInfo(str, " :hasAggs %s :hasSubLinks %s :unionClause ",
  185.  node->hasAggs ? "true" : "false",
  186.  node->hasSubLinks ? "true" : "false");
  187. _outNode(str, node->unionClause);
  188. appendStringInfo(str, " :intersectClause ");
  189. _outNode(str, node->intersectClause);
  190. appendStringInfo(str, " :limitOffset ");
  191. _outNode(str, node->limitOffset);
  192. appendStringInfo(str, " :limitCount ");
  193. _outNode(str, node->limitCount);
  194. appendStringInfo(str, " :rowMark ");
  195. _outNode(str, node->rowMark);
  196. }
  197. static void
  198. _outSortClause(StringInfo str, SortClause *node)
  199. {
  200. appendStringInfo(str, " SORTCLAUSE :resdom ");
  201. _outNode(str, node->resdom);
  202. appendStringInfo(str, " :opoid %u ", node->opoid);
  203. }
  204. static void
  205. _outGroupClause(StringInfo str, GroupClause *node)
  206. {
  207. appendStringInfo(str, " GROUPCLAUSE :grpOpoid %u :tleGroupref %d",
  208.  node->grpOpoid,
  209.  node->tleGroupref);
  210. }
  211. /*
  212.  * print the basic stuff of all nodes that inherit from Plan
  213.  */
  214. static void
  215. _outPlanInfo(StringInfo str, Plan *node)
  216. {
  217. appendStringInfo(str,
  218.   ":cost %g :size %d :width %d :state %s :qptargetlist ",
  219.  node->cost,
  220.  node->plan_size,
  221.  node->plan_width,
  222.  node->state ? "not-NULL" : "<>");
  223. _outNode(str, node->targetlist);
  224. appendStringInfo(str, " :qpqual ");
  225. _outNode(str, node->qual);
  226. appendStringInfo(str, " :lefttree ");
  227. _outNode(str, node->lefttree);
  228. appendStringInfo(str, " :righttree ");
  229. _outNode(str, node->righttree);
  230. appendStringInfo(str, " :extprm ");
  231. _outIntList(str, node->extParam);
  232. appendStringInfo(str, " :locprm ");
  233. _outIntList(str, node->locParam);
  234. appendStringInfo(str, " :initplan ");
  235. _outNode(str, node->initPlan);
  236. appendStringInfo(str, " :nprm %d ", node->nParamExec);
  237. }
  238. /*
  239.  * Stuff from plannodes.h
  240.  */
  241. static void
  242. _outPlan(StringInfo str, Plan *node)
  243. {
  244. appendStringInfo(str, " PLAN ");
  245. _outPlanInfo(str, (Plan *) node);
  246. }
  247. static void
  248. _outResult(StringInfo str, Result *node)
  249. {
  250. appendStringInfo(str, " RESULT ");
  251. _outPlanInfo(str, (Plan *) node);
  252. appendStringInfo(str, " :resconstantqual ");
  253. _outNode(str, node->resconstantqual);
  254. }
  255. /*
  256.  * Append is a subclass of Plan.
  257.  */
  258. static void
  259. _outAppend(StringInfo str, Append *node)
  260. {
  261. appendStringInfo(str, " APPEND ");
  262. _outPlanInfo(str, (Plan *) node);
  263. appendStringInfo(str, " :appendplans ");
  264. _outNode(str, node->appendplans);
  265. appendStringInfo(str, " :unionrtables ");
  266. _outNode(str, node->unionrtables);
  267. appendStringInfo(str,
  268.  " :inheritrelid %u :inheritrtable ",
  269.  node->inheritrelid);
  270. _outNode(str, node->inheritrtable);
  271. }
  272. /*
  273.  * Join is a subclass of Plan
  274.  */
  275. static void
  276. _outJoin(StringInfo str, Join *node)
  277. {
  278. appendStringInfo(str, " JOIN ");
  279. _outPlanInfo(str, (Plan *) node);
  280. }
  281. /*
  282.  * NestLoop is a subclass of Join
  283.  */
  284. static void
  285. _outNestLoop(StringInfo str, NestLoop *node)
  286. {
  287. appendStringInfo(str, " NESTLOOP ");
  288. _outPlanInfo(str, (Plan *) node);
  289. }
  290. /*
  291.  * MergeJoin is a subclass of Join
  292.  */
  293. static void
  294. _outMergeJoin(StringInfo str, MergeJoin *node)
  295. {
  296. appendStringInfo(str, " MERGEJOIN ");
  297. _outPlanInfo(str, (Plan *) node);
  298. appendStringInfo(str, " :mergeclauses ");
  299. _outNode(str, node->mergeclauses);
  300. }
  301. /*
  302.  * HashJoin is a subclass of Join.
  303.  */
  304. static void
  305. _outHashJoin(StringInfo str, HashJoin *node)
  306. {
  307. appendStringInfo(str, " HASHJOIN ");
  308. _outPlanInfo(str, (Plan *) node);
  309. appendStringInfo(str, " :hashclauses ");
  310. _outNode(str, node->hashclauses);
  311. appendStringInfo(str,
  312.  " :hashjoinop %u ",
  313.  node->hashjoinop);
  314. appendStringInfo(str,
  315.  " :hashdone %d ",
  316.  node->hashdone);
  317. }
  318. static void
  319. _outSubPlan(StringInfo str, SubPlan *node)
  320. {
  321. appendStringInfo(str, " SUBPLAN :plan ");
  322. _outNode(str, node->plan);
  323. appendStringInfo(str, " :planid %u :rtable ", node->plan_id);
  324. _outNode(str, node->rtable);
  325. appendStringInfo(str, " :setprm ");
  326. _outIntList(str, node->setParam);
  327. appendStringInfo(str, " :parprm ");
  328. _outIntList(str, node->parParam);
  329. appendStringInfo(str, " :slink ");
  330. _outNode(str, node->sublink);
  331. }
  332. /*
  333.  * Scan is a subclass of Node
  334.  */
  335. static void
  336. _outScan(StringInfo str, Scan *node)
  337. {
  338. appendStringInfo(str, " SCAN ");
  339. _outPlanInfo(str, (Plan *) node);
  340. appendStringInfo(str, " :scanrelid %u ", node->scanrelid);
  341. }
  342. /*
  343.  * SeqScan is a subclass of Scan
  344.  */
  345. static void
  346. _outSeqScan(StringInfo str, SeqScan *node)
  347. {
  348. appendStringInfo(str, " SEQSCAN ");
  349. _outPlanInfo(str, (Plan *) node);
  350. appendStringInfo(str, " :scanrelid %u ", node->scanrelid);
  351. }
  352. /*
  353.  * IndexScan is a subclass of Scan
  354.  */
  355. static void
  356. _outIndexScan(StringInfo str, IndexScan *node)
  357. {
  358. appendStringInfo(str, " INDEXSCAN ");
  359. _outPlanInfo(str, (Plan *) node);
  360. appendStringInfo(str, " :scanrelid %u :indxid ", node->scan.scanrelid);
  361. _outIntList(str, node->indxid);
  362. appendStringInfo(str, " :indxqual ");
  363. _outNode(str, node->indxqual);
  364. appendStringInfo(str, " :indxqualorig ");
  365. _outNode(str, node->indxqualorig);
  366. }
  367. /*
  368.  * Noname is a subclass of Plan
  369.  */
  370. static void
  371. _outNoname(StringInfo str, Noname *node)
  372. {
  373. appendStringInfo(str, " NONAME ");
  374. _outPlanInfo(str, (Plan *) node);
  375. appendStringInfo(str, " :nonameid %u :keycount %d ",
  376.  node->nonameid,
  377.  node->keycount);
  378. }
  379. /*
  380.  * Sort is a subclass of Noname
  381.  */
  382. static void
  383. _outSort(StringInfo str, Sort *node)
  384. {
  385. appendStringInfo(str, " SORT ");
  386. _outPlanInfo(str, (Plan *) node);
  387. appendStringInfo(str, " :nonameid %u :keycount %d ",
  388.  node->nonameid,
  389.  node->keycount);
  390. }
  391. static void
  392. _outAgg(StringInfo str, Agg *node)
  393. {
  394. appendStringInfo(str, " AGG ");
  395. _outPlanInfo(str, (Plan *) node);
  396. appendStringInfo(str, " :aggs ");
  397. _outNode(str, node->aggs);
  398. }
  399. static void
  400. _outGroup(StringInfo str, Group *node)
  401. {
  402. appendStringInfo(str, " GRP ");
  403. _outPlanInfo(str, (Plan *) node);
  404. /* the actual Group fields */
  405. appendStringInfo(str, " :numCols %d :tuplePerGroup %s ",
  406.  node->numCols,
  407.  node->tuplePerGroup ? "true" : "false");
  408. }
  409. /*
  410.  * For some reason, unique is a subclass of Noname.
  411.  */
  412. static void
  413. _outUnique(StringInfo str, Unique *node)
  414. {
  415. appendStringInfo(str, " UNIQUE ");
  416. _outPlanInfo(str, (Plan *) node);
  417. appendStringInfo(str, " :nonameid %u :keycount %d ",
  418.  node->nonameid,
  419.  node->keycount);
  420. }
  421. /*
  422.  * Hash is a subclass of Noname
  423.  */
  424. static void
  425. _outHash(StringInfo str, Hash *node)
  426. {
  427. appendStringInfo(str, " HASH ");
  428. _outPlanInfo(str, (Plan *) node);
  429. appendStringInfo(str, " :hashkey ");
  430. _outNode(str, node->hashkey);
  431. }
  432. /*****************************************************************************
  433.  *
  434.  * Stuff from primnodes.h.
  435.  *
  436.  *****************************************************************************/
  437. /*
  438.  * Resdom is a subclass of Node
  439.  */
  440. static void
  441. _outResdom(StringInfo str, Resdom *node)
  442. {
  443. appendStringInfo(str, " RESDOM :resno %d :restype %u :restypmod %d",
  444.  node->resno,
  445.  node->restype,
  446.  node->restypmod);
  447. appendStringInfo(str, " :resname "%s" :reskey %d :reskeyop %u",
  448.  stringStringInfo(node->resname),
  449.  node->reskey,
  450.  node->reskeyop);
  451. appendStringInfo(str, " :resgroupref %d :resjunk %s ",
  452.  node->resgroupref,
  453.  node->resjunk ? "true" : "false");
  454. }
  455. static void
  456. _outFjoin(StringInfo str, Fjoin *node)
  457. {
  458. int i;
  459. appendStringInfo(str, " FJOIN :initialized %s :nNodes %d ",
  460.  node->fj_initialized ? "true" : "false",
  461.  node->fj_nNodes);
  462. appendStringInfo(str, " :innerNode ");
  463. _outNode(str, node->fj_innerNode);
  464. appendStringInfo(str, " :results @ 0x%x :alwaysdone",
  465.  (int) node->fj_results);
  466. for (i = 0; i < node->fj_nNodes; i++)
  467. appendStringInfo(str, (node->fj_alwaysDone[i]) ? "true" : "false");
  468. }
  469. /*
  470.  * Expr is a subclass of Node
  471.  */
  472. static void
  473. _outExpr(StringInfo str, Expr *node)
  474. {
  475. char    *opstr = NULL;
  476. appendStringInfo(str, " EXPR :typeOid %u ",
  477.  node->typeOid);
  478. switch (node->opType)
  479. {
  480. case OP_EXPR:
  481. opstr = "op";
  482. break;
  483. case FUNC_EXPR:
  484. opstr = "func";
  485. break;
  486. case OR_EXPR:
  487. opstr = "or";
  488. break;
  489. case AND_EXPR:
  490. opstr = "and";
  491. break;
  492. case NOT_EXPR:
  493. opstr = "not";
  494. break;
  495. case SUBPLAN_EXPR:
  496. opstr = "subp";
  497. break;
  498. }
  499. appendStringInfo(str, " :opType %s :oper ", stringStringInfo(opstr));
  500. _outNode(str, node->oper);
  501. appendStringInfo(str, " :args ");
  502. _outNode(str, node->args);
  503. }
  504. /*
  505.  * Var is a subclass of Expr
  506.  */
  507. static void
  508. _outVar(StringInfo str, Var *node)
  509. {
  510. appendStringInfo(str,
  511. " VAR :varno %d :varattno %d :vartype %u :vartypmod %d ",
  512.  node->varno,
  513.  node->varattno,
  514.  node->vartype,
  515.  node->vartypmod);
  516. appendStringInfo(str, " :varlevelsup %u :varnoold %d :varoattno %d",
  517.  node->varlevelsup,
  518.  node->varnoold,
  519.  node->varoattno);
  520. }
  521. /*
  522.  * Const is a subclass of Expr
  523.  */
  524. static void
  525. _outConst(StringInfo str, Const *node)
  526. {
  527. appendStringInfo(str,
  528. " CONST :consttype %u :constlen %d :constisnull %s :constvalue ",
  529.  node->consttype,
  530.  node->constlen,
  531.  node->constisnull ? "true" : "false");
  532. if (node->constisnull)
  533. appendStringInfo(str, "<>");
  534. else
  535. _outDatum(str, node->constvalue, node->consttype);
  536. appendStringInfo(str, " :constbyval %s ",
  537.  node->constbyval ? "true" : "false");
  538. }
  539. /*
  540.  * Aggref
  541.  */
  542. static void
  543. _outAggref(StringInfo str, Aggref *node)
  544. {
  545. appendStringInfo(str,
  546.  " AGGREG :aggname %s :basetype %u :aggtype %u :target ",
  547.  stringStringInfo(node->aggname),
  548.  node->basetype,
  549.  node->aggtype);
  550. _outNode(str, node->target);
  551. appendStringInfo(str, ":aggno %d :usenulls %s",
  552.  node->aggno,
  553.  node->usenulls ? "true" : "false");
  554. }
  555. /*
  556.  * SubLink
  557.  */
  558. static void
  559. _outSubLink(StringInfo str, SubLink *node)
  560. {
  561. appendStringInfo(str,
  562.  " SUBLINK :subLinkType %d :useor %s :lefthand ",
  563.  node->subLinkType,
  564.  node->useor ? "true" : "false");
  565. _outNode(str, node->lefthand);
  566. appendStringInfo(str, " :oper ");
  567. _outNode(str, node->oper);
  568. appendStringInfo(str, " :subselect ");
  569. _outNode(str, node->subselect);
  570. }
  571. /*
  572.  * Array is a subclass of Expr
  573.  */
  574. static void
  575. _outArray(StringInfo str, Array *node)
  576. {
  577. int i;
  578. appendStringInfo(str,
  579.   " ARRAY :arrayelemtype %u :arrayelemlength %d :arrayelembyval %c ",
  580.  node->arrayelemtype,
  581.  node->arrayelemlength,
  582.  node->arrayelembyval ? 't' : 'f');
  583. appendStringInfo(str, " :arrayndim %d :arraylow ", node->arrayndim);
  584. for (i = 0; i < node->arrayndim; i++)
  585. appendStringInfo(str, " %d ", node->arraylow.indx[i]);
  586. appendStringInfo(str, " :arrayhigh ");
  587. for (i = 0; i < node->arrayndim; i++)
  588. appendStringInfo(str, " %d ", node->arrayhigh.indx[i]);
  589. appendStringInfo(str, " :arraylen %d ", node->arraylen);
  590. }
  591. /*
  592.  * ArrayRef is a subclass of Expr
  593.  */
  594. static void
  595. _outArrayRef(StringInfo str, ArrayRef *node)
  596. {
  597. appendStringInfo(str,
  598. " ARRAYREF :refelemtype %u :refattrlength $d :refelemlength %d ",
  599.  node->refelemtype,
  600.  node->refattrlength,
  601.  node->refelemlength);
  602. appendStringInfo(str, " :refelembyval %c :refupperindex ",
  603.  node->refelembyval ? 't' : 'f');
  604. _outNode(str, node->refupperindexpr);
  605. appendStringInfo(str, " :reflowerindex ");
  606. _outNode(str, node->reflowerindexpr);
  607. appendStringInfo(str, " :refexpr ");
  608. _outNode(str, node->refexpr);
  609. appendStringInfo(str, " :refassgnexpr ");
  610. _outNode(str, node->refassgnexpr);
  611. }
  612. /*
  613.  * Func is a subclass of Expr
  614.  */
  615. static void
  616. _outFunc(StringInfo str, Func *node)
  617. {
  618. appendStringInfo(str,
  619.    " FUNC :funcid %u :functype %u :funcisindex %s :funcsize %d ",
  620.  node->funcid,
  621.  node->functype,
  622.  node->funcisindex ? "true" : "false",
  623.  node->funcsize);
  624. appendStringInfo(str, " :func_fcache @ 0x%x :func_tlist ",
  625.  (int) node->func_fcache);
  626. _outNode(str, node->func_tlist);
  627. appendStringInfo(str, " :func_planlist ");
  628. _outNode(str, node->func_planlist);
  629. }
  630. /*
  631.  * Oper is a subclass of Expr
  632.  */
  633. static void
  634. _outOper(StringInfo str, Oper *node)
  635. {
  636. appendStringInfo(str,
  637.  " OPER :opno %u :opid %u :opresulttype %u ",
  638.  node->opno,
  639.  node->opid,
  640.  node->opresulttype);
  641. }
  642. /*
  643.  * Param is a subclass of Expr
  644.  */
  645. static void
  646. _outParam(StringInfo str, Param *node)
  647. {
  648. appendStringInfo(str,
  649.  " PARAM :paramkind %d :paramid %d :paramname %s :paramtype %u ",
  650.  node->paramkind,
  651.  node->paramid,
  652.  stringStringInfo(node->paramname),
  653.  node->paramtype);
  654. appendStringInfo(str, " :param_tlist ");
  655. _outNode(str, node->param_tlist);
  656. }
  657. /*
  658.  * Stuff from execnodes.h
  659.  */
  660. /*
  661.  * EState is a subclass of Node.
  662.  */
  663. static void
  664. _outEState(StringInfo str, EState *node)
  665. {
  666. appendStringInfo(str,
  667.  " ESTATE :direction %d :range_table ",
  668.  node->es_direction);
  669. _outNode(str, node->es_range_table);
  670. appendStringInfo(str, " :result_relation_info @ 0x%x ",
  671.  (int) (node->es_result_relation_info));
  672. }
  673. /*
  674.  * Stuff from relation.h
  675.  */
  676. static void
  677. _outRelOptInfo(StringInfo str, RelOptInfo *node)
  678. {
  679. appendStringInfo(str, " RELOPTINFO :relids ");
  680. _outIntList(str, node->relids);
  681. appendStringInfo(str,
  682.  " :indexed %s :pages %u :tuples %u :size %u :width %u :targetlist ",
  683.  node->indexed ? "true" : "false",
  684.  node->pages,
  685.  node->tuples,
  686.  node->size,
  687.  node->width);
  688. _outNode(str, node->targetlist);
  689. appendStringInfo(str, " :pathlist ");
  690. _outNode(str, node->pathlist);
  691. /*
  692.  * Not sure if these are nodes or not. They're declared as struct
  693.  * Path *. Since i don't know, i'll just print the addresses for now.
  694.  * This can be changed later, if necessary.
  695.  */
  696. appendStringInfo(str,
  697.  " :cheapestpath @ 0x%x :pruneable %s :restrictinfo ",
  698.  (int) node->cheapestpath,
  699.  node->pruneable ? "true" : "false");
  700. _outNode(str, node->restrictinfo);
  701. appendStringInfo(str, " :joininfo ");
  702. _outNode(str, node->joininfo);
  703. appendStringInfo(str, " :innerjoin ");
  704. _outNode(str, node->innerjoin);
  705. }
  706. /*
  707.  * TargetEntry is a subclass of Node.
  708.  */
  709. static void
  710. _outTargetEntry(StringInfo str, TargetEntry *node)
  711. {
  712. appendStringInfo(str, " TARGETENTRY :resdom ");
  713. _outNode(str, node->resdom);
  714. appendStringInfo(str, " :expr ");
  715. _outNode(str, node->expr);
  716. }
  717. static void
  718. _outRangeTblEntry(StringInfo str, RangeTblEntry *node)
  719. {
  720. appendStringInfo(str,
  721.  " RTE :relname %s :refname %s :relid %u :inh %s :inFromCl %s :skipAcl %s",
  722.  stringStringInfo(node->relname),
  723.  stringStringInfo(node->refname),
  724.  node->relid,
  725.  node->inh ? "true" : "false",
  726.  node->inFromCl ? "true" : "false",
  727.  node->skipAcl ? "true" : "false");
  728. }
  729. static void
  730. _outRowMark(StringInfo str, RowMark *node)
  731. {
  732. appendStringInfo(str, " ROWMARK :rti %u :info %u", node->rti, node->info);
  733. }
  734. /*
  735.  * Path is a subclass of Node.
  736.  */
  737. static void
  738. _outPathOrder(StringInfo str, PathOrder *node)
  739. {
  740. appendStringInfo(str, " PATHORDER :ordtype %d ",
  741.  node->ordtype);
  742. if (node->ordtype == SORTOP_ORDER)
  743. {
  744. int i;
  745. appendStringInfo(str, " :sortop ");
  746. if (node->ord.sortop == NULL)
  747. appendStringInfo(str, "<>");
  748. else
  749. {
  750. for (i = 0; node->ord.sortop[i] != 0; i++)
  751. appendStringInfo(str, " %d ", node->ord.sortop[i]);
  752. appendStringInfo(str, " %d ", 0);
  753. }
  754. }
  755. else
  756. {
  757. appendStringInfo(str, " :merge ");
  758. _outNode(str, node->ord.merge);
  759. }
  760. }
  761. /*
  762.  * Path is a subclass of Node.
  763.  */
  764. static void
  765. _outPath(StringInfo str, Path *node)
  766. {
  767. appendStringInfo(str, " PATH :pathtype %d :cost %f :pathkeys ",
  768.  node->pathtype,
  769.  node->path_cost);
  770. _outNode(str, node->pathkeys);
  771. appendStringInfo(str, " :pathorder ");
  772. _outNode(str, node->pathorder);
  773. }
  774. /*
  775.  * IndexPath is a subclass of Path.
  776.  */
  777. static void
  778. _outIndexPath(StringInfo str, IndexPath *node)
  779. {
  780. appendStringInfo(str,
  781.  " INDEXPATH :pathtype %d :cost %f :pathkeys ",
  782.  node->path.pathtype,
  783.  node->path.path_cost);
  784. _outNode(str, node->path.pathkeys);
  785. appendStringInfo(str, " :pathorder ");
  786. _outNode(str, node->path.pathorder);
  787. appendStringInfo(str, " :indexid ");
  788. _outIntList(str, node->indexid);
  789. appendStringInfo(str, " :indexqual ");
  790. _outNode(str, node->indexqual);
  791. }
  792. /*
  793.  * NestPath is a subclass of Path
  794.  */
  795. static void
  796. _outNestPath(StringInfo str, NestPath *node)
  797. {
  798. appendStringInfo(str,
  799.  " NESTPATH :pathtype %d :cost %f :pathkeys ",
  800.  node->path.pathtype,
  801.  node->path.path_cost);
  802. _outNode(str, node->path.pathkeys);
  803. appendStringInfo(str, " :pathorder ");
  804. _outNode(str, node->path.pathorder);
  805. appendStringInfo(str, " :pathinfo ");
  806. _outNode(str, node->pathinfo);
  807. /*
  808.  * Not sure if these are nodes; they're declared as "struct path *".
  809.  * For now, i'll just print the addresses.
  810.  */
  811. appendStringInfo(str,
  812.  " :outerjoinpath @ 0x%x :innerjoinpath @ 0x%x :outjoincost %f :joinid ",
  813.  (int) node->outerjoinpath,
  814.  (int) node->innerjoinpath,
  815.  node->path.outerjoincost);
  816. _outIntList(str, node->path.joinid);
  817. }
  818. /*
  819.  * MergePath is a subclass of NestPath.
  820.  */
  821. static void
  822. _outMergePath(StringInfo str, MergePath *node)
  823. {
  824. appendStringInfo(str,
  825.  " MERGEPATH :pathtype %d :cost %f :pathkeys ",
  826.  node->jpath.path.pathtype,
  827.  node->jpath.path.path_cost);
  828. _outNode(str, node->jpath.path.pathkeys);
  829. appendStringInfo(str, " :pathorder ");
  830. _outNode(str, node->jpath.path.pathorder);
  831. appendStringInfo(str, " :pathinfo ");
  832. _outNode(str, node->jpath.pathinfo);
  833. /*
  834.  * Not sure if these are nodes; they're declared as "struct path *".
  835.  * For now, i'll just print the addresses.
  836.  */
  837. appendStringInfo(str,
  838.  " :outerjoinpath @ 0x%x :innerjoinpath @ 0x%x :outerjoincost %f :joinid ",
  839.  (int) node->jpath.outerjoinpath,
  840.  (int) node->jpath.innerjoinpath,
  841.  (int) node->jpath.path.outerjoincost);
  842. _outIntList(str, node->jpath.path.joinid);
  843. appendStringInfo(str, " :path_mergeclauses ");
  844. _outNode(str, node->path_mergeclauses);
  845. appendStringInfo(str, " :outersortkeys ");
  846. _outNode(str, node->outersortkeys);
  847. appendStringInfo(str, " :innersortkeys ");
  848. _outNode(str, node->innersortkeys);
  849. }
  850. /*
  851.  * HashPath is a subclass of NestPath.
  852.  */
  853. static void
  854. _outHashPath(StringInfo str, HashPath *node)
  855. {
  856. appendStringInfo(str,
  857.  " HASHPATH :pathtype %d :cost %f :pathkeys ",
  858.  node->jpath.path.pathtype,
  859.  node->jpath.path.path_cost);
  860. _outNode(str, node->jpath.path.pathkeys);
  861. appendStringInfo(str, " :pathorder ");
  862. _outNode(str, node->jpath.path.pathorder);
  863. appendStringInfo(str, " :pathinfo ");
  864. _outNode(str, node->jpath.pathinfo);
  865. /*
  866.  * Not sure if these are nodes; they're declared as "struct path *".
  867.  * For now, i'll just print the addresses.
  868.  */
  869. appendStringInfo(str,
  870.  " :outerjoinpath @ 0x%x :innerjoinpath @ 0x%x :outerjoincost %f :joinid ",
  871.  (int) node->jpath.outerjoinpath,
  872.  (int) node->jpath.innerjoinpath,
  873.  node->jpath.path.outerjoincost);
  874. _outIntList(str, node->jpath.path.joinid);
  875. appendStringInfo(str, " :path_hashclauses ");
  876. _outNode(str, node->path_hashclauses);
  877. appendStringInfo(str, " :outerhashkeys ");
  878. _outNode(str, node->outerhashkeys);
  879. appendStringInfo(str, " :innerhashkeys ");
  880. _outNode(str, node->innerhashkeys);
  881. }
  882. /*
  883.  * OrderKey is a subclass of Node.
  884.  */
  885. static void
  886. _outOrderKey(StringInfo str, OrderKey *node)
  887. {
  888. appendStringInfo(str,
  889.  " ORDERKEY :attribute_number %d :array_index %d ",
  890.  node->attribute_number,
  891.  node->array_index);
  892. }
  893. /*
  894.  * JoinKey is a subclass of Node.
  895.  */
  896. static void
  897. _outJoinKey(StringInfo str, JoinKey *node)
  898. {
  899. appendStringInfo(str, " JOINKEY :outer ");
  900. _outNode(str, node->outer);
  901. appendStringInfo(str, " :inner ");
  902. _outNode(str, node->inner);
  903. }
  904. /*
  905.  * MergeOrder is a subclass of Node.
  906.  */
  907. static void
  908. _outMergeOrder(StringInfo str, MergeOrder *node)
  909. {
  910. appendStringInfo(str,
  911. " MERGEORDER :join_operator %u :left_operator %u :right_operator %u ",
  912.  node->join_operator,
  913.  node->left_operator,
  914.  node->right_operator);
  915. appendStringInfo(str,
  916.  " :left_type %u :right_type %u ",
  917.  node->left_type,
  918.  node->right_type);
  919. }
  920. /*
  921.  * RestrictInfo is a subclass of Node.
  922.  */
  923. static void
  924. _outRestrictInfo(StringInfo str, RestrictInfo *node)
  925. {
  926. appendStringInfo(str, " RESTRICTINFO :clause ");
  927. _outNode(str, node->clause);
  928. appendStringInfo(str,
  929.  " :selectivity %f :notclause %s :indexids ",
  930.  node->selectivity,
  931.  node->notclause ? "true" : "false");
  932. _outNode(str, node->indexids);
  933. appendStringInfo(str, " :mergejoinorder ");
  934. _outNode(str, node->mergejoinorder);
  935. appendStringInfo(str, " :hashjoinoperator %u ", node->hashjoinoperator);
  936. }
  937. /*
  938.  * JoinMethod is a subclass of Node.
  939.  */
  940. static void
  941. _outJoinMethod(StringInfo str, JoinMethod *node)
  942. {
  943. appendStringInfo(str, " JOINMETHOD :jmkeys ");
  944. _outNode(str, node->jmkeys);
  945. appendStringInfo(str, " :clauses ");
  946. _outNode(str, node->clauses);
  947. }
  948. /*
  949.  * HashInfo is a subclass of JoinMethod.
  950.  */
  951. static void
  952. _outHashInfo(StringInfo str, HashInfo *node)
  953. {
  954. appendStringInfo(str, " HASHINFO :hashop %u :jmkeys ", node->hashop);
  955. _outNode(str, node->jmethod.jmkeys);
  956. appendStringInfo(str, " :clauses ");
  957. _outNode(str, node->jmethod.clauses);
  958. }
  959. /*
  960.  * JoinInfo is a subclass of Node.
  961.  */
  962. static void
  963. _outJoinInfo(StringInfo str, JoinInfo *node)
  964. {
  965. appendStringInfo(str, " JINFO :unjoined_relids ");
  966. _outIntList(str, node->unjoined_relids);
  967. appendStringInfo(str, " :jinfo_restrictinfo ");
  968. _outNode(str, node->jinfo_restrictinfo);
  969. appendStringInfo(str, " :mergejoinable %s :hashjoinable %s ",
  970.  node->mergejoinable ? "true" : "false",
  971.  node->hashjoinable ? "true" : "false");
  972. }
  973. /*
  974.  * Print the value of a Datum given its type.
  975.  */
  976. static void
  977. _outDatum(StringInfo str, Datum value, Oid type)
  978. {
  979. char    *s;
  980. Size length,
  981. typeLength;
  982. bool byValue;
  983. int i;
  984. /*
  985.  * find some information about the type and the "real" length of the
  986.  * datum.
  987.  */
  988. byValue = get_typbyval(type);
  989. typeLength = get_typlen(type);
  990. length = datumGetSize(value, type, byValue, typeLength);
  991. if (byValue)
  992. {
  993. s = (char *) (&value);
  994. appendStringInfo(str, " %d [ ", length);
  995. for (i = 0; i < sizeof(Datum); i++)
  996. appendStringInfo(str, " %d ", (int) (s[i]));
  997. appendStringInfo(str, "] ");
  998. }
  999. else
  1000. { /* !byValue */
  1001. s = (char *) DatumGetPointer(value);
  1002. if (!PointerIsValid(s))
  1003. appendStringInfo(str, " 0 [ ] ");
  1004. else
  1005. {
  1006. /*
  1007.  * length is unsigned - very bad to do < comparison to -1
  1008.  * without casting it to int first!! -mer 8 Jan 1991
  1009.  */
  1010. if (((int) length) <= -1)
  1011. length = VARSIZE(s);
  1012. appendStringInfo(str, " %d [ ", length);
  1013. for (i = 0; i < length; i++)
  1014. appendStringInfo(str, " %d ", (int) (s[i]));
  1015. appendStringInfo(str, "] ");
  1016. }
  1017. }
  1018. }
  1019. static void
  1020. _outIter(StringInfo str, Iter *node)
  1021. {
  1022. appendStringInfo(str, " ITER :iterexpr ");
  1023. _outNode(str, node->iterexpr);
  1024. }
  1025. static void
  1026. _outStream(StringInfo str, Stream *node)
  1027. {
  1028. appendStringInfo(str,
  1029.  " STREAM :pathptr @ 0x%x :cinfo @ 0x%x :clausetype %d :upstream @ 0x%x ",
  1030.  (int) node->pathptr,
  1031.  (int) node->cinfo,
  1032.  (int) node->clausetype,
  1033.  (int) node->upstream);
  1034. appendStringInfo(str,
  1035.    " :downstream @ 0x%x :groupup %d :groupcost %f :groupsel %f ",
  1036.  (int) node->downstream,
  1037.  node->groupup,
  1038.  node->groupcost,
  1039.  node->groupsel);
  1040. }
  1041. static void
  1042. _outAExpr(StringInfo str, A_Expr *node)
  1043. {
  1044. appendStringInfo(str, "EXPR ");
  1045. switch (node->oper)
  1046. {
  1047. case AND:
  1048. appendStringInfo(str, "AND");
  1049. break;
  1050. case OR:
  1051. appendStringInfo(str, "OR");
  1052. break;
  1053. case NOT:
  1054. appendStringInfo(str, "NOT");
  1055. break;
  1056. case ISNULL:
  1057. appendStringInfo(str, "ISNULL");
  1058. break;
  1059. case NOTNULL:
  1060. appendStringInfo(str, "NOTNULL");
  1061. break;
  1062. default:
  1063. appendStringInfo(str, stringStringInfo(node->opname));
  1064. break;
  1065. }
  1066. _outNode(str, node->lexpr);
  1067. _outNode(str, node->rexpr);
  1068. return;
  1069. }
  1070. static void
  1071. _outValue(StringInfo str, Value *value)
  1072. {
  1073. switch (value->type)
  1074. {
  1075. case T_String:
  1076. appendStringInfo(str, " "%s" ", stringStringInfo(value->val.str));
  1077. break;
  1078. case T_Integer:
  1079. appendStringInfo(str, " %ld ", value->val.ival);
  1080. break;
  1081. case T_Float:
  1082. appendStringInfo(str, " %f ", value->val.dval);
  1083. break;
  1084. default:
  1085. break;
  1086. }
  1087. return;
  1088. }
  1089. static void
  1090. _outIdent(StringInfo str, Ident *node)
  1091. {
  1092. appendStringInfo(str, " IDENT "%s" ", stringStringInfo(node->name));
  1093. return;
  1094. }
  1095. static void
  1096. _outAttr(StringInfo str, Attr *node)
  1097. {
  1098. List    *l;
  1099. appendStringInfo(str, " ATTR "%s" ", stringStringInfo(node->relname));
  1100. appendStringInfo(str, "(");
  1101. foreach(l, node->attrs)
  1102. {
  1103. _outNode(str, lfirst(l));
  1104. if (lnext(l))
  1105. appendStringInfo(str, ",");
  1106. }
  1107. appendStringInfo(str, ")");
  1108. return;
  1109. }
  1110. static void
  1111. _outAConst(StringInfo str, A_Const *node)
  1112. {
  1113. appendStringInfo(str, "CONST ");
  1114. _outValue(str, &(node->val));
  1115. return;
  1116. }
  1117. static void
  1118. _outConstraint(StringInfo str, Constraint *node)
  1119. {
  1120. appendStringInfo(str, " %s :type", stringStringInfo(node->name));
  1121. switch (node->contype)
  1122. {
  1123. case CONSTR_PRIMARY:
  1124. appendStringInfo(str, " PRIMARY KEY ");
  1125. _outNode(str, node->keys);
  1126. break;
  1127. case CONSTR_CHECK:
  1128. appendStringInfo(str, " CHECK %s", stringStringInfo(node->def));
  1129. break;
  1130. case CONSTR_DEFAULT:
  1131. appendStringInfo(str, " DEFAULT %s", stringStringInfo(node->def));
  1132. break;
  1133. case CONSTR_NOTNULL:
  1134. appendStringInfo(str, " NOT NULL ");
  1135. break;
  1136. case CONSTR_UNIQUE:
  1137. appendStringInfo(str, " UNIQUE ");
  1138. _outNode(str, node->keys);
  1139. break;
  1140. default:
  1141. appendStringInfo(str, "<unrecognized constraint>");
  1142. break;
  1143. }
  1144. return;
  1145. }
  1146. static void
  1147. _outCaseExpr(StringInfo str, CaseExpr *node)
  1148. {
  1149. appendStringInfo(str, "CASE ");
  1150. _outNode(str, node->args);
  1151. appendStringInfo(str, " :default ");
  1152. _outNode(str, node->defresult);
  1153. return;
  1154. }
  1155. static void
  1156. _outCaseWhen(StringInfo str, CaseWhen *node)
  1157. {
  1158. appendStringInfo(str, " WHEN ");
  1159. _outNode(str, node->expr);
  1160. appendStringInfo(str, " :then ");
  1161. _outNode(str, node->result);
  1162. return;
  1163. }
  1164. /*
  1165.  * _outNode -
  1166.  *   converts a Node into ascii string and append it to 'str'
  1167.  */
  1168. static void
  1169. _outNode(StringInfo str, void *obj)
  1170. {
  1171. if (obj == NULL)
  1172. {
  1173. appendStringInfo(str, "<>");
  1174. return;
  1175. }
  1176. if (nodeTag(obj) == T_List)
  1177. {
  1178. List    *l;
  1179. appendStringInfo(str, "(");
  1180. foreach(l, (List *) obj)
  1181. {
  1182. _outNode(str, lfirst(l));
  1183. if (lnext(l))
  1184. appendStringInfo(str, " ");
  1185. }
  1186. appendStringInfo(str, ")");
  1187. }
  1188. else
  1189. {
  1190. appendStringInfo(str, "{");
  1191. switch (nodeTag(obj))
  1192. {
  1193. case T_CreateStmt:
  1194. _outCreateStmt(str, obj);
  1195. break;
  1196. case T_IndexStmt:
  1197. _outIndexStmt(str, obj);
  1198. break;
  1199. case T_ColumnDef:
  1200. _outColumnDef(str, obj);
  1201. break;
  1202. case T_TypeName:
  1203. _outTypeName(str, obj);
  1204. break;
  1205. case T_IndexElem:
  1206. _outIndexElem(str, obj);
  1207. break;
  1208. case T_Query:
  1209. _outQuery(str, obj);
  1210. break;
  1211. case T_SortClause:
  1212. _outSortClause(str, obj);
  1213. break;
  1214. case T_GroupClause:
  1215. _outGroupClause(str, obj);
  1216. break;
  1217. case T_Plan:
  1218. _outPlan(str, obj);
  1219. break;
  1220. case T_Result:
  1221. _outResult(str, obj);
  1222. break;
  1223. case T_Append:
  1224. _outAppend(str, obj);
  1225. break;
  1226. case T_Join:
  1227. _outJoin(str, obj);
  1228. break;
  1229. case T_NestLoop:
  1230. _outNestLoop(str, obj);
  1231. break;
  1232. case T_MergeJoin:
  1233. _outMergeJoin(str, obj);
  1234. break;
  1235. case T_HashJoin:
  1236. _outHashJoin(str, obj);
  1237. break;
  1238. case T_Scan:
  1239. _outScan(str, obj);
  1240. break;
  1241. case T_SeqScan:
  1242. _outSeqScan(str, obj);
  1243. break;
  1244. case T_IndexScan:
  1245. _outIndexScan(str, obj);
  1246. break;
  1247. case T_Noname:
  1248. _outNoname(str, obj);
  1249. break;
  1250. case T_Sort:
  1251. _outSort(str, obj);
  1252. break;
  1253. case T_Agg:
  1254. _outAgg(str, obj);
  1255. break;
  1256. case T_Group:
  1257. _outGroup(str, obj);
  1258. break;
  1259. case T_Unique:
  1260. _outUnique(str, obj);
  1261. break;
  1262. case T_Hash:
  1263. _outHash(str, obj);
  1264. break;
  1265. case T_SubPlan:
  1266. _outSubPlan(str, obj);
  1267. break;
  1268. case T_Resdom:
  1269. _outResdom(str, obj);
  1270. break;
  1271. case T_Fjoin:
  1272. _outFjoin(str, obj);
  1273. break;
  1274. case T_Expr:
  1275. _outExpr(str, obj);
  1276. break;
  1277. case T_Var:
  1278. _outVar(str, obj);
  1279. break;
  1280. case T_Const:
  1281. _outConst(str, obj);
  1282. break;
  1283. case T_Aggref:
  1284. _outAggref(str, obj);
  1285. break;
  1286. case T_SubLink:
  1287. _outSubLink(str, obj);
  1288. break;
  1289. case T_Array:
  1290. _outArray(str, obj);
  1291. break;
  1292. case T_ArrayRef:
  1293. _outArrayRef(str, obj);
  1294. break;
  1295. case T_Func:
  1296. _outFunc(str, obj);
  1297. break;
  1298. case T_Oper:
  1299. _outOper(str, obj);
  1300. break;
  1301. case T_Param:
  1302. _outParam(str, obj);
  1303. break;
  1304. case T_EState:
  1305. _outEState(str, obj);
  1306. break;
  1307. case T_RelOptInfo:
  1308. _outRelOptInfo(str, obj);
  1309. break;
  1310. case T_TargetEntry:
  1311. _outTargetEntry(str, obj);
  1312. break;
  1313. case T_RangeTblEntry:
  1314. _outRangeTblEntry(str, obj);
  1315. break;
  1316. case T_RowMark:
  1317. _outRowMark(str, obj);
  1318. break;
  1319. case T_PathOrder:
  1320. _outPathOrder(str, obj);
  1321. break;
  1322. case T_Path:
  1323. _outPath(str, obj);
  1324. break;
  1325. case T_IndexPath:
  1326. _outIndexPath(str, obj);
  1327. break;
  1328. case T_NestPath:
  1329. _outNestPath(str, obj);
  1330. break;
  1331. case T_MergePath:
  1332. _outMergePath(str, obj);
  1333. break;
  1334. case T_HashPath:
  1335. _outHashPath(str, obj);
  1336. break;
  1337. case T_OrderKey:
  1338. _outOrderKey(str, obj);
  1339. break;
  1340. case T_JoinKey:
  1341. _outJoinKey(str, obj);
  1342. break;
  1343. case T_MergeOrder:
  1344. _outMergeOrder(str, obj);
  1345. break;
  1346. case T_RestrictInfo:
  1347. _outRestrictInfo(str, obj);
  1348. break;
  1349. case T_JoinMethod:
  1350. _outJoinMethod(str, obj);
  1351. break;
  1352. case T_HashInfo:
  1353. _outHashInfo(str, obj);
  1354. break;
  1355. case T_JoinInfo:
  1356. _outJoinInfo(str, obj);
  1357. break;
  1358. case T_Iter:
  1359. _outIter(str, obj);
  1360. break;
  1361. case T_Stream:
  1362. _outStream(str, obj);
  1363. break;
  1364. case T_Integer:
  1365. case T_String:
  1366. case T_Float:
  1367. _outValue(str, obj);
  1368. break;
  1369. case T_A_Expr:
  1370. _outAExpr(str, obj);
  1371. break;
  1372. case T_Ident:
  1373. _outIdent(str, obj);
  1374. break;
  1375. case T_A_Const:
  1376. _outAConst(str, obj);
  1377. break;
  1378. case T_Constraint:
  1379. _outConstraint(str, obj);
  1380. break;
  1381. case T_CaseExpr:
  1382. _outCaseExpr(str, obj);
  1383. break;
  1384. case T_CaseWhen:
  1385. _outCaseWhen(str, obj);
  1386. break;
  1387. case T_VariableSetStmt:
  1388. break;
  1389. case T_SelectStmt:
  1390. _outSelectStmt(str, obj);
  1391. break;
  1392. case T_FuncCall:
  1393. _outFuncCall(str, obj);
  1394. break;
  1395. case T_Attr:
  1396. _outAttr(str, obj);
  1397. break;
  1398. default:
  1399. elog(NOTICE, "_outNode: don't know how to print type %d ",
  1400.  nodeTag(obj));
  1401. break;
  1402. }
  1403. appendStringInfo(str, "}");
  1404. }
  1405. return;
  1406. }
  1407. /*
  1408.  * nodeToString -
  1409.  *    returns the ascii representation of the Node as a palloc'd string
  1410.  */
  1411. char *
  1412. nodeToString(void *obj)
  1413. {
  1414. StringInfoData str;
  1415. /* see stringinfo.h for an explanation of this maneuver */
  1416. initStringInfo(&str);
  1417. _outNode(&str, obj);
  1418. return str.data;
  1419. }