mirror of
https://github.com/postgres/postgres.git
synced 2025-09-03 15:22:11 +03:00
Fix handling of R/W expanded datums that are passed to SQL functions.
fmgr_sql must make expanded-datum arguments read-only, because it's possible that the function body will pass the argument to more than one callee function. If one of those functions takes the datum's R/W property as license to scribble on it, then later callees will see an unexpected value, leading to wrong answers. From a performance standpoint, it'd be nice to skip this in the common case that the argument value is passed to only one callee. However, detecting that seems fairly hard, and certainly not something that I care to attempt in a back-patched bug fix. Per report from Adam Mackler. This has been broken since we invented expanded datums, so back-patch to all supported branches. Discussion: https://postgr.es/m/WScDU5qfoZ7PB2gXwNqwGGgDPmWzz08VdydcPFLhOwUKZcdWbblbo-0Lku-qhuEiZoXJ82jpiQU4hOjOcrevYEDeoAvz6nR0IU4IHhXnaCA=@mackler.email Discussion: https://postgr.es/m/187436.1660143060@sss.pgh.pa.us
This commit is contained in:
@@ -666,9 +666,25 @@ SELECT * FROM voidtest5(3);
|
||||
-----------
|
||||
(0 rows)
|
||||
|
||||
-- Regression tests for bugs:
|
||||
-- Check that arguments that are R/W expanded datums aren't corrupted by
|
||||
-- multiple uses. This test knows that array_append() returns a R/W datum
|
||||
-- and will modify a R/W array input in-place. We use SETOF to prevent
|
||||
-- inlining of the SQL function.
|
||||
CREATE FUNCTION double_append(anyarray, anyelement) RETURNS SETOF anyarray
|
||||
LANGUAGE SQL IMMUTABLE AS
|
||||
$$ SELECT array_append($1, $2) || array_append($1, $2) $$;
|
||||
SELECT double_append(array_append(ARRAY[q1], q2), q3)
|
||||
FROM (VALUES(1,2,3), (4,5,6)) v(q1,q2,q3);
|
||||
double_append
|
||||
---------------
|
||||
{1,2,3,1,2,3}
|
||||
{4,5,6,4,5,6}
|
||||
(2 rows)
|
||||
|
||||
-- Cleanup
|
||||
DROP SCHEMA temp_func_test CASCADE;
|
||||
NOTICE: drop cascades to 29 other objects
|
||||
NOTICE: drop cascades to 30 other objects
|
||||
DETAIL: drop cascades to function functest_a_1(text,date)
|
||||
drop cascades to function functest_a_2(text[])
|
||||
drop cascades to function functest_a_3()
|
||||
@@ -698,5 +714,6 @@ drop cascades to function voidtest2(integer,integer)
|
||||
drop cascades to function voidtest3(integer)
|
||||
drop cascades to function voidtest4(integer)
|
||||
drop cascades to function voidtest5(integer)
|
||||
drop cascades to function double_append(anyarray,anyelement)
|
||||
DROP USER regress_unpriv_user;
|
||||
RESET search_path;
|
||||
|
@@ -385,6 +385,19 @@ CREATE FUNCTION voidtest5(a int) RETURNS SETOF VOID LANGUAGE SQL AS
|
||||
$$ SELECT generate_series(1, a) $$ STABLE;
|
||||
SELECT * FROM voidtest5(3);
|
||||
|
||||
-- Regression tests for bugs:
|
||||
|
||||
-- Check that arguments that are R/W expanded datums aren't corrupted by
|
||||
-- multiple uses. This test knows that array_append() returns a R/W datum
|
||||
-- and will modify a R/W array input in-place. We use SETOF to prevent
|
||||
-- inlining of the SQL function.
|
||||
CREATE FUNCTION double_append(anyarray, anyelement) RETURNS SETOF anyarray
|
||||
LANGUAGE SQL IMMUTABLE AS
|
||||
$$ SELECT array_append($1, $2) || array_append($1, $2) $$;
|
||||
|
||||
SELECT double_append(array_append(ARRAY[q1], q2), q3)
|
||||
FROM (VALUES(1,2,3), (4,5,6)) v(q1,q2,q3);
|
||||
|
||||
-- Cleanup
|
||||
DROP SCHEMA temp_func_test CASCADE;
|
||||
DROP USER regress_unpriv_user;
|
||||
|
Reference in New Issue
Block a user