greenplumn appendonly_blkdir_udf 源码

  • 2022-08-18
  • 浏览 (500)

greenplumn appendonly_blkdir_udf 代码

文件路径:/src/backend/access/appendonly/appendonly_blkdir_udf.c

/*------------------------------------------------------------------------------
 *
 * AppendOnly_Blkdir UDFs
 *   User-defined functions (UDF) for support of append-only block directory
 *
 * Copyright (c) 2013-Present VMware, Inc. or its affiliates.
 *
 *
 * IDENTIFICATION
 *	    src/backend/access/appendonly/appendonly_blkdir_udf.c
 *
 *------------------------------------------------------------------------------
 */

#include "postgres.h"

#include "access/appendonly_visimap.h"
#include "access/table.h"
#include "catalog/aoblkdir.h"
#include "cdb/cdbappendonlyblockdirectory.h"
#include "cdb/cdbvars.h"
#include "funcapi.h"
#include "utils/snapmgr.h"

Datum gp_aoblkdir(PG_FUNCTION_ARGS);

/*
 * This UDF emits block directory entries for an AO/AOCO relation. It does so
 * by flattening the minipage column of ao_blkdir relations, yielding 1 minipage
 * entry / output row.
 *
 * Format:
 * tupleid | segno | columngroup_no | entry_no | first_row_no | file_offset | row_count
 *
 * This UDF also respects gp_select_invisible to report block directory entries
 * that are invisible. To determine invisible entries we can use the tupleid
 * projected here and tie it to the corresponding pg_aoblkdir tuple's xmax.
 */

