]> granicus.if.org Git - postgresql/commitdiff
Put back AcceptInvalidationMessages calls in heap_openrv(_extended).
authorTom Lane <tgl@sss.pgh.pa.us>
Wed, 19 Sep 2012 21:10:44 +0000 (17:10 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Wed, 19 Sep 2012 21:10:44 +0000 (17:10 -0400)
These calls were removed in commit 4240e429d0c2d889d0cda23c618f94e12c13ade7
as part of a general refactoring and improvement of DDL locking.  However,
there's a problem not solved by the rewrite, which is that GRANT/REVOKE
update pg_class.relacl without taking any particular lock on the target
table as such.  If another backend fails to do AcceptInvalidationMessages,
it won't notice a recently-committed change in ACLs.  Bug #7557 from Piotr
Czachur demonstrates that there's at least one code path in 9.2.0 in which
a command fails to do any AcceptInvalidationMessages calls at all, if the
current transaction already holds all the locks it will need.

Since we're hard up against the release deadline for 9.2.1, fix this by
putting back the AcceptInvalidationMessages calls in heap_openrv and
heap_openrv_extended, thereby restoring the historical behavior in this
area.  We ought to look for a more elegant and perhaps more bulletproof
solution, but there's no time for that right now.

src/backend/access/heap/heapam.c

index 9519e73e54c6c6315015a648ef0f958527315c80..3ecf5428a67f4de89871842662f95608068fad8a 100644 (file)
@@ -985,6 +985,20 @@ relation_openrv(const RangeVar *relation, LOCKMODE lockmode)
 {
        Oid                     relOid;
 
+       /*
+        * Check for shared-cache-inval messages before trying to open the
+        * relation.  This is needed even if we already hold a lock on the
+        * relation, because GRANT/REVOKE are executed without taking any lock on
+        * the target relation, and we want to be sure we see current ACL
+        * information.  We can skip this if asked for NoLock, on the assumption
+        * that such a call is not the first one in the current command, and so we
+        * should be reasonably up-to-date already.  (XXX this all could stand to
+        * be redesigned, but for the moment we'll keep doing this like it's been
+        * done historically.)
+        */
+       if (lockmode != NoLock)
+               AcceptInvalidationMessages();
+
        /* Look up and lock the appropriate relation using namespace search */
        relOid = RangeVarGetRelid(relation, lockmode, false);
 
@@ -1007,6 +1021,13 @@ relation_openrv_extended(const RangeVar *relation, LOCKMODE lockmode,
 {
        Oid                     relOid;
 
+       /*
+        * Check for shared-cache-inval messages before trying to open the
+        * relation.  See comments in relation_openrv().
+        */
+       if (lockmode != NoLock)
+               AcceptInvalidationMessages();
+
        /* Look up and lock the appropriate relation using namespace search */
        relOid = RangeVarGetRelid(relation, lockmode, missing_ok);