mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-24 01:29:19 +03:00 
			
		
		
		
	This is a back-patch of the v15-era commitf10f0ae42into older supported branches. The idea is to design out bugs in which an ill-timed relcache flush clears rel->rd_smgr partway through some code sequence that wasn't expecting that. We had another report today of a corner case that reliably crashes v14 under debug_discard_caches (nee CLOBBER_CACHE_ALWAYS), and therefore would crash once in a blue moon in the field. We're unlikely to get rid of all such code paths unless we adopt the more rigorous coding rules instituted byf10f0ae42. Therefore, even though this is a bit invasive, it's time to back-patch. Some comfort can be taken in the fact thatf10f0ae42has been in v15 for 16 months without problems. I left the RelationOpenSmgr macro present in the back branches, even though no core code should use it anymore, in order to not break third-party extensions in minor releases. Such extensions might opt to start using RelationGetSmgr instead, to reduce their code differential between v15 and earlier branches. This carries a hazard of failing to compile against headers from existing minor releases. However, once compiled the extension should work fine even with such releases, because RelationGetSmgr is a "static inline" function so it creates no link-time dependency. So depending on distribution practices, that might be an OK tradeoff. Per report from Spyridon Dimitrios Agathos. Original patch by Amul Sul. Discussion: https://postgr.es/m/CAFM5RaqdgyusQvmWkyPYaWMwoK5gigdtW-7HcgHgOeAw7mqJ_Q@mail.gmail.com Discussion: https://postgr.es/m/CANiYTQsU7yMFpQYnv=BrcRVqK_3U3mtAzAsJCaqtzsDHfsUbdQ@mail.gmail.com
		
			
				
	
	
		
			367 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			367 lines
		
	
	
		
			9.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*-------------------------------------------------------------------------
 | |
|  *
 | |
|  * blinsert.c
 | |
|  *		Bloom index build and insert functions.
 | |
|  *
 | |
|  * Copyright (c) 2016-2020, PostgreSQL Global Development Group
 | |
|  *
 | |
|  * IDENTIFICATION
 | |
|  *	  contrib/bloom/blinsert.c
 | |
|  *
 | |
|  *-------------------------------------------------------------------------
 | |
|  */
 | |
| #include "postgres.h"
 | |
| 
 | |
| #include "access/genam.h"
 | |
| #include "access/generic_xlog.h"
 | |
| #include "access/tableam.h"
 | |
| #include "bloom.h"
 | |
| #include "catalog/index.h"
 | |
| #include "miscadmin.h"
 | |
| #include "storage/bufmgr.h"
 | |
| #include "storage/indexfsm.h"
 | |
| #include "storage/smgr.h"
 | |
| #include "utils/memutils.h"
 | |
| #include "utils/rel.h"
 | |
| 
 | |
| PG_MODULE_MAGIC;
 | |
| 
 | |
| /*
 | |
|  * State of bloom index build.  We accumulate one page data here before
 | |
|  * flushing it to buffer manager.
 | |
|  */
 | |
| typedef struct
 | |
| {
 | |
| 	BloomState	blstate;		/* bloom index state */
 | |
| 	int64		indtuples;		/* total number of tuples indexed */
 | |
| 	MemoryContext tmpCtx;		/* temporary memory context reset after each
 | |
| 								 * tuple */
 | |
| 	PGAlignedBlock data;		/* cached page */
 | |
| 	int			count;			/* number of tuples in cached page */
 | |
| } BloomBuildState;
 | |
| 
 | |
| /*
 | |
|  * Flush page cached in BloomBuildState.
 | |
|  */
 | |
| static void
 | |
| flushCachedPage(Relation index, BloomBuildState *buildstate)
 | |
