|
|
|
|
/*-------------------------------------------------------------------------
|
|
|
|
|
*
|
|
|
|
|
* nodeMaterial.c
|
|
|
|
|
* Routines to handle materialization nodes.
|
|
|
|
|
*
|
|
|
|
|
* Portions Copyright (c) 1996-2018, PostgreSQL Global Development Group
|
|
|
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* IDENTIFICATION
|
|
|
|
|
* src/backend/executor/nodeMaterial.c
|
|
|
|
|
*
|
|
|
|
|
*-------------------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
/*
|
|
|
|
|
* INTERFACE ROUTINES
|
|
|
|
|
* ExecMaterial - materialize the result of a subplan
|
|
|
|
|
* ExecInitMaterial - initialize node and subnodes
|
|
|
|
|
* ExecEndMaterial - shutdown node and subnodes
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
#include "postgres.h"
|
|
|
|
|
|
|
|
|
|
#include "executor/executor.h"
|
|
|
|
|
#include "executor/nodeMaterial.h"
|
|
|
|
|
#include "miscadmin.h"
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecMaterial
|
|
|
|
|
*
|
|
|
|
|
* As long as we are at the end of the data collected in the tuplestore,
|
|
|
|
|
* we collect one new row from the subplan on each call, and stash it
|
|
|
|
|
* aside in the tuplestore before returning it. The tuplestore is
|
|
|
|
|
* only read if we are asked to scan backwards, rescan, or mark/restore.
|
|
|
|
|
*
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
static TupleTableSlot * /* result tuple from subplan */
|
|
|
|
|
ExecMaterial(PlanState *pstate)
|
|
|
|
|
{
|
|
|
|
|
MaterialState *node = castNode(MaterialState, pstate);
|
|
|
|
|
EState *estate;
|
|
|
|
|
ScanDirection dir;
|
|
|
|
|
bool forward;
|
|
|
|
|
Tuplestorestate *tuplestorestate;
|
|
|
|
|
bool eof_tuplestore;
|
|
|
|
|
TupleTableSlot *slot;
|
|
|
|
|
|
|
|
|
|
CHECK_FOR_INTERRUPTS();
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* get state info from node
|
|
|
|
|
*/
|
|
|
|
|
estate = node->ss.ps.state;
|
|
|
|
|
dir = estate->es_direction;
|
|
|
|
|
forward = ScanDirectionIsForward(dir);
|
|
|
|
|
tuplestorestate = node->tuplestorestate;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If first time through, and we need a tuplestore, initialize it.
|
|
|
|
|
*/
|
|
|
|
|
if (tuplestorestate == NULL && node->eflags != 0)
|
|
|
|
|
{
|
|
|
|
|
tuplestorestate = tuplestore_begin_heap(true, false, work_mem);
|
|
|
|
|
tuplestore_set_eflags(tuplestorestate, node->eflags);
|
|
|
|
|
if (node->eflags & EXEC_FLAG_MARK)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* Allocate a second read pointer to serve as the mark. We know it
|
|
|
|
|
* must have index 1, so needn't store that.
|
|
|
|
|
*/
|
|
|
|
|
int ptrno PG_USED_FOR_ASSERTS_ONLY;
|
|
|
|
|
|
|
|
|
|
ptrno = tuplestore_alloc_read_pointer(tuplestorestate,
|
|
|
|
|
node->eflags);
|
|
|
|
|
Assert(ptrno == 1);
|
|
|
|
|
}
|
|
|
|
|
node->tuplestorestate = tuplestorestate;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If we are not at the end of the tuplestore, or are going backwards, try
|
|
|
|
|
* to fetch a tuple from tuplestore.
|
|
|
|
|
*/
|
|
|
|
|
eof_tuplestore = (tuplestorestate == NULL) ||
|
|
|
|
|
tuplestore_ateof(tuplestorestate);
|
|
|
|
|
|
|
|
|
|
if (!forward && eof_tuplestore)
|
|
|
|
|
{
|
|
|
|
|
if (!node->eof_underlying)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* When reversing direction at tuplestore EOF, the first
|
|
|
|
|
* gettupleslot call will fetch the last-added tuple; but we want
|
|
|
|
|
* to return the one before that, if possible. So do an extra
|
|
|
|
|
* fetch.
|
|
|
|
|
*/
|
|
|
|
|
if (!tuplestore_advance(tuplestorestate, forward))
|
|
|
|
|
return NULL; /* the tuplestore must be empty */
|
|
|
|
|
}
|
|
|
|
|
eof_tuplestore = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If we can fetch another tuple from the tuplestore, return it.
|
|
|
|
|
*/
|
|
|
|
|
slot = node->ss.ps.ps_ResultTupleSlot;
|
|
|
|
|
if (!eof_tuplestore)
|
|
|
|
|
{
|
|
|
|
|
if (tuplestore_gettupleslot(tuplestorestate, forward, false, slot))
|
|
|
|
|
return slot;
|
|
|
|
|
if (forward)
|
|
|
|
|
eof_tuplestore = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If necessary, try to fetch another row from the subplan.
|
|
|
|
|
*
|
|
|
|
|
* Note: the eof_underlying state variable exists to short-circuit further
|
|
|
|
|
* subplan calls. It's not optional, unfortunately, because some plan
|
|
|
|
|
* node types are not robust about being called again when they've already
|
|
|
|
|
* returned NULL.
|
|
|
|
|
*/
|
|
|
|
|
if (eof_tuplestore && !node->eof_underlying)
|
|
|
|
|
{
|
|
|
|
|
PlanState *outerNode;
|
|
|
|
|
TupleTableSlot *outerslot;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* We can only get here with forward==true, so no need to worry about
|
|
|
|
|
* which direction the subplan will go.
|
|
|
|
|
*/
|
|
|
|
|
outerNode = outerPlanState(node);
|
|
|
|
|
outerslot = ExecProcNode(outerNode);
|
|
|
|
|
if (TupIsNull(outerslot))
|
|
|
|
|
{
|
|
|
|
|
node->eof_underlying = true;
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Append a copy of the returned tuple to tuplestore. NOTE: because
|
|
|
|
|
* the tuplestore is certainly in EOF state, its read position will
|
|
|
|
|
* move forward over the added tuple. This is what we want.
|
|
|
|
|
*/
|
|
|
|
|
if (tuplestorestate)
|
|
|
|
|
tuplestore_puttupleslot(tuplestorestate, outerslot);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* We can just return the subplan's returned tuple, without copying.
|
|
|
|
|
*/
|
|
|
|
|
return outerslot;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Nothing left ...
|
|
|
|
|
*/
|
|
|
|
|
return ExecClearTuple(slot);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecInitMaterial
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
MaterialState *
|
|
|
|
|
ExecInitMaterial(Material *node, EState *estate, int eflags)
|
|
|
|
|
{
|
|
|
|
|
MaterialState *matstate;
|
|
|
|
|
Plan *outerPlan;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* create state structure
|
|
|
|
|
*/
|
|
|
|
|
matstate = makeNode(MaterialState);
|
|
|
|
|
matstate->ss.ps.plan = (Plan *) node;
|
|
|
|
|
matstate->ss.ps.state = estate;
|
|
|
|
|
matstate->ss.ps.ExecProcNode = ExecMaterial;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* We must have a tuplestore buffering the subplan output to do backward
|
|
|
|
|
* scan or mark/restore. We also prefer to materialize the subplan output
|
|
|
|
|
* if we might be called on to rewind and replay it many times. However,
|
|
|
|
|
* if none of these cases apply, we can skip storing the data.
|
|
|
|
|
*/
|
|
|
|
|
matstate->eflags = (eflags & (EXEC_FLAG_REWIND |
|
|
|
|
|
EXEC_FLAG_BACKWARD |
|
|
|
|
|
EXEC_FLAG_MARK));
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Tuplestore's interpretation of the flag bits is subtly different from
|
|
|
|
|
* the general executor meaning: it doesn't think BACKWARD necessarily
|
|
|
|
|
* means "backwards all the way to start". If told to support BACKWARD we
|
|
|
|
|
* must include REWIND in the tuplestore eflags, else tuplestore_trim
|
|
|
|
|
* might throw away too much.
|
|
|
|
|
*/
|
|
|
|
|
if (eflags & EXEC_FLAG_BACKWARD)
|
|
|
|
|
matstate->eflags |= EXEC_FLAG_REWIND;
|
|
|
|
|
|
|
|
|
|
matstate->eof_underlying = false;
|
|
|
|
|
matstate->tuplestorestate = NULL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Miscellaneous initialization
|
|
|
|
|
*
|
|
|
|
|
* Materialization nodes don't need ExprContexts because they never call
|
|
|
|
|
* ExecQual or ExecProject.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* initialize child nodes
|
|
|
|
|
*
|
|
|
|
|
* We shield the child node from the need to support REWIND, BACKWARD, or
|
|
|
|
|
* MARK/RESTORE.
|
|
|
|
|
*/
|
|
|
|
|
eflags &= ~(EXEC_FLAG_REWIND | EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK);
|
|
|
|
|
|
|
|
|
|
outerPlan = outerPlan(node);
|
|
|
|
|
outerPlanState(matstate) = ExecInitNode(outerPlan, estate, eflags);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Initialize result type and slot. No need to initialize projection info
|
|
|
|
|
* because this node doesn't do projections.
|
|
|
|
|
*
|
|
|
|
|
* material nodes only return tuples from their materialized relation.
|
|
|
|
|
*/
|
Don't require return slots for nodes without projection.
In a lot of nodes the return slot is not required. That can either be
because the node doesn't do any projection (say an Append node), or
because the node does perform projections but the projection is
optimized away because the projection would yield an identical row.
Slots aren't that small, especially for wide rows, so it's worthwhile
to avoid creating them. It's not possible to just skip creating the
slot - it's currently used to determine the tuple descriptor returned
by ExecGetResultType(). So separate the determination of the result
type from the slot creation. The work previously done internally
ExecInitResultTupleSlotTL() can now also be done separately with
ExecInitResultTypeTL() and ExecInitResultSlot(). That way nodes that
aren't guaranteed to need a result slot, can use
ExecInitResultTypeTL() to determine the result type of the node, and
ExecAssignScanProjectionInfo() (via
ExecConditionalAssignProjectionInfo()) determines that a result slot
is needed, it is created with ExecInitResultSlot().
Besides the advantage of avoiding to create slots that then are
unused, this is necessary preparation for later patches around tuple
table slot abstraction. In particular separating the return descriptor
and slot is a prerequisite to allow JITing of tuple deforming with
knowledge of the underlying tuple format, and to avoid unnecessarily
creating JITed tuple deforming for virtual slots.
This commit removes a redundant argument from
ExecInitResultTupleSlotTL(). While this commit touches a lot of the
relevant lines anyway, it'd normally still not worthwhile to cause
breakage, except that aforementioned later commits will touch *all*
ExecInitResultTupleSlotTL() callers anyway (but fits worse
thematically).
Author: Andres Freund
Discussion: https://postgr.es/m/20181105210039.hh4vvi4vwoq5ba2q@alap3.anarazel.de
7 years ago
|
|
|
ExecInitResultTupleSlotTL(&matstate->ss.ps);
|
|
|
|
|
matstate->ss.ps.ps_ProjInfo = NULL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* initialize tuple type.
|
|
|
|
|
*/
|
|
|
|
|
ExecCreateScanSlotFromOuterPlan(estate, &matstate->ss);
|
|
|
|
|
|
|
|
|
|
return matstate;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecEndMaterial
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ExecEndMaterial(MaterialState *node)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* clean out the tuple table
|
|
|
|
|
*/
|
|
|
|
|
ExecClearTuple(node->ss.ss_ScanTupleSlot);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Release tuplestore resources
|
|
|
|
|
*/
|
|
|
|
|
if (node->tuplestorestate != NULL)
|
|
|
|
|
tuplestore_end(node->tuplestorestate);
|
|
|
|
|
node->tuplestorestate = NULL;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* shut down the subplan
|
|
|
|
|
*/
|
|
|
|
|
ExecEndNode(outerPlanState(node));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecMaterialMarkPos
|
|
|
|
|
*
|
|
|
|
|
* Calls tuplestore to save the current position in the stored file.
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ExecMaterialMarkPos(MaterialState *node)
|
|
|
|
|
{
|
|
|
|
|
Assert(node->eflags & EXEC_FLAG_MARK);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* if we haven't materialized yet, just return.
|
|
|
|
|
*/
|
|
|
|
|
if (!node->tuplestorestate)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* copy the active read pointer to the mark.
|
|
|
|
|
*/
|
|
|
|
|
tuplestore_copy_read_pointer(node->tuplestorestate, 0, 1);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* since we may have advanced the mark, try to truncate the tuplestore.
|
|
|
|
|
*/
|
|
|
|
|
tuplestore_trim(node->tuplestorestate);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecMaterialRestrPos
|
|
|
|
|
*
|
|
|
|
|
* Calls tuplestore to restore the last saved file position.
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ExecMaterialRestrPos(MaterialState *node)
|
|
|
|
|
{
|
|
|
|
|
Assert(node->eflags & EXEC_FLAG_MARK);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* if we haven't materialized yet, just return.
|
|
|
|
|
*/
|
|
|
|
|
if (!node->tuplestorestate)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* copy the mark to the active read pointer.
|
|
|
|
|
*/
|
|
|
|
|
tuplestore_copy_read_pointer(node->tuplestorestate, 1, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* ----------------------------------------------------------------
|
|
|
|
|
* ExecReScanMaterial
|
|
|
|
|
*
|
|
|
|
|
* Rescans the materialized relation.
|
|
|
|
|
* ----------------------------------------------------------------
|
|
|
|
|
*/
|
|
|
|
|
void
|
|
|
|
|
ExecReScanMaterial(MaterialState *node)
|
|
|
|
|
{
|
|
|
|
|
PlanState *outerPlan = outerPlanState(node);
|
|
|
|
|
|
|
|
|
|
ExecClearTuple(node->ss.ps.ps_ResultTupleSlot);
|
|
|
|
|
|
|
|
|
|
if (node->eflags != 0)
|
|
|
|
|
{
|
|
|
|
|
/*
|
|
|
|
|
* If we haven't materialized yet, just return. If outerplan's
|
|
|
|
|
* chgParam is not NULL then it will be re-scanned by ExecProcNode,
|
|
|
|
|
* else no reason to re-scan it at all.
|
|
|
|
|
*/
|
|
|
|
|
if (!node->tuplestorestate)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If subnode is to be rescanned then we forget previous stored
|
|
|
|
|
* results; we have to re-read the subplan and re-store. Also, if we
|
|
|
|
|
* told tuplestore it needn't support rescan, we lose and must
|
|
|
|
|
* re-read. (This last should not happen in common cases; else our
|
|
|
|
|
* caller lied by not passing EXEC_FLAG_REWIND to us.)
|
|
|
|
|
*
|
|
|
|
|
* Otherwise we can just rewind and rescan the stored output. The
|
|
|
|
|
* state of the subnode does not change.
|
|
|
|
|
*/
|
|
|
|
|
if (outerPlan->chgParam != NULL ||
|
|
|
|
|
(node->eflags & EXEC_FLAG_REWIND) == 0)
|
|
|
|
|
{
|
|
|
|
|
tuplestore_end(node->tuplestorestate);
|
|
|
|
|
node->tuplestorestate = NULL;
|
|
|
|
|
if (outerPlan->chgParam == NULL)
|
|
|
|
|
ExecReScan(outerPlan);
|
|
|
|
|
node->eof_underlying = false;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
tuplestore_rescan(node->tuplestorestate);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* In this case we are just passing on the subquery's output */
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* if chgParam of subnode is not null then plan will be re-scanned by
|
|
|
|
|
* first ExecProcNode.
|
|
|
|
|
*/
|
|
|
|
|
if (outerPlan->chgParam == NULL)
|
|
|
|
|
ExecReScan(outerPlan);
|
|
|
|
|
node->eof_underlying = false;
|
|
|
|
|
}
|
|
|
|
|
}
|