Update copyright for 2022
[pgsql.git] / src / include / utils / arrayaccess.h
blob7ad01d012d661c74cab312cb55ceef9bf6be6fa8
1 /*-------------------------------------------------------------------------
3 * arrayaccess.h
4 * Declarations for element-by-element access to Postgres arrays.
7 * Portions Copyright (c) 1996-2022, PostgreSQL Global Development Group
8 * Portions Copyright (c) 1994, Regents of the University of California
10 * src/include/utils/arrayaccess.h
12 *-------------------------------------------------------------------------
14 #ifndef ARRAYACCESS_H
15 #define ARRAYACCESS_H
17 #include "access/tupmacs.h"
18 #include "utils/array.h"
22 * Functions for iterating through elements of a flat or expanded array.
23 * These require a state struct "array_iter iter".
25 * Use "array_iter_setup(&iter, arrayptr);" to prepare to iterate, and
26 * "datumvar = array_iter_next(&iter, &isnullvar, index, ...);" to fetch
27 * the next element into datumvar/isnullvar.
28 * "index" must be the zero-origin element number; we make caller provide
29 * this since caller is generally counting the elements anyway. Despite
30 * that, these functions can only fetch elements sequentially.
33 typedef struct array_iter
35 /* datumptr being NULL or not tells if we have flat or expanded array */
37 /* Fields used when we have an expanded array */
38 Datum *datumptr; /* Pointer to Datum array */
39 bool *isnullptr; /* Pointer to isnull array */
41 /* Fields used when we have a flat array */
42 char *dataptr; /* Current spot in the data area */
43 bits8 *bitmapptr; /* Current byte of the nulls bitmap, or NULL */
44 int bitmask; /* mask for current bit in nulls bitmap */
45 } array_iter;
48 static inline void
49 array_iter_setup(array_iter *it, AnyArrayType *a)
51 if (VARATT_IS_EXPANDED_HEADER(a))
53 if (a->xpn.dvalues)
55 it->datumptr = a->xpn.dvalues;
56 it->isnullptr = a->xpn.dnulls;
57 /* we must fill all fields to prevent compiler warnings */
58 it->dataptr = NULL;
59 it->bitmapptr = NULL;
61 else
63 /* Work with flat array embedded in the expanded datum */
64 it->datumptr = NULL;
65 it->isnullptr = NULL;
66 it->dataptr = ARR_DATA_PTR(a->xpn.fvalue);
67 it->bitmapptr = ARR_NULLBITMAP(a->xpn.fvalue);
70 else
72 it->datumptr = NULL;
73 it->isnullptr = NULL;
74 it->dataptr = ARR_DATA_PTR((ArrayType *) a);
75 it->bitmapptr = ARR_NULLBITMAP((ArrayType *) a);
77 it->bitmask = 1;
80 static inline Datum
81 array_iter_next(array_iter *it, bool *isnull, int i,
82 int elmlen, bool elmbyval, char elmalign)
84 Datum ret;
86 if (it->datumptr)
88 ret = it->datumptr[i];
89 *isnull = it->isnullptr ? it->isnullptr[i] : false;
91 else
93 if (it->bitmapptr && (*(it->bitmapptr) & it->bitmask) == 0)
95 *isnull = true;
96 ret = (Datum) 0;
98 else
100 *isnull = false;
101 ret = fetch_att(it->dataptr, elmbyval, elmlen);
102 it->dataptr = att_addlength_pointer(it->dataptr, elmlen,
103 it->dataptr);
104 it->dataptr = (char *) att_align_nominal(it->dataptr, elmalign);
106 it->bitmask <<= 1;
107 if (it->bitmask == 0x100)
109 if (it->bitmapptr)
110 it->bitmapptr++;
111 it->bitmask = 1;
115 return ret;
118 #endif /* ARRAYACCESS_H */