| {
 | |
| 	Page		page;
 | |
| 	Buffer		buffer = BloomNewBuffer(index);
 | |
| 	GenericXLogState *state;
 | |
| 
 | |
| 	state = GenericXLogStart(index);
 | |
| 	page = GenericXLogRegisterBuffer(state, buffer, GENERIC_XLOG_FULL_IMAGE);
 | |
| 	memcpy(page, buildstate->data.data, BLCKSZ);
 | |
| 	GenericXLogFinish(state);
 | |
| 	UnlockReleaseBuffer(buffer);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * (Re)initialize cached page in BloomBuildState.
 | |
|  */
 | |
| static void
 | |
| initCachedPage(BloomBuildState *buildstate)
 | |
| {
 | |
| 	memset(buildstate->data.data, 0, BLCKSZ);
 | |
| 	BloomInitPage(buildstate->data.data, 0);
 | |
| 	buildstate->count = 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Per-tuple callback for table_index_build_scan.
 | |
|  */
 | |
| static void
 | |
| bloomBuildCallback(Relation index, ItemPointer tid, Datum *values,
 | |
| 				   bool *isnull, bool tupleIsAlive, void *state)
 | |
| {
 | |
| 	BloomBuildState *buildstate = (BloomBuildState *) state;
 | |
| 	MemoryContext oldCtx;
 | |
| 	BloomTuple *itup;
 | |
| 
 | |
| 	oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx);
 | |
| 
 | |
| 	itup = BloomFormTuple(&buildstate->blstate, tid, values, isnull);
 | |
| 
 | |
| 	/* Try to add next item to cached page */
 | |
| 	if (BloomPageAddItem(&buildstate->blstate, buildstate->data.data, itup))
 | |
| 	{
 | |
| 		/* Next item was added successfully */
 | |
| 		buildstate->count++;
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		/* Cached page is full, flush it out and make a new one */
 | |
| 		flushCachedPage(index, buildstate);
 | |
| 
 | |
| 		CHECK_FOR_INTERRUPTS();
 | |
| 
 | |
| 		initCachedPage(buildstate);
 | |
| 
 | |
| 		if (!BloomPageAddItem(&buildstate->blstate, buildstate->data.data, itup))
 | |
| 		{
 | |
| 			/* We shouldn't be here since we're inserting to the empty page */
 | |
| 			elog(ERROR, "could not add new bloom tuple to empty page");
 | |
| 		}
 | |
| 
 | |
| 		/* Next item was added successfully */
 | |
| 		buildstate->count++;
 | |
| 	}
 | |
| 
 | |
| 	/* Update total tuple count */
 | |
| 	buildstate->indtuples += 1;
 | |
| 
 | |
| 	MemoryContextSwitchTo(oldCtx);
 | |
| 	MemoryContextReset(buildstate->tmpCtx);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Build a new bloom index.
 | |
|  */
 | |
| IndexBuildResult *
 | |
| blbuild(Relation heap, Relation index, IndexInfo *indexInfo)
 | |
| {
 | |
| 	IndexBuildResult *result;
 | |
| 	double		reltuples;
 | |
| 	BloomBuildState buildstate;
 | |
| 
 | |
| 	if (RelationGetNumberOfBlocks(index) != 0)
 | |
| 		elog(ERROR, "index \"%s\" already contains data",
 | |
| 			 RelationGetRelationName(index));
 | |
| 
 | |
| 	/* Initialize the meta page */
 | |
| 	BloomInitMetapage(index);
 | |
| 
 | |
| 	/* Initialize the bloom build state */
 | |
| 	memset(&buildstate, 0, sizeof(buildstate));
 | |
| 	initBloomState(&buildstate.blstate, index);
 | |
| 	buildstate.tmpCtx = AllocSetContextCreate(CurrentMemoryContext,
 | |
| 											  "Bloom build temporary context",
 | |
| 											  ALLOCSET_DEFAULT_SIZES);
 | |
| 	initCachedPage(&buildstate);
 | |
| 
 | |
| 	/* Do the heap scan */
 | |
| 	reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
 | |
| 									   bloomBuildCallback, (void *) &buildstate,
 | |
| 									   NULL);
 | |
| 
 | |
| 	/* Flush last page if needed (it will be, unless heap was empty) */
 | |
| 	if (buildstate.count > 0)
 | |
| 		flushCachedPage(index, &buildstate);
 | |
| 
 | |
| 	MemoryContextDelete(buildstate.tmpCtx);
 | |
| 
 | |
| 	result = (IndexBuildResult *) palloc(sizeof(IndexBuildResult));
 | |
| 	result->heap_tuples = reltuples;
 | |
| 	result->index_tuples = buildstate.indtuples;
 | |
| 
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Build an empty bloom index in the initialization fork.
 | |
|  */
 | |
| void
 | |
| blbuildempty(Relation index)
 | |
| {
 | |
| 	Page		metapage;
 | |
| 
 | |
| 	/* Construct metapage. */
 | |
| 	metapage = (Page) palloc(BLCKSZ);
 | |
| 	BloomFillMetapage(index, metapage);
 | |
| 
 | |
| 	/*
 | |
| 	 * Write the page and log it.  It might seem that an immediate sync would
 | |
| 	 * be sufficient to guarantee that the file exists on disk, but recovery
 | |
| 	 * itself might remove it while replaying, for example, an
 | |
| 	 * XLOG_DBASE_CREATE or XLOG_TBLSPC_CREATE record.  Therefore, we need
 | |
| 	 * this even when wal_level=minimal.
 | |
| 	 */
 | |
| 	PageSetChecksumInplace(metapage, BLOOM_METAPAGE_BLKNO);
 | |
| 	smgrwrite(RelationGetSmgr(index), INIT_FORKNUM, BLOOM_METAPAGE_BLKNO,
 | |
| 			  (char *) metapage, true);
 | |
| 	log_newpage(&(RelationGetSmgr(index))->smgr_rnode.node, INIT_FORKNUM,
 | |
| 				BLOOM_METAPAGE_BLKNO, metapage, true);
 | |
| 
 | |
| 	/*
 | |
| 	 * An immediate sync is required even if we xlog'd the page, because the
 | |
| 	 * write did not go through shared_buffers and therefore a concurrent
 | |
| 	 * checkpoint may have moved the redo pointer past our xlog record.
 | |
| 	 */
 | |
| 	smgrimmedsync(RelationGetSmgr(index), INIT_FORKNUM);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Insert new tuple to the bloom index.
 | |
|  */
 | |
| bool
 | |
| blinsert(Relation index, Datum *values, bool *isnull,
 | |
| 		 ItemPointer ht_ctid, Relation heapRel,
 | |
| 		 IndexUniqueCheck checkUnique,
 | |
| 		 IndexInfo *indexInfo)
 | |
| {
 | |
| 	BloomState	blstate;
 | |
| 	BloomTuple *itup;
 | |
| 	MemoryContext oldCtx;
 | |
| 	MemoryContext insertCtx;
 | |
| 	BloomMetaPageData *metaData;
 | |
| 	Buffer		buffer,
 | |
| 				metaBuffer;
 | |
| 	Page		page,
 | |
| 				metaPage;
 | |
| 	BlockNumber blkno = InvalidBlockNumber;
 | |
| 	OffsetNumber nStart;
 | |
| 	GenericXLogState *state;
 | |
| 
 | |
| 	insertCtx = AllocSetContextCreate(CurrentMemoryContext,
 | |
| 									  "Bloom insert temporary context",
 | |
| 									  ALLOCSET_DEFAULT_SIZES);
 | |
| 
 | |
| 	oldCtx = MemoryContextSwitchTo(insertCtx);
 | |
| 
 | |
| 	initBloomState(&blstate, index);
 | |
| 	itup = BloomFormTuple(&blstate, ht_ctid, values, isnull);
 | |
| 
 | |
| 	/*
 | |
| 	 * At first, try to insert new tuple to the first page in notFullPage
 | |
| 	 * array.  If successful, we don't need to modify the meta page.
 | |
| 	 */
 | |
| 	metaBuffer = ReadBuffer(index, BLOOM_METAPAGE_BLKNO);
 | |
| 	LockBuffer(metaBuffer, BUFFER_LOCK_SHARE);
 | |
| 	metaData = BloomPageGetMeta(BufferGetPage(metaBuffer));
 | |
| 
 | |
| 	if (metaData->nEnd > metaData->nStart)
 | |
| 	{
 | |
| 		Page		page;
 | |
| 
 | |
| 		blkno = metaData->notFullPage[metaData->nStart];
 | |
| 		Assert(blkno != InvalidBlockNumber);
 | |
| 
 | |
| 		/* Don't hold metabuffer lock while doing insert */
 | |
| 		LockBuffer(metaBuffer, BUFFER_LOCK_UNLOCK);
 | |
| 
 | |
| 		buffer = ReadBuffer(index, blkno);
 | |
| 		LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE);
 | |
| 
 | |
| 		state = GenericXLogStart(index);
 | |
| 		page = GenericXLogRegisterBuffer(state, buffer, 0);
 | |
| 
 | |
| 		/*
 | |
| 		 * We might have found a page that was recently deleted by VACUUM.  If
 | |
| 		 * so, we can reuse it, but we must reinitialize it.
 | |
| 		 */
 | |
| 		if (PageIsNew(page) || BloomPageIsDeleted(page))
 | |
| 			BloomInitPage(page, 0);
 | |
| 
 | |
| 		if (BloomPageAddItem(&blstate, page, itup))
 | |
| 		{
 | |
| 			/* Success!  Apply the change, clean up, and exit */
 | |
| 			GenericXLogFinish(state);
 | |
| 			UnlockReleaseBuffer(buffer);
 | |
| 			ReleaseBuffer(metaBuffer);
 | |
| 			MemoryContextSwitchTo(oldCtx);
 | |
| 			MemoryContextDelete(insertCtx);
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		/* Didn't fit, must try other pages */
 | |
| 		GenericXLogAbort(state);
 | |
| 		UnlockReleaseBuffer(buffer);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		/* No entries in notFullPage */
 | |
| 		LockBuffer(metaBuffer, BUFFER_LOCK_UNLOCK);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * Try other pages in notFullPage array.  We will have to change nStart in
 | |
| 	 * metapage.  Thus, grab exclusive lock on metapage.
 | |
| 	 */
 | |
| 	LockBuffer(metaBuffer, BUFFER_LOCK_EXCLUSIVE);
 | |
| 
 | |
| 	/* nStart might have changed while we didn't have lock */
 | |
| 	nStart = metaData->nStart;
 | |
| 
 | |
| 	/* Skip first page if we already tried it above */
 | |
| 	if (nStart < metaData->nEnd &&
 | |
| 		blkno == metaData->notFullPage[nStart])
 | |
| 		nStart++;
 | |
| 
 | |
| 	/*
 | |
| 	 * This loop iterates for each page we try from the notFullPage array, and
 | |
| 	 * will also initialize a GenericXLogState for the fallback case of having
 | |
| 	 * to allocate a new page.
 | |
| 	 */
 | |
| 	for (;;)
 | |
| 	{
 | |
| 		state = GenericXLogStart(index);
 | |
| 
 | |
| 		/* get modifiable copy of metapage */
 | |
| 		metaPage = GenericXLogRegisterBuffer(state, metaBuffer, 0);
 | |
| 		metaData = BloomPageGetMeta(metaPage);
 | |
| 
 | |
| 		if (nStart >= metaData->nEnd)
 | |
| 			break;				/* no more entries in notFullPage array */
 | |
| 
 | |
| 		blkno = metaData->notFullPage[nStart];
 | |
| 		Assert(blkno != InvalidBlockNumber);
 | |
| 
 | |
| 		buffer = ReadBuffer(index, blkno);
 | |
| 		LockBuffer(buffer, BUFFER_LOCK_EXCLUSIVE);
 | |
| 		page = GenericXLogRegisterBuffer(state, buffer, 0);
 | |
| 
 | |
| 		/* Basically same logic as above */
 | |
| 		if (PageIsNew(page) || BloomPageIsDeleted(page))
 | |
| 			BloomInitPage(page, 0);
 | |
| 
 | |
| 		if (BloomPageAddItem(&blstate, page, itup))
 | |
| 		{
 | |
| 			/* Success!  Apply the changes, clean up, and exit */
 | |
| 			metaData->nStart = nStart;
 | |
| 			GenericXLogFinish(state);
 | |
| 			UnlockReleaseBuffer(buffer);
 | |
| 			UnlockReleaseBuffer(metaBuffer);
 | |
| 			MemoryContextSwitchTo(oldCtx);
 | |
| 			MemoryContextDelete(insertCtx);
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		/* Didn't fit, must try other pages */
 | |
| 		GenericXLogAbort(state);
 | |
| 		UnlockReleaseBuffer(buffer);
 | |
| 		nStart++;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * Didn't find place to insert in notFullPage array.  Allocate new page.
 | |
| 	 * (XXX is it good to do this while holding ex-lock on the metapage??)
 | |
| 	 */
 | |
| 	buffer = BloomNewBuffer(index);
 | |
| 
 | |
| 	page = GenericXLogRegisterBuffer(state, buffer, GENERIC_XLOG_FULL_IMAGE);
 | |
| 	BloomInitPage(page, 0);
 | |
| 
 | |
| 	if (!BloomPageAddItem(&blstate, page, itup))
 | |
| 	{
 | |
| 		/* We shouldn't be here since we're inserting to an empty page */
 | |
| 		elog(ERROR, "could not add new bloom tuple to empty page");
 | |
| 	}
 | |
| 
 | |
| 	/* Reset notFullPage array to contain just this new page */
 | |
| 	metaData->nStart = 0;
 | |
| 	metaData->nEnd = 1;
 | |
| 	metaData->notFullPage[0] = BufferGetBlockNumber(buffer);
 | |
| 
 | |
| 	/* Apply the changes, clean up, and exit */
 | |
| 	GenericXLogFinish(state);
 | |
| 
 | |
| 	UnlockReleaseBuffer(buffer);
 | |
| 	UnlockReleaseBuffer(metaBuffer);
 | |
| 
 | |
| 	MemoryContextSwitchTo(oldCtx);
 | |
| 	MemoryContextDelete(insertCtx);
 | |
| 
 | |
| 	return false;
 | |
| }
 |