]> granicus.if.org Git - postgresql/blob - src/backend/executor/nodeAppend.c
Add new palloc0 call as merge of palloc and MemSet(0).
[postgresql] / src / backend / executor / nodeAppend.c
1 /*-------------------------------------------------------------------------
2  *
3  * nodeAppend.c
4  *        routines to handle append nodes.
5  *
6  * Portions Copyright (c) 1996-2002, PostgreSQL Global Development Group
7  * Portions Copyright (c) 1994, Regents of the University of California
8  *
9  *
10  * IDENTIFICATION
11  *        $Header: /cvsroot/pgsql/src/backend/executor/nodeAppend.c,v 1.50 2002/11/13 00:39:47 momjian Exp $
12  *
13  *-------------------------------------------------------------------------
14  */
15 /* INTERFACE ROUTINES
16  *              ExecInitAppend  - initialize the append node
17  *              ExecProcAppend  - retrieve the next tuple from the node
18  *              ExecEndAppend   - shut down the append node
19  *              ExecReScanAppend - rescan the append node
20  *
21  *       NOTES
22  *              Each append node contains a list of one or more subplans which
23  *              must be iteratively processed (forwards or backwards).
24  *              Tuples are retrieved by executing the 'whichplan'th subplan
25  *              until the subplan stops returning tuples, at which point that
26  *              plan is shut down and the next started up.
27  *
28  *              Append nodes don't make use of their left and right
29  *              subtrees, rather they maintain a list of subplans so
30  *              a typical append node looks like this in the plan tree:
31  *
32  *                                 ...
33  *                                 /
34  *                              Append -------+------+------+--- nil
35  *                              /       \                 |              |              |
36  *                        nil   nil              ...    ...    ...
37  *                                                               subplans
38  *
39  *              Append nodes are currently used for unions, and to support
40  *              inheritance queries, where several relations need to be scanned.
41  *              For example, in our standard person/student/employee/student-emp
42  *              example, where student and employee inherit from person
43  *              and student-emp inherits from student and employee, the
44  *              query:
45  *
46  *                              retrieve (e.name) from e in person*
47  *
48  *              generates the plan:
49  *
50  *                                |
51  *                              Append -------+-------+--------+--------+
52  *                              /       \                 |               |                |            |
53  *                        nil   nil              Scan    Scan     Scan     Scan
54  *                                                        |               |                |            |
55  *                                                      person employee student student-emp
56  */
57
58 #include "postgres.h"
59
60 #include "access/heapam.h"
61 #include "executor/execdebug.h"
62 #include "executor/nodeAppend.h"
63 #include "parser/parsetree.h"
64
65 static bool exec_append_initialize_next(Append *node);
66
67
68 /* ----------------------------------------------------------------
69  *              exec_append_initialize_next
70  *
71  *              Sets up the append node state (i.e. the append state node)
72  *              for the "next" scan.
73  *
74  *              Returns t iff there is a "next" scan to process.
75  * ----------------------------------------------------------------
76  */
77 static bool
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         appendstate = node->appendstate;
89         whichplan = appendstate->as_whichplan;
90
91         if (whichplan < appendstate->as_firstplan)
92         {
93                 /*
94                  * if scanning in reverse, we start at the last scan in the list
95                  * and then proceed back to the first.. in any case we inform
96                  * ExecProcAppend that we are at the end of the line by returning
97                  * FALSE
98                  */
99                 appendstate->as_whichplan = appendstate->as_firstplan;
100                 return FALSE;
101         }
102         else if (whichplan > appendstate->as_lastplan)
103         {
104                 /*
105                  * as above, end the scan if we go beyond the last scan in our
106                  * list..
107                  */
108                 appendstate->as_whichplan = appendstate->as_lastplan;
109                 return FALSE;
110         }
111         else
112         {
113                 /*
114                  * initialize the scan
115                  *
116                  * If we are controlling the target relation, select the proper
117                  * active ResultRelInfo and junk filter for this target.
118                  */
119                 if (node->isTarget)
120                 {
121                         Assert(whichplan < estate->es_num_result_relations);
122                         estate->es_result_relation_info =
123                                 estate->es_result_relations + whichplan;
124                         estate->es_junkFilter =
125                                 estate->es_result_relation_info->ri_junkFilter;
126                 }
127
128                 return TRUE;
129         }
130 }
131
132 /* ----------------------------------------------------------------
133  *              ExecInitAppend
134  *
135  *              Begins all of the subscans of the append node, storing the
136  *              scan structures in the 'initialized' vector of the append-state
137  *              structure.
138  *
139  *         (This is potentially wasteful, since the entire result of the
140  *              append node may not be scanned, but this way all of the
141  *              structures get allocated in the executor's top level memory
142  *              block instead of that of the call to ExecProcAppend.)
143  *
144  *              Special case: during an EvalPlanQual recheck query of an inherited
145  *              target relation, we only want to initialize and scan the single
146  *              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         bool       *initialized;
156         int                     i;
157         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         node->plan.state = estate;
165
166         appendplans = node->appendplans;
167         nplans = length(appendplans);
168
169         initialized = (bool *) palloc0(nplans * sizeof(bool));
170
171         /*
172          * create new AppendState for our append node
173          */
174         appendstate = makeNode(AppendState);
175         appendstate->as_nplans = nplans;
176         appendstate->as_initialized = initialized;
177
178         node->appendstate = appendstate;
179
180         /*
181          * Do we want to scan just one subplan?  (Special case for
182          * EvalPlanQual) XXX pretty dirty way of determining that this case
183          * applies ...
184          */
185         if (node->isTarget && estate->es_evTuple != NULL)
186         {
187                 int                     tplan;
188
189                 tplan = estate->es_result_relation_info - estate->es_result_relations;
190                 Assert(tplan >= 0 && tplan < nplans);
191
192                 appendstate->as_firstplan = tplan;
193                 appendstate->as_lastplan = tplan;
194         }
195         else
196         {
197                 /* normal case, scan all subplans */
198                 appendstate->as_firstplan = 0;
199                 appendstate->as_lastplan = nplans - 1;
200         }
201
202         /*
203          * Miscellaneous initialization
204          *
205          * Append plans don't have expression contexts because they never call
206          * ExecQual or ExecProject.
207          */
208
209 #define APPEND_NSLOTS 1
210
211         /*
212          * append nodes still have Result slots, which hold pointers to
213          * tuples, so we have to initialize them.
214          */
215         ExecInitResultTupleSlot(estate, &appendstate->cstate);
216
217         /*
218          * call ExecInitNode on each of the plans to be executed and save the
219          * results into the array "initialized".  Note we *must* set
220          * estate->es_result_relation_info correctly while we initialize each
221          * sub-plan; ExecAssignResultTypeFromTL depends on that!
222          */
223         for (i = appendstate->as_firstplan; i <= appendstate->as_lastplan; i++)
224         {
225                 appendstate->as_whichplan = i;
226                 exec_append_initialize_next(node);
227
228                 initNode = (Plan *) nth(i, appendplans);
229                 initialized[i] = ExecInitNode(initNode, estate, (Plan *) node);
230         }
231
232         /*
233          * initialize tuple type
234          */
235         ExecAssignResultTypeFromTL((Plan *) node, &appendstate->cstate);
236         appendstate->cstate.cs_ProjInfo = NULL;
237
238         /*
239          * return the result from the first subplan's initialization
240          */
241         appendstate->as_whichplan = appendstate->as_firstplan;
242         exec_append_initialize_next(node);
243
244         return TRUE;
245 }
246
247 int
248 ExecCountSlotsAppend(Append *node)
249 {
250         List       *plan;
251         int                     nSlots = 0;
252
253         foreach(plan, node->appendplans)
254                 nSlots += ExecCountSlotsNode((Plan *) lfirst(plan));
255         return nSlots + APPEND_NSLOTS;
256 }
257
258 /* ----------------------------------------------------------------
259  *         ExecProcAppend
260  *
261  *              Handles the iteration over the multiple scans.
262  *
263  *         NOTE: Can't call this ExecAppend, that name is used in execMain.
264  * ----------------------------------------------------------------
265  */
266 TupleTableSlot *
267 ExecProcAppend(Append *node)
268 {
269         EState     *estate;
270         AppendState *appendstate;
271         int                     whichplan;
272         List       *appendplans;
273         Plan       *subnode;
274         TupleTableSlot *result;
275         TupleTableSlot *result_slot;
276         ScanDirection direction;
277
278         /*
279          * get information from the node
280          */
281         appendstate = node->appendstate;
282         estate = node->plan.state;
283         direction = estate->es_direction;
284         appendplans = node->appendplans;
285         whichplan = appendstate->as_whichplan;
286         result_slot = appendstate->cstate.cs_ResultTupleSlot;
287
288         /*
289          * figure out which subplan we are currently processing
290          */
291         subnode = (Plan *) nth(whichplan, appendplans);
292
293         if (subnode == NULL)
294                 elog(DEBUG1, "ExecProcAppend: subnode is NULL");
295
296         /*
297          * get a tuple from the subplan
298          */
299         result = ExecProcNode(subnode, (Plan *) node);
300
301         if (!TupIsNull(result))
302         {
303                 /*
304                  * if the subplan gave us something then place a copy of whatever
305                  * we get into our result slot and return it.
306                  *
307                  * Note we rely on the subplan to retain ownership of the tuple for
308                  * as long as we need it --- we don't copy it.
309                  */
310                 return ExecStoreTuple(result->val, result_slot, InvalidBuffer, false);
311         }
312         else
313         {
314                 /*
315                  * .. go on to the "next" subplan in the appropriate direction and
316                  * try processing again (recursively)
317                  */
318                 if (ScanDirectionIsForward(direction))
319                         appendstate->as_whichplan++;
320                 else
321                         appendstate->as_whichplan--;
322
323                 /*
324                  * return something from next node or an empty slot if all of our
325                  * subplans have been exhausted.
326                  */
327                 if (exec_append_initialize_next(node))
328                 {
329                         ExecSetSlotDescriptorIsNew(result_slot, true);
330                         return ExecProcAppend(node);
331                 }
332                 else
333                         return ExecClearTuple(result_slot);
334         }
335 }
336
337 /* ----------------------------------------------------------------
338  *              ExecEndAppend
339  *
340  *              Shuts down the subscans of the append node.
341  *
342  *              Returns nothing of interest.
343  * ----------------------------------------------------------------
344  */
345 void
346 ExecEndAppend(Append *node)
347 {
348         EState     *estate;
349         AppendState *appendstate;
350         int                     nplans;
351         List       *appendplans;
352         bool       *initialized;
353         int                     i;
354
355         /*
356          * get information from the node
357          */
358         appendstate = node->appendstate;
359         estate = node->plan.state;
360         appendplans = node->appendplans;
361         nplans = appendstate->as_nplans;
362         initialized = appendstate->as_initialized;
363
364         /*
365          * shut down each of the subscans
366          */
367         for (i = 0; i < nplans; i++)
368         {
369                 if (initialized[i])
370                         ExecEndNode((Plan *) nth(i, appendplans), (Plan *) node);
371         }
372 }
373
374 void
375 ExecReScanAppend(Append *node, ExprContext *exprCtxt, Plan *parent)
376 {
377         AppendState *appendstate = node->appendstate;
378         int                     i;
379
380         for (i = appendstate->as_firstplan; i <= appendstate->as_lastplan; i++)
381         {
382                 Plan       *subnode;
383
384                 subnode = (Plan *) nth(i, node->appendplans);
385
386                 /*
387                  * ExecReScan doesn't know about my subplans, so I have to do
388                  * changed-parameter signaling myself.
389                  */
390                 if (node->plan.chgParam != NULL)
391                         SetChangedParamList(subnode, node->plan.chgParam);
392
393                 /*
394                  * if chgParam of subnode is not null then plan will be re-scanned
395                  * by first ExecProcNode.
396                  */
397                 if (subnode->chgParam == NULL)
398                 {
399                         /* make sure estate is correct for this subnode (needed??) */
400                         appendstate->as_whichplan = i;
401                         exec_append_initialize_next(node);
402                         ExecReScan(subnode, exprCtxt, (Plan *) node);
403                 }
404         }
405         appendstate->as_whichplan = appendstate->as_firstplan;
406         exec_append_initialize_next(node);
407 }