Datum
gp_aoblkdir(PG_FUNCTION_ARGS)
{
	Oid       	aoRelOid = PG_GETARG_OID(0);
	HeapTuple 	tuple;

	typedef struct Context
	{
		Relation 				aorel;
		SysScanDesc 			scan;
		MinipagePerColumnGroup	currMinipage;
		bool					currMinipageValid;
		int 					currMinipageEntryIdx;
		Relation				blkdirrel;
	} Context;

	FuncCallContext *funcctx;
	Context			*context;

	if (SRF_IS_FIRSTCALL())
	{
		TupleDesc     tupdesc;
		MemoryContext oldcontext;
		Snapshot      sst;
		Oid           blkdirrelid;

		/* create a function context for cross-call persistence */
		funcctx = SRF_FIRSTCALL_INIT();

		/*
		 * switch to memory context appropriate for multiple function calls
		 */
		oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);

		/* build tupdesc for result tuples */
		tupdesc = CreateTemplateTupleDesc(7);
		TupleDescInitEntry(tupdesc, (AttrNumber) 1, "tupleid",
						   TIDOID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 2, "segno",
						   INT4OID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 3, "columngroup_no",
						   INT4OID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 4, "entry_no",
						   INT4OID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 5, "first_row_no",
						   INT8OID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 6, "file_offset",
						   INT8OID, -1, 0);
		TupleDescInitEntry(tupdesc, (AttrNumber) 7, "row_count",
						   INT8OID, -1, 0);
		funcctx->tuple_desc = BlessTupleDesc(tupdesc);

		/* initialize Context for SRF */
		context = (Context *) palloc0(sizeof(Context));
		context->aorel = table_open(aoRelOid, AccessShareLock);
		if (!RelationIsAppendOptimized(context->aorel))
			ereport(ERROR,
					(errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
						errmsg("function not supported on non append-optimized relation")));
		sst = GetLatestSnapshot();
		GetAppendOnlyEntryAuxOids(aoRelOid, sst,
								  NULL, &blkdirrelid, NULL,
								  NULL, NULL);
		sst = gp_select_invisible ? SnapshotAny : GetLatestSnapshot();
		if (blkdirrelid == InvalidOid)
			ereport(ERROR,
					(errmsg("appendoptimized relation doesn't have a block directory"),
					 errhint("relation must have or must have had an index")));
		context->blkdirrel = table_open(blkdirrelid, AccessShareLock);
		context->scan = systable_beginscan(context->blkdirrel,
										   InvalidOid,
										   false,
										   sst,
										   0,
										   NULL);
		context->currMinipage.minipage = palloc0(minipage_size(NUM_MINIPAGE_ENTRIES));
		context->currMinipageValid = false;
		context->currMinipageEntryIdx = -1;
		funcctx->user_fctx = (void *) context;

		MemoryContextSwitchTo(oldcontext);
	}

	funcctx = SRF_PERCALL_SETUP();
	context = (Context *) funcctx->user_fctx;

	if (!context->currMinipageValid)
	{
		Datum	minipage;
		bool	minipageNull;

		/* We need to fetch the next tuple from the blkdir relation */
		if (!systable_getnext(context->scan))
			goto srf_done;

		/* deform the tuple and populate slot->values/nulls */
		slot_getallattrs(context->scan->slot);

		minipage = slot_getattr(context->scan->slot, Anum_pg_aoblkdir_minipage, &minipageNull);
		/*
		 * There should not really be any NULL values. We opt to report it
		 * instead of ERRORing out.
		 */
		context->currMinipageValid = !minipageNull;
		if (context->currMinipageValid)
		{
			/*
			 * Cache the latest scanned minipage and use it to emit the next
			 * (context->currMinipage->numMinipageEntries) rows
			 */
			copy_out_minipage(&context->currMinipage, minipage, false);
			context->currMinipageEntryIdx = 0;
		}
	}

	{
		Datum          	values[7];
		bool			nulls[7];
		TupleTableSlot 	*slot = context->scan->slot;
		Datum          	result;

		values[0] = ItemPointerGetDatum(&slot->tts_tid);
		nulls[0] = false;

		values[1] = slot_getattr(slot, Anum_pg_aoblkdir_segno, &nulls[1]);
		values[2] = slot_getattr(slot, Anum_pg_aoblkdir_columngroupno, &nulls[2]);

		/* emit minipage entry */
		if (context->currMinipageValid)
		{
			MinipagePerColumnGroup *currMinipage = &context->currMinipage;
			MinipageEntry *minipageEntry;

			Assert(context->currMinipageEntryIdx < currMinipage->numMinipageEntries);

			minipageEntry = &currMinipage->minipage->entry[context->currMinipageEntryIdx];

			values[3] = context->currMinipageEntryIdx++;
			values[4] = Int64GetDatum(minipageEntry->firstRowNum);
			values[5] = Int64GetDatum(minipageEntry->fileOffset);
			values[6] = Int64GetDatum(minipageEntry->rowCount);

			nulls[3] = false;
			nulls[4] = false;
			nulls[5] = false;
			nulls[6] = false;

			context->currMinipageValid =
				(context->currMinipageEntryIdx != currMinipage->numMinipageEntries);
		}
		else
		{
			nulls[3] = true;
			nulls[4] = true;
			nulls[5] = true;
			nulls[6] = true;
		}

		tuple = heap_form_tuple(funcctx->tuple_desc, values, nulls);
		result = HeapTupleGetDatum(tuple);
		SRF_RETURN_NEXT(funcctx, result);
	}

srf_done:
	table_close(context->aorel, AccessShareLock);
	systable_endscan(context->scan);
	table_close(context->blkdirrel, AccessShareLock);
	pfree(context);
	funcctx->user_fctx = NULL;
	SRF_RETURN_DONE(funcctx);
}

相关信息

greenplumn 源码目录

相关文章

greenplumn aomd 源码

greenplumn aomd_filehandler 源码

greenplumn aosegfiles 源码

greenplumn appendonly_compaction 源码

greenplumn appendonly_visimap 源码

greenplumn appendonly_visimap_entry 源码

greenplumn appendonly_visimap_store 源码

greenplumn appendonly_visimap_udf 源码

greenplumn appendonlyam 源码

greenplumn appendonlyam_handler 源码

0  赞