nodeAppend.c 10.5 KB
Newer Older
1 2
/*-------------------------------------------------------------------------
 *
3
 * nodeAppend.c
4
 *	  routines to handle append nodes.
5
 *
B
Bruce Momjian 已提交
6
 * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
B
Add:  
Bruce Momjian 已提交
7
 * Portions Copyright (c) 1994, Regents of the University of California
8 9 10
 *
 *
 * IDENTIFICATION
B
Bruce Momjian 已提交
11
 *	  $Header: /cvsroot/pgsql/src/backend/executor/nodeAppend.c,v 1.45 2002/06/20 20:29:28 momjian Exp $
12 13 14 15
 *
 *-------------------------------------------------------------------------
 */
/* INTERFACE ROUTINES
16 17 18
 *		ExecInitAppend	- initialize the append node
 *		ExecProcAppend	- retrieve the next tuple from the node
 *		ExecEndAppend	- shut down the append node
19
 *		ExecReScanAppend - rescan the append node
20
 *
21 22 23 24 25 26
 *	 NOTES
 *		Each append node contains a list of one or more subplans which
 *		must be iteratively processed (forwards or backwards).
 *		Tuples are retrieved by executing the 'whichplan'th subplan
 *		until the subplan stops returning tuples, at which point that
 *		plan is shut down and the next started up.
27
 *
28 29 30
 *		Append nodes don't make use of their left and right
 *		subtrees, rather they maintain a list of subplans so
 *		a typical append node looks like this in the plan tree:
31
 *
32 33 34 35 36 37
 *				   ...
 *				   /
 *				Append -------+------+------+--- nil
 *				/	\		  |		 |		|
 *			  nil	nil		 ...	...    ...
 *								 subplans
38
 *
39 40
 *		Append nodes are currently used for unions, and to support
 *		inheritance queries, where several relations need to be scanned.
41 42 43 44
 *		For example, in our standard person/student/employee/student-emp
 *		example, where student and employee inherit from person
 *		and student-emp inherits from student and employee, the
 *		query:
45
 *
46
 *				retrieve (e.name) from e in person*
47
 *
48
 *		generates the plan:
49
 *
50 51 52 53 54 55
 *				  |
 *				Append -------+-------+--------+--------+
 *				/	\		  |		  |		   |		|
 *			  nil	nil		 Scan	 Scan	  Scan	   Scan
 *							  |		  |		   |		|
 *							person employee student student-emp
56
 */
57

58
#include "postgres.h"
59

60 61
#include "access/heapam.h"
#include "executor/execdebug.h"
62
#include "executor/nodeAppend.h"
B
Bruce Momjian 已提交
63
#include "parser/parsetree.h"
64

B
Bruce Momjian 已提交
65
static bool exec_append_initialize_next(Append *node);
66

67

68
/* ----------------------------------------------------------------
69
 *		exec_append_initialize_next
70 71 72 73 74
 *
 *		Sets up the append node state (i.e. the append state node)
 *		for the "next" scan.
 *
 *		Returns t iff there is a "next" scan to process.
75 76
 * ----------------------------------------------------------------
 */
77
static bool
B
Bruce Momjian 已提交
78
exec_append_initialize_next(Append *node)
79
{
80
	EState	   *estate;
81
	AppendState *appendstate;
82
	int			whichplan;
83

84 85
	/*
	 * get information from the append node
86 87
	 */
	estate = node->plan.state;
88 89
	appendstate = node->appendstate;
	whichplan = appendstate->as_whichplan;
90

91
	if (whichplan < appendstate->as_firstplan)
92
	{
93 94 95 96 97
		/*
		 * if scanning in reverse, we start at the last scan in the list
		 * and then proceed back to the first.. in any case we inform
		 * ExecProcAppend that we are at the end of the line by returning
		 * FALSE
98
		 */
99
		appendstate->as_whichplan = appendstate->as_firstplan;
100 101
		return FALSE;
	}
102
	else if (whichplan > appendstate->as_lastplan)
103
	{
104 105 106
		/*
		 * as above, end the scan if we go beyond the last scan in our
		 * list..
107
		 */
108
		appendstate->as_whichplan = appendstate->as_lastplan;
109 110 111 112
		return FALSE;
	}
	else
	{
113 114
		/*
		 * initialize the scan
115
		 *
116 117
		 * If we are controlling the target relation, select the proper
		 * active ResultRelInfo and junk filter for this target.
118
		 */
119
		if (node->isTarget)
120
		{
121 122 123 124 125
			Assert(whichplan < estate->es_num_result_relations);
			estate->es_result_relation_info =
				estate->es_result_relations + whichplan;
			estate->es_junkFilter =
				estate->es_result_relation_info->ri_junkFilter;
126
		}
127

128
		return TRUE;
129 130 131 132
	}
}

