]> granicus.if.org Git - postgresql/commitdiff
Preserve caller's memory context in ProcessCompletedNotifies().
authorTom Lane <tgl@sss.pgh.pa.us>
Fri, 27 May 2011 16:10:32 +0000 (12:10 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Fri, 27 May 2011 16:10:32 +0000 (12:10 -0400)
This is necessary to avoid long-term memory leakage, because the main loop
in PostgresMain expects to be executing in MessageContext, and hence is a
bit sloppy about freeing stuff that is only needed for the duration of
processing the current client message.  The known case of an actual leak
is when encoding conversion has to be done on the incoming command string,
but there might be others.  Per report from Per-Olov Esgard.

Back-patch to 9.0, where the bug was introduced by the LISTEN/NOTIFY
rewrite.

src/backend/commands/async.c

index 588e9f2b85da443c7fb290f8548ac3960fd8923f..02f8f9cd635d50db5c9e3f18b11adbaac6331fd6 100644 (file)
@@ -1090,6 +1090,7 @@ Exec_UnlistenAllCommit(void)
 void
 ProcessCompletedNotifies(void)
 {
+       MemoryContext caller_context;
        bool            signalled;
 
        /* Nothing to do if we didn't send any notifications */
@@ -1103,6 +1104,12 @@ ProcessCompletedNotifies(void)
         */
        backendHasSentNotifications = false;
 
+       /*
+        * We must preserve the caller's memory context (probably MessageContext)
+        * across the transaction we do here.
+        */
+       caller_context = CurrentMemoryContext;
+
        if (Trace_notify)
                elog(DEBUG1, "ProcessCompletedNotifies");
 
@@ -1135,6 +1142,8 @@ ProcessCompletedNotifies(void)
 
        CommitTransactionCommand();
 
+       MemoryContextSwitchTo(caller_context);
+
        /* We don't need pq_flush() here since postgres.c will do one shortly */
 }