Skip to content

Commit

Permalink
Fix hang issue due to result node don't squelch outer node explicitly
Browse files Browse the repository at this point in the history
For a result node with one-time filter, if it's outer plan is not
empty and contains a motion node, then it needs to squelch the outer
node explicitly if the one-time filter check is false. This is necessary
espically for motion node under it, ExecSquelchNode() force a stop
message so the interconnect sender don't stuck at recending or polling
ACKs.
  • Loading branch information
pengzhout committed Jun 7, 2018
1 parent f48cd72 commit 2c011ce
Show file tree
Hide file tree
Showing 3 changed files with 55 additions and 0 deletions.
9 changes: 9 additions & 0 deletions src/backend/executor/nodeResult.c
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,15 @@ ExecResult(ResultState *node)
node->rs_checkqual = false;
if (!qualResult)
{
/*
* CDB: We'll read no more from outer subtree. To keep our
* sibling QEs from being starved, tell source QEs not to clog
* up the pipeline with our never-to-be-consumed data.
*/
PlanState *outerPlan = outerPlanState(node);
if (outerPlan)
ExecSquelchNode(outerPlan);

return NULL;
}
}
Expand Down
25 changes: 25 additions & 0 deletions src/test/regress/expected/ic.out
Original file line number Diff line number Diff line change
Expand Up @@ -502,3 +502,28 @@ NOTICE: Success:
t
(1 row)

-- Use WITH RECURSIVE to construct a one-time filter result node that executed
-- on QD, meanwhile, let the result node has a outer node which contain motion.
-- It's used to test that result node on QD can send a stop message to sender in
-- one-time filter case.
RESET gp_interconnect_snd_queue_depth;
RESET gp_interconnect_queue_depth;
CREATE TABLE recursive_table_ic (a INT) DISTRIBUTED BY (a);
-- Insert enough data so interconnect sender don't quit earlier.
INSERT INTO recursive_table_ic SELECT * FROM generate_series(20, 30000);
WITH RECURSIVE
r(i) AS (
SELECT 1
),
y(i) AS (
SELECT 1
UNION ALL
SELECT i + 1 FROM y, recursive_table_ic WHERE NOT EXISTS (SELECT * FROM r LIMIT 10)
)
SELECT * FROM y LIMIT 10;
i
---
1
(1 row)

DROP TABLE recursive_table_ic;
21 changes: 21 additions & 0 deletions src/test/regress/sql/ic.sql
Original file line number Diff line number Diff line change
Expand Up @@ -212,3 +212,24 @@ SELECT gp_inject_fault('interconnect_setup_palloc', 'error', 1);
SELECT * FROM a;
DROP TABLE a;
SELECT gp_inject_fault('interconnect_setup_palloc', 'reset', 1);

-- Use WITH RECURSIVE to construct a one-time filter result node that executed
-- on QD, meanwhile, let the result node has a outer node which contain motion.
-- It's used to test that result node on QD can send a stop message to sender in
-- one-time filter case.
RESET gp_interconnect_snd_queue_depth;
RESET gp_interconnect_queue_depth;
CREATE TABLE recursive_table_ic (a INT) DISTRIBUTED BY (a);
-- Insert enough data so interconnect sender don't quit earlier.
INSERT INTO recursive_table_ic SELECT * FROM generate_series(20, 30000);
WITH RECURSIVE
r(i) AS (
SELECT 1
),
y(i) AS (
SELECT 1
UNION ALL
SELECT i + 1 FROM y, recursive_table_ic WHERE NOT EXISTS (SELECT * FROM r LIMIT 10)
)
SELECT * FROM y LIMIT 10;
DROP TABLE recursive_table_ic;

0 comments on commit 2c011ce

Please sign in to comment.