Save/restore SPI's global variables in SPI_connect() and SPI_finish().
authorTom Lane
Sat, 8 Sep 2018 00:09:57 +0000 (20:09 -0400)
committerTom Lane
Sat, 8 Sep 2018 00:09:57 +0000 (20:09 -0400)
This patch removes two sources of interference between nominally
independent functions when one SPI-using function calls another,
perhaps without knowing that it does so.

Chapman Flack pointed out that xml.c's query_to_xml_internal() expects
SPI_tuptable and SPI_processed to stay valid across datatype output
function calls; but it's possible that such a call could involve
re-entrant use of SPI.  It seems likely that there are similar hazards
elsewhere, if not in the core code then in third-party SPI users.
Previously SPI_finish() reset SPI's API globals to zeroes/nulls, which
would typically make for a crash in such a situation.  Restoring them
to the values they had at SPI_connect() seems like a considerably more
useful behavior, and it still meets the design goal of not leaving any
dangling pointers to tuple tables of the function being exited.

Also, cause SPI_connect() to reset these variables to zeroes/nulls after
saving them.  This prevents interference in the opposite direction: it's
possible that a SPI-using function that's only ever been tested standalone
contains assumptions that these variables start out as zeroes.  That was
the case as long as you were the outermost SPI user, but not so much for
an inner user.  Now it's consistent.

Report and fix suggestion by Chapman Flack, actual patch by me.
Back-patch to all supported branches.

Discussion: https://postgr.es/m/9fa25bef-2e4f-1c32-22a4-3ad0723c4a17@anastigmatix.net

src/backend/executor/spi.c
src/include/executor/spi_priv.h

index ffa190aa83872a0151aedd5602d6b06df20967dc..99021891c15864cebd6d42a02033aa28a663c911 100644 (file)
 #include "utils/typcache.h"
 
 
+/*
+ * These global variables are part of the API for various SPI functions
+ * (a horrible API choice, but it's too late now).  To reduce the risk of
+ * interference between different SPI callers, we save and restore them
+ * when entering/exiting a SPI nesting level.
+ */
 uint64     SPI_processed = 0;
 Oid            SPI_lastoid = InvalidOid;
 SPITupleTable *SPI_tuptable = NULL;
-int            SPI_result;
+int            SPI_result = 0;
 
 static _SPI_connection *_SPI_stack = NULL;
 static _SPI_connection *_SPI_current = NULL;
@@ -132,6 +138,10 @@ SPI_connect(void)
    _SPI_current->procCxt = NULL;       /* in case we fail to create 'em */
    _SPI_current->execCxt = NULL;
    _SPI_current->connectSubid = GetCurrentSubTransactionId();
+   _SPI_current->outer_processed = SPI_processed;
+   _SPI_current->outer_lastoid = SPI_lastoid;
+   _SPI_current->outer_tuptable = SPI_tuptable;
+   _SPI_current->outer_result = SPI_result;
 
    /*
     * Create memory contexts for this procedure
@@ -150,6 +160,15 @@ SPI_connect(void)
    /* ... and switch to procedure's context */
    _SPI_current->savedcxt = MemoryContextSwitchTo(_SPI_current->procCxt);
 
+   /*
+    * Reset API global variables so that current caller cannot accidentally
+    * depend on state of an outer caller.
+    */
+   SPI_processed = 0;
+   SPI_lastoid = InvalidOid;
+   SPI_tuptable = NULL;
+   SPI_result = 0;
+
    return SPI_OK_CONNECT;
 }
 
@@ -172,12 +191,13 @@ SPI_finish(void)
    _SPI_current->procCxt = NULL;
 
    /*
-    * Reset result variables, especially SPI_tuptable which is probably
+    * Restore outer API variables, especially SPI_tuptable which is probably
     * pointing at a just-deleted tuptable
     */
-   SPI_processed = 0;
-   SPI_lastoid = InvalidOid;
-   SPI_tuptable = NULL;
+   SPI_processed = _SPI_current->outer_processed;
+   SPI_lastoid = _SPI_current->outer_lastoid;
+   SPI_tuptable = _SPI_current->outer_tuptable;
+   SPI_result = _SPI_current->outer_result;
 
    /*
     * After _SPI_begin_call _SPI_connected == _SPI_curid. Now we are closing
@@ -214,9 +234,11 @@ AtEOXact_SPI(bool isCommit)
    _SPI_current = _SPI_stack = NULL;
    _SPI_stack_depth = 0;
    _SPI_connected = _SPI_curid = -1;
+   /* Reset API global variables, too */
    SPI_processed = 0;
    SPI_lastoid = InvalidOid;
    SPI_tuptable = NULL;
+   SPI_result = 0;
 }
 
 /*
@@ -254,19 +276,21 @@ AtEOSubXact_SPI(bool isCommit, SubTransactionId mySubid)
        }
 
        /*
-        * Pop the stack entry and reset global variables.  Unlike
+        * Restore outer global variables and pop the stack entry.  Unlike
         * SPI_finish(), we don't risk switching to memory contexts that might
         * be already gone.
         */
+       SPI_processed = connection->outer_processed;
+       SPI_lastoid = connection->outer_lastoid;
+       SPI_tuptable = connection->outer_tuptable;
+       SPI_result = connection->outer_result;
+
        _SPI_connected--;
        _SPI_curid = _SPI_connected;
        if (_SPI_connected == -1)
            _SPI_current = NULL;
        else
            _SPI_current = &(_SPI_stack[_SPI_connected]);
-       SPI_processed = 0;
-       SPI_lastoid = InvalidOid;
-       SPI_tuptable = NULL;
    }
 
    if (found && isCommit)
index 5936d86f7071684f9b8e91b5d44e77908a2c6e27..a6db0c7ad95b797308ad4f52bfe0375e525132a6 100644 (file)
@@ -34,6 +34,12 @@ typedef struct
 
    /* subtransaction in which current Executor call was started */
    SubTransactionId execSubid;
+
+   /* saved values of API global variables for previous nesting level */
+   uint64      outer_processed;
+   Oid         outer_lastoid;
+   SPITupleTable *outer_tuptable;
+   int         outer_result;
 } _SPI_connection;
 
 /*