]> granicus.if.org Git - postgresql/commitdiff
Don't lock tables in RelationGetPartitionDispatchInfo.
authorRobert Haas <rhaas@postgresql.org>
Thu, 17 Aug 2017 19:39:17 +0000 (15:39 -0400)
committerRobert Haas <rhaas@postgresql.org>
Thu, 17 Aug 2017 19:47:29 +0000 (15:47 -0400)
Instead, lock them in the caller using find_all_inheritors so that
they get locked in the standard order, minimizing deadlock risks.

Also in RelationGetPartitionDispatchInfo, avoid opening tables which
are not partitioned; there's no need.

Amit Langote, reviewed by Ashutosh Bapat and Amit Khandekar

Discussion: http://postgr.es/m/91b36fa1-c197-b72f-ca6e-56c593bae68c@lab.ntt.co.jp

src/backend/catalog/partition.c
src/backend/executor/execMain.c
src/include/catalog/partition.h

index 71bc4b3d105c443bf646b80c64e0550f8a43eee6..21901380cb250f09553220c8d59ff2760480dd7c 100644 (file)
@@ -1000,12 +1000,16 @@ get_partition_qual_relid(Oid relid)
  * RelationGetPartitionDispatchInfo
  *             Returns information necessary to route tuples down a partition tree
  *
- * All the partitions will be locked with lockmode, unless it is NoLock.
- * A list of the OIDs of all the leaf partitions of rel is returned in
- * *leaf_part_oids.
+ * The number of elements in the returned array (that is, the number of
+ * PartitionDispatch objects for the partitioned tables in the partition tree)
+ * is returned in *num_parted and a list of the OIDs of all the leaf
+ * partitions of rel is returned in *leaf_part_oids.
+ *
+ * All the relations in the partition tree (including 'rel') must have been
+ * locked (using at least the AccessShareLock) by the caller.
  */
 PartitionDispatch *
-RelationGetPartitionDispatchInfo(Relation rel, int lockmode,
+RelationGetPartitionDispatchInfo(Relation rel,
                                                                 int *num_parted, List **leaf_part_oids)
 {
        PartitionDispatchData **pd;
@@ -1020,14 +1024,18 @@ RelationGetPartitionDispatchInfo(Relation rel, int lockmode,
                                offset;
 
        /*
-        * Lock partitions and make a list of the partitioned ones to prepare
-        * their PartitionDispatch objects below.
+        * We rely on the relcache to traverse the partition tree to build both
+        * the leaf partition OIDs list and the array of PartitionDispatch objects
+        * for the partitioned tables in the tree.  That means every partitioned
+        * table in the tree must be locked, which is fine since we require the
+        * caller to lock all the partitions anyway.
         *
-        * Cannot use find_all_inheritors() here, because then the order of OIDs
-        * in parted_rels list would be unknown, which does not help, because we
-        * assign indexes within individual PartitionDispatch in an order that is
-        * predetermined (determined by the order of OIDs in individual partition
-        * descriptors).
+        * For every partitioned table in the tree, starting with the root
+        * partitioned table, add its relcache entry to parted_rels, while also
+        * queuing its partitions (in the order in which they appear in the
+        * partition descriptor) to be looked at later in the same loop.  This is
+        * a bit tricky but works because the foreach() macro doesn't fetch the
+        * next list element until the bottom of the loop.
         */
        *num_parted = 1;
        parted_rels = list_make1(rel);
@@ -1036,29 +1044,24 @@ RelationGetPartitionDispatchInfo(Relation rel, int lockmode,
        APPEND_REL_PARTITION_OIDS(rel, all_parts, all_parents);
        forboth(lc1, all_parts, lc2, all_parents)
        {
-               Relation        partrel = heap_open(lfirst_oid(lc1), lockmode);
+               Oid                     partrelid = lfirst_oid(lc1);
                Relation        parent = lfirst(lc2);
-               PartitionDesc partdesc = RelationGetPartitionDesc(partrel);
 
-               /*
-                * If this partition is a partitioned table, add its children to the
-                * end of the list, so that they are processed as well.
-                */
-               if (partdesc)
+               if (get_rel_relkind(partrelid) == RELKIND_PARTITIONED_TABLE)
                {
+                       /*
+                        * Already locked by the caller.  Note that it is the
+                        * responsibility of the caller to close the below relcache entry,
+                        * once done using the information being collected here (for
+                        * example, in ExecEndModifyTable).
+                        */
+                       Relation        partrel = heap_open(partrelid, NoLock);
+
                        (*num_parted)++;
                        parted_rels = lappend(parted_rels, partrel);
                        parted_rel_parents = lappend(parted_rel_parents, parent);
                        APPEND_REL_PARTITION_OIDS(partrel, all_parts, all_parents);
                }
-               else
-                       heap_close(partrel, NoLock);
-
-               /*
-                * We keep the partitioned ones open until we're done using the
-                * information being collected here (for example, see
-                * ExecEndModifyTable).
-                */
        }
 
        /*
index 6671a25ffb367e299d528ba3029843e4080f676e..74071eede6e30adf61402c477fe14894883fb6b1 100644 (file)
@@ -43,6 +43,7 @@
 #include "access/xact.h"
 #include "catalog/namespace.h"
 #include "catalog/partition.h"
+#include "catalog/pg_inherits_fn.h"
 #include "catalog/pg_publication.h"
 #include "commands/matview.h"
 #include "commands/trigger.h"
@@ -3249,9 +3250,12 @@ ExecSetupPartitionTupleRouting(Relation rel,
        int                     i;
        ResultRelInfo *leaf_part_rri;
 
-       /* Get the tuple-routing information and lock partitions */
-       *pd = RelationGetPartitionDispatchInfo(rel, RowExclusiveLock, num_parted,
-                                                                                  &leaf_parts);
+       /*
+        * Get the information about the partition tree after locking all the
+        * partitions.
+        */
+       (void) find_all_inheritors(RelationGetRelid(rel), RowExclusiveLock, NULL);
+       *pd = RelationGetPartitionDispatchInfo(rel, num_parted, &leaf_parts);
        *num_partitions = list_length(leaf_parts);
        *partitions = (ResultRelInfo *) palloc(*num_partitions *
                                                                                   sizeof(ResultRelInfo));
index 434ded37d74f07cc2c90e4bd7c9d02138f6a1746..7e415fe8727b15d2bbc1ef34212f529c94425fb1 100644 (file)
@@ -87,8 +87,7 @@ extern Expr *get_partition_qual_relid(Oid relid);
 
 /* For tuple routing */
 extern PartitionDispatch *RelationGetPartitionDispatchInfo(Relation rel,
-                                                                int lockmode, int *num_parted,
-                                                                List **leaf_part_oids);
+                                                                int *num_parted, List **leaf_part_oids);
 extern void FormPartitionKeyDatum(PartitionDispatch pd,
                                          TupleTableSlot *slot,
                                          EState *estate,