Fix unsafe order of operations in foreign-table DDL commands.
authorTom Lane
Sun, 14 Aug 2011 19:40:21 +0000 (15:40 -0400)
committerTom Lane
Sun, 14 Aug 2011 19:40:21 +0000 (15:40 -0400)
When updating or deleting a system catalog tuple, it's necessary to acquire
RowExclusiveLock on the catalog before looking up the tuple; otherwise a
concurrent VACUUM FULL on the catalog might move the tuple to a different
TID before we can apply the update.  Coding patterns that find the tuple
via a table scan aren't at risk here, but when obtaining the tuple from a
catalog cache, correct ordering is important; and several routines in
foreigncmds.c got it wrong.  Noted while running the regression tests in
parallel with VACUUM FULL of assorted system catalogs.

For consistency I moved all the heap_open calls to the starts of their
functions, including a couple for which there was no actual bug.

Back-patch to 8.4 where foreigncmds.c was added.

src/backend/commands/foreigncmds.c

index d16932ba6a025049f78235b13d528295f69671c5..3a3c131366829018913e8a0b7afa12661718c371 100644 (file)
@@ -215,6 +215,8 @@ AlterForeignDataWrapperOwner(const char *name, Oid newOwnerId)
    Oid         fdwId;
    Form_pg_foreign_data_wrapper form;
 
+   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
+
    /* Must be a superuser to change a FDW owner */
    if (!superuser())
        ereport(ERROR,
@@ -231,8 +233,6 @@ AlterForeignDataWrapperOwner(const char *name, Oid newOwnerId)
                        name),
        errhint("The owner of a foreign-data wrapper must be a superuser.")));
 
-   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
-
    tup = SearchSysCacheCopy1(FOREIGNDATAWRAPPERNAME, CStringGetDatum(name));
 
    if (!HeapTupleIsValid(tup))
@@ -432,6 +432,8 @@ CreateForeignDataWrapper(CreateFdwStmt *stmt)
    ObjectAddress myself;
    ObjectAddress referenced;
 
+   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
+
    /* Must be super user */
    if (!superuser())
        ereport(ERROR,
@@ -455,8 +457,6 @@ CreateForeignDataWrapper(CreateFdwStmt *stmt)
    /*
     * Insert tuple into pg_foreign_data_wrapper.
     */
-   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
-
    memset(values, 0, sizeof(values));
    memset(nulls, false, sizeof(nulls));
 
@@ -545,6 +545,8 @@ AlterForeignDataWrapper(AlterFdwStmt *stmt)
    Oid         fdwhandler;
    Oid         fdwvalidator;
 
+   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
+
    /* Must be super user */
    if (!superuser())
        ereport(ERROR,
@@ -635,9 +637,6 @@ AlterForeignDataWrapper(AlterFdwStmt *stmt)
    }
 
    /* Everything looks good - update the tuple */
-
-   rel = heap_open(ForeignDataWrapperRelationId, RowExclusiveLock);
-
    tp = heap_modify_tuple(tp, RelationGetDescr(rel),
                           repl_val, repl_null, repl_repl);
 
@@ -773,6 +772,8 @@ CreateForeignServer(CreateForeignServerStmt *stmt)
    ObjectAddress referenced;
    ForeignDataWrapper *fdw;
 
+   rel = heap_open(ForeignServerRelationId, RowExclusiveLock);
+
    /* For now the owner cannot be specified on create. Use effective user ID. */
    ownerId = GetUserId();
 
@@ -798,8 +799,6 @@ CreateForeignServer(CreateForeignServerStmt *stmt)
    /*
     * Insert tuple into pg_foreign_server.
     */
-   rel = heap_open(ForeignServerRelationId, RowExclusiveLock);
-
    memset(values, 0, sizeof(values));
    memset(nulls, false, sizeof(nulls));
 
@@ -880,6 +879,8 @@ AlterForeignServer(AlterForeignServerStmt *stmt)
    Oid         srvId;
    Form_pg_foreign_server srvForm;
 
+   rel = heap_open(ForeignServerRelationId, RowExclusiveLock);
+
    tp = SearchSysCacheCopy1(FOREIGNSERVERNAME,
                             CStringGetDatum(stmt->servername));
 
@@ -945,9 +946,6 @@ AlterForeignServer(AlterForeignServerStmt *stmt)
    }
 
    /* Everything looks good - update the tuple */
-
-   rel = heap_open(ForeignServerRelationId, RowExclusiveLock);
-
    tp = heap_modify_tuple(tp, RelationGetDescr(rel),
                           repl_val, repl_null, repl_repl);
 
@@ -1068,6 +1066,8 @@ CreateUserMapping(CreateUserMappingStmt *stmt)
    ForeignServer *srv;
    ForeignDataWrapper *fdw;
 
+   rel = heap_open(UserMappingRelationId, RowExclusiveLock);
+
    useId = GetUserOidFromMapping(stmt->username, false);
 
    /* Check that the server exists. */
@@ -1093,8 +1093,6 @@ CreateUserMapping(CreateUserMappingStmt *stmt)
    /*
     * Insert tuple into pg_user_mapping.
     */
-   rel = heap_open(UserMappingRelationId, RowExclusiveLock);
-
    memset(values, 0, sizeof(values));
    memset(nulls, false, sizeof(nulls));
 
@@ -1161,6 +1159,8 @@ AlterUserMapping(AlterUserMappingStmt *stmt)
    Oid         umId;
    ForeignServer *srv;
 
+   rel = heap_open(UserMappingRelationId, RowExclusiveLock);
+
    useId = GetUserOidFromMapping(stmt->username, false);
    srv = GetForeignServerByName(stmt->servername, false);
 
@@ -1218,9 +1218,6 @@ AlterUserMapping(AlterUserMappingStmt *stmt)
    }
 
    /* Everything looks good - update the tuple */
-
-   rel = heap_open(UserMappingRelationId, RowExclusiveLock);
-
    tp = heap_modify_tuple(tp, RelationGetDescr(rel),
                           repl_val, repl_null, repl_repl);
 
@@ -1344,11 +1341,13 @@ CreateForeignTable(CreateForeignTableStmt *stmt, Oid relid)
    ForeignServer *server;
 
    /*
-    * Advance command counter to ensure the pg_attribute tuple visible; the
-    * tuple might be updated to add constraints in previous step.
+    * Advance command counter to ensure the pg_attribute tuple is visible;
+    * the tuple might be updated to add constraints in previous step.
     */
    CommandCounterIncrement();
 
+   ftrel = heap_open(ForeignTableRelationId, RowExclusiveLock);
+
    /*
     * For now the owner cannot be specified on create. Use effective user ID.
     */
@@ -1368,8 +1367,6 @@ CreateForeignTable(CreateForeignTableStmt *stmt, Oid relid)
    /*
     * Insert tuple into pg_foreign_table.
     */
-   ftrel = heap_open(ForeignTableRelationId, RowExclusiveLock);
-
    memset(values, 0, sizeof(values));
    memset(nulls, false, sizeof(nulls));