/* ----------------------------------------------------------------
133 134 135 136 137
 *		ExecInitAppend
 *
 *		Begins all of the subscans of the append node, storing the
 *		scan structures in the 'initialized' vector of the append-state
 *		structure.
138
 *
139 140 141 142 143
 *	   (This is potentially wasteful, since the entire result of the
 *		append node may not be scanned, but this way all of the
 *		structures get allocated in the executor's top level memory
 *		block instead of that of the call to ExecProcAppend.)
 *
144 145 146
 *		Special case: during an EvalPlanQual recheck query of an inherited
 *		target relation, we only want to initialize and scan the single
 *		subplan that corresponds to the target relation being checked.
147 148 149
 * ----------------------------------------------------------------
 */
bool
150
ExecInitAppend(Append *node, EState *estate, Plan *parent)
151
{
152
	AppendState *appendstate;
153
	int			nplans;
154
	List	   *appendplans;
155 156 157
	bool	   *initialized;
	int			i;
	Plan	   *initNode;
158 159

	CXT1_printf("ExecInitAppend: context is %d\n", CurrentMemoryContext);
160

161 162
	/*
	 * assign execution state to node and get information for append state
163 164 165
	 */
	node->plan.state = estate;

166 167
	appendplans = node->appendplans;
	nplans = length(appendplans);
168 169

	initialized = (bool *) palloc(nplans * sizeof(bool));
170
	MemSet(initialized, 0, nplans * sizeof(bool));
171

172 173
	/*
	 * create new AppendState for our append node
174
	 */
175 176 177
	appendstate = makeNode(AppendState);
	appendstate->as_nplans = nplans;
	appendstate->as_initialized = initialized;
178

179
	node->appendstate = appendstate;
180

181
	/*
182 183 184
	 * Do we want to scan just one subplan?  (Special case for
	 * EvalPlanQual) XXX pretty dirty way of determining that this case
	 * applies ...
185 186 187
	 */
	if (node->isTarget && estate->es_evTuple != NULL)
	{
188
		int			tplan;
189 190 191 192 193 194 195 196 197 198 199 200 201 202

		tplan = estate->es_result_relation_info - estate->es_result_relations;
		Assert(tplan >= 0 && tplan < nplans);

		appendstate->as_firstplan = tplan;
		appendstate->as_lastplan = tplan;
	}
	else
	{
		/* normal case, scan all subplans */
		appendstate->as_firstplan = 0;
		appendstate->as_lastplan = nplans - 1;
	}

203 204
	/*
	 * Miscellaneous initialization
205
	 *
206 207
	 * Append plans don't have expression contexts because they never call
	 * ExecQual or ExecProject.
208 209
	 */

210
#define APPEND_NSLOTS 1
211 212 213 214

	/*
	 * append nodes still have Result slots, which hold pointers to
	 * tuples, so we have to initialize them.
215
	 */
216
	ExecInitResultTupleSlot(estate, &appendstate->cstate);
217

218
	/*
219
	 * call ExecInitNode on each of the plans to be executed and save the
220
	 * results into the array "initialized"
221
	 */
222
	for (i = appendstate->as_firstplan; i <= appendstate->as_lastplan; i++)
223
	{
224
		appendstate->as_whichplan = i;
225 226
		exec_append_initialize_next(node);

227
		initNode = (Plan *) nth(i, appendplans);
228
		initialized[i] = ExecInitNode(initNode, estate, (Plan *) node);
229
	}
230

231 232
	/*
	 * initialize tuple type
233
	 */
234
	ExecAssignResultTypeFromTL((Plan *) node, &appendstate->cstate);
235
	appendstate->cstate.cs_ProjInfo = NULL;
236

237 238
	/*
	 * return the result from the first subplan's initialization
239
	 */
240
	appendstate->as_whichplan = appendstate->as_firstplan;
241
	exec_append_initialize_next(node);
242

243
	return TRUE;
244 245 246
}

int
B
Bruce Momjian 已提交
247
ExecCountSlotsAppend(Append *node)
248
{
249 250
	List	   *plan;
	int			nSlots = 0;
251

252
	foreach(plan, node->appendplans)
253 254
		nSlots += ExecCountSlotsNode((Plan *) lfirst(plan));
	return nSlots + APPEND_NSLOTS;
255 256 257
}

/* ----------------------------------------------------------------
258 259 260 261
 *	   ExecProcAppend
 *
 *		Handles the iteration over the multiple scans.
 *
262
 *	   NOTE: Can't call this ExecAppend, that name is used in execMain.
263 264 265
 * ----------------------------------------------------------------
 */
