Allow subquery pullup to wrap a PlaceHolderVar in another one.
authorTom Lane <[email protected]>
Thu, 11 Jan 2024 20:28:13 +0000 (15:28 -0500)
committerTom Lane <[email protected]>
Thu, 11 Jan 2024 20:28:22 +0000 (15:28 -0500)
The code for wrapping subquery output expressions in PlaceHolderVars
believed that if the expression already was a PlaceHolderVar, it was
never necessary to wrap that in another one.  That's wrong if the
expression is underneath an outer join and involves a lateral
reference to outside that scope: failing to add an additional PHV
risks evaluating the expression at the wrong place and hence not
forcing it to null when the outer join should do so.  This is an
oversight in commit 9e7e29c75, which added logic to forcibly wrap
lateral-reference Vars in PlaceHolderVars, but didn't see that the
adjacent case for PlaceHolderVars needed the same treatment.

The test case we have for this doesn't fail before 4be058fe9, but now
that I see the problem I wonder if it is possible to demonstrate
related errors before that.  That's moot though, since all such
branches are out of support.

Per bug #18284 from Holger Reise.  Back- to all supported
branches.

Discussion: https://postgr.es/m/18284-47505a20c23647f8@postgresql.org

src/backend/optimizer/prep/prepjointree.c
src/test/regress/expected/join.out
src/test/regress/sql/join.sql

index d0df5374ef6cc32fe88a8d94b39d2ab6a46d41ff..aa83dd3636fe89dafac35170d250baafd0d7e1c5 100644 (file)
@@ -2435,8 +2435,13 @@ pullup_replace_vars_callback(Var *var,
            else if (newnode && IsA(newnode, PlaceHolderVar) &&
                     ((PlaceHolderVar *) newnode)->phlevelsup == 0)
            {
-               /* No need to wrap a PlaceHolderVar with another one, either */
-               wrap = false;
+               /* The same rules apply for a PlaceHolderVar */
+               if (rcon->target_rte->lateral &&
+                   !bms_is_subset(((PlaceHolderVar *) newnode)->phrels,
+                                  rcon->relids))
+                   wrap = true;
+               else
+                   wrap = false;
            }
            else
            {
index 321b6a7147bd6c2b00bdb456312852c6fc18ed88..e0418a3ae7549e1692648f28cf2a9e9be557e3ce 100644 (file)
@@ -7980,6 +7980,33 @@ select * from
          Output: (COALESCE((COALESCE(b.q2, '42'::bigint)), d.q2))
 (24 rows)
 
+-- another case requiring nested PlaceHolderVars
+explain (verbose, costs off)
+select * from
+  (select 0 as val0) as ss0
+  left join (select 1 as val) as ss1 on true
+  left join lateral (select ss1.val as val_filtered where false) as ss2 on true;
+           QUERY PLAN           
+--------------------------------
+ Nested Loop Left Join
+   Output: 0, (1), ((1))
+   Join Filter: false
+   ->  Result
+         Output: 1
+   ->  Result
+         Output: (1)
+         One-Time Filter: false
+(8 rows)
+
+select * from
+  (select 0 as val0) as ss0
+  left join (select 1 as val) as ss1 on true
+  left join lateral (select ss1.val as val_filtered where false) as ss2 on true;
+ val0 | val | val_filtered 
+------+-----+--------------
+    0 |   1 |             
+(1 row)
+
 -- case that breaks the old ph_may_need optimization
 explain (verbose, costs off)
 select c.*,a.*,ss1.q1,ss2.q1,ss3.* from
index 6d066ef9528b4369c97cf3bc2eff6d34bd63a6b7..e272ff5c14c5ae04d206d78b83b84fa4719028ef 100644 (file)
@@ -2950,6 +2950,18 @@ select * from
   ) on c.q2 = ss2.q1,
   lateral (select ss2.y offset 0) ss3;
 
+-- another case requiring nested PlaceHolderVars
+explain (verbose, costs off)
+select * from
+  (select 0 as val0) as ss0
+  left join (select 1 as val) as ss1 on true
+  left join lateral (select ss1.val as val_filtered where false) as ss2 on true;
+
+select * from
+  (select 0 as val0) as ss0
+  left join (select 1 as val) as ss1 on true
+  left join lateral (select ss1.val as val_filtered where false) as ss2 on true;
+
 -- case that breaks the old ph_may_need optimization
 explain (verbose, costs off)
 select c.*,a.*,ss1.q1,ss2.q1,ss3.* from