mirror of
https://github.com/postgres/postgres.git
synced 2025-04-24 10:47:04 +03:00
entry that has rules. This allows us to release the rule parsetrees on relcache flush without needing a working freeObject() routine. Formerly, the rule trees were leaked permanently at relcache flush. Also, clean up handling of rule creation and deletion --- there was not sufficient locking of the relation being modified, and there was no reliable notification of other backends that a relcache reload was needed. Also, clean up relcache.c code so that scans of system tables needed to load a relcache entry are done in the caller's memory context, not in CacheMemoryContext. This prevents any un-pfreed memory from those scans from becoming a permanent memory leak.
181 lines
4.6 KiB
C
181 lines
4.6 KiB
C
/*-------------------------------------------------------------------------
|
|
*
|
|
* rewriteRemove.c
|
|
* routines for removing rewrite rules
|
|
*
|
|
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
|
|
* Portions Copyright (c) 1994, Regents of the University of California
|
|
*
|
|
*
|
|
* IDENTIFICATION
|
|
* $Header: /cvsroot/pgsql/src/backend/rewrite/rewriteRemove.c,v 1.38 2000/06/30 07:04:23 tgl Exp $
|
|
*
|
|
*-------------------------------------------------------------------------
|
|
*/
|
|
|
|
|
|
#include "postgres.h"
|
|
|
|
#include "utils/builtins.h"
|
|
#include "access/heapam.h"
|
|
#include "catalog/catname.h"
|
|
#include "catalog/pg_rewrite.h"
|
|
#include "commands/comment.h"
|
|
#include "rewrite/rewriteRemove.h"
|
|
#include "rewrite/rewriteSupport.h"
|
|
#include "utils/fmgroids.h"
|
|
#include "utils/syscache.h"
|
|
|
|
/*-----------------------------------------------------------------------
|
|
* RewriteGetRuleEventRel
|
|
*-----------------------------------------------------------------------
|
|
*/
|
|
char *
|
|
RewriteGetRuleEventRel(char *rulename)
|
|
{
|
|
HeapTuple htup;
|
|
Oid eventrel;
|
|
|
|
htup = SearchSysCacheTuple(RULENAME,
|
|
PointerGetDatum(rulename),
|
|
0, 0, 0);
|
|
if (!HeapTupleIsValid(htup))
|
|
elog(ERROR, "Rule or view '%s' not found",
|
|
((!strncmp(rulename, "_RET", 4)) ? (rulename + 4) : rulename));
|
|
eventrel = ((Form_pg_rewrite) GETSTRUCT(htup))->ev_class;
|
|
htup = SearchSysCacheTuple(RELOID,
|
|
PointerGetDatum(eventrel),
|
|
0, 0, 0);
|
|
if (!HeapTupleIsValid(htup))
|
|
elog(ERROR, "Class '%u' not found", eventrel);
|
|
|
|
return NameStr(((Form_pg_class) GETSTRUCT(htup))->relname);
|
|
}
|
|
|
|
/*
|
|
* RemoveRewriteRule
|
|
*
|
|
* Delete a rule given its rulename.
|
|
*/
|
|
void
|
|
RemoveRewriteRule(char *ruleName)
|
|
{
|
|
Relation RewriteRelation;
|
|
Relation event_relation;
|
|
HeapTuple tuple;
|
|
Oid ruleId;
|
|
Oid eventRelationOid;
|
|
bool hasMoreRules;
|
|
|
|
/*
|
|
* Open the pg_rewrite relation.
|
|
*/
|
|
RewriteRelation = heap_openr(RewriteRelationName, RowExclusiveLock);
|
|
|
|
/*
|
|
* Find the tuple for the target rule.
|
|
*/
|
|
tuple = SearchSysCacheTupleCopy(RULENAME,
|
|
PointerGetDatum(ruleName),
|
|
0, 0, 0);
|
|
|
|
/*
|
|
* complain if no rule with such name existed
|
|
*/
|
|
if (!HeapTupleIsValid(tuple))
|
|
{
|
|
heap_close(RewriteRelation, RowExclusiveLock);
|
|
elog(ERROR, "Rule '%s' not found\n", ruleName);
|
|
}
|
|
|
|
/*
|
|
* Save the OID of the rule (i.e. the tuple's OID) and the event
|
|
* relation's OID
|
|
*/
|
|
ruleId = tuple->t_data->t_oid;
|
|
eventRelationOid = ((Form_pg_rewrite) GETSTRUCT(tuple))->ev_class;
|
|
|
|
/*
|
|
* We had better grab AccessExclusiveLock so that we know no other
|
|
* rule additions/deletions are going on for this relation. Else
|
|
* we cannot set relhasrules correctly. Besides, we don't want to
|
|
* be changing the ruleset while queries are executing on the rel.
|
|
*/
|
|
event_relation = heap_open(eventRelationOid, AccessExclusiveLock);
|
|
|
|
hasMoreRules = event_relation->rd_rules != NULL &&
|
|
event_relation->rd_rules->numLocks > 1;
|
|
|
|
/*
|
|
* Delete any comments associated with this rule
|
|
*/
|
|
DeleteComments(ruleId);
|
|
|
|
/*
|
|
* Now delete the pg_rewrite tuple for the rule
|
|
*/
|
|
heap_delete(RewriteRelation, &tuple->t_self, NULL);
|
|
|
|
heap_freetuple(tuple);
|
|
|
|
heap_close(RewriteRelation, RowExclusiveLock);
|
|
|
|
/*
|
|
* Set pg_class 'relhasrules' field correctly for event relation.
|
|
*
|
|
* Important side effect: an SI notice is broadcast to force all
|
|
* backends (including me!) to update relcache entries with the
|
|
* new rule set. Therefore, must do this even if relhasrules is
|
|
* still true!
|
|
*/
|
|
setRelhasrulesInRelation(eventRelationOid, hasMoreRules);
|
|
|
|
/* Close rel, but keep lock till commit... */
|
|
heap_close(event_relation, NoLock);
|
|
}
|
|
|
|
/*
|
|
* RelationRemoveRules -
|
|
* removes all rules associated with the relation when the relation is
|
|
* being removed.
|
|
*/
|
|
void
|
|
RelationRemoveRules(Oid relid)
|
|
{
|
|
Relation RewriteRelation = NULL;
|
|
HeapScanDesc scanDesc = NULL;
|
|
ScanKeyData scanKeyData;
|
|
HeapTuple tuple = NULL;
|
|
|
|
/*
|
|
* Open the pg_rewrite relation.
|
|
*/
|
|
RewriteRelation = heap_openr(RewriteRelationName, RowExclusiveLock);
|
|
|
|
/*
|
|
* Scan the RuleRelation ('pg_rewrite') for all the tuples that has
|
|
* the same ev_class as relid (the relation to be removed).
|
|
*/
|
|
ScanKeyEntryInitialize(&scanKeyData,
|
|
0,
|
|
Anum_pg_rewrite_ev_class,
|
|
F_OIDEQ,
|
|
ObjectIdGetDatum(relid));
|
|
scanDesc = heap_beginscan(RewriteRelation,
|
|
0, SnapshotNow, 1, &scanKeyData);
|
|
|
|
while (HeapTupleIsValid(tuple = heap_getnext(scanDesc, 0)))
|
|
{
|
|
|
|
/*** Delete any comments associated with this relation ***/
|
|
|
|
DeleteComments(tuple->t_data->t_oid);
|
|
|
|
heap_delete(RewriteRelation, &tuple->t_self, NULL);
|
|
|
|
}
|
|
|
|
heap_endscan(scanDesc);
|
|
heap_close(RewriteRelation, RowExclusiveLock);
|
|
}
|