indextuple.c

来自「postgresql8.3.4源码,开源数据库」· C语言 代码 · 共 449 行

C
449
字号
/*------------------------------------------------------------------------- * * indextuple.c *	   This file contains index tuple accessor and mutator routines, *	   as well as various tuple utilities. * * Portions Copyright (c) 1996-2008, PostgreSQL Global Development Group * Portions Copyright (c) 1994, Regents of the University of California * * * IDENTIFICATION *	  $PostgreSQL: pgsql/src/backend/access/common/indextuple.c,v 1.85 2008/01/01 19:45:45 momjian Exp $ * *------------------------------------------------------------------------- */#include "postgres.h"#include "access/heapam.h"#include "access/itup.h"#include "access/tuptoaster.h"/* ---------------------------------------------------------------- *				  index_ tuple interface routines * ---------------------------------------------------------------- *//* ---------------- *		index_form_tuple * ---------------- */IndexTupleindex_form_tuple(TupleDesc tupleDescriptor,				 Datum *values,				 bool *isnull){	char	   *tp;				/* tuple pointer */	IndexTuple	tuple;			/* return tuple */	Size		size,				data_size,				hoff;	int			i;	unsigned short infomask = 0;	bool		hasnull = false;	uint16		tupmask = 0;	int			numberOfAttributes = tupleDescriptor->natts;#ifdef TOAST_INDEX_HACK	Datum		untoasted_values[INDEX_MAX_KEYS];	bool		untoasted_free[INDEX_MAX_KEYS];#endif	if (numberOfAttributes > INDEX_MAX_KEYS)		ereport(ERROR,				(errcode(ERRCODE_TOO_MANY_COLUMNS),				 errmsg("number of index columns (%d) exceeds limit (%d)",						numberOfAttributes, INDEX_MAX_KEYS)));#ifdef TOAST_INDEX_HACK	for (i = 0; i < numberOfAttributes; i++)	{		Form_pg_attribute att = tupleDescriptor->attrs[i];		untoasted_values[i] = values[i];		untoasted_free[i] = false;		/* Do nothing if value is NULL or not of varlena type */		if (isnull[i] || att->attlen != -1)			continue;		/*		 * If value is stored EXTERNAL, must fetch it so we are not depending		 * on outside storage.	This should be improved someday.		 */		if (VARATT_IS_EXTERNAL(values[i]))		{			untoasted_values[i] =				PointerGetDatum(heap_tuple_fetch_attr((struct varlena *)												DatumGetPointer(values[i])));			untoasted_free[i] = true;		}		/*		 * If value is above size target, and is of a compressible datatype,		 * try to compress it in-line.		 */		if (!VARATT_IS_EXTENDED(untoasted_values[i]) &&			VARSIZE(untoasted_values[i]) > TOAST_INDEX_TARGET &&			(att->attstorage == 'x' || att->attstorage == 'm'))		{			Datum		cvalue = toast_compress_datum(untoasted_values[i]);			if (DatumGetPointer(cvalue) != NULL)			{				/* successful compression */				if (untoasted_free[i])					pfree(DatumGetPointer(untoasted_values[i]));				untoasted_values[i] = cvalue;				untoasted_free[i] = true;			}		}	}#endif	for (i = 0; i < numberOfAttributes; i++)	{		if (isnull[i])		{			hasnull = true;			break;		}	}	if (hasnull)		infomask |= INDEX_NULL_MASK;	hoff = IndexInfoFindDataOffset(infomask);#ifdef TOAST_INDEX_HACK	data_size = heap_compute_data_size(tupleDescriptor,									   untoasted_values, isnull);#else	data_size = heap_compute_data_size(tupleDescriptor,									   values, isnull);#endif	size = hoff + data_size;	size = MAXALIGN(size);		/* be conservative */	tp = (char *) palloc0(size);	tuple = (IndexTuple) tp;	heap_fill_tuple(tupleDescriptor,#ifdef TOAST_INDEX_HACK					untoasted_values,#else					values,#endif					isnull,					(char *) tp + hoff,					data_size,					&tupmask,					(hasnull ? (bits8 *) tp + sizeof(IndexTupleData) : NULL));#ifdef TOAST_INDEX_HACK	for (i = 0; i < numberOfAttributes; i++)	{		if (untoasted_free[i])			pfree(DatumGetPointer(untoasted_values[i]));	}#endif	/*	 * We do this because heap_fill_tuple wants to initialize a "tupmask"	 * which is used for HeapTuples, but we want an indextuple infomask. The	 * only relevant info is the "has variable attributes" field. We have	 * already set the hasnull bit above.	 */	if (tupmask & HEAP_HASVARWIDTH)		infomask |= INDEX_VAR_MASK;	/*	 * Here we make sure that the size will fit in the field reserved for it	 * in t_info.	 */	if ((size & INDEX_SIZE_MASK) != size)		ereport(ERROR,				(errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),				 errmsg("index row requires %lu bytes, maximum size is %lu",						(unsigned long) size,						(unsigned long) INDEX_SIZE_MASK)));	infomask |= size;	/*	 * initialize metadata	 */	tuple->t_info = infomask;	return tuple;}/* ---------------- *		nocache_index_getattr * *		This gets called from index_getattr() macro, and only in cases *		where we can't use cacheoffset and the value is not null. * *		This caches attribute offsets in the attribute descriptor. * *		An alternative way to speed things up would be to cache offsets *		with the tuple, but that seems more difficult unless you take *		the storage hit of actually putting those offsets into the *		tuple you send to disk.  Yuck. * *		This scheme will be slightly slower than that, but should *		perform well for queries which hit large #'s of tuples.  After *		you cache the offsets once, examining all the other tuples using *		the same attribute descriptor will go much quicker. -cim 5/4/91 * ---------------- */Datumnocache_index_getattr(IndexTuple tup,					  int attnum,					  TupleDesc tupleDesc,					  bool *isnull){	Form_pg_attribute *att = tupleDesc->attrs;	char	   *tp;				/* ptr to data part of tuple */	bits8	   *bp = NULL;		/* ptr to null bitmap in tuple */	bool		slow = false;	/* do we have to walk attrs? */	int			data_off;		/* tuple data offset */	int			off;			/* current offset within data */	(void) isnull;				/* not used */	/* ----------------	 *	 Three cases:	 *	 *	 1: No nulls and no variable-width attributes.	 *	 2: Has a null or a var-width AFTER att.	 *	 3: Has nulls or var-widths BEFORE att.	 * ----------------	 */#ifdef IN_MACRO/* This is handled in the macro */	Assert(PointerIsValid(isnull));	Assert(attnum > 0);	*isnull = false;#endif	data_off = IndexInfoFindDataOffset(tup->t_info);	attnum--;	if (!IndexTupleHasNulls(tup))	{#ifdef IN_MACRO/* This is handled in the macro */		if (att[attnum]->attcacheoff >= 0)		{			return fetchatt(att[attnum],							(char *) tup + data_off +							att[attnum]->attcacheoff);		}#endif	}	else	{		/*		 * there's a null somewhere in the tuple		 *		 * check to see if desired att is null		 */		/* XXX "knows" t_bits are just after fixed tuple header! */		bp = (bits8 *) ((char *) tup + sizeof(IndexTupleData));#ifdef IN_MACRO/* This is handled in the macro */		if (att_isnull(attnum, bp))		{			*isnull = true;			return (Datum) NULL;		}#endif		/*		 * Now check to see if any preceding bits are null...		 */		{			int			byte = attnum >> 3;			int			finalbit = attnum & 0x07;			/* check for nulls "before" final bit of last byte */			if ((~bp[byte]) & ((1 << finalbit) - 1))				slow = true;			else			{				/* check for nulls in any "earlier" bytes */				int			i;				for (i = 0; i < byte; i++)				{					if (bp[i] != 0xFF)					{						slow = true;						break;					}				}			}		}	}	tp = (char *) tup + data_off;	if (!slow)	{		/*		 * If we get here, there are no nulls up to and including the target		 * attribute.  If we have a cached offset, we can use it.		 */		if (att[attnum]->attcacheoff >= 0)		{			return fetchatt(att[attnum],							tp + att[attnum]->attcacheoff);		}		/*		 * Otherwise, check for non-fixed-length attrs up to and including		 * target.	If there aren't any, it's safe to cheaply initialize the		 * cached offsets for these attrs.		 */		if (IndexTupleHasVarwidths(tup))		{			int			j;			for (j = 0; j <= attnum; j++)			{				if (att[j]->attlen <= 0)				{					slow = true;					break;				}			}		}	}	if (!slow)	{		int			natts = tupleDesc->natts;		int			j = 1;		/*		 * If we get here, we have a tuple with no nulls or var-widths up to		 * and including the target attribute, so we can use the cached offset		 * ... only we don't have it yet, or we'd not have got here.  Since		 * it's cheap to compute offsets for fixed-width columns, we take the		 * opportunity to initialize the cached offsets for *all* the leading		 * fixed-width columns, in hope of avoiding future visits to this		 * routine.		 */		att[0]->attcacheoff = 0;		/* we might have set some offsets in the slow path previously */		while (j < natts && att[j]->attcacheoff > 0)			j++;		off = att[j - 1]->attcacheoff + att[j - 1]->attlen;		for (; j < natts; j++)		{			if (att[j]->attlen <= 0)				break;			off = att_align_nominal(off, att[j]->attalign);			att[j]->attcacheoff = off;			off += att[j]->attlen;		}		Assert(j > attnum);		off = att[attnum]->attcacheoff;	}	else	{		bool		usecache = true;		int			i;		/*		 * Now we know that we have to walk the tuple CAREFULLY.  But we still		 * might be able to cache some offsets for next time.		 *		 * Note - This loop is a little tricky.  For each non-null attribute,		 * we have to first account for alignment padding before the attr,		 * then advance over the attr based on its length.	Nulls have no		 * storage and no alignment padding either.  We can use/set		 * attcacheoff until we reach either a null or a var-width attribute.		 */		off = 0;		for (i = 0;; i++)		/* loop exit is at "break" */		{			if (IndexTupleHasNulls(tup) && att_isnull(i, bp))			{				usecache = false;				continue;		/* this cannot be the target att */			}			/* If we know the next offset, we can skip the rest */			if (usecache && att[i]->attcacheoff >= 0)				off = att[i]->attcacheoff;			else if (att[i]->attlen == -1)			{				/*				 * We can only cache the offset for a varlena attribute if the				 * offset is already suitably aligned, so that there would be				 * no pad bytes in any case: then the offset will be valid for				 * either an aligned or unaligned value.				 */				if (usecache &&					off == att_align_nominal(off, att[i]->attalign))					att[i]->attcacheoff = off;				else				{					off = att_align_pointer(off, att[i]->attalign, -1,											tp + off);					usecache = false;				}			}			else			{				/* not varlena, so safe to use att_align_nominal */				off = att_align_nominal(off, att[i]->attalign);				if (usecache)					att[i]->attcacheoff = off;			}			if (i == attnum)				break;			off = att_addlength_pointer(off, att[i]->attlen, tp + off);			if (usecache && att[i]->attlen <= 0)				usecache = false;		}	}	return fetchatt(att[attnum], tp + off);}/* * Create a palloc'd copy of an index tuple. */IndexTupleCopyIndexTuple(IndexTuple source){	IndexTuple	result;	Size		size;	size = IndexTupleSize(source);	result = (IndexTuple) palloc(size);	memcpy(result, source, size);	return result;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?