Commit 53e75768 authored by Tom Lane's avatar Tom Lane

Make NestLoop plan nodes pass outer-relation variables into their inner

relation using the general PARAM_EXEC executor parameter mechanism, rather
than the ad-hoc kluge of passing the outer tuple down through ExecReScan.
The previous method was hard to understand and could never be extended to
handle parameters coming from multiple join levels.  This patch doesn't
change the set of possible plans nor have any significant performance effect,
but it's necessary infrastructure for future generalization of the concept
of an inner indexscan plan.

ExecReScan's second parameter is now unused, so it's removed.
parent 5a348935
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/backend/executor/execAmi.c,v 1.108 2010/02/14 18:42:14 rhaas Exp $ * $PostgreSQL: pgsql/src/backend/executor/execAmi.c,v 1.109 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -59,17 +59,9 @@ static bool IndexSupportsBackwardScan(Oid indexid); ...@@ -59,17 +59,9 @@ static bool IndexSupportsBackwardScan(Oid indexid);
* *
* Note that if the plan node has parameters that have changed value, * Note that if the plan node has parameters that have changed value,
* the output might be different from last time. * the output might be different from last time.
*
* The second parameter is currently only used to pass a NestLoop plan's
* econtext down to its inner child plan, in case that is an indexscan that
* needs access to variables of the current outer tuple. (The handling of
* this parameter is currently pretty inconsistent: some callers pass NULL
* and some pass down their parent's value; so don't rely on it in other
* situations. It'd probably be better to remove the whole thing and use
* the generalized parameter mechanism instead.)
*/ */
void void
ExecReScan(PlanState *node, ExprContext *exprCtxt) ExecReScan(PlanState *node)
{ {
/* If collecting timing stats, update them */ /* If collecting timing stats, update them */
if (node->instrument) if (node->instrument)
...@@ -126,119 +118,119 @@ ExecReScan(PlanState *node, ExprContext *exprCtxt) ...@@ -126,119 +118,119 @@ ExecReScan(PlanState *node, ExprContext *exprCtxt)
switch (nodeTag(node)) switch (nodeTag(node))
{ {
case T_ResultState: case T_ResultState:
ExecReScanResult((ResultState *) node, exprCtxt); ExecReScanResult((ResultState *) node);
break; break;
case T_ModifyTableState: case T_ModifyTableState:
ExecReScanModifyTable((ModifyTableState *) node, exprCtxt); ExecReScanModifyTable((ModifyTableState *) node);
break; break;
case T_AppendState: case T_AppendState:
ExecReScanAppend((AppendState *) node, exprCtxt); ExecReScanAppend((AppendState *) node);
break; break;
case T_RecursiveUnionState: case T_RecursiveUnionState:
ExecRecursiveUnionReScan((RecursiveUnionState *) node, exprCtxt); ExecReScanRecursiveUnion((RecursiveUnionState *) node);
break; break;
case T_BitmapAndState: case T_BitmapAndState:
ExecReScanBitmapAnd((BitmapAndState *) node, exprCtxt); ExecReScanBitmapAnd((BitmapAndState *) node);
break; break;
case T_BitmapOrState: case T_BitmapOrState:
ExecReScanBitmapOr((BitmapOrState *) node, exprCtxt); ExecReScanBitmapOr((BitmapOrState *) node);
break; break;
case T_SeqScanState: case T_SeqScanState:
ExecSeqReScan((SeqScanState *) node, exprCtxt); ExecReScanSeqScan((SeqScanState *) node);
break; break;
case T_IndexScanState: case T_IndexScanState:
ExecIndexReScan((IndexScanState *) node, exprCtxt); ExecReScanIndexScan((IndexScanState *) node);
break; break;
case T_BitmapIndexScanState: case T_BitmapIndexScanState:
ExecBitmapIndexReScan((BitmapIndexScanState *) node, exprCtxt); ExecReScanBitmapIndexScan((BitmapIndexScanState *) node);
break; break;
case T_BitmapHeapScanState: case T_BitmapHeapScanState:
ExecBitmapHeapReScan((BitmapHeapScanState *) node, exprCtxt); ExecReScanBitmapHeapScan((BitmapHeapScanState *) node);
break; break;
case T_TidScanState: case T_TidScanState:
ExecTidReScan((TidScanState *) node, exprCtxt); ExecReScanTidScan((TidScanState *) node);
break; break;
case T_SubqueryScanState: case T_SubqueryScanState:
ExecSubqueryReScan((SubqueryScanState *) node, exprCtxt); ExecReScanSubqueryScan((SubqueryScanState *) node);
break; break;
case T_FunctionScanState: case T_FunctionScanState:
ExecFunctionReScan((FunctionScanState *) node, exprCtxt); ExecReScanFunctionScan((FunctionScanState *) node);
break; break;
case T_ValuesScanState: case T_ValuesScanState:
ExecValuesReScan((ValuesScanState *) node, exprCtxt); ExecReScanValuesScan((ValuesScanState *) node);
break; break;
case T_CteScanState: case T_CteScanState:
ExecCteScanReScan((CteScanState *) node, exprCtxt); ExecReScanCteScan((CteScanState *) node);
break; break;
case T_WorkTableScanState: case T_WorkTableScanState:
ExecWorkTableScanReScan((WorkTableScanState *) node, exprCtxt); ExecReScanWorkTableScan((WorkTableScanState *) node);
break; break;
case T_NestLoopState: case T_NestLoopState:
ExecReScanNestLoop((NestLoopState *) node, exprCtxt); ExecReScanNestLoop((NestLoopState *) node);
break; break;
case T_MergeJoinState: case T_MergeJoinState:
ExecReScanMergeJoin((MergeJoinState *) node, exprCtxt); ExecReScanMergeJoin((MergeJoinState *) node);
break; break;
case T_HashJoinState: case T_HashJoinState:
ExecReScanHashJoin((HashJoinState *) node, exprCtxt); ExecReScanHashJoin((HashJoinState *) node);
break; break;
case T_MaterialState: case T_MaterialState:
ExecMaterialReScan((MaterialState *) node, exprCtxt); ExecReScanMaterial((MaterialState *) node);
break; break;
case T_SortState: case T_SortState:
ExecReScanSort((SortState *) node, exprCtxt); ExecReScanSort((SortState *) node);
break; break;
case T_GroupState: case T_GroupState:
ExecReScanGroup((GroupState *) node, exprCtxt); ExecReScanGroup((GroupState *) node);
break; break;
case T_AggState: case T_AggState:
ExecReScanAgg((AggState *) node, exprCtxt); ExecReScanAgg((AggState *) node);
break; break;
case T_WindowAggState: case T_WindowAggState:
ExecReScanWindowAgg((WindowAggState *) node, exprCtxt); ExecReScanWindowAgg((WindowAggState *) node);
break; break;
case T_UniqueState: case T_UniqueState:
ExecReScanUnique((UniqueState *) node, exprCtxt); ExecReScanUnique((UniqueState *) node);
break; break;
case T_HashState: case T_HashState:
ExecReScanHash((HashState *) node, exprCtxt); ExecReScanHash((HashState *) node);
break; break;
case T_SetOpState: case T_SetOpState:
ExecReScanSetOp((SetOpState *) node, exprCtxt); ExecReScanSetOp((SetOpState *) node);
break; break;
case T_LockRowsState: case T_LockRowsState:
ExecReScanLockRows((LockRowsState *) node, exprCtxt); ExecReScanLockRows((LockRowsState *) node);
break; break;
case T_LimitState: case T_LimitState:
ExecReScanLimit((LimitState *) node, exprCtxt); ExecReScanLimit((LimitState *) node);
break; break;
default: default:
......
...@@ -26,7 +26,7 @@ ...@@ -26,7 +26,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/execMain.c,v 1.350 2010/07/09 14:06:01 rhaas Exp $ * $PostgreSQL: pgsql/src/backend/executor/execMain.c,v 1.351 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -400,7 +400,7 @@ ExecutorRewind(QueryDesc *queryDesc) ...@@ -400,7 +400,7 @@ ExecutorRewind(QueryDesc *queryDesc)
/* /*
* rescan plan * rescan plan
*/ */
ExecReScan(queryDesc->planstate, NULL); ExecReScan(queryDesc->planstate);
MemoryContextSwitchTo(oldcontext); MemoryContextSwitchTo(oldcontext);
} }
......
...@@ -12,7 +12,7 @@ ...@@ -12,7 +12,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/execProcnode.c,v 1.70 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/execProcnode.c,v 1.71 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -341,7 +341,7 @@ ExecProcNode(PlanState *node) ...@@ -341,7 +341,7 @@ ExecProcNode(PlanState *node)
CHECK_FOR_INTERRUPTS(); CHECK_FOR_INTERRUPTS();
if (node->chgParam != NULL) /* something changed */ if (node->chgParam != NULL) /* something changed */
ExecReScan(node, NULL); /* let ReScan handle this */ ExecReScan(node); /* let ReScan handle this */
if (node->instrument) if (node->instrument)
InstrStartNode(node->instrument); InstrStartNode(node->instrument);
...@@ -504,7 +504,7 @@ MultiExecProcNode(PlanState *node) ...@@ -504,7 +504,7 @@ MultiExecProcNode(PlanState *node)
CHECK_FOR_INTERRUPTS(); CHECK_FOR_INTERRUPTS();
if (node->chgParam != NULL) /* something changed */ if (node->chgParam != NULL) /* something changed */
ExecReScan(node, NULL); /* let ReScan handle this */ ExecReScan(node); /* let ReScan handle this */
switch (nodeTag(node)) switch (nodeTag(node))
{ {
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.263 2010/02/26 02:00:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/execQual.c,v 1.264 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -78,7 +78,9 @@ static Datum ExecEvalWholeRowSlow(ExprState *exprstate, ExprContext *econtext, ...@@ -78,7 +78,9 @@ static Datum ExecEvalWholeRowSlow(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone); bool *isNull, ExprDoneCond *isDone);
static Datum ExecEvalConst(ExprState *exprstate, ExprContext *econtext, static Datum ExecEvalConst(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone); bool *isNull, ExprDoneCond *isDone);
static Datum ExecEvalParam(ExprState *exprstate, ExprContext *econtext, static Datum ExecEvalParamExec(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone);
static Datum ExecEvalParamExtern(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone); bool *isNull, ExprDoneCond *isDone);
static void init_fcache(Oid foid, FuncExprState *fcache, static void init_fcache(Oid foid, FuncExprState *fcache,
MemoryContext fcacheCxt, bool needDescForSets); MemoryContext fcacheCxt, bool needDescForSets);
...@@ -961,80 +963,87 @@ ExecEvalConst(ExprState *exprstate, ExprContext *econtext, ...@@ -961,80 +963,87 @@ ExecEvalConst(ExprState *exprstate, ExprContext *econtext,
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecEvalParam * ExecEvalParamExec
* *
* Returns the value of a parameter. A param node contains * Returns the value of a PARAM_EXEC parameter.
* something like ($.name) and the expression context contains
* the current parameter bindings (name = "sam") (age = 34)...
* so our job is to find and return the appropriate datum ("sam").
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
static Datum static Datum
ExecEvalParam(ExprState *exprstate, ExprContext *econtext, ExecEvalParamExec(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone) bool *isNull, ExprDoneCond *isDone)
{ {
Param *expression = (Param *) exprstate->expr; Param *expression = (Param *) exprstate->expr;
int thisParamId = expression->paramid; int thisParamId = expression->paramid;
ParamExecData *prm;
if (isDone) if (isDone)
*isDone = ExprSingleResult; *isDone = ExprSingleResult;
if (expression->paramkind == PARAM_EXEC) /*
* PARAM_EXEC params (internal executor parameters) are stored in the
* ecxt_param_exec_vals array, and can be accessed by array index.
*/
prm = &(econtext->ecxt_param_exec_vals[thisParamId]);
if (prm->execPlan != NULL)
{ {
/* /* Parameter not evaluated yet, so go do it */
* PARAM_EXEC params (internal executor parameters) are stored in the ExecSetParamPlan(prm->execPlan, econtext);
* ecxt_param_exec_vals array, and can be accessed by array index. /* ExecSetParamPlan should have processed this param... */
*/ Assert(prm->execPlan == NULL);
ParamExecData *prm;
prm = &(econtext->ecxt_param_exec_vals[thisParamId]);
if (prm->execPlan != NULL)
{
/* Parameter not evaluated yet, so go do it */
ExecSetParamPlan(prm->execPlan, econtext);
/* ExecSetParamPlan should have processed this param... */
Assert(prm->execPlan == NULL);
}
*isNull = prm->isnull;
return prm->value;
} }
else *isNull = prm->isnull;
{ return prm->value;
/* }
* PARAM_EXTERN parameters must be sought in ecxt_param_list_info.
*/
ParamListInfo paramInfo = econtext->ecxt_param_list_info;
Assert(expression->paramkind == PARAM_EXTERN); /* ----------------------------------------------------------------
if (paramInfo && * ExecEvalParamExtern
thisParamId > 0 && thisParamId <= paramInfo->numParams) *
{ * Returns the value of a PARAM_EXTERN parameter.
ParamExternData *prm = &paramInfo->params[thisParamId - 1]; * ----------------------------------------------------------------
*/
static Datum
ExecEvalParamExtern(ExprState *exprstate, ExprContext *econtext,
bool *isNull, ExprDoneCond *isDone)
{
Param *expression = (Param *) exprstate->expr;
int thisParamId = expression->paramid;
ParamListInfo paramInfo = econtext->ecxt_param_list_info;
/* give hook a chance in case parameter is dynamic */ if (isDone)
if (!OidIsValid(prm->ptype) && paramInfo->paramFetch != NULL) *isDone = ExprSingleResult;
(*paramInfo->paramFetch) (paramInfo, thisParamId);
if (OidIsValid(prm->ptype)) /*
{ * PARAM_EXTERN parameters must be sought in ecxt_param_list_info.
/* safety check in case hook did something unexpected */ */
if (prm->ptype != expression->paramtype) if (paramInfo &&
ereport(ERROR, thisParamId > 0 && thisParamId <= paramInfo->numParams)
(errcode(ERRCODE_DATATYPE_MISMATCH), {
errmsg("type of parameter %d (%s) does not match that when preparing the plan (%s)", ParamExternData *prm = &paramInfo->params[thisParamId - 1];
thisParamId,
format_type_be(prm->ptype),
format_type_be(expression->paramtype))));
*isNull = prm->isnull; /* give hook a chance in case parameter is dynamic */
return prm->value; if (!OidIsValid(prm->ptype) && paramInfo->paramFetch != NULL)
} (*paramInfo->paramFetch) (paramInfo, thisParamId);
if (OidIsValid(prm->ptype))
{
/* safety check in case hook did something unexpected */
if (prm->ptype != expression->paramtype)
ereport(ERROR,
(errcode(ERRCODE_DATATYPE_MISMATCH),
errmsg("type of parameter %d (%s) does not match that when preparing the plan (%s)",
thisParamId,
format_type_be(prm->ptype),
format_type_be(expression->paramtype))));
*isNull = prm->isnull;
return prm->value;
} }
ereport(ERROR,
(errcode(ERRCODE_UNDEFINED_OBJECT),
errmsg("no value found for parameter %d", thisParamId)));
return (Datum) 0; /* keep compiler quiet */
} }
ereport(ERROR,
(errcode(ERRCODE_UNDEFINED_OBJECT),
errmsg("no value found for parameter %d", thisParamId)));
return (Datum) 0; /* keep compiler quiet */
} }
...@@ -4228,7 +4237,19 @@ ExecInitExpr(Expr *node, PlanState *parent) ...@@ -4228,7 +4237,19 @@ ExecInitExpr(Expr *node, PlanState *parent)
break; break;
case T_Param: case T_Param:
state = (ExprState *) makeNode(ExprState); state = (ExprState *) makeNode(ExprState);
state->evalfunc = ExecEvalParam; switch (((Param *) node)->paramkind)
{
case PARAM_EXEC:
state->evalfunc = ExecEvalParamExec;
break;
case PARAM_EXTERN:
state->evalfunc = ExecEvalParamExtern;
break;
default:
elog(ERROR, "unrecognized paramkind: %d",
(int) ((Param *) node)->paramkind);
break;
}
break; break;
case T_CoerceToDomainValue: case T_CoerceToDomainValue:
state = (ExprState *) makeNode(ExprState); state = (ExprState *) makeNode(ExprState);
......
...@@ -71,7 +71,7 @@ ...@@ -71,7 +71,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeAgg.c,v 1.175 2010/02/26 02:00:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeAgg.c,v 1.176 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1886,7 +1886,7 @@ ExecEndAgg(AggState *node) ...@@ -1886,7 +1886,7 @@ ExecEndAgg(AggState *node)
} }
void void
ExecReScanAgg(AggState *node, ExprContext *exprCtxt) ExecReScanAgg(AggState *node)
{ {
ExprContext *econtext = node->ss.ps.ps_ExprContext; ExprContext *econtext = node->ss.ps.ps_ExprContext;
int aggno; int aggno;
...@@ -1911,7 +1911,7 @@ ExecReScanAgg(AggState *node, ExprContext *exprCtxt) ...@@ -1911,7 +1911,7 @@ ExecReScanAgg(AggState *node, ExprContext *exprCtxt)
* parameter changes, then we can just rescan the existing hash table; * parameter changes, then we can just rescan the existing hash table;
* no need to build it again. * no need to build it again.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
{ {
ResetTupleHashIterator(node->hashtable, &node->hashiter); ResetTupleHashIterator(node->hashtable, &node->hashiter);
return; return;
...@@ -1967,8 +1967,8 @@ ExecReScanAgg(AggState *node, ExprContext *exprCtxt) ...@@ -1967,8 +1967,8 @@ ExecReScanAgg(AggState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ss.ps.lefttree);
} }
/* /*
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeAppend.c,v 1.77 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeAppend.c,v 1.78 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -263,7 +263,7 @@ ExecEndAppend(AppendState *node) ...@@ -263,7 +263,7 @@ ExecEndAppend(AppendState *node)
} }
void void
ExecReScanAppend(AppendState *node, ExprContext *exprCtxt) ExecReScanAppend(AppendState *node)
{ {
int i; int i;
...@@ -280,12 +280,10 @@ ExecReScanAppend(AppendState *node, ExprContext *exprCtxt) ...@@ -280,12 +280,10 @@ ExecReScanAppend(AppendState *node, ExprContext *exprCtxt)
/* /*
* If chgParam of subnode is not null then plan will be re-scanned by * If chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. However, if caller is passing us an exprCtxt * first ExecProcNode.
* then forcibly rescan all the subnodes now, so that we can pass the
* exprCtxt down to the subnodes (needed for appendrel indexscan).
*/ */
if (subnode->chgParam == NULL || exprCtxt != NULL) if (subnode->chgParam == NULL)
ExecReScan(subnode, exprCtxt); ExecReScan(subnode);
} }
node->as_whichplan = 0; node->as_whichplan = 0;
exec_append_initialize_next(node); exec_append_initialize_next(node);
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeBitmapAnd.c,v 1.13 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeBitmapAnd.c,v 1.14 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -185,7 +185,7 @@ ExecEndBitmapAnd(BitmapAndState *node) ...@@ -185,7 +185,7 @@ ExecEndBitmapAnd(BitmapAndState *node)
} }
void void
ExecReScanBitmapAnd(BitmapAndState *node, ExprContext *exprCtxt) ExecReScanBitmapAnd(BitmapAndState *node)
{ {
int i; int i;
...@@ -201,9 +201,10 @@ ExecReScanBitmapAnd(BitmapAndState *node, ExprContext *exprCtxt) ...@@ -201,9 +201,10 @@ ExecReScanBitmapAnd(BitmapAndState *node, ExprContext *exprCtxt)
UpdateChangedParamSet(subnode, node->ps.chgParam); UpdateChangedParamSet(subnode, node->ps.chgParam);
/* /*
* Always rescan the inputs immediately, to ensure we can pass down * If chgParam of subnode is not null then plan will be re-scanned by
* any outer tuple that might be used in index quals. * first ExecProcNode.
*/ */
ExecReScan(subnode, exprCtxt); if (subnode->chgParam == NULL)
ExecReScan(subnode);
} }
} }
...@@ -21,7 +21,7 @@ ...@@ -21,7 +21,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeBitmapHeapscan.c,v 1.38 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeBitmapHeapscan.c,v 1.39 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -30,7 +30,7 @@ ...@@ -30,7 +30,7 @@
* ExecBitmapHeapScan scans a relation using bitmap info * ExecBitmapHeapScan scans a relation using bitmap info
* ExecBitmapHeapNext workhorse for above * ExecBitmapHeapNext workhorse for above
* ExecInitBitmapHeapScan creates and initializes state info. * ExecInitBitmapHeapScan creates and initializes state info.
* ExecBitmapHeapReScan prepares to rescan the plan. * ExecReScanBitmapHeapScan prepares to rescan the plan.
* ExecEndBitmapHeapScan releases all storage. * ExecEndBitmapHeapScan releases all storage.
*/ */
#include "postgres.h" #include "postgres.h"
...@@ -420,24 +420,12 @@ ExecBitmapHeapScan(BitmapHeapScanState *node) ...@@ -420,24 +420,12 @@ ExecBitmapHeapScan(BitmapHeapScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecBitmapHeapReScan(node) * ExecReScanBitmapHeapScan(node)
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecBitmapHeapReScan(BitmapHeapScanState *node, ExprContext *exprCtxt) ExecReScanBitmapHeapScan(BitmapHeapScanState *node)
{ {
/*
* If we are being passed an outer tuple, link it into the "regular"
* per-tuple econtext for possible qual eval.
*/
if (exprCtxt != NULL)
{
ExprContext *stdecontext;
stdecontext = node->ss.ps.ps_ExprContext;
stdecontext->ecxt_outertuple = exprCtxt->ecxt_outertuple;
}
/* rescan to release any page pin */ /* rescan to release any page pin */
heap_rescan(node->ss.ss_currentScanDesc, NULL); heap_rescan(node->ss.ss_currentScanDesc, NULL);
...@@ -455,10 +443,11 @@ ExecBitmapHeapReScan(BitmapHeapScanState *node, ExprContext *exprCtxt) ...@@ -455,10 +443,11 @@ ExecBitmapHeapReScan(BitmapHeapScanState *node, ExprContext *exprCtxt)
ExecScanReScan(&node->ss); ExecScanReScan(&node->ss);
/* /*
* Always rescan the input immediately, to ensure we can pass down any * if chgParam of subnode is not null then plan will be re-scanned by
* outer tuple that might be used in index quals. * first ExecProcNode.
*/ */
ExecReScan(outerPlanState(node), exprCtxt); if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(node->ss.ps.lefttree);
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeBitmapIndexscan.c,v 1.33 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeBitmapIndexscan.c,v 1.34 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
* INTERFACE ROUTINES * INTERFACE ROUTINES
* MultiExecBitmapIndexScan scans a relation using index. * MultiExecBitmapIndexScan scans a relation using index.
* ExecInitBitmapIndexScan creates and initializes state info. * ExecInitBitmapIndexScan creates and initializes state info.
* ExecBitmapIndexReScan prepares to rescan the plan. * ExecReScanBitmapIndexScan prepares to rescan the plan.
* ExecEndBitmapIndexScan releases all storage. * ExecEndBitmapIndexScan releases all storage.
*/ */
#include "postgres.h" #include "postgres.h"
...@@ -60,7 +60,7 @@ MultiExecBitmapIndexScan(BitmapIndexScanState *node) ...@@ -60,7 +60,7 @@ MultiExecBitmapIndexScan(BitmapIndexScanState *node)
if (!node->biss_RuntimeKeysReady && if (!node->biss_RuntimeKeysReady &&
(node->biss_NumRuntimeKeys != 0 || node->biss_NumArrayKeys != 0)) (node->biss_NumRuntimeKeys != 0 || node->biss_NumArrayKeys != 0))
{ {
ExecReScan((PlanState *) node, NULL); ExecReScan((PlanState *) node);
doscan = node->biss_RuntimeKeysReady; doscan = node->biss_RuntimeKeysReady;
} }
else else
...@@ -106,39 +106,28 @@ MultiExecBitmapIndexScan(BitmapIndexScanState *node) ...@@ -106,39 +106,28 @@ MultiExecBitmapIndexScan(BitmapIndexScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecBitmapIndexReScan(node) * ExecReScanBitmapIndexScan(node)
* *
* Recalculates the value of the scan keys whose value depends on * Recalculates the values of any scan keys whose value depends on
* information known at runtime and rescans the indexed relation. * information known at runtime, then rescans the indexed relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecBitmapIndexReScan(BitmapIndexScanState *node, ExprContext *exprCtxt) ExecReScanBitmapIndexScan(BitmapIndexScanState *node)
{ {
ExprContext *econtext; ExprContext *econtext = node->biss_RuntimeContext;
econtext = node->biss_RuntimeContext; /* context for runtime keys */
/*
* Reset the runtime-key context so we don't leak memory as each outer
* tuple is scanned. Note this assumes that we will recalculate *all*
* runtime keys on each call.
*/
if (econtext) if (econtext)
{
/*
* If we are being passed an outer tuple, save it for runtime key
* calc.
*/
if (exprCtxt != NULL)
econtext->ecxt_outertuple = exprCtxt->ecxt_outertuple;
/*
* Reset the runtime-key context so we don't leak memory as each outer
* tuple is scanned. Note this assumes that we will recalculate *all*
* runtime keys on each call.
*/
ResetExprContext(econtext); ResetExprContext(econtext);
}
/* /*
* If we are doing runtime key calculations (ie, the index keys depend on * If we are doing runtime key calculations (ie, any of the index key
* data from an outer scan), compute the new key values. * values weren't simple Consts), compute the new key values.
* *
* Array keys are also treated as runtime keys; note that if we return * Array keys are also treated as runtime keys; note that if we return
* with biss_RuntimeKeysReady still false, then there is an empty array * with biss_RuntimeKeysReady still false, then there is an empty array
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeBitmapOr.c,v 1.12 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeBitmapOr.c,v 1.13 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -201,7 +201,7 @@ ExecEndBitmapOr(BitmapOrState *node) ...@@ -201,7 +201,7 @@ ExecEndBitmapOr(BitmapOrState *node)
} }
void void
ExecReScanBitmapOr(BitmapOrState *node, ExprContext *exprCtxt) ExecReScanBitmapOr(BitmapOrState *node)
{ {
int i; int i;
...@@ -217,9 +217,10 @@ ExecReScanBitmapOr(BitmapOrState *node, ExprContext *exprCtxt) ...@@ -217,9 +217,10 @@ ExecReScanBitmapOr(BitmapOrState *node, ExprContext *exprCtxt)
UpdateChangedParamSet(subnode, node->ps.chgParam); UpdateChangedParamSet(subnode, node->ps.chgParam);
/* /*
* Always rescan the inputs immediately, to ensure we can pass down * If chgParam of subnode is not null then plan will be re-scanned by
* any outer tuple that might be used in index quals. * first ExecProcNode.
*/ */
ExecReScan(subnode, exprCtxt); if (subnode->chgParam == NULL)
ExecReScan(subnode);
} }
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeCtescan.c,v 1.8 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeCtescan.c,v 1.9 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -298,13 +298,13 @@ ExecEndCteScan(CteScanState *node) ...@@ -298,13 +298,13 @@ ExecEndCteScan(CteScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecCteScanReScan * ExecReScanCteScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecCteScanReScan(CteScanState *node, ExprContext *exprCtxt) ExecReScanCteScan(CteScanState *node)
{ {
Tuplestorestate *tuplestorestate = node->leader->cte_table; Tuplestorestate *tuplestorestate = node->leader->cte_table;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeFunctionscan.c,v 1.55 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeFunctionscan.c,v 1.56 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -18,7 +18,7 @@ ...@@ -18,7 +18,7 @@
* ExecFunctionNext retrieve next tuple in sequential order. * ExecFunctionNext retrieve next tuple in sequential order.
* ExecInitFunctionScan creates and initializes a functionscan node. * ExecInitFunctionScan creates and initializes a functionscan node.
* ExecEndFunctionScan releases any storage allocated. * ExecEndFunctionScan releases any storage allocated.
* ExecFunctionReScan rescans the function * ExecReScanFunctionScan rescans the function
*/ */
#include "postgres.h" #include "postgres.h"
...@@ -255,13 +255,13 @@ ExecEndFunctionScan(FunctionScanState *node) ...@@ -255,13 +255,13 @@ ExecEndFunctionScan(FunctionScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecFunctionReScan * ExecReScanFunctionScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecFunctionReScan(FunctionScanState *node, ExprContext *exprCtxt) ExecReScanFunctionScan(FunctionScanState *node)
{ {
ExecClearTuple(node->ss.ps.ps_ResultTupleSlot); ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* locate group boundaries. * locate group boundaries.
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeGroup.c,v 1.77 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeGroup.c,v 1.78 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -277,14 +277,18 @@ ExecEndGroup(GroupState *node) ...@@ -277,14 +277,18 @@ ExecEndGroup(GroupState *node)
} }
void void
ExecReScanGroup(GroupState *node, ExprContext *exprCtxt) ExecReScanGroup(GroupState *node)
{ {
node->grp_done = FALSE; node->grp_done = FALSE;
node->ss.ps.ps_TupFromTlist = false; node->ss.ps.ps_TupFromTlist = false;
/* must clear first tuple */ /* must clear first tuple */
ExecClearTuple(node->ss.ss_ScanTupleSlot); ExecClearTuple(node->ss.ss_ScanTupleSlot);
if (((PlanState *) node)->lefttree && /*
((PlanState *) node)->lefttree->chgParam == NULL) * if chgParam of subnode is not null then plan will be re-scanned by
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); * first ExecProcNode.
*/
if (node->ss.ps.lefttree &&
node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(node->ss.ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeHash.c,v 1.129 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeHash.c,v 1.130 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -961,14 +961,14 @@ ExecHashTableReset(HashJoinTable hashtable) ...@@ -961,14 +961,14 @@ ExecHashTableReset(HashJoinTable hashtable)
} }
void void
ExecReScanHash(HashState *node, ExprContext *exprCtxt) ExecReScanHash(HashState *node)
{ {
/* /*
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeHashjoin.c,v 1.103 2010/01/02 16:57:41 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeHashjoin.c,v 1.104 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -842,7 +842,7 @@ ExecHashJoinGetSavedTuple(HashJoinState *hjstate, ...@@ -842,7 +842,7 @@ ExecHashJoinGetSavedTuple(HashJoinState *hjstate,
void void
ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt) ExecReScanHashJoin(HashJoinState *node)
{ {
/* /*
* In a multi-batch join, we currently have to do rescans the hard way, * In a multi-batch join, we currently have to do rescans the hard way,
...@@ -854,7 +854,7 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt) ...@@ -854,7 +854,7 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt)
if (node->hj_HashTable != NULL) if (node->hj_HashTable != NULL)
{ {
if (node->hj_HashTable->nbatch == 1 && if (node->hj_HashTable->nbatch == 1 &&
((PlanState *) node)->righttree->chgParam == NULL) node->js.ps.righttree->chgParam == NULL)
{ {
/* /*
* okay to reuse the hash table; needn't rescan inner, either. * okay to reuse the hash table; needn't rescan inner, either.
...@@ -880,8 +880,8 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt) ...@@ -880,8 +880,8 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned * if chgParam of subnode is not null then plan will be re-scanned
* by first ExecProcNode. * by first ExecProcNode.
*/ */
if (((PlanState *) node)->righttree->chgParam == NULL) if (node->js.ps.righttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->righttree, exprCtxt); ExecReScan(node->js.ps.righttree);
} }
} }
...@@ -900,6 +900,6 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt) ...@@ -900,6 +900,6 @@ ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->js.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->js.ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeIndexscan.c,v 1.139 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeIndexscan.c,v 1.140 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -17,7 +17,7 @@ ...@@ -17,7 +17,7 @@
* ExecIndexScan scans a relation using indices * ExecIndexScan scans a relation using indices
* ExecIndexNext using index to retrieve next tuple * ExecIndexNext using index to retrieve next tuple
* ExecInitIndexScan creates and initializes state info. * ExecInitIndexScan creates and initializes state info.
* ExecIndexReScan rescans the indexed relation. * ExecReScanIndexScan rescans the indexed relation.
* ExecEndIndexScan releases all storage. * ExecEndIndexScan releases all storage.
* ExecIndexMarkPos marks scan position. * ExecIndexMarkPos marks scan position.
* ExecIndexRestrPos restores scan position. * ExecIndexRestrPos restores scan position.
...@@ -141,7 +141,7 @@ ExecIndexScan(IndexScanState *node) ...@@ -141,7 +141,7 @@ ExecIndexScan(IndexScanState *node)
* If we have runtime keys and they've not already been set up, do it now. * If we have runtime keys and they've not already been set up, do it now.
*/ */
if (node->iss_NumRuntimeKeys != 0 && !node->iss_RuntimeKeysReady) if (node->iss_NumRuntimeKeys != 0 && !node->iss_RuntimeKeysReady)
ExecReScan((PlanState *) node, NULL); ExecReScan((PlanState *) node);
return ExecScan(&node->ss, return ExecScan(&node->ss,
(ExecScanAccessMtd) IndexNext, (ExecScanAccessMtd) IndexNext,
...@@ -149,54 +149,35 @@ ExecIndexScan(IndexScanState *node) ...@@ -149,54 +149,35 @@ ExecIndexScan(IndexScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecIndexReScan(node) * ExecReScanIndexScan(node)
*
* Recalculates the values of any scan keys whose value depends on
* information known at runtime, then rescans the indexed relation.
* *
* Recalculates the value of the scan keys whose value depends on
* information known at runtime and rescans the indexed relation.
* Updating the scan key was formerly done separately in * Updating the scan key was formerly done separately in
* ExecUpdateIndexScanKeys. Integrating it into ReScan makes * ExecUpdateIndexScanKeys. Integrating it into ReScan makes
* rescans of indices and relations/general streams more uniform. * rescans of indices and relations/general streams more uniform.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecIndexReScan(IndexScanState *node, ExprContext *exprCtxt) ExecReScanIndexScan(IndexScanState *node)
{ {
ExprContext *econtext;
econtext = node->iss_RuntimeContext; /* context for runtime keys */
if (econtext)
{
/*
* If we are being passed an outer tuple, save it for runtime key
* calc. We also need to link it into the "regular" per-tuple
* econtext, so it can be used during indexqualorig evaluations.
*/
if (exprCtxt != NULL)
{
ExprContext *stdecontext;
econtext->ecxt_outertuple = exprCtxt->ecxt_outertuple;
stdecontext = node->ss.ps.ps_ExprContext;
stdecontext->ecxt_outertuple = exprCtxt->ecxt_outertuple;
}
/*
* Reset the runtime-key context so we don't leak memory as each outer
* tuple is scanned. Note this assumes that we will recalculate *all*
* runtime keys on each call.
*/
ResetExprContext(econtext);
}
/* /*
* If we are doing runtime key calculations (ie, the index keys depend on * If we are doing runtime key calculations (ie, any of the index key
* data from an outer scan), compute the new key values * values weren't simple Consts), compute the new key values. But first,
* reset the context so we don't leak memory as each outer tuple is
* scanned. Note this assumes that we will recalculate *all* runtime keys
* on each call.
*/ */
if (node->iss_NumRuntimeKeys != 0) if (node->iss_NumRuntimeKeys != 0)
{
ExprContext *econtext = node->iss_RuntimeContext;
ResetExprContext(econtext);
ExecIndexEvalRuntimeKeys(econtext, ExecIndexEvalRuntimeKeys(econtext,
node->iss_RuntimeKeys, node->iss_RuntimeKeys,
node->iss_NumRuntimeKeys); node->iss_NumRuntimeKeys);
}
node->iss_RuntimeKeysReady = true; node->iss_RuntimeKeysReady = true;
/* reset index scan */ /* reset index scan */
...@@ -229,11 +210,11 @@ ExecIndexEvalRuntimeKeys(ExprContext *econtext, ...@@ -229,11 +210,11 @@ ExecIndexEvalRuntimeKeys(ExprContext *econtext,
/* /*
* For each run-time key, extract the run-time expression and evaluate * For each run-time key, extract the run-time expression and evaluate
* it with respect to the current outer tuple. We then stick the * it with respect to the current context. We then stick the result
* result into the proper scan key. * into the proper scan key.
* *
* Note: the result of the eval could be a pass-by-ref value that's * Note: the result of the eval could be a pass-by-ref value that's
* stored in the outer scan's tuple, not in * stored in some outer scan's tuple, not in
* econtext->ecxt_per_tuple_memory. We assume that the outer tuple * econtext->ecxt_per_tuple_memory. We assume that the outer tuple
* will stay put throughout our scan. If this is wrong, we could copy * will stay put throughout our scan. If this is wrong, we could copy
* the result into our context explicitly, but I think that's not * the result into our context explicitly, but I think that's not
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeLimit.c,v 1.41 2010/01/02 16:57:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeLimit.c,v 1.42 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -404,7 +404,7 @@ ExecEndLimit(LimitState *node) ...@@ -404,7 +404,7 @@ ExecEndLimit(LimitState *node)
void void
ExecReScanLimit(LimitState *node, ExprContext *exprCtxt) ExecReScanLimit(LimitState *node)
{ {
/* /*
* Recompute limit/offset in case parameters changed, and reset the state * Recompute limit/offset in case parameters changed, and reset the state
...@@ -417,6 +417,6 @@ ExecReScanLimit(LimitState *node, ExprContext *exprCtxt) ...@@ -417,6 +417,6 @@ ExecReScanLimit(LimitState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeLockRows.c,v 1.4 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeLockRows.c,v 1.5 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -339,12 +339,12 @@ ExecEndLockRows(LockRowsState *node) ...@@ -339,12 +339,12 @@ ExecEndLockRows(LockRowsState *node)
void void
ExecReScanLockRows(LockRowsState *node, ExprContext *exprCtxt) ExecReScanLockRows(LockRowsState *node)
{ {
/* /*
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeMaterial.c,v 1.71 2010/01/02 16:57:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeMaterial.c,v 1.72 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -309,22 +309,22 @@ ExecMaterialRestrPos(MaterialState *node) ...@@ -309,22 +309,22 @@ ExecMaterialRestrPos(MaterialState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecMaterialReScan * ExecReScanMaterial
* *
* Rescans the materialized relation. * Rescans the materialized relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt) ExecReScanMaterial(MaterialState *node)
{ {
ExecClearTuple(node->ss.ps.ps_ResultTupleSlot); ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
if (node->eflags != 0) if (node->eflags != 0)
{ {
/* /*
* If we haven't materialized yet, just return. If outerplan' chgParam * If we haven't materialized yet, just return. If outerplan's
* is not NULL then it will be re-scanned by ExecProcNode, else - no * chgParam is not NULL then it will be re-scanned by ExecProcNode,
* reason to re-scan it at all. * else no reason to re-scan it at all.
*/ */
if (!node->tuplestorestate) if (!node->tuplestorestate)
return; return;
...@@ -339,13 +339,13 @@ ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt) ...@@ -339,13 +339,13 @@ ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt)
* Otherwise we can just rewind and rescan the stored output. The * Otherwise we can just rewind and rescan the stored output. The
* state of the subnode does not change. * state of the subnode does not change.
*/ */
if (((PlanState *) node)->lefttree->chgParam != NULL || if (node->ss.ps.lefttree->chgParam != NULL ||
(node->eflags & EXEC_FLAG_REWIND) == 0) (node->eflags & EXEC_FLAG_REWIND) == 0)
{ {
tuplestore_end(node->tuplestorestate); tuplestore_end(node->tuplestorestate);
node->tuplestorestate = NULL; node->tuplestorestate = NULL;
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ss.ps.lefttree);
node->eof_underlying = false; node->eof_underlying = false;
} }
else else
...@@ -359,8 +359,8 @@ ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt) ...@@ -359,8 +359,8 @@ ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ss.ps.lefttree);
node->eof_underlying = false; node->eof_underlying = false;
} }
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeMergejoin.c,v 1.103 2010/07/06 19:18:56 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeMergejoin.c,v 1.104 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1680,7 +1680,7 @@ ExecEndMergeJoin(MergeJoinState *node) ...@@ -1680,7 +1680,7 @@ ExecEndMergeJoin(MergeJoinState *node)
} }
void void
ExecReScanMergeJoin(MergeJoinState *node, ExprContext *exprCtxt) ExecReScanMergeJoin(MergeJoinState *node)
{ {
ExecClearTuple(node->mj_MarkedTupleSlot); ExecClearTuple(node->mj_MarkedTupleSlot);
...@@ -1695,9 +1695,9 @@ ExecReScanMergeJoin(MergeJoinState *node, ExprContext *exprCtxt) ...@@ -1695,9 +1695,9 @@ ExecReScanMergeJoin(MergeJoinState *node, ExprContext *exprCtxt)
* if chgParam of subnodes is not null then plans will be re-scanned by * if chgParam of subnodes is not null then plans will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->js.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->js.ps.lefttree);
if (((PlanState *) node)->righttree->chgParam == NULL) if (node->js.ps.righttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->righttree, exprCtxt); ExecReScan(node->js.ps.righttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeModifyTable.c,v 1.7 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeModifyTable.c,v 1.8 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1045,7 +1045,7 @@ ExecEndModifyTable(ModifyTableState *node) ...@@ -1045,7 +1045,7 @@ ExecEndModifyTable(ModifyTableState *node)
} }
void void
ExecReScanModifyTable(ModifyTableState *node, ExprContext *exprCtxt) ExecReScanModifyTable(ModifyTableState *node)
{ {
/* /*
* Currently, we don't need to support rescan on ModifyTable nodes. The * Currently, we don't need to support rescan on ModifyTable nodes. The
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeNestloop.c,v 1.55 2010/01/02 16:57:44 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeNestloop.c,v 1.56 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -59,6 +59,7 @@ ...@@ -59,6 +59,7 @@
TupleTableSlot * TupleTableSlot *
ExecNestLoop(NestLoopState *node) ExecNestLoop(NestLoopState *node)
{ {
NestLoop *nl;
PlanState *innerPlan; PlanState *innerPlan;
PlanState *outerPlan; PlanState *outerPlan;
TupleTableSlot *outerTupleSlot; TupleTableSlot *outerTupleSlot;
...@@ -66,12 +67,14 @@ ExecNestLoop(NestLoopState *node) ...@@ -66,12 +67,14 @@ ExecNestLoop(NestLoopState *node)
List *joinqual; List *joinqual;
List *otherqual; List *otherqual;
ExprContext *econtext; ExprContext *econtext;
ListCell *lc;
/* /*
* get information from the node * get information from the node
*/ */
ENL1_printf("getting info from node"); ENL1_printf("getting info from node");
nl = (NestLoop *) node->js.ps.plan;
joinqual = node->js.joinqual; joinqual = node->js.joinqual;
otherqual = node->js.ps.qual; otherqual = node->js.ps.qual;
outerPlan = outerPlanState(node); outerPlan = outerPlanState(node);
...@@ -134,16 +137,33 @@ ExecNestLoop(NestLoopState *node) ...@@ -134,16 +137,33 @@ ExecNestLoop(NestLoopState *node)
node->nl_MatchedOuter = false; node->nl_MatchedOuter = false;
/* /*
* now rescan the inner plan * fetch the values of any outer Vars that must be passed to
* the inner scan, and store them in the appropriate PARAM_EXEC
* slots.
*/ */
ENL1_printf("rescanning inner plan"); foreach(lc, nl->nestParams)
{
NestLoopParam *nlp = (NestLoopParam *) lfirst(lc);
int paramno = nlp->paramno;
ParamExecData *prm;
prm = &(econtext->ecxt_param_exec_vals[paramno]);
/* Param value should be an OUTER var */
Assert(nlp->paramval->varno == OUTER);
Assert(nlp->paramval->varattno > 0);
prm->value = slot_getattr(outerTupleSlot,
nlp->paramval->varattno,
&(prm->isnull));
/* Flag parameter value as changed */
innerPlan->chgParam = bms_add_member(innerPlan->chgParam,
paramno);
}
/* /*
* The scan key of the inner plan might depend on the current * now rescan the inner plan
* outer tuple (e.g. in index scans), that's why we pass our expr
* context.
*/ */
ExecReScan(innerPlan, econtext); ENL1_printf("rescanning inner plan");
ExecReScan(innerPlan);
} }
/* /*
...@@ -308,15 +328,18 @@ ExecInitNestLoop(NestLoop *node, EState *estate, int eflags) ...@@ -308,15 +328,18 @@ ExecInitNestLoop(NestLoop *node, EState *estate, int eflags)
/* /*
* initialize child nodes * initialize child nodes
* *
* Tell the inner child that cheap rescans would be good. (This is * If we have no parameters to pass into the inner rel from the outer,
* unnecessary if we are doing nestloop with inner indexscan, because the * tell the inner child that cheap rescans would be good. If we do have
* rescan will always be with a fresh parameter --- but since * such parameters, then there is no point in REWIND support at all in
* nodeIndexscan doesn't actually care about REWIND, there's no point in * the inner child, because it will always be rescanned with fresh
* dealing with that refinement.) * parameter values.
*/ */
outerPlanState(nlstate) = ExecInitNode(outerPlan(node), estate, eflags); outerPlanState(nlstate) = ExecInitNode(outerPlan(node), estate, eflags);
innerPlanState(nlstate) = ExecInitNode(innerPlan(node), estate, if (node->nestParams == NIL)
eflags | EXEC_FLAG_REWIND); eflags |= EXEC_FLAG_REWIND;
else
eflags &= ~EXEC_FLAG_REWIND;
innerPlanState(nlstate) = ExecInitNode(innerPlan(node), estate, eflags);
/* /*
* tuple table initialization * tuple table initialization
...@@ -395,18 +418,22 @@ ExecEndNestLoop(NestLoopState *node) ...@@ -395,18 +418,22 @@ ExecEndNestLoop(NestLoopState *node)
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecReScanNestLoop(NestLoopState *node, ExprContext *exprCtxt) ExecReScanNestLoop(NestLoopState *node)
{ {
PlanState *outerPlan = outerPlanState(node); PlanState *outerPlan = outerPlanState(node);
/* /*
* If outerPlan->chgParam is not null then plan will be automatically * If outerPlan->chgParam is not null then plan will be automatically
* re-scanned by first ExecProcNode. innerPlan is re-scanned for each new * re-scanned by first ExecProcNode.
* outer tuple and MUST NOT be re-scanned from here or you'll get troubles
* from inner index scans when outer Vars are used as run-time keys...
*/ */
if (outerPlan->chgParam == NULL) if (outerPlan->chgParam == NULL)
ExecReScan(outerPlan, exprCtxt); ExecReScan(outerPlan);
/*
* innerPlan is re-scanned for each new outer tuple and MUST NOT be
* re-scanned from here or you'll get troubles from inner index scans when
* outer Vars are used as run-time keys...
*/
node->js.ps.ps_TupFromTlist = false; node->js.ps.ps_TupFromTlist = false;
node->nl_NeedNewOuter = true; node->nl_NeedNewOuter = true;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeRecursiveunion.c,v 1.6 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeRecursiveunion.c,v 1.7 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -299,13 +299,13 @@ ExecEndRecursiveUnion(RecursiveUnionState *node) ...@@ -299,13 +299,13 @@ ExecEndRecursiveUnion(RecursiveUnionState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecRecursiveUnionReScan * ExecReScanRecursiveUnion
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecRecursiveUnionReScan(RecursiveUnionState *node, ExprContext *exprCtxt) ExecReScanRecursiveUnion(RecursiveUnionState *node)
{ {
PlanState *outerPlan = outerPlanState(node); PlanState *outerPlan = outerPlanState(node);
PlanState *innerPlan = innerPlanState(node); PlanState *innerPlan = innerPlanState(node);
...@@ -323,7 +323,7 @@ ExecRecursiveUnionReScan(RecursiveUnionState *node, ExprContext *exprCtxt) ...@@ -323,7 +323,7 @@ ExecRecursiveUnionReScan(RecursiveUnionState *node, ExprContext *exprCtxt)
* non-recursive term. * non-recursive term.
*/ */
if (outerPlan->chgParam == NULL) if (outerPlan->chgParam == NULL)
ExecReScan(outerPlan, exprCtxt); ExecReScan(outerPlan);
/* Release any hashtable storage */ /* Release any hashtable storage */
if (node->tableContext) if (node->tableContext)
......
...@@ -38,7 +38,7 @@ ...@@ -38,7 +38,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeResult.c,v 1.45 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeResult.c,v 1.46 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -292,7 +292,7 @@ ExecEndResult(ResultState *node) ...@@ -292,7 +292,7 @@ ExecEndResult(ResultState *node)
} }
void void
ExecReScanResult(ResultState *node, ExprContext *exprCtxt) ExecReScanResult(ResultState *node)
{ {
node->rs_done = false; node->rs_done = false;
node->ps.ps_TupFromTlist = false; node->ps.ps_TupFromTlist = false;
...@@ -300,11 +300,9 @@ ExecReScanResult(ResultState *node, ExprContext *exprCtxt) ...@@ -300,11 +300,9 @@ ExecReScanResult(ResultState *node, ExprContext *exprCtxt)
/* /*
* If chgParam of subnode is not null then plan will be re-scanned by * If chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. However, if caller is passing us an exprCtxt then * first ExecProcNode.
* forcibly rescan the subnode now, so that we can pass the exprCtxt down
* to the subnode (needed for gated indexscan).
*/ */
if (node->ps.lefttree && if (node->ps.lefttree &&
(node->ps.lefttree->chgParam == NULL || exprCtxt != NULL)) node->ps.lefttree->chgParam == NULL)
ExecReScan(node->ps.lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeSeqscan.c,v 1.70 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeSeqscan.c,v 1.71 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -18,7 +18,7 @@ ...@@ -18,7 +18,7 @@
* ExecSeqNext retrieve next tuple in sequential order. * ExecSeqNext retrieve next tuple in sequential order.
* ExecInitSeqScan creates and initializes a seqscan node. * ExecInitSeqScan creates and initializes a seqscan node.
* ExecEndSeqScan releases any storage allocated. * ExecEndSeqScan releases any storage allocated.
* ExecSeqReScan rescans the relation * ExecReScanSeqScan rescans the relation
* ExecSeqMarkPos marks scan position * ExecSeqMarkPos marks scan position
* ExecSeqRestrPos restores scan position * ExecSeqRestrPos restores scan position
*/ */
...@@ -255,13 +255,13 @@ ExecEndSeqScan(SeqScanState *node) ...@@ -255,13 +255,13 @@ ExecEndSeqScan(SeqScanState *node)
*/ */
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecSeqReScan * ExecReScanSeqScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecSeqReScan(SeqScanState *node, ExprContext *exprCtxt) ExecReScanSeqScan(SeqScanState *node)
{ {
HeapScanDesc scan; HeapScanDesc scan;
......
...@@ -37,7 +37,7 @@ ...@@ -37,7 +37,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeSetOp.c,v 1.33 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeSetOp.c,v 1.34 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -597,7 +597,7 @@ ExecEndSetOp(SetOpState *node) ...@@ -597,7 +597,7 @@ ExecEndSetOp(SetOpState *node)
void void
ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt) ExecReScanSetOp(SetOpState *node)
{ {
ExecClearTuple(node->ps.ps_ResultTupleSlot); ExecClearTuple(node->ps.ps_ResultTupleSlot);
node->setop_done = false; node->setop_done = false;
...@@ -619,7 +619,7 @@ ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt) ...@@ -619,7 +619,7 @@ ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt)
* parameter changes, then we can just rescan the existing hash table; * parameter changes, then we can just rescan the existing hash table;
* no need to build it again. * no need to build it again.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
{ {
ResetTupleHashIterator(node->hashtable, &node->hashiter); ResetTupleHashIterator(node->hashtable, &node->hashiter);
return; return;
...@@ -648,6 +648,6 @@ ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt) ...@@ -648,6 +648,6 @@ ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeSort.c,v 1.67 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeSort.c,v 1.68 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -287,11 +287,11 @@ ExecSortRestrPos(SortState *node) ...@@ -287,11 +287,11 @@ ExecSortRestrPos(SortState *node)
} }
void void
ExecReScanSort(SortState *node, ExprContext *exprCtxt) ExecReScanSort(SortState *node)
{ {
/* /*
* If we haven't sorted yet, just return. If outerplan' chgParam is not * If we haven't sorted yet, just return. If outerplan's chgParam is not
* NULL then it will be re-scanned by ExecProcNode, else - no reason to * NULL then it will be re-scanned by ExecProcNode, else no reason to
* re-scan it at all. * re-scan it at all.
*/ */
if (!node->sort_Done) if (!node->sort_Done)
...@@ -307,7 +307,7 @@ ExecReScanSort(SortState *node, ExprContext *exprCtxt) ...@@ -307,7 +307,7 @@ ExecReScanSort(SortState *node, ExprContext *exprCtxt)
* *
* Otherwise we can just rewind and rescan the sorted output. * Otherwise we can just rewind and rescan the sorted output.
*/ */
if (((PlanState *) node)->lefttree->chgParam != NULL || if (node->ss.ps.lefttree->chgParam != NULL ||
node->bounded != node->bounded_Done || node->bounded != node->bounded_Done ||
node->bound != node->bound_Done || node->bound != node->bound_Done ||
!node->randomAccess) !node->randomAccess)
...@@ -320,8 +320,8 @@ ExecReScanSort(SortState *node, ExprContext *exprCtxt) ...@@ -320,8 +320,8 @@ ExecReScanSort(SortState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ss.ps.lefttree);
} }
else else
tuplesort_rescan((Tuplesortstate *) node->tuplesortstate); tuplesort_rescan((Tuplesortstate *) node->tuplesortstate);
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeSubplan.c,v 1.101 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeSubplan.c,v 1.102 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -256,7 +256,7 @@ ExecScanSubPlan(SubPlanState *node, ...@@ -256,7 +256,7 @@ ExecScanSubPlan(SubPlanState *node,
/* /*
* Now that we've set up its parameters, we can reset the subplan. * Now that we've set up its parameters, we can reset the subplan.
*/ */
ExecReScan(planstate, NULL); ExecReScan(planstate);
/* /*
* For all sublink types except EXPR_SUBLINK and ARRAY_SUBLINK, the result * For all sublink types except EXPR_SUBLINK and ARRAY_SUBLINK, the result
...@@ -508,7 +508,7 @@ buildSubPlanHash(SubPlanState *node, ExprContext *econtext) ...@@ -508,7 +508,7 @@ buildSubPlanHash(SubPlanState *node, ExprContext *econtext)
/* /*
* Reset subplan to start. * Reset subplan to start.
*/ */
ExecReScan(planstate, NULL); ExecReScan(planstate);
/* /*
* Scan the subplan and load the hash table(s). Note that when there are * Scan the subplan and load the hash table(s). Note that when there are
...@@ -884,7 +884,7 @@ ExecInitSubPlan(SubPlan *subplan, PlanState *parent) ...@@ -884,7 +884,7 @@ ExecInitSubPlan(SubPlan *subplan, PlanState *parent)
* *
* Executes an InitPlan subplan and sets its output parameters. * Executes an InitPlan subplan and sets its output parameters.
* *
* This is called from ExecEvalParam() when the value of a PARAM_EXEC * This is called from ExecEvalParamExec() when the value of a PARAM_EXEC
* parameter is requested and the param's execPlan field is set (indicating * parameter is requested and the param's execPlan field is set (indicating
* that the param has not yet been evaluated). This allows lazy evaluation * that the param has not yet been evaluated). This allows lazy evaluation
* of initplans: we don't run the subplan until/unless we need its output. * of initplans: we don't run the subplan until/unless we need its output.
......
...@@ -12,7 +12,7 @@ ...@@ -12,7 +12,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeSubqueryscan.c,v 1.45 2010/02/26 02:00:42 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeSubqueryscan.c,v 1.46 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -22,7 +22,7 @@ ...@@ -22,7 +22,7 @@
* ExecSubqueryNext retrieve next tuple in sequential order. * ExecSubqueryNext retrieve next tuple in sequential order.
* ExecInitSubqueryScan creates and initializes a subqueryscan node. * ExecInitSubqueryScan creates and initializes a subqueryscan node.
* ExecEndSubqueryScan releases any storage allocated. * ExecEndSubqueryScan releases any storage allocated.
* ExecSubqueryReScan rescans the relation * ExecReScanSubqueryScan rescans the relation
* *
*/ */
#include "postgres.h" #include "postgres.h"
...@@ -187,13 +187,13 @@ ExecEndSubqueryScan(SubqueryScanState *node) ...@@ -187,13 +187,13 @@ ExecEndSubqueryScan(SubqueryScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecSubqueryReScan * ExecReScanSubqueryScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecSubqueryReScan(SubqueryScanState *node, ExprContext *exprCtxt) ExecReScanSubqueryScan(SubqueryScanState *node)
{ {
ExecScanReScan(&node->ss); ExecScanReScan(&node->ss);
...@@ -210,5 +210,5 @@ ExecSubqueryReScan(SubqueryScanState *node, ExprContext *exprCtxt) ...@@ -210,5 +210,5 @@ ExecSubqueryReScan(SubqueryScanState *node, ExprContext *exprCtxt)
* first ExecProcNode. * first ExecProcNode.
*/ */
if (node->subplan->chgParam == NULL) if (node->subplan->chgParam == NULL)
ExecReScan(node->subplan, NULL); ExecReScan(node->subplan);
} }
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeTidscan.c,v 1.65 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeTidscan.c,v 1.66 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -17,7 +17,7 @@ ...@@ -17,7 +17,7 @@
* *
* ExecTidScan scans a relation using tids * ExecTidScan scans a relation using tids
* ExecInitTidScan creates and initializes state info. * ExecInitTidScan creates and initializes state info.
* ExecTidReScan rescans the tid relation. * ExecReScanTidScan rescans the tid relation.
* ExecEndTidScan releases all storage. * ExecEndTidScan releases all storage.
* ExecTidMarkPos marks scan position. * ExecTidMarkPos marks scan position.
* ExecTidRestrPos restores scan position. * ExecTidRestrPos restores scan position.
...@@ -398,17 +398,12 @@ ExecTidScan(TidScanState *node) ...@@ -398,17 +398,12 @@ ExecTidScan(TidScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecTidReScan(node) * ExecReScanTidScan(node)
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecTidReScan(TidScanState *node, ExprContext *exprCtxt) ExecReScanTidScan(TidScanState *node)
{ {
/* If we are being passed an outer tuple, save it for runtime key calc */
if (exprCtxt != NULL)
node->ss.ps.ps_ExprContext->ecxt_outertuple =
exprCtxt->ecxt_outertuple;
if (node->tss_TidList) if (node->tss_TidList)
pfree(node->tss_TidList); pfree(node->tss_TidList);
node->tss_TidList = NULL; node->tss_TidList = NULL;
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeUnique.c,v 1.63 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeUnique.c,v 1.64 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -184,7 +184,7 @@ ExecEndUnique(UniqueState *node) ...@@ -184,7 +184,7 @@ ExecEndUnique(UniqueState *node)
void void
ExecReScanUnique(UniqueState *node, ExprContext *exprCtxt) ExecReScanUnique(UniqueState *node)
{ {
/* must clear result tuple so first input tuple is returned */ /* must clear result tuple so first input tuple is returned */
ExecClearTuple(node->ps.ps_ResultTupleSlot); ExecClearTuple(node->ps.ps_ResultTupleSlot);
...@@ -193,6 +193,6 @@ ExecReScanUnique(UniqueState *node, ExprContext *exprCtxt) ...@@ -193,6 +193,6 @@ ExecReScanUnique(UniqueState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ps.lefttree);
} }
...@@ -9,7 +9,7 @@ ...@@ -9,7 +9,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeValuesscan.c,v 1.12 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeValuesscan.c,v 1.13 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
* ExecValuesNext retrieve next tuple in sequential order. * ExecValuesNext retrieve next tuple in sequential order.
* ExecInitValuesScan creates and initializes a valuesscan node. * ExecInitValuesScan creates and initializes a valuesscan node.
* ExecEndValuesScan releases any storage allocated. * ExecEndValuesScan releases any storage allocated.
* ExecValuesReScan rescans the values list * ExecReScanValuesScan rescans the values list
*/ */
#include "postgres.h" #include "postgres.h"
...@@ -319,13 +319,13 @@ ExecValuesRestrPos(ValuesScanState *node) ...@@ -319,13 +319,13 @@ ExecValuesRestrPos(ValuesScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecValuesReScan * ExecReScanValuesScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecValuesReScan(ValuesScanState *node, ExprContext *exprCtxt) ExecReScanValuesScan(ValuesScanState *node)
{ {
ExecClearTuple(node->ss.ps.ps_ResultTupleSlot); ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
......
...@@ -27,7 +27,7 @@ ...@@ -27,7 +27,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeWindowAgg.c,v 1.13 2010/03/21 00:17:58 petere Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeWindowAgg.c,v 1.14 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -1664,7 +1664,7 @@ ExecEndWindowAgg(WindowAggState *node) ...@@ -1664,7 +1664,7 @@ ExecEndWindowAgg(WindowAggState *node)
* ----------------- * -----------------
*/ */
void void
ExecReScanWindowAgg(WindowAggState *node, ExprContext *exprCtxt) ExecReScanWindowAgg(WindowAggState *node)
{ {
ExprContext *econtext = node->ss.ps.ps_ExprContext; ExprContext *econtext = node->ss.ps.ps_ExprContext;
...@@ -1691,8 +1691,8 @@ ExecReScanWindowAgg(WindowAggState *node, ExprContext *exprCtxt) ...@@ -1691,8 +1691,8 @@ ExecReScanWindowAgg(WindowAggState *node, ExprContext *exprCtxt)
* if chgParam of subnode is not null then plan will be re-scanned by * if chgParam of subnode is not null then plan will be re-scanned by
* first ExecProcNode. * first ExecProcNode.
*/ */
if (((PlanState *) node)->lefttree->chgParam == NULL) if (node->ss.ps.lefttree->chgParam == NULL)
ExecReScan(((PlanState *) node)->lefttree, exprCtxt); ExecReScan(node->ss.ps.lefttree);
} }
/* /*
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/executor/nodeWorktablescan.c,v 1.10 2010/01/02 16:57:45 momjian Exp $ * $PostgreSQL: pgsql/src/backend/executor/nodeWorktablescan.c,v 1.11 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -203,13 +203,13 @@ ExecEndWorkTableScan(WorkTableScanState *node) ...@@ -203,13 +203,13 @@ ExecEndWorkTableScan(WorkTableScanState *node)
} }
/* ---------------------------------------------------------------- /* ----------------------------------------------------------------
* ExecWorkTableScanReScan * ExecReScanWorkTableScan
* *
* Rescans the relation. * Rescans the relation.
* ---------------------------------------------------------------- * ----------------------------------------------------------------
*/ */
void void
ExecWorkTableScanReScan(WorkTableScanState *node, ExprContext *exprCtxt) ExecReScanWorkTableScan(WorkTableScanState *node)
{ {
ExecClearTuple(node->ss.ps.ps_ResultTupleSlot); ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
......
...@@ -15,7 +15,7 @@ ...@@ -15,7 +15,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/copyfuncs.c,v 1.464 2010/02/26 02:00:43 momjian Exp $ * $PostgreSQL: pgsql/src/backend/nodes/copyfuncs.c,v 1.465 2010/07/12 17:01:05 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -564,6 +564,11 @@ _copyNestLoop(NestLoop *from) ...@@ -564,6 +564,11 @@ _copyNestLoop(NestLoop *from)
*/ */
CopyJoinFields((Join *) from, (Join *) newnode); CopyJoinFields((Join *) from, (Join *) newnode);
/*
* copy remainder of node
*/
COPY_NODE_FIELD(nestParams);
return newnode; return newnode;
} }
...@@ -844,6 +849,20 @@ _copyLimit(Limit *from) ...@@ -844,6 +849,20 @@ _copyLimit(Limit *from)
return newnode; return newnode;
} }
/*
* _copyNestLoopParam
*/
static NestLoopParam *
_copyNestLoopParam(NestLoopParam *from)
{
NestLoopParam *newnode = makeNode(NestLoopParam);
COPY_SCALAR_FIELD(paramno);
COPY_NODE_FIELD(paramval);
return newnode;
}
/* /*
* _copyPlanRowMark * _copyPlanRowMark
*/ */
...@@ -3671,6 +3690,9 @@ copyObject(void *from) ...@@ -3671,6 +3690,9 @@ copyObject(void *from)
case T_Limit: case T_Limit:
retval = _copyLimit(from); retval = _copyLimit(from);
break; break;
case T_NestLoopParam:
retval = _copyNestLoopParam(from);
break;
case T_PlanRowMark: case T_PlanRowMark:
retval = _copyPlanRowMark(from); retval = _copyPlanRowMark(from);
break; break;
......
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.385 2010/03/30 21:58:10 tgl Exp $ * $PostgreSQL: pgsql/src/backend/nodes/outfuncs.c,v 1.386 2010/07/12 17:01:05 tgl Exp $
* *
* NOTES * NOTES
* Every node type that can appear in stored rules' parsetrees *must* * Every node type that can appear in stored rules' parsetrees *must*
...@@ -517,6 +517,8 @@ _outNestLoop(StringInfo str, NestLoop *node) ...@@ -517,6 +517,8 @@ _outNestLoop(StringInfo str, NestLoop *node)
WRITE_NODE_TYPE("NESTLOOP"); WRITE_NODE_TYPE("NESTLOOP");
_outJoinPlanInfo(str, (Join *) node); _outJoinPlanInfo(str, (Join *) node);
WRITE_NODE_FIELD(nestParams);
} }
static void static void
...@@ -748,6 +750,15 @@ _outLimit(StringInfo str, Limit *node) ...@@ -748,6 +750,15 @@ _outLimit(StringInfo str, Limit *node)
WRITE_NODE_FIELD(limitCount); WRITE_NODE_FIELD(limitCount);
} }
static void
_outNestLoopParam(StringInfo str, NestLoopParam *node)
{
WRITE_NODE_TYPE("NESTLOOPPARAM");
WRITE_INT_FIELD(paramno);
WRITE_NODE_FIELD(paramval);
}
static void static void
_outPlanRowMark(StringInfo str, PlanRowMark *node) _outPlanRowMark(StringInfo str, PlanRowMark *node)
{ {
...@@ -1565,6 +1576,8 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node) ...@@ -1565,6 +1576,8 @@ _outPlannerInfo(StringInfo str, PlannerInfo *node)
WRITE_BOOL_FIELD(hasPseudoConstantQuals); WRITE_BOOL_FIELD(hasPseudoConstantQuals);
WRITE_BOOL_FIELD(hasRecursion); WRITE_BOOL_FIELD(hasRecursion);
WRITE_INT_FIELD(wt_param_id); WRITE_INT_FIELD(wt_param_id);
WRITE_BITMAPSET_FIELD(curOuterRels);
WRITE_NODE_FIELD(curOuterParams);
} }
static void static void
...@@ -2562,6 +2575,9 @@ _outNode(StringInfo str, void *obj) ...@@ -2562,6 +2575,9 @@ _outNode(StringInfo str, void *obj)
case T_Limit: case T_Limit:
_outLimit(str, obj); _outLimit(str, obj);
break; break;
case T_NestLoopParam:
_outNestLoopParam(str, obj);
break;
case T_PlanRowMark: case T_PlanRowMark:
_outPlanRowMark(str, obj); _outPlanRowMark(str, obj);
break; break;
......
This diff is collapsed.
This diff is collapsed.
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/optimizer/plan/subselect.c,v 1.162 2010/04/19 00:55:25 rhaas Exp $ * $PostgreSQL: pgsql/src/backend/optimizer/plan/subselect.c,v 1.163 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -83,30 +83,20 @@ static bool finalize_primnode(Node *node, finalize_primnode_context *context); ...@@ -83,30 +83,20 @@ static bool finalize_primnode(Node *node, finalize_primnode_context *context);
/* /*
* Generate a Param node to replace the given Var, * Select a PARAM_EXEC number to identify the given Var.
* which is expected to have varlevelsup > 0 (ie, it is not local). * If the Var already has a param slot, return that one.
*/ */
static Param * static int
replace_outer_var(PlannerInfo *root, Var *var) assign_param_for_var(PlannerInfo *root, Var *var)
{ {
Param *retval;
ListCell *ppl; ListCell *ppl;
PlannerParamItem *pitem; PlannerParamItem *pitem;
Index abslevel; Index abslevel;
int i; int i;
Assert(var->varlevelsup > 0 && var->varlevelsup < root->query_level);
abslevel = root->query_level - var->varlevelsup; abslevel = root->query_level - var->varlevelsup;
/* /* If there's already a paramlist entry for this same Var, just use it */
* If there's already a paramlist entry for this same Var, just use it.
* NOTE: in sufficiently complex querytrees, it is possible for the same
* varno/abslevel to refer to different RTEs in different parts of the
* parsetree, so that different fields might end up sharing the same Param
* number. As long as we check the vartype/typmod as well, I believe that
* this sort of aliasing will cause no trouble. The correct field should
* get stored into the Param slot at execution in each part of the tree.
*/
i = 0; i = 0;
foreach(ppl, root->glob->paramlist) foreach(ppl, root->glob->paramlist)
{ {
...@@ -119,24 +109,76 @@ replace_outer_var(PlannerInfo *root, Var *var) ...@@ -119,24 +109,76 @@ replace_outer_var(PlannerInfo *root, Var *var)
pvar->varattno == var->varattno && pvar->varattno == var->varattno &&
pvar->vartype == var->vartype && pvar->vartype == var->vartype &&
pvar->vartypmod == var->vartypmod) pvar->vartypmod == var->vartypmod)
break; return i;
} }
i++; i++;
} }
if (!ppl) /* Nope, so make a new one */
{ var = (Var *) copyObject(var);
/* Nope, so make a new one */ var->varlevelsup = 0;
var = (Var *) copyObject(var);
var->varlevelsup = 0;
pitem = makeNode(PlannerParamItem); pitem = makeNode(PlannerParamItem);
pitem->item = (Node *) var; pitem->item = (Node *) var;
pitem->abslevel = abslevel; pitem->abslevel = abslevel;
root->glob->paramlist = lappend(root->glob->paramlist, pitem); root->glob->paramlist = lappend(root->glob->paramlist, pitem);
/* i is already the correct index for the new item */
} /* i is already the correct list index for the new item */
return i;
}
/*
* Generate a Param node to replace the given Var,
* which is expected to have varlevelsup > 0 (ie, it is not local).
*/
static Param *
replace_outer_var(PlannerInfo *root, Var *var)
{
Param *retval;
int i;
Assert(var->varlevelsup > 0 && var->varlevelsup < root->query_level);
/*
* Find the Var in root->glob->paramlist, or add it if not present.
*
* NOTE: in sufficiently complex querytrees, it is possible for the same
* varno/abslevel to refer to different RTEs in different parts of the
* parsetree, so that different fields might end up sharing the same Param
* number. As long as we check the vartype/typmod as well, I believe that
* this sort of aliasing will cause no trouble. The correct field should
* get stored into the Param slot at execution in each part of the tree.
*/
i = assign_param_for_var(root, var);
retval = makeNode(Param);
retval->paramkind = PARAM_EXEC;
retval->paramid = i;
retval->paramtype = var->vartype;
retval->paramtypmod = var->vartypmod;
retval->location = -1;
return retval;
}
/*
* Generate a Param node to replace the given Var, which will be supplied
* from an upper NestLoop join node.
*
* Because we allow nestloop and subquery Params to alias each other,
* this is effectively the same as replace_outer_var, except that we expect
* the Var to be local to the current query level.
*/
Param *
assign_nestloop_param(PlannerInfo *root, Var *var)
{
Param *retval;
int i;
Assert(var->varlevelsup == 0);
i = assign_param_for_var(root, var);
retval = makeNode(Param); retval = makeNode(Param);
retval->paramkind = PARAM_EXEC; retval->paramkind = PARAM_EXEC;
...@@ -1773,8 +1815,9 @@ SS_finalize_plan(PlannerInfo *root, Plan *plan, bool attach_initplans) ...@@ -1773,8 +1815,9 @@ SS_finalize_plan(PlannerInfo *root, Plan *plan, bool attach_initplans)
* *
* Note: this is a bit overly generous since some parameters of upper * Note: this is a bit overly generous since some parameters of upper
* query levels might belong to query subtrees that don't include this * query levels might belong to query subtrees that don't include this
* query. However, valid_params is only a debugging crosscheck, so it * query, or might be nestloop params that won't be passed down at all.
* doesn't seem worth expending lots of cycles to try to be exact. * However, valid_params is only a debugging crosscheck, so it doesn't
* seem worth expending lots of cycles to try to be exact.
*/ */
valid_params = bms_copy(initSetParam); valid_params = bms_copy(initSetParam);
paramid = 0; paramid = 0;
...@@ -1849,6 +1892,8 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params, ...@@ -1849,6 +1892,8 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params,
{ {
finalize_primnode_context context; finalize_primnode_context context;
int locally_added_param; int locally_added_param;
Bitmapset *nestloop_params;
Bitmapset *child_params;
if (plan == NULL) if (plan == NULL)
return NULL; return NULL;
...@@ -1856,6 +1901,7 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params, ...@@ -1856,6 +1901,7 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params,
context.root = root; context.root = root;
context.paramids = NULL; /* initialize set to empty */ context.paramids = NULL; /* initialize set to empty */
locally_added_param = -1; /* there isn't one */ locally_added_param = -1; /* there isn't one */
nestloop_params = NULL; /* there aren't any */
/* /*
* When we call finalize_primnode, context.paramids sets are automatically * When we call finalize_primnode, context.paramids sets are automatically
...@@ -2056,8 +2102,20 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params, ...@@ -2056,8 +2102,20 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params,
break; break;
case T_NestLoop: case T_NestLoop:
finalize_primnode((Node *) ((Join *) plan)->joinqual, {
&context); ListCell *l;
finalize_primnode((Node *) ((Join *) plan)->joinqual,
&context);
/* collect set of params that will be passed to right child */
foreach(l, ((NestLoop *) plan)->nestParams)
{
NestLoopParam *nlp = (NestLoopParam *) lfirst(l);
nestloop_params = bms_add_member(nestloop_params,
nlp->paramno);
}
}
break; break;
case T_MergeJoin: case T_MergeJoin:
...@@ -2120,17 +2178,32 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params, ...@@ -2120,17 +2178,32 @@ finalize_plan(PlannerInfo *root, Plan *plan, Bitmapset *valid_params,
} }
/* Process left and right child plans, if any */ /* Process left and right child plans, if any */
context.paramids = bms_add_members(context.paramids, child_params = finalize_plan(root,
finalize_plan(root, plan->lefttree,
plan->lefttree, valid_params,
valid_params, scan_params);
scan_params)); context.paramids = bms_add_members(context.paramids, child_params);
context.paramids = bms_add_members(context.paramids, if (nestloop_params)
finalize_plan(root, {
plan->righttree, /* right child can reference nestloop_params as well as valid_params */
valid_params, child_params = finalize_plan(root,
scan_params)); plan->righttree,
bms_union(nestloop_params, valid_params),
scan_params);
/* ... and they don't count as parameters used at my level */
child_params = bms_difference(child_params, nestloop_params);
bms_free(nestloop_params);
}
else
{
/* easy case */
child_params = finalize_plan(root,
plan->righttree,
valid_params,
scan_params);
}
context.paramids = bms_add_members(context.paramids, child_params);
/* /*
* Any locally generated parameter doesn't count towards its generating * Any locally generated parameter doesn't count towards its generating
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/executor.h,v 1.169 2010/07/09 14:06:01 rhaas Exp $ * $PostgreSQL: pgsql/src/include/executor/executor.h,v 1.170 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -82,7 +82,7 @@ extern PGDLLIMPORT ExecutorCheckPerms_hook_type ExecutorCheckPerms_hook; ...@@ -82,7 +82,7 @@ extern PGDLLIMPORT ExecutorCheckPerms_hook_type ExecutorCheckPerms_hook;
/* /*
* prototypes from functions in execAmi.c * prototypes from functions in execAmi.c
*/ */
extern void ExecReScan(PlanState *node, ExprContext *exprCtxt); extern void ExecReScan(PlanState *node);
extern void ExecMarkPos(PlanState *node); extern void ExecMarkPos(PlanState *node);
extern void ExecRestrPos(PlanState *node); extern void ExecRestrPos(PlanState *node);
extern bool ExecSupportsMarkRestore(NodeTag plantype); extern bool ExecSupportsMarkRestore(NodeTag plantype);
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeAgg.h,v 1.32 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeAgg.h,v 1.33 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,7 +19,7 @@ ...@@ -19,7 +19,7 @@
extern AggState *ExecInitAgg(Agg *node, EState *estate, int eflags); extern AggState *ExecInitAgg(Agg *node, EState *estate, int eflags);
extern TupleTableSlot *ExecAgg(AggState *node); extern TupleTableSlot *ExecAgg(AggState *node);
extern void ExecEndAgg(AggState *node); extern void ExecEndAgg(AggState *node);
extern void ExecReScanAgg(AggState *node, ExprContext *exprCtxt); extern void ExecReScanAgg(AggState *node);
extern Size hash_agg_entry_size(int numAggs); extern Size hash_agg_entry_size(int numAggs);
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeAppend.h,v 1.30 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeAppend.h,v 1.31 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern AppendState *ExecInitAppend(Append *node, EState *estate, int eflags); extern AppendState *ExecInitAppend(Append *node, EState *estate, int eflags);
extern TupleTableSlot *ExecAppend(AppendState *node); extern TupleTableSlot *ExecAppend(AppendState *node);
extern void ExecEndAppend(AppendState *node); extern void ExecEndAppend(AppendState *node);
extern void ExecReScanAppend(AppendState *node, ExprContext *exprCtxt); extern void ExecReScanAppend(AppendState *node);
#endif /* NODEAPPEND_H */ #endif /* NODEAPPEND_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeBitmapAnd.h,v 1.8 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeBitmapAnd.h,v 1.9 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern BitmapAndState *ExecInitBitmapAnd(BitmapAnd *node, EState *estate, int eflags); extern BitmapAndState *ExecInitBitmapAnd(BitmapAnd *node, EState *estate, int eflags);
extern Node *MultiExecBitmapAnd(BitmapAndState *node); extern Node *MultiExecBitmapAnd(BitmapAndState *node);
extern void ExecEndBitmapAnd(BitmapAndState *node); extern void ExecEndBitmapAnd(BitmapAndState *node);
extern void ExecReScanBitmapAnd(BitmapAndState *node, ExprContext *exprCtxt); extern void ExecReScanBitmapAnd(BitmapAndState *node);
#endif /* NODEBITMAPAND_H */ #endif /* NODEBITMAPAND_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeBitmapHeapscan.h,v 1.8 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeBitmapHeapscan.h,v 1.9 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern BitmapHeapScanState *ExecInitBitmapHeapScan(BitmapHeapScan *node, EState *estate, int eflags); extern BitmapHeapScanState *ExecInitBitmapHeapScan(BitmapHeapScan *node, EState *estate, int eflags);
extern TupleTableSlot *ExecBitmapHeapScan(BitmapHeapScanState *node); extern TupleTableSlot *ExecBitmapHeapScan(BitmapHeapScanState *node);
extern void ExecEndBitmapHeapScan(BitmapHeapScanState *node); extern void ExecEndBitmapHeapScan(BitmapHeapScanState *node);
extern void ExecBitmapHeapReScan(BitmapHeapScanState *node, ExprContext *exprCtxt); extern void ExecReScanBitmapHeapScan(BitmapHeapScanState *node);
#endif /* NODEBITMAPHEAPSCAN_H */ #endif /* NODEBITMAPHEAPSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeBitmapIndexscan.h,v 1.8 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeBitmapIndexscan.h,v 1.9 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern BitmapIndexScanState *ExecInitBitmapIndexScan(BitmapIndexScan *node, EState *estate, int eflags); extern BitmapIndexScanState *ExecInitBitmapIndexScan(BitmapIndexScan *node, EState *estate, int eflags);
extern Node *MultiExecBitmapIndexScan(BitmapIndexScanState *node); extern Node *MultiExecBitmapIndexScan(BitmapIndexScanState *node);
extern void ExecEndBitmapIndexScan(BitmapIndexScanState *node); extern void ExecEndBitmapIndexScan(BitmapIndexScanState *node);
extern void ExecBitmapIndexReScan(BitmapIndexScanState *node, ExprContext *exprCtxt); extern void ExecReScanBitmapIndexScan(BitmapIndexScanState *node);
#endif /* NODEBITMAPINDEXSCAN_H */ #endif /* NODEBITMAPINDEXSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeBitmapOr.h,v 1.8 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeBitmapOr.h,v 1.9 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern BitmapOrState *ExecInitBitmapOr(BitmapOr *node, EState *estate, int eflags); extern BitmapOrState *ExecInitBitmapOr(BitmapOr *node, EState *estate, int eflags);
extern Node *MultiExecBitmapOr(BitmapOrState *node); extern Node *MultiExecBitmapOr(BitmapOrState *node);
extern void ExecEndBitmapOr(BitmapOrState *node); extern void ExecEndBitmapOr(BitmapOrState *node);
extern void ExecReScanBitmapOr(BitmapOrState *node, ExprContext *exprCtxt); extern void ExecReScanBitmapOr(BitmapOrState *node);
#endif /* NODEBITMAPOR_H */ #endif /* NODEBITMAPOR_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeCtescan.h,v 1.4 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeCtescan.h,v 1.5 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern CteScanState *ExecInitCteScan(CteScan *node, EState *estate, int eflags); extern CteScanState *ExecInitCteScan(CteScan *node, EState *estate, int eflags);
extern TupleTableSlot *ExecCteScan(CteScanState *node); extern TupleTableSlot *ExecCteScan(CteScanState *node);
extern void ExecEndCteScan(CteScanState *node); extern void ExecEndCteScan(CteScanState *node);
extern void ExecCteScanReScan(CteScanState *node, ExprContext *exprCtxt); extern void ExecReScanCteScan(CteScanState *node);
#endif /* NODECTESCAN_H */ #endif /* NODECTESCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeFunctionscan.h,v 1.15 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeFunctionscan.h,v 1.16 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern FunctionScanState *ExecInitFunctionScan(FunctionScan *node, EState *estate, int eflags); extern FunctionScanState *ExecInitFunctionScan(FunctionScan *node, EState *estate, int eflags);
extern TupleTableSlot *ExecFunctionScan(FunctionScanState *node); extern TupleTableSlot *ExecFunctionScan(FunctionScanState *node);
extern void ExecEndFunctionScan(FunctionScanState *node); extern void ExecEndFunctionScan(FunctionScanState *node);
extern void ExecFunctionReScan(FunctionScanState *node, ExprContext *exprCtxt); extern void ExecReScanFunctionScan(FunctionScanState *node);
#endif /* NODEFUNCTIONSCAN_H */ #endif /* NODEFUNCTIONSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeGroup.h,v 1.35 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeGroup.h,v 1.36 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern GroupState *ExecInitGroup(Group *node, EState *estate, int eflags); extern GroupState *ExecInitGroup(Group *node, EState *estate, int eflags);
extern TupleTableSlot *ExecGroup(GroupState *node); extern TupleTableSlot *ExecGroup(GroupState *node);
extern void ExecEndGroup(GroupState *node); extern void ExecEndGroup(GroupState *node);
extern void ExecReScanGroup(GroupState *node, ExprContext *exprCtxt); extern void ExecReScanGroup(GroupState *node);
#endif /* NODEGROUP_H */ #endif /* NODEGROUP_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeHash.h,v 1.49 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeHash.h,v 1.50 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -20,7 +20,7 @@ extern HashState *ExecInitHash(Hash *node, EState *estate, int eflags); ...@@ -20,7 +20,7 @@ extern HashState *ExecInitHash(Hash *node, EState *estate, int eflags);
extern TupleTableSlot *ExecHash(HashState *node); extern TupleTableSlot *ExecHash(HashState *node);
extern Node *MultiExecHash(HashState *node); extern Node *MultiExecHash(HashState *node);
extern void ExecEndHash(HashState *node); extern void ExecEndHash(HashState *node);
extern void ExecReScanHash(HashState *node, ExprContext *exprCtxt); extern void ExecReScanHash(HashState *node);
extern HashJoinTable ExecHashTableCreate(Hash *node, List *hashOperators); extern HashJoinTable ExecHashTableCreate(Hash *node, List *hashOperators);
extern void ExecHashTableDestroy(HashJoinTable hashtable); extern void ExecHashTableDestroy(HashJoinTable hashtable);
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeHashjoin.h,v 1.40 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeHashjoin.h,v 1.41 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -20,7 +20,7 @@ ...@@ -20,7 +20,7 @@
extern HashJoinState *ExecInitHashJoin(HashJoin *node, EState *estate, int eflags); extern HashJoinState *ExecInitHashJoin(HashJoin *node, EState *estate, int eflags);
extern TupleTableSlot *ExecHashJoin(HashJoinState *node); extern TupleTableSlot *ExecHashJoin(HashJoinState *node);
extern void ExecEndHashJoin(HashJoinState *node); extern void ExecEndHashJoin(HashJoinState *node);
extern void ExecReScanHashJoin(HashJoinState *node, ExprContext *exprCtxt); extern void ExecReScanHashJoin(HashJoinState *node);
extern void ExecHashJoinSaveTuple(MinimalTuple tuple, uint32 hashvalue, extern void ExecHashJoinSaveTuple(MinimalTuple tuple, uint32 hashvalue,
BufFile **fileptr); BufFile **fileptr);
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeIndexscan.h,v 1.36 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeIndexscan.h,v 1.37 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,7 +21,7 @@ extern TupleTableSlot *ExecIndexScan(IndexScanState *node); ...@@ -21,7 +21,7 @@ extern TupleTableSlot *ExecIndexScan(IndexScanState *node);
extern void ExecEndIndexScan(IndexScanState *node); extern void ExecEndIndexScan(IndexScanState *node);
extern void ExecIndexMarkPos(IndexScanState *node); extern void ExecIndexMarkPos(IndexScanState *node);
extern void ExecIndexRestrPos(IndexScanState *node); extern void ExecIndexRestrPos(IndexScanState *node);
extern void ExecIndexReScan(IndexScanState *node, ExprContext *exprCtxt); extern void ExecReScanIndexScan(IndexScanState *node);
/* routines exported to share code with nodeBitmapIndexscan.c */ /* routines exported to share code with nodeBitmapIndexscan.c */
extern void ExecIndexBuildScanKeys(PlanState *planstate, Relation index, extern void ExecIndexBuildScanKeys(PlanState *planstate, Relation index,
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeLimit.h,v 1.18 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeLimit.h,v 1.19 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern LimitState *ExecInitLimit(Limit *node, EState *estate, int eflags); extern LimitState *ExecInitLimit(Limit *node, EState *estate, int eflags);
extern TupleTableSlot *ExecLimit(LimitState *node); extern TupleTableSlot *ExecLimit(LimitState *node);
extern void ExecEndLimit(LimitState *node); extern void ExecEndLimit(LimitState *node);
extern void ExecReScanLimit(LimitState *node, ExprContext *exprCtxt); extern void ExecReScanLimit(LimitState *node);
#endif /* NODELIMIT_H */ #endif /* NODELIMIT_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeLockRows.h,v 1.2 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeLockRows.h,v 1.3 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern LockRowsState *ExecInitLockRows(LockRows *node, EState *estate, int eflags); extern LockRowsState *ExecInitLockRows(LockRows *node, EState *estate, int eflags);
extern TupleTableSlot *ExecLockRows(LockRowsState *node); extern TupleTableSlot *ExecLockRows(LockRowsState *node);
extern void ExecEndLockRows(LockRowsState *node); extern void ExecEndLockRows(LockRowsState *node);
extern void ExecReScanLockRows(LockRowsState *node, ExprContext *exprCtxt); extern void ExecReScanLockRows(LockRowsState *node);
#endif /* NODELOCKROWS_H */ #endif /* NODELOCKROWS_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeMaterial.h,v 1.30 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeMaterial.h,v 1.31 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecMaterial(MaterialState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecMaterial(MaterialState *node);
extern void ExecEndMaterial(MaterialState *node); extern void ExecEndMaterial(MaterialState *node);
extern void ExecMaterialMarkPos(MaterialState *node); extern void ExecMaterialMarkPos(MaterialState *node);
extern void ExecMaterialRestrPos(MaterialState *node); extern void ExecMaterialRestrPos(MaterialState *node);
extern void ExecMaterialReScan(MaterialState *node, ExprContext *exprCtxt); extern void ExecReScanMaterial(MaterialState *node);
#endif /* NODEMATERIAL_H */ #endif /* NODEMATERIAL_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeMergejoin.h,v 1.29 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeMergejoin.h,v 1.30 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern MergeJoinState *ExecInitMergeJoin(MergeJoin *node, EState *estate, int eflags); extern MergeJoinState *ExecInitMergeJoin(MergeJoin *node, EState *estate, int eflags);
extern TupleTableSlot *ExecMergeJoin(MergeJoinState *node); extern TupleTableSlot *ExecMergeJoin(MergeJoinState *node);
extern void ExecEndMergeJoin(MergeJoinState *node); extern void ExecEndMergeJoin(MergeJoinState *node);
extern void ExecReScanMergeJoin(MergeJoinState *node, ExprContext *exprCtxt); extern void ExecReScanMergeJoin(MergeJoinState *node);
#endif /* NODEMERGEJOIN_H */ #endif /* NODEMERGEJOIN_H */
...@@ -6,7 +6,7 @@ ...@@ -6,7 +6,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeModifyTable.h,v 1.2 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeModifyTable.h,v 1.3 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -18,6 +18,6 @@ ...@@ -18,6 +18,6 @@
extern ModifyTableState *ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags); extern ModifyTableState *ExecInitModifyTable(ModifyTable *node, EState *estate, int eflags);
extern TupleTableSlot *ExecModifyTable(ModifyTableState *node); extern TupleTableSlot *ExecModifyTable(ModifyTableState *node);
extern void ExecEndModifyTable(ModifyTableState *node); extern void ExecEndModifyTable(ModifyTableState *node);
extern void ExecReScanModifyTable(ModifyTableState *node, ExprContext *exprCtxt); extern void ExecReScanModifyTable(ModifyTableState *node);
#endif /* NODEMODIFYTABLE_H */ #endif /* NODEMODIFYTABLE_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeNestloop.h,v 1.30 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeNestloop.h,v 1.31 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern NestLoopState *ExecInitNestLoop(NestLoop *node, EState *estate, int eflags); extern NestLoopState *ExecInitNestLoop(NestLoop *node, EState *estate, int eflags);
extern TupleTableSlot *ExecNestLoop(NestLoopState *node); extern TupleTableSlot *ExecNestLoop(NestLoopState *node);
extern void ExecEndNestLoop(NestLoopState *node); extern void ExecEndNestLoop(NestLoopState *node);
extern void ExecReScanNestLoop(NestLoopState *node, ExprContext *exprCtxt); extern void ExecReScanNestLoop(NestLoopState *node);
#endif /* NODENESTLOOP_H */ #endif /* NODENESTLOOP_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeRecursiveunion.h,v 1.4 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeRecursiveunion.h,v 1.5 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern RecursiveUnionState *ExecInitRecursiveUnion(RecursiveUnion *node, EState *estate, int eflags); extern RecursiveUnionState *ExecInitRecursiveUnion(RecursiveUnion *node, EState *estate, int eflags);
extern TupleTableSlot *ExecRecursiveUnion(RecursiveUnionState *node); extern TupleTableSlot *ExecRecursiveUnion(RecursiveUnionState *node);
extern void ExecEndRecursiveUnion(RecursiveUnionState *node); extern void ExecEndRecursiveUnion(RecursiveUnionState *node);
extern void ExecRecursiveUnionReScan(RecursiveUnionState *node, ExprContext *exprCtxt); extern void ExecReScanRecursiveUnion(RecursiveUnionState *node);
#endif /* NODERECURSIVEUNION_H */ #endif /* NODERECURSIVEUNION_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeResult.h,v 1.28 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeResult.h,v 1.29 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecResult(ResultState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecResult(ResultState *node);
extern void ExecEndResult(ResultState *node); extern void ExecEndResult(ResultState *node);
extern void ExecResultMarkPos(ResultState *node); extern void ExecResultMarkPos(ResultState *node);
extern void ExecResultRestrPos(ResultState *node); extern void ExecResultRestrPos(ResultState *node);
extern void ExecReScanResult(ResultState *node, ExprContext *exprCtxt); extern void ExecReScanResult(ResultState *node);
#endif /* NODERESULT_H */ #endif /* NODERESULT_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeSeqscan.h,v 1.29 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeSeqscan.h,v 1.30 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecSeqScan(SeqScanState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecSeqScan(SeqScanState *node);
extern void ExecEndSeqScan(SeqScanState *node); extern void ExecEndSeqScan(SeqScanState *node);
extern void ExecSeqMarkPos(SeqScanState *node); extern void ExecSeqMarkPos(SeqScanState *node);
extern void ExecSeqRestrPos(SeqScanState *node); extern void ExecSeqRestrPos(SeqScanState *node);
extern void ExecSeqReScan(SeqScanState *node, ExprContext *exprCtxt); extern void ExecReScanSeqScan(SeqScanState *node);
#endif /* NODESEQSCAN_H */ #endif /* NODESEQSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeSetOp.h,v 1.18 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeSetOp.h,v 1.19 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern SetOpState *ExecInitSetOp(SetOp *node, EState *estate, int eflags); extern SetOpState *ExecInitSetOp(SetOp *node, EState *estate, int eflags);
extern TupleTableSlot *ExecSetOp(SetOpState *node); extern TupleTableSlot *ExecSetOp(SetOpState *node);
extern void ExecEndSetOp(SetOpState *node); extern void ExecEndSetOp(SetOpState *node);
extern void ExecReScanSetOp(SetOpState *node, ExprContext *exprCtxt); extern void ExecReScanSetOp(SetOpState *node);
#endif /* NODESETOP_H */ #endif /* NODESETOP_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeSort.h,v 1.27 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeSort.h,v 1.28 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecSort(SortState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecSort(SortState *node);
extern void ExecEndSort(SortState *node); extern void ExecEndSort(SortState *node);
extern void ExecSortMarkPos(SortState *node); extern void ExecSortMarkPos(SortState *node);
extern void ExecSortRestrPos(SortState *node); extern void ExecSortRestrPos(SortState *node);
extern void ExecReScanSort(SortState *node, ExprContext *exprCtxt); extern void ExecReScanSort(SortState *node);
#endif /* NODESORT_H */ #endif /* NODESORT_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeSubqueryscan.h,v 1.18 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeSubqueryscan.h,v 1.19 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern SubqueryScanState *ExecInitSubqueryScan(SubqueryScan *node, EState *estate, int eflags); extern SubqueryScanState *ExecInitSubqueryScan(SubqueryScan *node, EState *estate, int eflags);
extern TupleTableSlot *ExecSubqueryScan(SubqueryScanState *node); extern TupleTableSlot *ExecSubqueryScan(SubqueryScanState *node);
extern void ExecEndSubqueryScan(SubqueryScanState *node); extern void ExecEndSubqueryScan(SubqueryScanState *node);
extern void ExecSubqueryReScan(SubqueryScanState *node, ExprContext *exprCtxt); extern void ExecReScanSubqueryScan(SubqueryScanState *node);
#endif /* NODESUBQUERYSCAN_H */ #endif /* NODESUBQUERYSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeTidscan.h,v 1.22 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeTidscan.h,v 1.23 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecTidScan(TidScanState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecTidScan(TidScanState *node);
extern void ExecEndTidScan(TidScanState *node); extern void ExecEndTidScan(TidScanState *node);
extern void ExecTidMarkPos(TidScanState *node); extern void ExecTidMarkPos(TidScanState *node);
extern void ExecTidRestrPos(TidScanState *node); extern void ExecTidRestrPos(TidScanState *node);
extern void ExecTidReScan(TidScanState *node, ExprContext *exprCtxt); extern void ExecReScanTidScan(TidScanState *node);
#endif /* NODETIDSCAN_H */ #endif /* NODETIDSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeUnique.h,v 1.27 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeUnique.h,v 1.28 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern UniqueState *ExecInitUnique(Unique *node, EState *estate, int eflags); extern UniqueState *ExecInitUnique(Unique *node, EState *estate, int eflags);
extern TupleTableSlot *ExecUnique(UniqueState *node); extern TupleTableSlot *ExecUnique(UniqueState *node);
extern void ExecEndUnique(UniqueState *node); extern void ExecEndUnique(UniqueState *node);
extern void ExecReScanUnique(UniqueState *node, ExprContext *exprCtxt); extern void ExecReScanUnique(UniqueState *node);
#endif /* NODEUNIQUE_H */ #endif /* NODEUNIQUE_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeValuesscan.h,v 1.7 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeValuesscan.h,v 1.8 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecValuesScan(ValuesScanState *node); ...@@ -21,6 +21,6 @@ extern TupleTableSlot *ExecValuesScan(ValuesScanState *node);
extern void ExecEndValuesScan(ValuesScanState *node); extern void ExecEndValuesScan(ValuesScanState *node);
extern void ExecValuesMarkPos(ValuesScanState *node); extern void ExecValuesMarkPos(ValuesScanState *node);
extern void ExecValuesRestrPos(ValuesScanState *node); extern void ExecValuesRestrPos(ValuesScanState *node);
extern void ExecValuesReScan(ValuesScanState *node, ExprContext *exprCtxt); extern void ExecReScanValuesScan(ValuesScanState *node);
#endif /* NODEVALUESSCAN_H */ #endif /* NODEVALUESSCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeWindowAgg.h,v 1.4 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeWindowAgg.h,v 1.5 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern WindowAggState *ExecInitWindowAgg(WindowAgg *node, EState *estate, int eflags); extern WindowAggState *ExecInitWindowAgg(WindowAgg *node, EState *estate, int eflags);
extern TupleTableSlot *ExecWindowAgg(WindowAggState *node); extern TupleTableSlot *ExecWindowAgg(WindowAggState *node);
extern void ExecEndWindowAgg(WindowAggState *node); extern void ExecEndWindowAgg(WindowAggState *node);
extern void ExecReScanWindowAgg(WindowAggState *node, ExprContext *exprCtxt); extern void ExecReScanWindowAgg(WindowAggState *node);
#endif /* NODEWINDOWAGG_H */ #endif /* NODEWINDOWAGG_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/executor/nodeWorktablescan.h,v 1.4 2010/01/02 16:58:03 momjian Exp $ * $PostgreSQL: pgsql/src/include/executor/nodeWorktablescan.h,v 1.5 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -19,6 +19,6 @@ ...@@ -19,6 +19,6 @@
extern WorkTableScanState *ExecInitWorkTableScan(WorkTableScan *node, EState *estate, int eflags); extern WorkTableScanState *ExecInitWorkTableScan(WorkTableScan *node, EState *estate, int eflags);
extern TupleTableSlot *ExecWorkTableScan(WorkTableScanState *node); extern TupleTableSlot *ExecWorkTableScan(WorkTableScanState *node);
extern void ExecEndWorkTableScan(WorkTableScanState *node); extern void ExecEndWorkTableScan(WorkTableScanState *node);
extern void ExecWorkTableScanReScan(WorkTableScanState *node, ExprContext *exprCtxt); extern void ExecReScanWorkTableScan(WorkTableScanState *node);
#endif /* NODEWORKTABLESCAN_H */ #endif /* NODEWORKTABLESCAN_H */
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/nodes.h,v 1.234 2010/03/28 22:59:33 tgl Exp $ * $PostgreSQL: pgsql/src/include/nodes/nodes.h,v 1.235 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -74,6 +74,7 @@ typedef enum NodeTag ...@@ -74,6 +74,7 @@ typedef enum NodeTag
T_LockRows, T_LockRows,
T_Limit, T_Limit,
/* these aren't subclasses of Plan: */ /* these aren't subclasses of Plan: */
T_NestLoopParam,
T_PlanRowMark, T_PlanRowMark,
T_PlanInvalItem, T_PlanInvalItem,
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/plannodes.h,v 1.117 2010/02/26 02:01:25 momjian Exp $ * $PostgreSQL: pgsql/src/include/nodes/plannodes.h,v 1.118 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -432,13 +432,26 @@ typedef struct Join ...@@ -432,13 +432,26 @@ typedef struct Join
/* ---------------- /* ----------------
* nest loop join node * nest loop join node
*
* The nestParams list identifies any executor Params that must be passed
* into execution of the inner subplan carrying values from the current row
* of the outer subplan. Currently we restrict these values to be simple
* Vars, but perhaps someday that'd be worth relaxing.
* ---------------- * ----------------
*/ */
typedef struct NestLoop typedef struct NestLoop
{ {
Join join; Join join;
List *nestParams; /* list of NestLoopParam nodes */
} NestLoop; } NestLoop;
typedef struct NestLoopParam
{
NodeTag type;
int paramno; /* number of the PARAM_EXEC Param to set */
Var *paramval; /* outer-relation Var to assign to Param */
} NestLoopParam;
/* ---------------- /* ----------------
* merge join node * merge join node
* *
......
...@@ -10,7 +10,7 @@ ...@@ -10,7 +10,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/primnodes.h,v 1.156 2010/02/26 02:01:25 momjian Exp $ * $PostgreSQL: pgsql/src/include/nodes/primnodes.h,v 1.157 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -175,7 +175,8 @@ typedef struct Const ...@@ -175,7 +175,8 @@ typedef struct Const
* Such parameters are numbered from 1 to n. * Such parameters are numbered from 1 to n.
* *
* PARAM_EXEC: The parameter is an internal executor parameter, used * PARAM_EXEC: The parameter is an internal executor parameter, used
* for passing values into and out of sub-queries. * for passing values into and out of sub-queries or from
* nestloop joins to their inner scans.
* For historical reasons, such parameters are numbered from 0. * For historical reasons, such parameters are numbered from 0.
* These numbers are independent of PARAM_EXTERN numbers. * These numbers are independent of PARAM_EXTERN numbers.
* *
......
...@@ -7,7 +7,7 @@ ...@@ -7,7 +7,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/nodes/relation.h,v 1.187 2010/07/06 19:19:00 momjian Exp $ * $PostgreSQL: pgsql/src/include/nodes/relation.h,v 1.188 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -209,6 +209,10 @@ typedef struct PlannerInfo ...@@ -209,6 +209,10 @@ typedef struct PlannerInfo
int wt_param_id; /* PARAM_EXEC ID for the work table */ int wt_param_id; /* PARAM_EXEC ID for the work table */
struct Plan *non_recursive_plan; /* plan for non-recursive term */ struct Plan *non_recursive_plan; /* plan for non-recursive term */
/* These fields are workspace for createplan.c */
Relids curOuterRels; /* outer rels above current node */
List *curOuterParams; /* not-yet-assigned NestLoopParams */
/* optional private data for join_search_hook, e.g., GEQO */ /* optional private data for join_search_hook, e.g., GEQO */
void *join_search_private; void *join_search_private;
} PlannerInfo; } PlannerInfo;
...@@ -1330,16 +1334,20 @@ typedef struct PlaceHolderInfo ...@@ -1330,16 +1334,20 @@ typedef struct PlaceHolderInfo
/* /*
* glob->paramlist keeps track of the PARAM_EXEC slots that we have decided * glob->paramlist keeps track of the PARAM_EXEC slots that we have decided
* we need for the query. At runtime these slots are used to pass values * we need for the query. At runtime these slots are used to pass values
* either down into subqueries (for outer references in subqueries) or up out * around from one plan node to another. They can be used to pass values
* of subqueries (for the results of a subplan). The n'th entry in the list * down into subqueries (for outer references in subqueries), or up out of
* (n counts from 0) corresponds to Param->paramid = n. * subqueries (for the results of a subplan), or from a NestLoop plan node
* into its inner relation (when the inner scan is parameterized with values
* from the outer relation). The n'th entry in the list (n counts from 0)
* corresponds to Param->paramid = n.
* *
* Each paramlist item shows the absolute query level it is associated with, * Each paramlist item shows the absolute query level it is associated with,
* where the outermost query is level 1 and nested subqueries have higher * where the outermost query is level 1 and nested subqueries have higher
* numbers. The item the parameter slot represents can be one of three kinds: * numbers. The item the parameter slot represents can be one of three kinds:
* *
* A Var: the slot represents a variable of that level that must be passed * A Var: the slot represents a variable of that level that must be passed
* down because subqueries have outer references to it. The varlevelsup * down because subqueries have outer references to it, or must be passed
* from a NestLoop node of that level to its inner scan. The varlevelsup
* value in the Var will always be zero. * value in the Var will always be zero.
* *
* An Aggref (with an expression tree representing its argument): the slot * An Aggref (with an expression tree representing its argument): the slot
...@@ -1352,7 +1360,13 @@ typedef struct PlaceHolderInfo ...@@ -1352,7 +1360,13 @@ typedef struct PlaceHolderInfo
* to the parent query of the subplan. * to the parent query of the subplan.
* *
* Note: we detect duplicate Var parameters and coalesce them into one slot, * Note: we detect duplicate Var parameters and coalesce them into one slot,
* but we do not do this for Aggref or Param slots. * but we do not bother to do this for Aggrefs, and it would be incorrect
* to do so for Param slots. Duplicate detection is actually *necessary*
* in the case of NestLoop parameters since it serves to match up the usage
* of a Param (in the inner scan) with the assignment of the value (in the
* NestLoop node). This might result in the same PARAM_EXEC slot being used
* by multiple NestLoop nodes or SubPlan nodes, but no harm is done since
* the same value would be assigned anyway.
*/ */
typedef struct PlannerParamItem typedef struct PlannerParamItem
{ {
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
* Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group * Portions Copyright (c) 1996-2010, PostgreSQL Global Development Group
* Portions Copyright (c) 1994, Regents of the University of California * Portions Copyright (c) 1994, Regents of the University of California
* *
* $PostgreSQL: pgsql/src/include/optimizer/subselect.h,v 1.39 2010/01/02 16:58:07 momjian Exp $ * $PostgreSQL: pgsql/src/include/optimizer/subselect.h,v 1.40 2010/07/12 17:01:06 tgl Exp $
* *
*------------------------------------------------------------------------- *-------------------------------------------------------------------------
*/ */
...@@ -29,6 +29,7 @@ extern void SS_finalize_plan(PlannerInfo *root, Plan *plan, ...@@ -29,6 +29,7 @@ extern void SS_finalize_plan(PlannerInfo *root, Plan *plan,
bool attach_initplans); bool attach_initplans);
extern Param *SS_make_initplan_from_plan(PlannerInfo *root, Plan *plan, extern Param *SS_make_initplan_from_plan(PlannerInfo *root, Plan *plan,
Oid resulttype, int32 resulttypmod); Oid resulttype, int32 resulttypmod);
extern Param *assign_nestloop_param(PlannerInfo *root, Var *var);
extern int SS_assign_special_param(PlannerInfo *root); extern int SS_assign_special_param(PlannerInfo *root);
#endif /* SUBSELECT_H */ #endif /* SUBSELECT_H */
...@@ -281,7 +281,7 @@ INSERT INTO foorescan values(5007,5,'abc.5007.5'); ...@@ -281,7 +281,7 @@ INSERT INTO foorescan values(5007,5,'abc.5007.5');
INSERT INTO foorescan values(5008,5,'abc.5008.5'); INSERT INTO foorescan values(5008,5,'abc.5008.5');
INSERT INTO foorescan values(5009,5,'abc.5009.5'); INSERT INTO foorescan values(5009,5,'abc.5009.5');
CREATE FUNCTION foorescan(int,int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid >= $1 and fooid < $2 ;' LANGUAGE SQL; CREATE FUNCTION foorescan(int,int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid >= $1 and fooid < $2 ;' LANGUAGE SQL;
--invokes ExecFunctionReScan --invokes ExecReScanFunctionScan
SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,5004)) ORDER BY 1,2; SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,5004)) ORDER BY 1,2;
fooid | foosubid | fooname fooid | foosubid | fooname
-------+----------+------------ -------+----------+------------
...@@ -298,7 +298,7 @@ SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,500 ...@@ -298,7 +298,7 @@ SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,500
(10 rows) (10 rows)
CREATE VIEW vw_foorescan AS SELECT * FROM foorescan(5002,5004); CREATE VIEW vw_foorescan AS SELECT * FROM foorescan(5002,5004);
--invokes ExecFunctionReScan --invokes ExecReScanFunctionScan
SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM vw_foorescan) ORDER BY 1,2; SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM vw_foorescan) ORDER BY 1,2;
fooid | foosubid | fooname fooid | foosubid | fooname
-------+----------+------------ -------+----------+------------
...@@ -323,7 +323,7 @@ INSERT INTO barrescan values(5006); ...@@ -323,7 +323,7 @@ INSERT INTO barrescan values(5006);
INSERT INTO barrescan values(5007); INSERT INTO barrescan values(5007);
INSERT INTO barrescan values(5008); INSERT INTO barrescan values(5008);
CREATE FUNCTION foorescan(int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid = $1;' LANGUAGE SQL; CREATE FUNCTION foorescan(int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid = $1;' LANGUAGE SQL;
--invokes ExecFunctionReScan with chgParam != NULL --invokes ExecReScanFunctionScan with chgParam != NULL
SELECT f.* FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) ORDER BY 1,2; SELECT f.* FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) ORDER BY 1,2;
fooid | foosubid | fooname fooid | foosubid | fooname
-------+----------+------------ -------+----------+------------
......
...@@ -164,12 +164,12 @@ INSERT INTO foorescan values(5009,5,'abc.5009.5'); ...@@ -164,12 +164,12 @@ INSERT INTO foorescan values(5009,5,'abc.5009.5');
CREATE FUNCTION foorescan(int,int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid >= $1 and fooid < $2 ;' LANGUAGE SQL; CREATE FUNCTION foorescan(int,int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid >= $1 and fooid < $2 ;' LANGUAGE SQL;
--invokes ExecFunctionReScan --invokes ExecReScanFunctionScan
SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,5004)) ORDER BY 1,2; SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM foorescan(5002,5004)) ORDER BY 1,2;
CREATE VIEW vw_foorescan AS SELECT * FROM foorescan(5002,5004); CREATE VIEW vw_foorescan AS SELECT * FROM foorescan(5002,5004);
--invokes ExecFunctionReScan --invokes ExecReScanFunctionScan
SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM vw_foorescan) ORDER BY 1,2; SELECT * FROM foorescan f WHERE f.fooid IN (SELECT fooid FROM vw_foorescan) ORDER BY 1,2;
CREATE TABLE barrescan (fooid int primary key); CREATE TABLE barrescan (fooid int primary key);
...@@ -182,7 +182,7 @@ INSERT INTO barrescan values(5008); ...@@ -182,7 +182,7 @@ INSERT INTO barrescan values(5008);
CREATE FUNCTION foorescan(int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid = $1;' LANGUAGE SQL; CREATE FUNCTION foorescan(int) RETURNS setof foorescan AS 'SELECT * FROM foorescan WHERE fooid = $1;' LANGUAGE SQL;
--invokes ExecFunctionReScan with chgParam != NULL --invokes ExecReScanFunctionScan with chgParam != NULL
SELECT f.* FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) ORDER BY 1,2; SELECT f.* FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) ORDER BY 1,2;
SELECT b.fooid, max(f.foosubid) FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) GROUP BY b.fooid ORDER BY 1,2; SELECT b.fooid, max(f.foosubid) FROM barrescan b, foorescan f WHERE f.fooid = b.fooid AND b.fooid IN (SELECT fooid FROM foorescan(b.fooid)) GROUP BY b.fooid ORDER BY 1,2;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment