From 8bfb65df7bf2c24a12d0c38e5a91015b3f1066b1 Mon Sep 17 00:00:00 2001 From: Nathan Bossart Date: Wed, 3 Jun 2026 13:26:05 -0500 Subject: [PATCH v4 1/1] Alert client when prepared statements are deallocated. While trying to take our own advice and teach libpq's large object interface to use prepared statements instead of fast-path function calls, I discovered an interesting problem: a user could deallocate prepared statements (e.g., via DISCARD ALL) without libpq knowing, causing the large object functions to subsequently fail. Of the various options to deal with this problem, the cleanest one seems to be to solve the general problem and notify clients when a prepared statement is deallocated. This allows different layers of the client stack to notice when their statements are deallocated and to preemptively re-prepare them before reuse. This commit adds this behavior via a new protocol extension that libpq enables by default, but that can be disabled via a connection parameter. libpq offers the ability to register callback functions that are executed immediately upon receiving a deallocation notification, as well as a function that returns whether the server is configured to send these notifications. This is primarily intended for use in a follow-up commit that teaches the libpq large object interface to use prepared statements, but the new message type and libpq interface may be useful elsewhere. Reviewed-by: Jacob Champion Reviewed-by: Zsolt Parragi Discussion: https://postgr.es/m/ahm_4eOKkkKJ3Gds%40nathan --- doc/src/sgml/libpq.sgml | 91 +++++++++++++++++++ doc/src/sgml/protocol.sgml | 62 ++++++++++++- src/backend/commands/prepare.c | 33 +++++++ src/backend/tcop/backend_startup.c | 13 ++- src/bin/pg_upgrade/dump.c | 6 +- src/bin/pg_upgrade/server.c | 3 + src/bin/pg_upgrade/task.c | 3 + src/bin/pg_upgrade/version.c | 3 +- src/include/libpq/libpq-be.h | 1 + src/include/libpq/protocol.h | 1 + src/interfaces/libpq/exports.txt | 2 + src/interfaces/libpq/fe-connect.c | 63 +++++++++++++ src/interfaces/libpq/fe-protocol3.c | 56 ++++++++++++ src/interfaces/libpq/fe-trace.c | 10 ++ src/interfaces/libpq/libpq-fe.h | 9 ++ src/interfaces/libpq/libpq-int.h | 6 ++ .../libpq_pipeline/traces/prepared.trace | 1 + 17 files changed, 354 insertions(+), 9 deletions(-) diff --git a/doc/src/sgml/libpq.sgml b/doc/src/sgml/libpq.sgml index 7d3c3bb66d8..f76b31264f5 100644 --- a/doc/src/sgml/libpq.sgml +++ b/doc/src/sgml/libpq.sgml @@ -2297,6 +2297,20 @@ postgresql://%2Fvar%2Flib%2Fpostgresql/dbname + + report_prep_stmt_dealloc + + + If set to 1 (the default), + libpq requests the + _pq_.report_prep_stmt_dealloc + protocol extension, so that the server reports when prepared statements + are deallocated. Set it to 0 to suppress the + request. + + + + scram_client_key @@ -7253,6 +7267,83 @@ typedef struct pgNotify + + Prepared Statement Deallocation Notifications + + + prepared statement deallocation + in libpq + + + + The server can notify the client whenever a named prepared statement is + deallocated by + DEALLOCATE, + DISCARD, + , or + . This is useful when multiple + layers of client code share a connection and one drops a statement another + prepared. This behavior is negotiated through the + _pq_.report_prep_stmt_dealloc + protocol extension, which is only available on servers running + PostgreSQL 20 and later. + By default, libpq requests this protocol + extension; this can be disabled with the + connection + parameter. + + + + Notifications are delivered to callbacks registered with + PQaddPrepStmtDeallocCallback. + + + + PQaddPrepStmtDeallocCallbackPQaddPrepStmtDeallocCallback + + + Registers a callback to be invoked immediately upon receiving a prepared + statement deallocation notification. The value of + arg is passed unaltered to the callback. The + name argument will contain the name of the + deallocated prepared statement, or an empty string if all were + deallocated. Callbacks run while libpq + processes incoming data, so they must not call any + libpq functions on the same + conn, and they must not assume that + name survives after returning (copy it if it is + needed later). Returns 1 on success or + 0 if the callback could not be registered (e.g., due + to running out of memory). + + +typedef void (*PQprepStmtDeallocCallback) (PGconn *conn, void *arg, const char *name); + +int PQaddPrepStmtDeallocCallback(PGconn *conn, PQprepStmtDeallocCallback cb, void *arg); + + + + + + + PQprepStmtDeallocReportingPQprepStmtDeallocReporting + + + Returns 1 if the server accepted the + _pq_.report_prep_stmt_dealloc protocol extension. + Otherwise, returns 0 to indicate that no prepared + statement deallocation notifications will be sent. + + +int PQprepStmtDeallocReporting(PGconn *conn); + + + + + + + + Functions Associated with the <command>COPY</command> Command diff --git a/doc/src/sgml/protocol.sgml b/doc/src/sgml/protocol.sgml index 49f81676712..70ecc451efd 100644 --- a/doc/src/sgml/protocol.sgml +++ b/doc/src/sgml/protocol.sgml @@ -346,8 +346,15 @@ - - (No supported protocol extensions are currently defined.) + _pq_.report_prep_stmt_dealloc + none + PostgreSQL 20 and later + When negotiated, the server sends a + PrepStmtDealloc + message whenever a named prepared statement is deallocated by + DEALLOCATE, + DISCARD, or a + Close message. @@ -1587,6 +1594,20 @@ SELCT 1/0; point in the protocol. + + + If the client requested the + _pq_.report_prep_stmt_dealloc + protocol extension, the backend sends a PrepStmtDealloc message whenever a + named prepared statement is deallocated by + DEALLOCATE, + DISCARD, or a + Close message. This + alerts the client that a statement it prepared is gone (e.g., if another + layer of the client stack dropped it) and that it must be re-prepared + before reuse. The message carries the deallocated statement's name, or an + empty string to mean that all prepared statements were deallocated. + @@ -5873,6 +5894,43 @@ psql "dbname=postgres replication=database" -c "IDENTIFY_SYSTEM;" + + PrepStmtDealloc (B) + + + + Byte1('i') + + + Identifies the message as a prepared statement deallocation + notification. This is sent only if the client requested the + _pq_.report_prep_stmt_dealloc protocol extension. + + + + + + Int32 + + + Length of message contents in bytes, including self. + + + + + + String + + + The name of the deallocated prepared statement. An empty string + indicates that all prepared statements were deallocated. + + + + + + + Query (F) diff --git a/src/backend/commands/prepare.c b/src/backend/commands/prepare.c index 876aad2100a..4e7af002d6d 100644 --- a/src/backend/commands/prepare.c +++ b/src/backend/commands/prepare.c @@ -26,6 +26,9 @@ #include "commands/explain_state.h" #include "commands/prepare.h" #include "funcapi.h" +#include "libpq/libpq.h" +#include "libpq/pqformat.h" +#include "miscadmin.h" #include "nodes/nodeFuncs.h" #include "parser/parse_coerce.h" #include "parser/parse_collate.h" @@ -512,6 +515,26 @@ DeallocateQuery(DeallocateStmt *stmt) DropAllPreparedStatements(); } +/* + * Tell the client that a prepared statement has been deallocated (an empty + * string means all of them). Only sent to clients that requested the + * _pq_.report_prep_stmt_dealloc protocol extension. + */ +static void +SendStmtDeallocMsg(const char *name) +{ + StringInfoData buf; + + if (whereToSendOutput != DestRemote) + return; + if (!MyProcPort || !MyProcPort->report_prep_stmt_dealloc) + return; + + pq_beginmessage(&buf, PqMsg_PrepStmtDealloc); + pq_sendstring(&buf, name); + pq_endmessage(&buf); +} + /* * Internal version of DEALLOCATE * @@ -532,6 +555,9 @@ DropPreparedStatement(const char *stmt_name, bool showError) /* Now we can remove the hash table entry */ hash_search(prepared_queries, entry->stmt_name, HASH_REMOVE, NULL); + + /* Alert the client */ + SendStmtDeallocMsg(stmt_name); } } @@ -543,6 +569,7 @@ DropAllPreparedStatements(void) { HASH_SEQ_STATUS seq; PreparedStatement *entry; + bool found = false; /* nothing cached */ if (!prepared_queries) @@ -552,12 +579,18 @@ DropAllPreparedStatements(void) hash_seq_init(&seq, prepared_queries); while ((entry = hash_seq_search(&seq)) != NULL) { + found = true; + /* Release the plancache entry */ DropCachedPlan(entry->plansource); /* Now we can remove the hash table entry */ hash_search(prepared_queries, entry->stmt_name, HASH_REMOVE, NULL); } + + /* Alert the client */ + if (found) + SendStmtDeallocMsg(""); } /* diff --git a/src/backend/tcop/backend_startup.c b/src/backend/tcop/backend_startup.c index 25205cee0fa..a9158e86ea1 100644 --- a/src/backend/tcop/backend_startup.c +++ b/src/backend/tcop/backend_startup.c @@ -806,12 +806,15 @@ retry: else if (strncmp(nameptr, "_pq_.", 5) == 0) { /* - * Any option beginning with _pq_. is reserved for use as a - * protocol-level option, but at present no such options are - * defined. + * Options beginning with _pq_. are protocol extensions. + * Recognized ones are handled here; report the rest as + * unsupported. */ - unrecognized_protocol_options = - lappend(unrecognized_protocol_options, pstrdup(nameptr)); + if (strcmp(nameptr, "_pq_.report_prep_stmt_dealloc") == 0) + port->report_prep_stmt_dealloc = true; + else + unrecognized_protocol_options = + lappend(unrecognized_protocol_options, pstrdup(nameptr)); } else { diff --git a/src/bin/pg_upgrade/dump.c b/src/bin/pg_upgrade/dump.c index f47c8d06211..8d542903cb6 100644 --- a/src/bin/pg_upgrade/dump.c +++ b/src/bin/pg_upgrade/dump.c @@ -24,7 +24,8 @@ generate_old_dump(void) "\"%s/pg_dumpall\" %s%s --globals-only --quote-all-identifiers " "--binary-upgrade %s --no-sync -f \"%s/%s\"", new_cluster.bindir, cluster_conn_opts(&old_cluster), - protocol_negotiation_supported(&old_cluster) ? "" : " -d \"max_protocol_version=3.0\"", + protocol_negotiation_supported(&old_cluster) ? + "" : " -d \"max_protocol_version=3.0 report_prep_stmt_dealloc=0\"", log_opts.verbose ? "--verbose" : "", log_opts.dumpdir, GLOBALS_DUMP_FILE); @@ -45,7 +46,10 @@ generate_old_dump(void) appendPQExpBufferStr(&connstr, "dbname="); appendConnStrVal(&connstr, old_db->db_name); if (!protocol_negotiation_supported(&old_cluster)) + { appendPQExpBufferStr(&connstr, " max_protocol_version=3.0"); + appendPQExpBufferStr(&connstr, " report_prep_stmt_dealloc=0"); + } initPQExpBuffer(&escaped_connstr); appendShellString(&escaped_connstr, connstr.data); diff --git a/src/bin/pg_upgrade/server.c b/src/bin/pg_upgrade/server.c index 5d81e4e95b8..841802b2b14 100644 --- a/src/bin/pg_upgrade/server.c +++ b/src/bin/pg_upgrade/server.c @@ -72,7 +72,10 @@ get_db_conn(ClusterInfo *cluster, const char *db_name) appendConnStrVal(&conn_opts, cluster->sockdir); } if (!protocol_negotiation_supported(cluster)) + { appendPQExpBufferStr(&conn_opts, " max_protocol_version=3.0"); + appendPQExpBufferStr(&conn_opts, " report_prep_stmt_dealloc=0"); + } conn = PQconnectdb(conn_opts.data); termPQExpBuffer(&conn_opts); diff --git a/src/bin/pg_upgrade/task.c b/src/bin/pg_upgrade/task.c index b6eb29e1f3a..4c64bbabdcf 100644 --- a/src/bin/pg_upgrade/task.c +++ b/src/bin/pg_upgrade/task.c @@ -189,7 +189,10 @@ start_conn(const ClusterInfo *cluster, UpgradeTaskSlot *slot) appendConnStrVal(&conn_opts, cluster->sockdir); } if (!protocol_negotiation_supported(cluster)) + { appendPQExpBufferStr(&conn_opts, " max_protocol_version=3.0"); + appendPQExpBufferStr(&conn_opts, " report_prep_stmt_dealloc=0"); + } slot->conn = PQconnectStart(conn_opts.data); diff --git a/src/bin/pg_upgrade/version.c b/src/bin/pg_upgrade/version.c index 047670d4acb..9a86b9a8a55 100644 --- a/src/bin/pg_upgrade/version.c +++ b/src/bin/pg_upgrade/version.c @@ -30,7 +30,8 @@ jsonb_9_4_check_applicable(ClusterInfo *cluster) /* * Older servers can't support newer protocol versions, so their connection - * strings will need to lock max_protocol_version to 3.0. + * strings will need to lock max_protocol_version to 3.0 and disable prepared + * statement deallocation reports. */ bool protocol_negotiation_supported(const ClusterInfo *cluster) diff --git a/src/include/libpq/libpq-be.h b/src/include/libpq/libpq-be.h index 921b2daa4ff..1fbf08fc420 100644 --- a/src/include/libpq/libpq-be.h +++ b/src/include/libpq/libpq-be.h @@ -151,6 +151,7 @@ typedef struct Port char *user_name; char *cmdline_options; List *guc_options; + bool report_prep_stmt_dealloc; /* * The startup packet application name, only used here for the "connection diff --git a/src/include/libpq/protocol.h b/src/include/libpq/protocol.h index eae8f0e7238..aee0675594d 100644 --- a/src/include/libpq/protocol.h +++ b/src/include/libpq/protocol.h @@ -53,6 +53,7 @@ #define PqMsg_FunctionCallResponse 'V' #define PqMsg_CopyBothResponse 'W' #define PqMsg_ReadyForQuery 'Z' +#define PqMsg_PrepStmtDealloc 'i' #define PqMsg_NoData 'n' #define PqMsg_PortalSuspended 's' #define PqMsg_ParameterDescription 't' diff --git a/src/interfaces/libpq/exports.txt b/src/interfaces/libpq/exports.txt index 1e3d5bd5867..9413412e65d 100644 --- a/src/interfaces/libpq/exports.txt +++ b/src/interfaces/libpq/exports.txt @@ -211,3 +211,5 @@ PQdefaultAuthDataHook 208 PQfullProtocolVersion 209 appendPQExpBufferVA 210 PQgetThreadLock 211 +PQaddPrepStmtDeallocCallback 212 +PQprepStmtDeallocReporting 213 diff --git a/src/interfaces/libpq/fe-connect.c b/src/interfaces/libpq/fe-connect.c index 4272d386e64..1f0219d09a3 100644 --- a/src/interfaces/libpq/fe-connect.c +++ b/src/interfaces/libpq/fe-connect.c @@ -421,6 +421,10 @@ static const internalPQconninfoOption PQconninfoOptions[] = { "SSL-Key-Log-File", "D", 64, offsetof(struct pg_conn, sslkeylogfile)}, + {"report_prep_stmt_dealloc", NULL, "1", NULL, + "Report-Prepared-Statement-Deallocations", "", 1, + offsetof(struct pg_conn, report_prep_stmt_dealloc)}, + /* Terminating entry --- MUST BE LAST */ {NULL, NULL, NULL, NULL, NULL, NULL, 0} @@ -708,6 +712,7 @@ pqDropServerData(PGconn *conn) free(conn->write_err_msg); conn->write_err_msg = NULL; conn->oauth_want_retry = false; + conn->prepStmtDeallocReporting = false; /* * Cancel connections need to retain their be_pid and be_cancel_key across @@ -5178,6 +5183,11 @@ freePGconn(PGconn *conn) free(conn->rowBuf); termPQExpBuffer(&conn->errorMessage); termPQExpBuffer(&conn->workBuffer); + free(conn->report_prep_stmt_dealloc); + if (conn->prepStmtDeallocCallbacks) + free(conn->prepStmtDeallocCallbacks); + if (conn->prepStmtDeallocCallbackArgs) + free(conn->prepStmtDeallocCallbackArgs); free(conn); } @@ -8426,3 +8436,56 @@ PQgetThreadLock(void) Assert(pg_g_threadlock); return pg_g_threadlock; } + +/* + * Registers a callback to be invoked whenever the server reports a prepared + * statement deallocation. arg is passed through to the callback unaltered. + */ +int +PQaddPrepStmtDeallocCallback(PGconn *conn, PQprepStmtDeallocCallback cb, + void *arg) +{ + int i; + PQprepStmtDeallocCallback *new_cbs; + void **new_args; + + if (!conn) + return 0; + + i = conn->nPrepStmtDeallocCallbacks; + + new_cbs = realloc(conn->prepStmtDeallocCallbacks, + (i + 1) * sizeof(PQprepStmtDeallocCallback)); + if (!new_cbs) + { + libpq_append_conn_error(conn, "out of memory"); + return 0; + } + conn->prepStmtDeallocCallbacks = new_cbs; + + new_args = realloc(conn->prepStmtDeallocCallbackArgs, + (i + 1) * sizeof(void *)); + if (!new_args) + { + libpq_append_conn_error(conn, "out of memory"); + return 0; + } + conn->prepStmtDeallocCallbackArgs = new_args; + + new_cbs[i] = cb; + new_args[i] = arg; + conn->nPrepStmtDeallocCallbacks = i + 1; + + return 1; +} + +/* + * Returns true if the server accepted the _pq_.report_prep_stmt_dealloc + * extension. If false, no notifications will arrive and the caller must + * re-prepare on error. + */ +int +PQprepStmtDeallocReporting(PGconn *conn) +{ + return conn && conn->prepStmtDeallocReporting; +} diff --git a/src/interfaces/libpq/fe-protocol3.c b/src/interfaces/libpq/fe-protocol3.c index 78ffb1025d0..f5cc361fb73 100644 --- a/src/interfaces/libpq/fe-protocol3.c +++ b/src/interfaces/libpq/fe-protocol3.c @@ -61,6 +61,30 @@ static size_t build_startup_packet(const PGconn *conn, char *packet, const PQEnvironmentOption *options); +/* + * Read a PrepStmtDealloc message and invoke the registered callbacks. Broken + * out as a subroutine since it can occur in several places. + * + * Entry: 'i' message type and length already consumed. + * Exit: 0 on success, EOF if not enough data. + */ +static int +getPrepStmtDealloc(PGconn *conn) +{ + if (pqGets(&conn->workBuffer, conn)) + return EOF; + + for (int i = 0; i < conn->nPrepStmtDeallocCallbacks; i++) + { + PQprepStmtDeallocCallback cb = conn->prepStmtDeallocCallbacks[i]; + void *arg = conn->prepStmtDeallocCallbackArgs[i]; + + cb(conn, arg, conn->workBuffer.data); + } + + return 0; +} + /* * parseInput: if appropriate, parse input data from backend * until input is exhausted or a stopping state is reached. @@ -184,6 +208,11 @@ pqParseInput3(PGconn *conn) if (getParameterStatus(conn)) return; } + else if (id == PqMsg_PrepStmtDealloc) + { + if (getPrepStmtDealloc(conn)) + return; + } else { /* Any other case is unexpected and we summarily skip it */ @@ -305,6 +334,10 @@ pqParseInput3(PGconn *conn) if (getParameterStatus(conn)) return; break; + case PqMsg_PrepStmtDealloc: + if (getPrepStmtDealloc(conn)) + return; + break; case PqMsg_BackendKeyData: /* @@ -1545,6 +1578,8 @@ pqGetNegotiateProtocolVersion3(PGconn *conn) { found_test_protocol_negotiation = true; } + else if (strcmp(conn->workBuffer.data, "_pq_.report_prep_stmt_dealloc") == 0) + conn->prepStmtDeallocReporting = false; else { libpq_append_conn_error(conn, "received invalid protocol negotiation message: server reported an unsupported parameter that was not requested (\"%s\")", @@ -1906,6 +1941,10 @@ getCopyDataMessage(PGconn *conn) if (getParameterStatus(conn)) return 0; break; + case PqMsg_PrepStmtDealloc: + if (getPrepStmtDealloc(conn)) + return 0; + break; case PqMsg_CopyData: return msgLength; case PqMsg_CopyDone: @@ -2410,6 +2449,10 @@ pqFunctionCall3(PGconn *conn, Oid fnid, if (getParameterStatus(conn)) continue; break; + case PqMsg_PrepStmtDealloc: + if (getPrepStmtDealloc(conn)) + continue; + break; default: /* The backend violates the protocol. */ libpq_append_conn_error(conn, "protocol error: id=0x%x", id); @@ -2451,6 +2494,15 @@ pqBuildStartupPacket3(PGconn *conn, int *packetlen, char *startpacket; size_t len; + /* + * Initialize prepStmtDeallocReporting based on whether the client + * requested the protocol extension. pqGetNegotiateProtocolVersion3() + * clears this if the server rejects it. + */ + if (conn->report_prep_stmt_dealloc && + strcmp(conn->report_prep_stmt_dealloc, "1") == 0) + conn->prepStmtDeallocReporting = true; + len = build_startup_packet(conn, NULL, options); if (len == 0 || len > INT_MAX) return NULL; @@ -2525,6 +2577,10 @@ build_startup_packet(const PGconn *conn, char *packet, if (conn->client_encoding_initial && conn->client_encoding_initial[0]) ADD_STARTUP_OPTION("client_encoding", conn->client_encoding_initial); + /* Ask the server to report prepared statement deallocations. */ + if (conn->prepStmtDeallocReporting) + ADD_STARTUP_OPTION("_pq_.report_prep_stmt_dealloc", ""); + /* * Add the test_protocol_negotiation option when greasing, to test that * servers properly report unsupported protocol options in addition to diff --git a/src/interfaces/libpq/fe-trace.c b/src/interfaces/libpq/fe-trace.c index c348b08c39b..401e83d95e8 100644 --- a/src/interfaces/libpq/fe-trace.c +++ b/src/interfaces/libpq/fe-trace.c @@ -543,6 +543,13 @@ pqTraceOutput_ParameterStatus(FILE *f, const char *message, int *cursor) pqTraceOutputString(f, message, cursor, false); } +static void +pqTraceOutput_PrepStmtDealloc(FILE *f, const char *message, int *cursor) +{ + fprintf(f, "PrepStmtDealloc\t"); + pqTraceOutputString(f, message, cursor, false); +} + static void pqTraceOutput_ParameterDescription(FILE *f, const char *message, int *cursor, bool regress) { @@ -793,6 +800,9 @@ pqTraceOutputMessage(PGconn *conn, const char *message, bool toServer) else pqTraceOutput_ParameterStatus(conn->Pfdebug, message, &logCursor); break; + case PqMsg_PrepStmtDealloc: + pqTraceOutput_PrepStmtDealloc(conn->Pfdebug, message, &logCursor); + break; case PqMsg_ParameterDescription: pqTraceOutput_ParameterDescription(conn->Pfdebug, message, &logCursor, regress); break; diff --git a/src/interfaces/libpq/libpq-fe.h b/src/interfaces/libpq/libpq-fe.h index 8ecb9b4a4c7..6cec59c939c 100644 --- a/src/interfaces/libpq/libpq-fe.h +++ b/src/interfaces/libpq/libpq-fe.h @@ -486,6 +486,15 @@ typedef void (*pgthreadlock_t) (int acquire); extern pgthreadlock_t PQregisterThreadLock(pgthreadlock_t newhandler); extern pgthreadlock_t PQgetThreadLock(void); +/* callbacks for prepared statement deallocation notifications */ +typedef void (*PQprepStmtDeallocCallback) (PGconn *conn, void *arg, + const char *name); + +extern int PQaddPrepStmtDeallocCallback(PGconn *conn, + PQprepStmtDeallocCallback cb, + void *arg); +extern int PQprepStmtDeallocReporting(PGconn *conn); + /* === in fe-trace.c === */ extern void PQtrace(PGconn *conn, FILE *debug_port); extern void PQuntrace(PGconn *conn); diff --git a/src/interfaces/libpq/libpq-int.h b/src/interfaces/libpq/libpq-int.h index 461b39620c3..c4f264cf603 100644 --- a/src/interfaces/libpq/libpq-int.h +++ b/src/interfaces/libpq/libpq-int.h @@ -432,6 +432,7 @@ struct pg_conn char *scram_client_key; /* base64-encoded SCRAM client key */ char *scram_server_key; /* base64-encoded SCRAM server key */ char *sslkeylogfile; /* where should the client write ssl keylogs */ + char *report_prep_stmt_dealloc; /* request pstmt dealloc reports */ bool cancelRequest; /* true if this connection is used to send a * cancel request, instead of being a normal @@ -532,6 +533,11 @@ struct pg_conn void (*cleanup_async_auth) (PGconn *conn); pgsocket altsock; /* alternative socket for client to poll */ + /* prepared statement deallocation notifications */ + bool prepStmtDeallocReporting; + PQprepStmtDeallocCallback *prepStmtDeallocCallbacks; + void **prepStmtDeallocCallbackArgs; + int nPrepStmtDeallocCallbacks; /* Transient state needed while establishing connection */ PGTargetServerType target_server_type; /* desired session properties */ diff --git a/src/test/modules/libpq_pipeline/traces/prepared.trace b/src/test/modules/libpq_pipeline/traces/prepared.trace index aeb5de109e0..ef383f20697 100644 --- a/src/test/modules/libpq_pipeline/traces/prepared.trace +++ b/src/test/modules/libpq_pipeline/traces/prepared.trace @@ -7,6 +7,7 @@ B 113 RowDescription 4 "?column?" NNNN 0 NNNN 4 -1 0 "?column?" NNNN 0 NNNN 655 B 5 ReadyForQuery I F 16 Close S "select_one" F 4 Sync +B 15 PrepStmtDealloc "select_one" B 4 CloseComplete B 5 ReadyForQuery I F 16 Describe S "select_one" -- 2.50.1 (Apple Git-155)