Commit 2abfd9d5 authored by Tom Lane's avatar Tom Lane

Second try at fixing O(N^2) problem in foreign key references.

This replaces ill-fated commit 5ddc7288,
which was reverted because it broke active uses of FK cache entries.  In
this patch, we still do nothing more to invalidatable cache entries than
mark them as needing revalidation, so we won't break active uses.  To keep
down the overhead of InvalidateConstraintCacheCallBack(), keep a list of
just the currently-valid cache entries.  (The entries are large enough that
some added space for list links doesn't seem like a big problem.)  This
would still be O(N^2) when there are many valid entries, though, so when
the list gets too long, just force the "sinval reset" behavior to remove
everything from the list.  I set the threshold at 1000 entries, somewhat
arbitrarily.  Possibly that could be fine-tuned later.  Another item for
future study is whether it's worth adding reference counting so that we
could safely remove invalidated entries.  As-is, problem cases are likely
to end up with large and mostly invalid FK caches.

Like the previous attempt, backpatch to 9.3.

Jan Wieck and Tom Lane
parent 77130fc1
...@@ -40,6 +40,7 @@ ...@@ -40,6 +40,7 @@
#include "commands/trigger.h" #include "commands/trigger.h"
#include "executor/executor.h" #include "executor/executor.h"
#include "executor/spi.h" #include "executor/spi.h"
#include "lib/ilist.h"
#include "parser/parse_coerce.h" #include "parser/parse_coerce.h"
#include "parser/parse_relation.h" #include "parser/parse_relation.h"
#include "miscadmin.h" #include "miscadmin.h"
...@@ -125,6 +126,7 @@ typedef struct RI_ConstraintInfo ...@@ -125,6 +126,7 @@ typedef struct RI_ConstraintInfo
* PK) */ * PK) */
Oid ff_eq_oprs[RI_MAX_NUMKEYS]; /* equality operators (FK = Oid ff_eq_oprs[RI_MAX_NUMKEYS]; /* equality operators (FK =
* FK) */ * FK) */
dlist_node valid_link; /* Link in list of valid entries */
} RI_ConstraintInfo; } RI_ConstraintInfo;
...@@ -185,6 +187,8 @@ typedef struct RI_CompareHashEntry ...@@ -185,6 +187,8 @@ typedef struct RI_CompareHashEntry
static HTAB *ri_constraint_cache = NULL; static HTAB *ri_constraint_cache = NULL;
static HTAB *ri_query_cache = NULL; static HTAB *ri_query_cache = NULL;
static HTAB *ri_compare_cache = NULL; static HTAB *ri_compare_cache = NULL;
static dlist_head ri_constraint_cache_valid_list;
static int ri_constraint_cache_valid_count = 0;
/* ---------- /* ----------
...@@ -2924,6 +2928,13 @@ ri_LoadConstraintInfo(Oid constraintOid) ...@@ -2924,6 +2928,13 @@ ri_LoadConstraintInfo(Oid constraintOid)
ReleaseSysCache(tup); ReleaseSysCache(tup);
/*
* For efficient processing of invalidation messages below, we keep a
* doubly-linked list, and a count, of all currently valid entries.
*/
dlist_push_tail(&ri_constraint_cache_valid_list, &riinfo->valid_link);
ri_constraint_cache_valid_count++;
riinfo->valid = true; riinfo->valid = true;
return riinfo; return riinfo;
...@@ -2936,21 +2947,41 @@ ri_LoadConstraintInfo(Oid constraintOid) ...@@ -2936,21 +2947,41 @@ ri_LoadConstraintInfo(Oid constraintOid)
* gets enough update traffic that it's probably worth being smarter. * gets enough update traffic that it's probably worth being smarter.
* Invalidate any ri_constraint_cache entry associated with the syscache * Invalidate any ri_constraint_cache entry associated with the syscache
* entry with the specified hash value, or all entries if hashvalue == 0. * entry with the specified hash value, or all entries if hashvalue == 0.
*
* Note: at the time a cache invalidation message is processed there may be
* active references to the cache. Because of this we never remove entries
* from the cache, but only mark them invalid, which is harmless to active
* uses. (Any query using an entry should hold a lock sufficient to keep that
* data from changing under it --- but we may get cache flushes anyway.)
*/ */
static void static void
InvalidateConstraintCacheCallBack(Datum arg, int cacheid, uint32 hashvalue) InvalidateConstraintCacheCallBack(Datum arg, int cacheid, uint32 hashvalue)
{ {
HASH_SEQ_STATUS status; dlist_mutable_iter iter;
RI_ConstraintInfo *hentry;
Assert(ri_constraint_cache != NULL); Assert(ri_constraint_cache != NULL);
hash_seq_init(&status, ri_constraint_cache); /*
while ((hentry = (RI_ConstraintInfo *) hash_seq_search(&status)) != NULL) * If the list of currently valid entries gets excessively large, we mark
* them all invalid so we can empty the list. This arrangement avoids
* O(N^2) behavior in situations where a session touches many foreign keys
* and also does many ALTER TABLEs, such as a restore from pg_dump.
*/
if (ri_constraint_cache_valid_count > 1000)
hashvalue = 0; /* pretend it's a cache reset */
dlist_foreach_modify(iter, &ri_constraint_cache_valid_list)
{
RI_ConstraintInfo *riinfo = dlist_container(RI_ConstraintInfo,
valid_link, iter.cur);
if (hashvalue == 0 || riinfo->oidHashValue == hashvalue)
{ {
if (hentry->valid && riinfo->valid = false;
(hashvalue == 0 || hentry->oidHashValue == hashvalue)) /* Remove invalidated entries from the list, too */
hentry->valid = false; dlist_delete(iter.cur);
ri_constraint_cache_valid_count--;
}
} }
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment