]> granicus.if.org Git - postgresql/commitdiff
Fix issues with checks for unsupported transaction states in Hot Standby.
authorTom Lane <tgl@sss.pgh.pa.us>
Fri, 24 Aug 2012 17:09:04 +0000 (13:09 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Fri, 24 Aug 2012 17:09:04 +0000 (13:09 -0400)
The GUC check hooks for transaction_read_only and transaction_isolation
tried to check RecoveryInProgress(), so as to disallow setting read/write
mode or serializable isolation level (respectively) in hot standby
sessions.  However, GUC check hooks can be called in many situations where
we're not connected to shared memory at all, resulting in a crash in
RecoveryInProgress().  Among other cases, this results in EXEC_BACKEND
builds crashing during child process start if default_transaction_isolation
is serializable, as reported by Heikki Linnakangas.  Protect those calls
by silently allowing any setting when not inside a transaction; which is
okay anyway since these GUCs are always reset at start of transaction.

Also, add a check to GetSerializableTransactionSnapshot() to complain
if we are in hot standby.  We need that check despite the one in
check_XactIsoLevel() because default_transaction_isolation could be
serializable.  We don't want to complain any sooner than this in such
cases, since that would prevent running transactions at all in such a
state; but a transaction can be run, if SET TRANSACTION ISOLATION is done
before setting a snapshot.  Per report some months ago from Robert Haas.

Back-patch to 9.1, since these problems were introduced by the SSI patch.

Kevin Grittner and Tom Lane, with ideas from Heikki Linnakangas

src/backend/commands/variable.c
src/backend/storage/lmgr/predicate.c
src/backend/utils/init/postinit.c

index 0e9eb3a64f5db66dd2fca68252a597e68defdbf8..5d894ba77f74d02637008deedd4ca2919d8af6b8 100644 (file)
@@ -533,11 +533,16 @@ show_log_timezone(void)
  * read-only may be changed to read-write only when in a top-level transaction
  * that has not yet taken an initial snapshot. Can't do it in a hot standby
  * slave, either.
+ *
+ * If we are not in a transaction at all, just allow the change; it means
+ * nothing since XactReadOnly will be reset by the next StartTransaction().
+ * The IsTransactionState() test protects us against trying to check
+ * RecoveryInProgress() in contexts where shared memory is not accessible.
  */
 bool
 check_transaction_read_only(bool *newval, void **extra, GucSource source)
 {
-       if (*newval == false && XactReadOnly)
+       if (*newval == false && XactReadOnly && IsTransactionState())
        {
                /* Can't go to r/w mode inside a r/o transaction */
                if (IsSubTransaction())
@@ -556,6 +561,7 @@ check_transaction_read_only(bool *newval, void **extra, GucSource source)
                /* Can't go to r/w mode while recovery is still active */
                if (RecoveryInProgress())
                {
+                       GUC_check_errcode(ERRCODE_FEATURE_NOT_SUPPORTED);
                        GUC_check_errmsg("cannot set transaction read-write mode during recovery");
                        return false;
                }
@@ -569,6 +575,8 @@ check_transaction_read_only(bool *newval, void **extra, GucSource source)
  *
  * We allow idempotent changes at any time, but otherwise this can only be
  * changed in a toplevel transaction that has not yet taken a snapshot.
+ *
+ * As in check_transaction_read_only, allow it if not inside a transaction.
  */
 bool
 check_XactIsoLevel(char **newval, void **extra, GucSource source)
@@ -598,7 +606,7 @@ check_XactIsoLevel(char **newval, void **extra, GucSource source)
        else
                return false;
 
-       if (newXactIsoLevel != XactIsoLevel)
+       if (newXactIsoLevel != XactIsoLevel && IsTransactionState())
        {
                if (FirstSnapshotSet)
                {
@@ -616,6 +624,7 @@ check_XactIsoLevel(char **newval, void **extra, GucSource source)
                /* Can't go to serializable mode while recovery is still active */
                if (newXactIsoLevel == XACT_SERIALIZABLE && RecoveryInProgress())
                {
+                       GUC_check_errcode(ERRCODE_FEATURE_NOT_SUPPORTED);
                        GUC_check_errmsg("cannot use serializable mode in a hot standby");
                        GUC_check_errhint("You can use REPEATABLE READ instead.");
                        return false;
index b22faf9607d87f1a2bc7ce306e923dc12b8cfcc7..8b5a95ceaa0ff460d34231dadc2d3562fa7b0c63 100644 (file)
@@ -1571,6 +1571,19 @@ GetSerializableTransactionSnapshot(Snapshot snapshot)
 {
        Assert(IsolationIsSerializable());
 
+       /*
+        * Can't use serializable mode while recovery is still active, as it is,
+        * for example, on a hot standby.  We could get here despite the check
+        * in check_XactIsoLevel() if default_transaction_isolation is set to
+        * serializable, so phrase the hint accordingly.
+        */
+       if (RecoveryInProgress())
+               ereport(ERROR,
+                               (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
+                                errmsg("cannot use serializable mode in a hot standby"),
+                                errdetail("\"default_transaction_isolation\" is set to \"serializable\"."),
+                                errhint("You can use \"SET default_transaction_isolation = 'repeatable read'\" to change the default.")));
+
        /*
         * A special optimization is available for SERIALIZABLE READ ONLY
         * DEFERRABLE transactions -- we can wait for a suitable snapshot and
index 6a3fc6f693f66304a487032e2099d1660590d836..bef301a79c8fea1f602f5bd8b340bb4950d285fa 100644 (file)
@@ -584,6 +584,15 @@ InitPostgres(const char *in_dbname, Oid dboid, const char *username,
                /* statement_timestamp must be set for timeouts to work correctly */
                SetCurrentStatementStartTimestamp();
                StartTransactionCommand();
+
+               /*
+                * transaction_isolation will have been set to the default by the
+                * above.  If the default is "serializable", and we are in hot
+                * standby, we will fail if we don't change it to something lower.
+                * Fortunately, "read committed" is plenty good enough.
+                */
+               XactIsoLevel = XACT_READ_COMMITTED;
+
                (void) GetTransactionSnapshot();
        }