Fix possible crash in pg_dump with identity sequences.
[pgsql.git] / src / include / access / itup.h
blob94885751e590f93eba33dc7ebfd198de9df047c2
1 /*-------------------------------------------------------------------------
3 * itup.h
4 * POSTGRES index tuple definitions.
7 * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
8 * Portions Copyright (c) 1994, Regents of the University of California
10 * src/include/access/itup.h
12 *-------------------------------------------------------------------------
14 #ifndef ITUP_H
15 #define ITUP_H
17 #include "access/tupdesc.h"
18 #include "access/tupmacs.h"
19 #include "storage/bufpage.h"
20 #include "storage/itemptr.h"
23 * Index tuple header structure
25 * All index tuples start with IndexTupleData. If the HasNulls bit is set,
26 * this is followed by an IndexAttributeBitMapData. The index attribute
27 * values follow, beginning at a MAXALIGN boundary.
29 * Note that the space allocated for the bitmap does not vary with the number
30 * of attributes; that is because we don't have room to store the number of
31 * attributes in the header. Given the MAXALIGN constraint there's no space
32 * savings to be had anyway, for usual values of INDEX_MAX_KEYS.
35 typedef struct IndexTupleData
37 ItemPointerData t_tid; /* reference TID to heap tuple */
39 /* ---------------
40 * t_info is laid out in the following fashion:
42 * 15th (high) bit: has nulls
43 * 14th bit: has var-width attributes
44 * 13th bit: AM-defined meaning
45 * 12-0 bit: size of tuple
46 * ---------------
49 unsigned short t_info; /* various info about tuple */
51 } IndexTupleData; /* MORE DATA FOLLOWS AT END OF STRUCT */
53 typedef IndexTupleData *IndexTuple;
55 typedef struct IndexAttributeBitMapData
57 bits8 bits[(INDEX_MAX_KEYS + 8 - 1) / 8];
58 } IndexAttributeBitMapData;
60 typedef IndexAttributeBitMapData * IndexAttributeBitMap;
63 * t_info manipulation macros
65 #define INDEX_SIZE_MASK 0x1FFF
66 #define INDEX_AM_RESERVED_BIT 0x2000 /* reserved for index-AM specific
67 * usage */
68 #define INDEX_VAR_MASK 0x4000
69 #define INDEX_NULL_MASK 0x8000
71 #define IndexTupleSize(itup) ((Size) ((itup)->t_info & INDEX_SIZE_MASK))
72 #define IndexTupleHasNulls(itup) ((((IndexTuple) (itup))->t_info & INDEX_NULL_MASK))
73 #define IndexTupleHasVarwidths(itup) ((((IndexTuple) (itup))->t_info & INDEX_VAR_MASK))
76 /* routines in indextuple.c */
77 extern IndexTuple index_form_tuple(TupleDesc tupleDescriptor,
78 const Datum *values, const bool *isnull);
79 extern IndexTuple index_form_tuple_context(TupleDesc tupleDescriptor,
80 const Datum *values, const bool *isnull,
81 MemoryContext context);
82 extern Datum nocache_index_getattr(IndexTuple tup, int attnum,
83 TupleDesc tupleDesc);
84 extern void index_deform_tuple(IndexTuple tup, TupleDesc tupleDescriptor,
85 Datum *values, bool *isnull);
86 extern void index_deform_tuple_internal(TupleDesc tupleDescriptor,
87 Datum *values, bool *isnull,
88 char *tp, bits8 *bp, int hasnulls);
89 extern IndexTuple CopyIndexTuple(IndexTuple source);
90 extern IndexTuple index_truncate_tuple(TupleDesc sourceDescriptor,
91 IndexTuple source, int leavenatts);
95 * Takes an infomask as argument (primarily because this needs to be usable
96 * at index_form_tuple time so enough space is allocated).
98 static inline Size
99 IndexInfoFindDataOffset(unsigned short t_info)
101 if (!(t_info & INDEX_NULL_MASK))
102 return MAXALIGN(sizeof(IndexTupleData));
103 else
104 return MAXALIGN(sizeof(IndexTupleData) + sizeof(IndexAttributeBitMapData));
107 #ifndef FRONTEND
109 /* ----------------
110 * index_getattr
112 * This gets called many times, so we macro the cacheable and NULL
113 * lookups, and call nocache_index_getattr() for the rest.
115 * ----------------
117 static inline Datum
118 index_getattr(IndexTuple tup, int attnum, TupleDesc tupleDesc, bool *isnull)
120 Assert(PointerIsValid(isnull));
121 Assert(attnum > 0);
123 *isnull = false;
125 if (!IndexTupleHasNulls(tup))
127 if (TupleDescAttr(tupleDesc, attnum - 1)->attcacheoff >= 0)
129 return fetchatt(TupleDescAttr(tupleDesc, attnum - 1),
130 (char *) tup + IndexInfoFindDataOffset(tup->t_info)
131 + TupleDescAttr(tupleDesc, attnum - 1)->attcacheoff);
133 else
134 return nocache_index_getattr(tup, attnum, tupleDesc);
136 else
138 if (att_isnull(attnum - 1, (bits8 *) tup + sizeof(IndexTupleData)))
140 *isnull = true;
141 return (Datum) NULL;
143 else
144 return nocache_index_getattr(tup, attnum, tupleDesc);
148 #endif
151 * MaxIndexTuplesPerPage is an upper bound on the number of tuples that can
152 * fit on one index page. An index tuple must have either data or a null
153 * bitmap, so we can safely assume it's at least 1 byte bigger than a bare
154 * IndexTupleData struct. We arrive at the divisor because each tuple
155 * must be maxaligned, and it must have an associated line pointer.
157 * To be index-type-independent, this does not account for any special space
158 * on the page, and is thus conservative.
160 * Note: in btree non-leaf pages, the first tuple has no key (it's implicitly
161 * minus infinity), thus breaking the "at least 1 byte bigger" assumption.
162 * On such a page, N tuples could take one MAXALIGN quantum less space than
163 * estimated here, seemingly allowing one more tuple than estimated here.
164 * But such a page always has at least MAXALIGN special space, so we're safe.
166 #define MaxIndexTuplesPerPage \
167 ((int) ((BLCKSZ - SizeOfPageHeaderData) / \
168 (MAXALIGN(sizeof(IndexTupleData) + 1) + sizeof(ItemIdData))))
170 #endif /* ITUP_H */