mirror of
https://git.postgresql.org/git/postgresql.git
synced 2025-01-12 18:34:36 +08:00
Change executor to just Assert that table locks were already obtained.
Instead of locking tables during executor startup, just Assert that suitable locks were obtained already during the parse/plan pipeline (or re-obtained by the plan cache). This must be so, else we have a hazard that concurrent DDL has invalidated the plan. This is pretty inefficient as well as undercommented, but it's all going to go away shortly, so I didn't try hard. This commit is just another attempt to use the buildfarm to see if we've missed anything in the plan to simplify the executor's table management. Note that the change needed here in relation_open() exposes that parallel workers now really are accessing tables without holding any lock of their own, whereas they were not doing that before this commit. This does not give me a warm fuzzy feeling about that aspect of parallel query; it does not seem like a good design, and we now know that it's had exactly no actual testing. I think that we should modify parallel query so that that change can be reverted. Discussion: https://postgr.es/m/468c85d9-540e-66a2-1dde-fec2b741e688@lab.ntt.co.jp
This commit is contained in:
parent
c03c1449c0
commit
9a3cebeaa7
@ -1140,9 +1140,13 @@ relation_open(Oid relationId, LOCKMODE lockmode)
|
|||||||
/*
|
/*
|
||||||
* If we didn't get the lock ourselves, assert that caller holds one,
|
* If we didn't get the lock ourselves, assert that caller holds one,
|
||||||
* except in bootstrap mode where no locks are used.
|
* except in bootstrap mode where no locks are used.
|
||||||
|
*
|
||||||
|
* Also, parallel workers currently assume that their parent holds locks
|
||||||
|
* for tables used in the parallel query (a mighty shaky assumption).
|
||||||
*/
|
*/
|
||||||
Assert(lockmode != NoLock ||
|
Assert(lockmode != NoLock ||
|
||||||
IsBootstrapProcessingMode() ||
|
IsBootstrapProcessingMode() ||
|
||||||
|
IsParallelWorker() ||
|
||||||
CheckRelationLockedByMe(r, AccessShareLock, true));
|
CheckRelationLockedByMe(r, AccessShareLock, true));
|
||||||
|
|
||||||
/* Make note that we've accessed a temporary relation */
|
/* Make note that we've accessed a temporary relation */
|
||||||
|
@ -849,8 +849,8 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
Relation resultRelation;
|
Relation resultRelation;
|
||||||
|
|
||||||
resultRelationOid = getrelid(resultRelationIndex, rangeTable);
|
resultRelationOid = getrelid(resultRelationIndex, rangeTable);
|
||||||
Assert(rt_fetch(resultRelationIndex, rangeTable)->rellockmode == RowExclusiveLock);
|
resultRelation = heap_open(resultRelationOid, NoLock);
|
||||||
resultRelation = heap_open(resultRelationOid, RowExclusiveLock);
|
Assert(CheckRelationLockedByMe(resultRelation, RowExclusiveLock, true));
|
||||||
|
|
||||||
InitResultRelInfo(resultRelInfo,
|
InitResultRelInfo(resultRelInfo,
|
||||||
resultRelation,
|
resultRelation,
|
||||||
@ -890,8 +890,8 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
Relation resultRelDesc;
|
Relation resultRelDesc;
|
||||||
|
|
||||||
resultRelOid = getrelid(resultRelIndex, rangeTable);
|
resultRelOid = getrelid(resultRelIndex, rangeTable);
|
||||||
Assert(rt_fetch(resultRelIndex, rangeTable)->rellockmode == RowExclusiveLock);
|
resultRelDesc = heap_open(resultRelOid, NoLock);
|
||||||
resultRelDesc = heap_open(resultRelOid, RowExclusiveLock);
|
Assert(CheckRelationLockedByMe(resultRelDesc, RowExclusiveLock, true));
|
||||||
InitResultRelInfo(resultRelInfo,
|
InitResultRelInfo(resultRelInfo,
|
||||||
resultRelDesc,
|
resultRelDesc,
|
||||||
lfirst_int(l),
|
lfirst_int(l),
|
||||||
@ -903,7 +903,8 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
estate->es_root_result_relations = resultRelInfos;
|
estate->es_root_result_relations = resultRelInfos;
|
||||||
estate->es_num_root_result_relations = num_roots;
|
estate->es_num_root_result_relations = num_roots;
|
||||||
|
|
||||||
/* Simply lock the rest of them. */
|
/* Simply check the rest of them are locked. */
|
||||||
|
#ifdef USE_ASSERT_CHECKING
|
||||||
foreach(l, plannedstmt->nonleafResultRelations)
|
foreach(l, plannedstmt->nonleafResultRelations)
|
||||||
{
|
{
|
||||||
Index resultRelIndex = lfirst_int(l);
|
Index resultRelIndex = lfirst_int(l);
|
||||||
@ -912,11 +913,15 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
if (!list_member_int(plannedstmt->rootResultRelations,
|
if (!list_member_int(plannedstmt->rootResultRelations,
|
||||||
resultRelIndex))
|
resultRelIndex))
|
||||||
{
|
{
|
||||||
Assert(rt_fetch(resultRelIndex, rangeTable)->rellockmode == RowExclusiveLock);
|
Relation resultRelDesc;
|
||||||
LockRelationOid(getrelid(resultRelIndex, rangeTable),
|
|
||||||
RowExclusiveLock);
|
resultRelDesc = heap_open(getrelid(resultRelIndex, rangeTable),
|
||||||
|
NoLock);
|
||||||
|
Assert(CheckRelationLockedByMe(resultRelDesc, RowExclusiveLock, true));
|
||||||
|
heap_close(resultRelDesc, NoLock);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -945,7 +950,6 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
{
|
{
|
||||||
PlanRowMark *rc = (PlanRowMark *) lfirst(l);
|
PlanRowMark *rc = (PlanRowMark *) lfirst(l);
|
||||||
Oid relid;
|
Oid relid;
|
||||||
LOCKMODE rellockmode;
|
|
||||||
Relation relation;
|
Relation relation;
|
||||||
ExecRowMark *erm;
|
ExecRowMark *erm;
|
||||||
|
|
||||||
@ -963,8 +967,10 @@ InitPlan(QueryDesc *queryDesc, int eflags)
|
|||||||
case ROW_MARK_SHARE:
|
case ROW_MARK_SHARE:
|
||||||
case ROW_MARK_KEYSHARE:
|
case ROW_MARK_KEYSHARE:
|
||||||
case ROW_MARK_REFERENCE:
|
case ROW_MARK_REFERENCE:
|
||||||
rellockmode = rt_fetch(rc->rti, rangeTable)->rellockmode;
|
relation = heap_open(relid, NoLock);
|
||||||
relation = heap_open(relid, rellockmode);
|
Assert(CheckRelationLockedByMe(relation,
|
||||||
|
rt_fetch(rc->rti, rangeTable)->rellockmode,
|
||||||
|
true));
|
||||||
break;
|
break;
|
||||||
case ROW_MARK_COPY:
|
case ROW_MARK_COPY:
|
||||||
/* no physical table access is required */
|
/* no physical table access is required */
|
||||||
|
@ -42,6 +42,7 @@
|
|||||||
|
|
||||||
#include "postgres.h"
|
#include "postgres.h"
|
||||||
|
|
||||||
|
#include "access/parallel.h"
|
||||||
#include "access/relscan.h"
|
#include "access/relscan.h"
|
||||||
#include "access/transam.h"
|
#include "access/transam.h"
|
||||||
#include "executor/executor.h"
|
#include "executor/executor.h"
|
||||||
@ -648,12 +649,14 @@ ExecOpenScanRelation(EState *estate, Index scanrelid, int eflags)
|
|||||||
{
|
{
|
||||||
Relation rel;
|
Relation rel;
|
||||||
Oid reloid;
|
Oid reloid;
|
||||||
LOCKMODE lockmode;
|
|
||||||
|
|
||||||
/* Open the relation and acquire lock as needed */
|
/* Open the relation and verify lock was obtained upstream */
|
||||||
reloid = getrelid(scanrelid, estate->es_range_table);
|
reloid = getrelid(scanrelid, estate->es_range_table);
|
||||||
lockmode = rt_fetch(scanrelid, estate->es_range_table)->rellockmode;
|
rel = heap_open(reloid, NoLock);
|
||||||
rel = heap_open(reloid, lockmode);
|
Assert(IsParallelWorker() ||
|
||||||
|
CheckRelationLockedByMe(rel,
|
||||||
|
rt_fetch(scanrelid, estate->es_range_table)->rellockmode,
|
||||||
|
true));
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Complain if we're attempting a scan of an unscannable relation, except
|
* Complain if we're attempting a scan of an unscannable relation, except
|
||||||
|
Loading…
Reference in New Issue
Block a user