Commit c4427226 authored by Tomas Vondra's avatar Tomas Vondra

Fix handling of REWIND/MARK/BACKWARD in incremental sort

The executor flags were not handled entirely correctly, although the
bugs were mostly harmless and it was mostly comment inaccuracy. We don't
need to strip any of the flags for child nodes.

Incremental sort does not support backward scans of mark/restore, so
MARK/BACKWARDS flags should not be possible. So we simply ensure this
using an assert, and we don't bother removing them when initializing
the child node.

With REWIND it's a bit less clear - incremental sort does not support
REWIND, but there is no way to signal this - it's legal to just ignore
the flag. We however continue passing the flag to child nodes, because
they might be useful to leverage that.

Reported-by: Michael Paquier
Author: James Coleman
Reviewed-by: Tomas Vondra
Discussion: https://postgr.es/m/20200414065336.GI1492@paquier.xyz
parent 6c298881
...@@ -986,12 +986,11 @@ ExecInitIncrementalSort(IncrementalSort *node, EState *estate, int eflags) ...@@ -986,12 +986,11 @@ ExecInitIncrementalSort(IncrementalSort *node, EState *estate, int eflags)
SO_printf("ExecInitIncrementalSort: initializing sort node\n"); SO_printf("ExecInitIncrementalSort: initializing sort node\n");
/* /*
* Incremental sort can't be used with either EXEC_FLAG_REWIND, * Incremental sort can't be used with EXEC_FLAG_BACKWARD or EXEC_FLAG_MARK,
* EXEC_FLAG_BACKWARD or EXEC_FLAG_MARK, because we only one of many sort * because the current sort state contains only one sort batch rather than
* batches in the current sort state. * the full result set.
*/ */
Assert((eflags & (EXEC_FLAG_BACKWARD | Assert((eflags & (EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK)) == 0);
EXEC_FLAG_MARK)) == 0);
/* Initialize state structure. */ /* Initialize state structure. */
incrsortstate = makeNode(IncrementalSortState); incrsortstate = makeNode(IncrementalSortState);
...@@ -1041,11 +1040,11 @@ ExecInitIncrementalSort(IncrementalSort *node, EState *estate, int eflags) ...@@ -1041,11 +1040,11 @@ ExecInitIncrementalSort(IncrementalSort *node, EState *estate, int eflags)
/* /*
* Initialize child nodes. * Initialize child nodes.
* *
* We shield the child node from the need to support REWIND, BACKWARD, or * Incremental sort does not support backwards scans and mark/restore, so
* MARK/RESTORE. * we don't bother removing the flags from eflags here. We allow passing
* a REWIND flag, because although incremental sort can't use it, the child
* nodes may be able to do something more useful.
*/ */
eflags &= ~(EXEC_FLAG_REWIND | EXEC_FLAG_BACKWARD | EXEC_FLAG_MARK);
outerPlanState(incrsortstate) = ExecInitNode(outerPlan(node), estate, eflags); outerPlanState(incrsortstate) = ExecInitNode(outerPlan(node), estate, eflags);
/* /*
...@@ -1126,7 +1125,8 @@ ExecReScanIncrementalSort(IncrementalSortState *node) ...@@ -1126,7 +1125,8 @@ ExecReScanIncrementalSort(IncrementalSortState *node)
* store all tuples at once for the full sort. * store all tuples at once for the full sort.
* *
* So even if EXEC_FLAG_REWIND is set we just reset all of our state and * So even if EXEC_FLAG_REWIND is set we just reset all of our state and
* reexecute the sort along with the child node below us. * re-execute the sort along with the child node. Incremental sort itself
* can't do anything smarter, but maybe the child nodes can.
* *
* In theory if we've only filled the full sort with one batch (and haven't * In theory if we've only filled the full sort with one batch (and haven't
* reset it for a new batch yet) then we could efficiently rewind, but * reset it for a new batch yet) then we could efficiently rewind, but
......
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