mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-25 13:17:41 +03:00 
			
		
		
		
	Retry after buffer locking failure during SPGiST index creation.
The original coding thought this case was impossible, but it can happen if the bgwriter or checkpointer processes decide to write out an index page while creation is still proceeding, leading to a bogus "unexpected spgdoinsert() failure" error. Problem reported by Jonathan S. Katz. Teodor Sigaev
This commit is contained in:
		| @@ -1946,9 +1946,12 @@ spgdoinsert(Relation index, SpGistState *state, | |||||||
| 			 * Attempt to acquire lock on child page.  We must beware of | 			 * Attempt to acquire lock on child page.  We must beware of | ||||||
| 			 * deadlock against another insertion process descending from that | 			 * deadlock against another insertion process descending from that | ||||||
| 			 * page to our parent page (see README).  If we fail to get lock, | 			 * page to our parent page (see README).  If we fail to get lock, | ||||||
| 			 * abandon the insertion and tell our caller to start over.  XXX | 			 * abandon the insertion and tell our caller to start over. | ||||||
| 			 * this could be improved; perhaps it'd be worth sleeping a bit | 			 * | ||||||
| 			 * before giving up? | 			 * XXX this could be improved, because failing to get lock on a | ||||||
|  | 			 * buffer is not proof of a deadlock situation; the lock might be | ||||||
|  | 			 * held by a reader, or even just background writer/checkpointer | ||||||
|  | 			 * process.  Perhaps it'd be worth retrying after sleeping a bit? | ||||||
| 			 */ | 			 */ | ||||||
| 			if (!ConditionalLockBuffer(current.buffer)) | 			if (!ConditionalLockBuffer(current.buffer)) | ||||||
| 			{ | 			{ | ||||||
|   | |||||||
| @@ -45,10 +45,17 @@ spgistBuildCallback(Relation index, HeapTuple htup, Datum *values, | |||||||
| 	/* Work in temp context, and reset it after each tuple */ | 	/* Work in temp context, and reset it after each tuple */ | ||||||
| 	oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx); | 	oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx); | ||||||
|  |  | ||||||
| 	/* No concurrent insertions can be happening, so failure is unexpected */ | 	/* | ||||||
| 	if (!spgdoinsert(index, &buildstate->spgstate, &htup->t_self, | 	 * Even though no concurrent insertions can be happening, we still might | ||||||
|  | 	 * get a buffer-locking failure due to bgwriter or checkpointer taking a | ||||||
|  | 	 * lock on some buffer.  So we need to be willing to retry.  We can flush | ||||||
|  | 	 * any temp data when retrying. | ||||||
|  | 	 */ | ||||||
|  | 	while (!spgdoinsert(index, &buildstate->spgstate, &htup->t_self, | ||||||
| 						*values, *isnull)) | 						*values, *isnull)) | ||||||
| 		elog(ERROR, "unexpected spgdoinsert() failure"); | 	{ | ||||||
|  | 		MemoryContextReset(buildstate->tmpCtx); | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	MemoryContextSwitchTo(oldCtx); | 	MemoryContextSwitchTo(oldCtx); | ||||||
| 	MemoryContextReset(buildstate->tmpCtx); | 	MemoryContextReset(buildstate->tmpCtx); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user