TupleTableSlot *
B
Bruce Momjian 已提交
266
ExecProcAppend(Append *node)
267
{
268
	EState	   *estate;
269
	AppendState *appendstate;
270
	int			whichplan;
271
	List	   *appendplans;
272
	Plan	   *subnode;
273 274
	TupleTableSlot *result;
	TupleTableSlot *result_slot;
275
	ScanDirection direction;
276

277 278
	/*
	 * get information from the node
279
	 */
280
	appendstate = node->appendstate;
281 282
	estate = node->plan.state;
	direction = estate->es_direction;
283 284 285
	appendplans = node->appendplans;
	whichplan = appendstate->as_whichplan;
	result_slot = appendstate->cstate.cs_ResultTupleSlot;
286

287 288
	/*
	 * figure out which subplan we are currently processing
289
	 */
290
	subnode = (Plan *) nth(whichplan, appendplans);
291 292

	if (subnode == NULL)
293
		elog(LOG, "ExecProcAppend: subnode is NULL");
294

295 296
	/*
	 * get a tuple from the subplan
297 298 299 300 301
	 */
	result = ExecProcNode(subnode, (Plan *) node);

	if (!TupIsNull(result))
	{
302 303 304
		/*
		 * if the subplan gave us something then place a copy of whatever
		 * we get into our result slot and return it.
305
		 *
306 307
		 * Note we rely on the subplan to retain ownership of the tuple for
		 * as long as we need it --- we don't copy it.
308
		 */
309
		return ExecStoreTuple(result->val, result_slot, InvalidBuffer, false);
310 311 312
	}
	else
	{
313 314 315
		/*
		 * .. go on to the "next" subplan in the appropriate direction and
		 * try processing again (recursively)
316 317
		 */
		if (ScanDirectionIsForward(direction))
318
			appendstate->as_whichplan++;
319
		else
320
			appendstate->as_whichplan--;
321

322 323 324
		/*
		 * return something from next node or an empty slot if all of our
		 * subplans have been exhausted.
325 326 327 328
		 */
		if (exec_append_initialize_next(node))
		{
			ExecSetSlotDescriptorIsNew(result_slot, true);
329
			return ExecProcAppend(node);
330 331 332 333
		}
		else
			return ExecClearTuple(result_slot);
	}
334 335 336
}

/* ----------------------------------------------------------------
337 338 339 340 341
 *		ExecEndAppend
 *
 *		Shuts down the subscans of the append node.
 *
 *		Returns nothing of interest.
342 343 344
 * ----------------------------------------------------------------
 */
void
B
Bruce Momjian 已提交
345
ExecEndAppend(Append *node)
346
{
347
	EState	   *estate;
348
	AppendState *appendstate;
349
	int			nplans;
350
	List	   *appendplans;
351 352
	bool	   *initialized;
	int			i;
353

354 355
	/*
	 * get information from the node
356
	 */
357
	appendstate = node->appendstate;
358
	estate = node->plan.state;
359 360 361
	appendplans = node->appendplans;
	nplans = appendstate->as_nplans;
	initialized = appendstate->as_initialized;
362

363 364
	/*
	 * shut down each of the subscans
365 366 367
	 */
	for (i = 0; i < nplans; i++)
	{
368
		if (initialized[i])
369
			ExecEndNode((Plan *) nth(i, appendplans), (Plan *) node);
370 371
	}
}
372

373 374 375 376
void
ExecReScanAppend(Append *node, ExprContext *exprCtxt, Plan *parent)
{
	AppendState *appendstate = node->appendstate;
377
	int			i;
378

379
	for (i = appendstate->as_firstplan; i <= appendstate->as_lastplan; i++)
380
	{
381
		Plan	   *subnode;
382

383
		subnode = (Plan *) nth(i, node->appendplans);
384

385 386 387 388 389 390
		/*
		 * ExecReScan doesn't know about my subplans, so I have to do
		 * changed-parameter signaling myself.
		 */
		if (node->plan.chgParam != NULL)
			SetChangedParamList(subnode, node->plan.chgParam);
391

392
		/*
393 394
		 * if chgParam of subnode is not null then plan will be re-scanned
		 * by first ExecProcNode.
395 396
		 */
		if (subnode->chgParam == NULL)
397
		{
398 399
			/* make sure estate is correct for this subnode (needed??) */
			appendstate->as_whichplan = i;
400
			exec_append_initialize_next(node);
401
			ExecReScan(subnode, exprCtxt, (Plan *) node);
402 403
		}
	}
404
	appendstate->as_whichplan = appendstate->as_firstplan;
405 406
	exec_append_initialize_next(node);
}