This changes the argtypes argument of SPI_prepare(),
SPI_prepare_cursor(), SPI_cursor_open_with_args(), and
SPI_execute_with_args() from Oid *argtypes to const Oid *argtypes.
The underlying functions were already receptive to that, so this
doesn't require any significant changes beyond the function signatures
and some internal variables.
Commit 28972b6fc3dc recently introduced a case where a const had to be
cast away before calling these functions. This is fixed here.
From ad0857ce3c257ddc022ca48d93300d4f9957a489 Mon Sep 17 00:00:00 2001
From: Peter Eisentraut <[email protected]>
Date: Sat, 25 Apr 2026 23:44:36 +0200
Subject: [PATCH] Make SPI_prepare argtypes argument const
This changes the argtypes argument of SPI_prepare(),
SPI_prepare_cursor(), SPI_cursor_open_with_args(), and
SPI_execute_with_args() from Oid *argtypes to const Oid *argtypes.
The underlying functions were already receptive to that, so this
doesn't require any significant changes beyond the function signatures
and some internal variables.
Commit 28972b6fc3dc recently introduced a case where a const had to be
cast away before calling these functions. This is fixed here.
---
contrib/postgres_fdw/postgres_fdw.c | 6 +++---
doc/src/sgml/spi.sgml | 8 ++++----
src/backend/executor/spi.c | 26 ++++++++++++++++----------
src/backend/utils/adt/ri_triggers.c | 4 ++--
src/backend/utils/cache/plancache.c | 2 +-
src/include/executor/spi.h | 8 ++++----
src/include/executor/spi_priv.h | 2 +-
src/include/utils/plancache.h | 2 +-
8 files changed, 32 insertions(+), 26 deletions(-)
diff --git a/contrib/postgres_fdw/postgres_fdw.c
b/contrib/postgres_fdw/postgres_fdw.c
index 0a589f8db74..322212552ee 100644
--- a/contrib/postgres_fdw/postgres_fdw.c
+++ b/contrib/postgres_fdw/postgres_fdw.c
@@ -6164,12 +6164,12 @@ import_fetched_statistics(const char *schemaname,
Assert(PQntuples(remstats->att) >= 1);
attimport_plan = SPI_prepare(attimport_sql,
ATTIMPORT_SQL_NUM_FIELDS,
- (Oid
*) attimport_argtypes);
+
attimport_argtypes);
if (attimport_plan == NULL)
elog(ERROR, "failed to prepare attimport_sql query");
attclear_plan = SPI_prepare(attclear_sql,
ATTCLEAR_SQL_NUM_FIELDS,
- (Oid *)
attclear_argtypes);
+
attclear_argtypes);
if (attclear_plan == NULL)
elog(ERROR, "failed to prepare attclear_sql query");
@@ -6247,7 +6247,7 @@ import_fetched_statistics(const char *schemaname,
spirc = SPI_execute_with_args(relimport_sql,
RELIMPORT_SQL_NUM_FIELDS,
- (Oid *)
relimport_argtypes,
+
relimport_argtypes,
values,
nulls, false, 1);
if (spirc != SPI_OK_SELECT)
elog(ERROR, "failed to execute relimport_sql query for foreign
table \"%s.%s\"",
diff --git a/doc/src/sgml/spi.sgml b/doc/src/sgml/spi.sgml
index e30d0962ae7..043c6f00039 100644
--- a/doc/src/sgml/spi.sgml
+++ b/doc/src/sgml/spi.sgml
@@ -845,7 +845,7 @@ <title>Return Value</title>
<refsynopsisdiv>
<synopsis>
int SPI_execute_with_args(const char *<parameter>command</parameter>,
- int <parameter>nargs</parameter>, Oid
*<parameter>argtypes</parameter>,
+ int <parameter>nargs</parameter>, const Oid
*<parameter>argtypes</parameter>,
const Datum *<parameter>values</parameter>, const
char *<parameter>nulls</parameter>,
bool <parameter>read_only</parameter>, long
<parameter>count</parameter>)
</synopsis>
@@ -997,7 +997,7 @@ <title>Return Value</title>
<refsynopsisdiv>
<synopsis>
-SPIPlanPtr SPI_prepare(const char * <parameter>command</parameter>, int
<parameter>nargs</parameter>, Oid * <parameter>argtypes</parameter>)
+SPIPlanPtr SPI_prepare(const char * <parameter>command</parameter>, int
<parameter>nargs</parameter>, const Oid * <parameter>argtypes</parameter>)
</synopsis>
</refsynopsisdiv>
@@ -1160,7 +1160,7 @@ <title>Notes</title>
<refsynopsisdiv>
<synopsis>
SPIPlanPtr SPI_prepare_cursor(const char * <parameter>command</parameter>, int
<parameter>nargs</parameter>,
- Oid * <parameter>argtypes</parameter>, int
<parameter>cursorOptions</parameter>)
+ const Oid * <parameter>argtypes</parameter>, int
<parameter>cursorOptions</parameter>)
</synopsis>
</refsynopsisdiv>
@@ -2316,7 +2316,7 @@ <title>Return Value</title>
<synopsis>
Portal SPI_cursor_open_with_args(const char *<parameter>name</parameter>,
const char *<parameter>command</parameter>,
- int <parameter>nargs</parameter>, Oid
*<parameter>argtypes</parameter>,
+ int <parameter>nargs</parameter>, const Oid
*<parameter>argtypes</parameter>,
const Datum *<parameter>values</parameter>,
const char *<parameter>nulls</parameter>,
bool <parameter>read_only</parameter>, int
<parameter>cursorOptions</parameter>)
</synopsis>
diff --git a/src/backend/executor/spi.c b/src/backend/executor/spi.c
index 52f3b11301c..d884c962f14 100644
--- a/src/backend/executor/spi.c
+++ b/src/backend/executor/spi.c
@@ -68,7 +68,7 @@ static int _SPI_execute_plan(SPIPlanPtr plan, const
SPIExecuteOptions *options,
Snapshot snapshot,
Snapshot crosscheck_snapshot,
bool fire_triggers);
-static ParamListInfo _SPI_convert_params(int nargs, Oid *argtypes,
+static ParamListInfo _SPI_convert_params(int nargs, const Oid *argtypes,
const Datum *Values, const char *Nulls);
static int _SPI_pquery(QueryDesc *queryDesc, bool fire_triggers, uint64
tcount);
@@ -811,7 +811,7 @@ SPI_execute_snapshot(SPIPlanPtr plan,
*/
int
SPI_execute_with_args(const char *src,
- int nargs, Oid *argtypes,
+ int nargs, const Oid *argtypes,
const Datum *Values, const char
*Nulls,
bool read_only, long tcount)
{
@@ -858,13 +858,13 @@ SPI_execute_with_args(const char *src,
}
SPIPlanPtr
-SPI_prepare(const char *src, int nargs, Oid *argtypes)
+SPI_prepare(const char *src, int nargs, const Oid *argtypes)
{
return SPI_prepare_cursor(src, nargs, argtypes, 0);
}
SPIPlanPtr
-SPI_prepare_cursor(const char *src, int nargs, Oid *argtypes,
+SPI_prepare_cursor(const char *src, int nargs, const Oid *argtypes,
int cursorOptions)
{
_SPI_plan plan;
@@ -1472,7 +1472,7 @@ SPI_cursor_open(const char *name, SPIPlanPtr plan,
Portal
SPI_cursor_open_with_args(const char *name,
const char *src,
- int nargs, Oid *argtypes,
+ int nargs, const Oid
*argtypes,
Datum *Values, const char
*Nulls,
bool read_only, int
cursorOptions)
{
@@ -2846,7 +2846,7 @@ _SPI_execute_plan(SPIPlanPtr plan, const
SPIExecuteOptions *options,
* Convert arrays of query parameters to form wanted by planner and executor
*/
static ParamListInfo
-_SPI_convert_params(int nargs, Oid *argtypes,
+_SPI_convert_params(int nargs, const Oid *argtypes,
const Datum *Values, const char *Nulls)
{
ParamListInfo paramLI;
@@ -3170,8 +3170,11 @@ _SPI_make_plan_non_temp(SPIPlanPtr plan)
newplan->nargs = plan->nargs;
if (plan->nargs > 0)
{
- newplan->argtypes = palloc_array(Oid, plan->nargs);
- memcpy(newplan->argtypes, plan->argtypes, plan->nargs *
sizeof(Oid));
+ Oid *newplan_argtypes;
+
+ newplan_argtypes = palloc_array(Oid, plan->nargs);
+ memcpy(newplan_argtypes, plan->argtypes, plan->nargs *
sizeof(Oid));
+ newplan->argtypes = newplan_argtypes;
}
else
newplan->argtypes = NULL;
@@ -3235,8 +3238,11 @@ _SPI_save_plan(SPIPlanPtr plan)
newplan->nargs = plan->nargs;
if (plan->nargs > 0)
{
- newplan->argtypes = palloc_array(Oid, plan->nargs);
- memcpy(newplan->argtypes, plan->argtypes, plan->nargs *
sizeof(Oid));
+ Oid *newplan_argtypes;
+
+ newplan_argtypes = palloc_array(Oid, plan->nargs);
+ memcpy(newplan_argtypes, plan->argtypes, plan->nargs *
sizeof(Oid));
+ newplan->argtypes = newplan_argtypes;
}
else
newplan->argtypes = NULL;
diff --git a/src/backend/utils/adt/ri_triggers.c
b/src/backend/utils/adt/ri_triggers.c
index dc89c686394..4741b559c61 100644
--- a/src/backend/utils/adt/ri_triggers.c
+++ b/src/backend/utils/adt/ri_triggers.c
@@ -301,7 +301,7 @@ static RI_ConstraintInfo *ri_FetchConstraintInfo(Trigger
*trigger,
Relation trig_rel, bool rel_is_pk);
static RI_ConstraintInfo *ri_LoadConstraintInfo(Oid constraintOid);
static Oid get_ri_constraint_root(Oid constrOid);
-static SPIPlanPtr ri_PlanCheck(const char *querystr, int nargs, Oid *argtypes,
+static SPIPlanPtr ri_PlanCheck(const char *querystr, int nargs, const Oid
*argtypes,
RI_QueryKey *qkey,
Relation fk_rel, Relation pk_rel);
static bool ri_PerformCheck(const RI_ConstraintInfo *riinfo,
RI_QueryKey *qkey,
SPIPlanPtr qplan,
@@ -2582,7 +2582,7 @@ InvalidateConstraintCacheCallBack(Datum arg,
SysCacheIdentifier cacheid,
* Prepare execution plan for a query to enforce an RI restriction
*/
static SPIPlanPtr
-ri_PlanCheck(const char *querystr, int nargs, Oid *argtypes,
+ri_PlanCheck(const char *querystr, int nargs, const Oid *argtypes,
RI_QueryKey *qkey, Relation fk_rel, Relation pk_rel)
{
SPIPlanPtr qplan;
diff --git a/src/backend/utils/cache/plancache.c
b/src/backend/utils/cache/plancache.c
index 698e7c1aa22..26f1bd64515 100644
--- a/src/backend/utils/cache/plancache.c
+++ b/src/backend/utils/cache/plancache.c
@@ -393,7 +393,7 @@ void
CompleteCachedPlan(CachedPlanSource *plansource,
List *querytree_list,
MemoryContext querytree_context,
- Oid *param_types,
+ const Oid *param_types,
int num_params,
ParserSetupHook parserSetup,
void *parserSetupArg,
diff --git a/src/include/executor/spi.h b/src/include/executor/spi.h
index f4985cb715d..e809ba18098 100644
--- a/src/include/executor/spi.h
+++ b/src/include/executor/spi.h
@@ -127,11 +127,11 @@ extern int SPI_execute_snapshot(SPIPlanPtr plan,
Snapshot
crosscheck_snapshot,
bool
read_only, bool fire_triggers, long tcount);
extern int SPI_execute_with_args(const char *src,
- int nargs,
Oid *argtypes,
+ int nargs,
const Oid *argtypes,
const Datum
*Values, const char *Nulls,
bool
read_only, long tcount);
-extern SPIPlanPtr SPI_prepare(const char *src, int nargs, Oid *argtypes);
-extern SPIPlanPtr SPI_prepare_cursor(const char *src, int nargs, Oid *argtypes,
+extern SPIPlanPtr SPI_prepare(const char *src, int nargs, const Oid *argtypes);
+extern SPIPlanPtr SPI_prepare_cursor(const char *src, int nargs, const Oid
*argtypes,
int
cursorOptions);
extern SPIPlanPtr SPI_prepare_extended(const char *src,
const SPIPrepareOptions *options);
@@ -175,7 +175,7 @@ extern Portal SPI_cursor_open(const char *name, SPIPlanPtr
plan,
const Datum *Values,
const char *Nulls, bool read_only);
extern Portal SPI_cursor_open_with_args(const char *name,
const char *src,
-
int nargs, Oid *argtypes,
+
int nargs, const Oid *argtypes,
Datum *Values, const char *Nulls,
bool read_only, int cursorOptions);
extern Portal SPI_cursor_open_with_paramlist(const char *name, SPIPlanPtr plan,
diff --git a/src/include/executor/spi_priv.h b/src/include/executor/spi_priv.h
index 240383b97b9..9fb82ad0984 100644
--- a/src/include/executor/spi_priv.h
+++ b/src/include/executor/spi_priv.h
@@ -97,7 +97,7 @@ typedef struct _SPI_plan
RawParseMode parse_mode; /* raw_parser() mode */
int cursor_options; /* Cursor options used for
planning */
int nargs; /* number of plan
arguments */
- Oid *argtypes; /* Argument types (NULL if
nargs is 0) */
+ const Oid *argtypes; /* Argument types (NULL if nargs is 0)
*/
ParserSetupHook parserSetup; /* alternative parameter spec method */
void *parserSetupArg;
} _SPI_plan;
diff --git a/src/include/utils/plancache.h b/src/include/utils/plancache.h
index 7a4a85c8038..a0355e79c28 100644
--- a/src/include/utils/plancache.h
+++ b/src/include/utils/plancache.h
@@ -214,7 +214,7 @@ extern CachedPlanSource *CreateOneShotCachedPlan(RawStmt
*raw_parse_tree,
extern void CompleteCachedPlan(CachedPlanSource *plansource,
List *querytree_list,
MemoryContext
querytree_context,
- Oid *param_types,
+ const Oid
*param_types,
int num_params,
ParserSetupHook
parserSetup,
void *parserSetupArg,
--
2.54.0