/* Reset the queue and re-add procs in the desired order */
dclist_init(waitQueue);
for (int j = 0; j < nProcs; j++)
- dclist_push_tail(waitQueue, &procs[j]->links);
+ dclist_push_tail(waitQueue, &procs[j]->waitLink);
#ifdef DEBUG_DEADLOCK
PrintLockQueue(lock, "rearranged to:");
* If the process is waiting, there is an outgoing waits-for edge to each
* process that blocks it.
*/
- if (checkProc->links.next != NULL && checkProc->waitLock != NULL &&
+ if (!dlist_node_is_detached(&checkProc->waitLink) &&
FindLockCycleRecurseMember(checkProc, checkProc, depth, softEdges,
nSoftEdges))
return true;
memberProc = dlist_container(PGPROC, lockGroupLink, iter.cur);
- if (memberProc->links.next != NULL && memberProc->waitLock != NULL &&
+ if (!dlist_node_is_detached(&memberProc->waitLink) && memberProc->waitLock != NULL &&
memberProc != checkProc &&
FindLockCycleRecurseMember(memberProc, checkProc, depth, softEdges,
nSoftEdges))
{
dclist_foreach(proc_iter, waitQueue)
{
- proc = dlist_container(PGPROC, links, proc_iter.cur);
+ proc = dlist_container(PGPROC, waitLink, proc_iter.cur);
if (proc->lockGroupLeader == checkProcLeader)
lastGroupMember = proc;
{
PGPROC *leader;
- proc = dlist_container(PGPROC, links, proc_iter.cur);
+ proc = dlist_container(PGPROC, waitLink, proc_iter.cur);
leader = proc->lockGroupLeader == NULL ? proc :
proc->lockGroupLeader;
i = 0;
dclist_foreach(proc_iter, waitQueue)
{
- proc = dlist_container(PGPROC, links, proc_iter.cur);
+ proc = dlist_container(PGPROC, waitLink, proc_iter.cur);
topoProcs[i++] = proc;
}
Assert(i == queue_size);
dclist_foreach(proc_iter, waitQueue)
{
- PGPROC *proc = dlist_container(PGPROC, links, proc_iter.cur);
+ PGPROC *proc = dlist_container(PGPROC, waitLink, proc_iter.cur);
printf(" %d", proc->pid);
}
if (i < MaxConnections)
{
/* PGPROC for normal backend, add to freeProcs list */
- dlist_push_tail(&ProcGlobal->freeProcs, &proc->links);
+ dlist_push_tail(&ProcGlobal->freeProcs, &proc->freeProcsLink);
proc->procgloballist = &ProcGlobal->freeProcs;
}
else if (i < MaxConnections + autovacuum_worker_slots + NUM_SPECIAL_WORKER_PROCS)
{
/* PGPROC for AV or special worker, add to autovacFreeProcs list */
- dlist_push_tail(&ProcGlobal->autovacFreeProcs, &proc->links);
+ dlist_push_tail(&ProcGlobal->autovacFreeProcs, &proc->freeProcsLink);
proc->procgloballist = &ProcGlobal->autovacFreeProcs;
}
else if (i < MaxConnections + autovacuum_worker_slots + NUM_SPECIAL_WORKER_PROCS + max_worker_processes)
{
/* PGPROC for bgworker, add to bgworkerFreeProcs list */
- dlist_push_tail(&ProcGlobal->bgworkerFreeProcs, &proc->links);
+ dlist_push_tail(&ProcGlobal->bgworkerFreeProcs, &proc->freeProcsLink);
proc->procgloballist = &ProcGlobal->bgworkerFreeProcs;
}
else if (i < MaxBackends)
{
/* PGPROC for walsender, add to walsenderFreeProcs list */
- dlist_push_tail(&ProcGlobal->walsenderFreeProcs, &proc->links);
+ dlist_push_tail(&ProcGlobal->walsenderFreeProcs, &proc->freeProcsLink);
proc->procgloballist = &ProcGlobal->walsenderFreeProcs;
}
if (!dlist_is_empty(procgloballist))
{
- MyProc = dlist_container(PGPROC, links, dlist_pop_head_node(procgloballist));
+ MyProc = dlist_container(PGPROC, freeProcsLink, dlist_pop_head_node(procgloballist));
SpinLockRelease(&ProcGlobal->freeProcsLock);
}
else
* Initialize all fields of MyProc, except for those previously
* initialized by InitProcGlobal.
*/
- dlist_node_init(&MyProc->links);
+ dlist_node_init(&MyProc->freeProcsLink);
MyProc->waitStatus = PROC_WAIT_STATUS_OK;
MyProc->fpVXIDLock = false;
MyProc->fpLocalTransactionId = InvalidLocalTransactionId;
MyProc->lwWaiting = LW_WS_NOT_WAITING;
MyProc->lwWaitMode = 0;
MyProc->waitLock = NULL;
+ dlist_node_init(&MyProc->waitLink);
MyProc->waitProcLock = NULL;
pg_atomic_write_u64(&MyProc->waitStart, 0);
#ifdef USE_ASSERT_CHECKING
* Initialize all fields of MyProc, except for those previously
* initialized by InitProcGlobal.
*/
- dlist_node_init(&MyProc->links);
+ dlist_node_init(&MyProc->freeProcsLink);
MyProc->waitStatus = PROC_WAIT_STATUS_OK;
MyProc->fpVXIDLock = false;
MyProc->fpLocalTransactionId = InvalidLocalTransactionId;
MyProc->lwWaiting = LW_WS_NOT_WAITING;
MyProc->lwWaitMode = 0;
MyProc->waitLock = NULL;
+ dlist_node_init(&MyProc->waitLink);
MyProc->waitProcLock = NULL;
pg_atomic_write_u64(&MyProc->waitStart, 0);
#ifdef USE_ASSERT_CHECKING
partitionLock = LockHashPartitionLock(lockAwaited->hashcode);
LWLockAcquire(partitionLock, LW_EXCLUSIVE);
- if (!dlist_node_is_detached(&MyProc->links))
+ if (!dlist_node_is_detached(&MyProc->waitLink))
{
/* We could not have been granted the lock yet */
RemoveFromWaitQueue(MyProc, lockAwaited->hashcode);
/* Leader exited first; return its PGPROC. */
SpinLockAcquire(&ProcGlobal->freeProcsLock);
- dlist_push_head(procgloballist, &leader->links);
+ dlist_push_head(procgloballist, &leader->freeProcsLink);
SpinLockRelease(&ProcGlobal->freeProcsLock);
}
}
Assert(dlist_is_empty(&proc->lockGroupMembers));
/* Return PGPROC structure (and semaphore) to appropriate freelist */
- dlist_push_tail(procgloballist, &proc->links);
+ dlist_push_tail(procgloballist, &proc->freeProcsLink);
}
/* Update shared estimate of spins_per_delay */
dclist_foreach(iter, waitQueue)
{
- PGPROC *proc = dlist_container(PGPROC, links, iter.cur);
+ PGPROC *proc = dlist_container(PGPROC, waitLink, iter.cur);
/*
* If we're part of the same locking group as this waiter, its
* Insert self into queue, at the position determined above.
*/
if (insert_before)
- dclist_insert_before(waitQueue, &insert_before->links, &MyProc->links);
+ dclist_insert_before(waitQueue, &insert_before->waitLink, &MyProc->waitLink);
else
- dclist_push_tail(waitQueue, &MyProc->links);
+ dclist_push_tail(waitQueue, &MyProc->waitLink);
lock->waitMask |= LOCKBIT_ON(lockmode);
/*
* ProcWakeup -- wake up a process by setting its latch.
*
- * Also remove the process from the wait queue and set its links invalid.
+ * Also remove the process from the wait queue and set its waitLink invalid.
*
* The appropriate lock partition lock must be held by caller.
*
void
ProcWakeup(PGPROC *proc, ProcWaitStatus waitStatus)
{
- if (dlist_node_is_detached(&proc->links))
+ if (dlist_node_is_detached(&proc->waitLink))
return;
Assert(proc->waitStatus == PROC_WAIT_STATUS_WAITING);
/* Remove process from wait queue */
- dclist_delete_from_thoroughly(&proc->waitLock->waitProcs, &proc->links);
+ dclist_delete_from_thoroughly(&proc->waitLock->waitProcs, &proc->waitLink);
/* Clean up process' state and pass it the ok/fail signal */
proc->waitLock = NULL;
dclist_foreach_modify(miter, waitQueue)
{
- PGPROC *proc = dlist_container(PGPROC, links, miter.cur);
+ PGPROC *proc = dlist_container(PGPROC, waitLink, miter.cur);
LOCKMODE lockmode = proc->waitLockMode;
/*
* We check by looking to see if we've been unlinked from the wait queue.
* This is safe because we hold the lock partition lock.
*/
- if (MyProc->links.prev == NULL ||
- MyProc->links.next == NULL)
+ if (dlist_node_is_detached(&MyProc->waitLink))
{
result = DS_NO_DEADLOCK;
goto check_done;
* Each backend has a PGPROC struct in shared memory. There is also a list of
* currently-unused PGPROC structs that will be reallocated to new backends.
*
- * links: list link for any list the PGPROC is in. When waiting for a lock,
- * the PGPROC is linked into that lock's waitProcs queue. A recycled PGPROC
- * is linked into ProcGlobal's freeProcs list.
- *
* Note: twophase.c also sets up a dummy PGPROC struct for each currently
* prepared transaction. These PGPROCs appear in the ProcArray data structure
* so that the prepared transactions appear to be still running and are
*/
struct PGPROC
{
- dlist_node links; /* list link if process is in a list */
dlist_head *procgloballist; /* procglobal list that owns this PGPROC */
+ dlist_node freeProcsLink; /* link in procgloballist, when in recycled
+ * state */
PGSemaphore sem; /* ONE semaphore to sleep on */
ProcWaitStatus waitStatus;
/* Info about lock the process is currently waiting for, if any. */
/* waitLock and waitProcLock are NULL if not currently waiting. */
LOCK *waitLock; /* Lock object we're sleeping on ... */
+ dlist_node waitLink; /* position in waitLock->waitProcs queue */
PROCLOCK *waitProcLock; /* Per-holder info for awaited lock */
LOCKMODE waitLockMode; /* type of lock we're waiting for */
LOCKMASK heldLocks; /* bitmask for lock types already held on this