Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * arrayfuncs.c
4 : * Support functions for arrays.
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : *
10 : * IDENTIFICATION
11 : * src/backend/utils/adt/arrayfuncs.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 : #include "postgres.h"
16 :
17 : #include <ctype.h>
18 : #include <math.h>
19 :
20 : #include "access/transam.h"
21 : #include "catalog/pg_type.h"
22 : #include "common/int.h"
23 : #include "funcapi.h"
24 : #include "libpq/pqformat.h"
25 : #include "nodes/nodeFuncs.h"
26 : #include "nodes/supportnodes.h"
27 : #include "optimizer/optimizer.h"
28 : #include "parser/scansup.h"
29 : #include "port/pg_bitutils.h"
30 : #include "utils/array.h"
31 : #include "utils/arrayaccess.h"
32 : #include "utils/builtins.h"
33 : #include "utils/datum.h"
34 : #include "utils/fmgroids.h"
35 : #include "utils/lsyscache.h"
36 : #include "utils/memutils.h"
37 : #include "utils/selfuncs.h"
38 : #include "utils/typcache.h"
39 :
40 :
41 : /*
42 : * GUC parameter
43 : */
44 : bool Array_nulls = true;
45 :
46 : /*
47 : * Local definitions
48 : */
49 : #define ASSGN "="
50 :
51 : #define AARR_FREE_IF_COPY(array,n) \
52 : do { \
53 : if (!VARATT_IS_EXPANDED_HEADER(array)) \
54 : PG_FREE_IF_COPY(array, n); \
55 : } while (0)
56 :
57 : /* ReadArrayToken return type */
58 : typedef enum
59 : {
60 : ATOK_LEVEL_START,
61 : ATOK_LEVEL_END,
62 : ATOK_DELIM,
63 : ATOK_ELEM,
64 : ATOK_ELEM_NULL,
65 : ATOK_ERROR,
66 : } ArrayToken;
67 :
68 : /* Working state for array_iterate() */
69 : typedef struct ArrayIteratorData
70 : {
71 : /* basic info about the array, set up during array_create_iterator() */
72 : ArrayType *arr; /* array we're iterating through */
73 : bits8 *nullbitmap; /* its null bitmap, if any */
74 : int nitems; /* total number of elements in array */
75 : int16 typlen; /* element type's length */
76 : bool typbyval; /* element type's byval property */
77 : char typalign; /* element type's align property */
78 :
79 : /* information about the requested slice size */
80 : int slice_ndim; /* slice dimension, or 0 if not slicing */
81 : int slice_len; /* number of elements per slice */
82 : int *slice_dims; /* slice dims array */
83 : int *slice_lbound; /* slice lbound array */
84 : Datum *slice_values; /* workspace of length slice_len */
85 : bool *slice_nulls; /* workspace of length slice_len */
86 :
87 : /* current position information, updated on each iteration */
88 : char *data_ptr; /* our current position in the array */
89 : int current_item; /* the item # we're at in the array */
90 : } ArrayIteratorData;
91 :
92 : static bool ReadArrayDimensions(char **srcptr, int *ndim_p,
93 : int *dim, int *lBound,
94 : const char *origStr, Node *escontext);
95 : static bool ReadDimensionInt(char **srcptr, int *result,
96 : const char *origStr, Node *escontext);
97 : static bool ReadArrayStr(char **srcptr,
98 : FmgrInfo *inputproc, Oid typioparam, int32 typmod,
99 : char typdelim,
100 : int typlen, bool typbyval, char typalign,
101 : int *ndim_p, int *dim,
102 : int *nitems_p,
103 : Datum **values_p, bool **nulls_p,
104 : const char *origStr, Node *escontext);
105 : static ArrayToken ReadArrayToken(char **srcptr, StringInfo elembuf, char typdelim,
106 : const char *origStr, Node *escontext);
107 : static void ReadArrayBinary(StringInfo buf, int nitems,
108 : FmgrInfo *receiveproc, Oid typioparam, int32 typmod,
109 : int typlen, bool typbyval, char typalign,
110 : Datum *values, bool *nulls,
111 : bool *hasnulls, int32 *nbytes);
112 : static Datum array_get_element_expanded(Datum arraydatum,
113 : int nSubscripts, int *indx,
114 : int arraytyplen,
115 : int elmlen, bool elmbyval, char elmalign,
116 : bool *isNull);
117 : static Datum array_set_element_expanded(Datum arraydatum,
118 : int nSubscripts, int *indx,
119 : Datum dataValue, bool isNull,
120 : int arraytyplen,
121 : int elmlen, bool elmbyval, char elmalign);
122 : static bool array_get_isnull(const bits8 *nullbitmap, int offset);
123 : static void array_set_isnull(bits8 *nullbitmap, int offset, bool isNull);
124 : static Datum ArrayCast(char *value, bool byval, int len);
125 : static int ArrayCastAndSet(Datum src,
126 : int typlen, bool typbyval, char typalign,
127 : char *dest);
128 : static char *array_seek(char *ptr, int offset, bits8 *nullbitmap, int nitems,
129 : int typlen, bool typbyval, char typalign);
130 : static int array_nelems_size(char *ptr, int offset, bits8 *nullbitmap,
131 : int nitems, int typlen, bool typbyval, char typalign);
132 : static int array_copy(char *destptr, int nitems,
133 : char *srcptr, int offset, bits8 *nullbitmap,
134 : int typlen, bool typbyval, char typalign);
135 : static int array_slice_size(char *arraydataptr, bits8 *arraynullsptr,
136 : int ndim, int *dim, int *lb,
137 : int *st, int *endp,
138 : int typlen, bool typbyval, char typalign);
139 : static void array_extract_slice(ArrayType *newarray,
140 : int ndim, int *dim, int *lb,
141 : char *arraydataptr, bits8 *arraynullsptr,
142 : int *st, int *endp,
143 : int typlen, bool typbyval, char typalign);
144 : static void array_insert_slice(ArrayType *destArray, ArrayType *origArray,
145 : ArrayType *srcArray,
146 : int ndim, int *dim, int *lb,
147 : int *st, int *endp,
148 : int typlen, bool typbyval, char typalign);
149 : static int array_cmp(FunctionCallInfo fcinfo);
150 : static ArrayType *create_array_envelope(int ndims, int *dimv, int *lbsv, int nbytes,
151 : Oid elmtype, int dataoffset);
152 : static ArrayType *array_fill_internal(ArrayType *dims, ArrayType *lbs,
153 : Datum value, bool isnull, Oid elmtype,
154 : FunctionCallInfo fcinfo);
155 : static ArrayType *array_replace_internal(ArrayType *array,
156 : Datum search, bool search_isnull,
157 : Datum replace, bool replace_isnull,
158 : bool remove, Oid collation,
159 : FunctionCallInfo fcinfo);
160 : static int width_bucket_array_float8(Datum operand, ArrayType *thresholds);
161 : static int width_bucket_array_fixed(Datum operand,
162 : ArrayType *thresholds,
163 : Oid collation,
164 : TypeCacheEntry *typentry);
165 : static int width_bucket_array_variable(Datum operand,
166 : ArrayType *thresholds,
167 : Oid collation,
168 : TypeCacheEntry *typentry);
169 :
170 :
171 : /*
172 : * array_in :
173 : * converts an array from the external format in "string" to
174 : * its internal format.
175 : *
176 : * return value :
177 : * the internal representation of the input array
178 : */
179 : Datum
180 237212 : array_in(PG_FUNCTION_ARGS)
181 : {
182 237212 : char *string = PG_GETARG_CSTRING(0); /* external form */
183 237212 : Oid element_type = PG_GETARG_OID(1); /* type of an array
184 : * element */
185 237212 : int32 typmod = PG_GETARG_INT32(2); /* typmod for array elements */
186 237212 : Node *escontext = fcinfo->context;
187 : int typlen;
188 : bool typbyval;
189 : char typalign;
190 : char typdelim;
191 : Oid typioparam;
192 : char *p;
193 : int nitems;
194 : Datum *values;
195 : bool *nulls;
196 : bool hasnulls;
197 : int32 nbytes;
198 : int32 dataoffset;
199 : ArrayType *retval;
200 : int ndim,
201 : dim[MAXDIM],
202 : lBound[MAXDIM];
203 : ArrayMetaState *my_extra;
204 :
205 : /*
206 : * We arrange to look up info about element type, including its input
207 : * conversion proc, only once per series of calls, assuming the element
208 : * type doesn't change underneath us.
209 : */
210 237212 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
211 237212 : if (my_extra == NULL)
212 : {
213 75080 : fcinfo->flinfo->fn_extra = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
214 : sizeof(ArrayMetaState));
215 75080 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
216 75080 : my_extra->element_type = ~element_type;
217 : }
218 :
219 237212 : if (my_extra->element_type != element_type)
220 : {
221 : /*
222 : * Get info about element type, including its input conversion proc
223 : */
224 75112 : get_type_io_data(element_type, IOFunc_input,
225 : &my_extra->typlen, &my_extra->typbyval,
226 : &my_extra->typalign, &my_extra->typdelim,
227 : &my_extra->typioparam, &my_extra->typiofunc);
228 75112 : fmgr_info_cxt(my_extra->typiofunc, &my_extra->proc,
229 75112 : fcinfo->flinfo->fn_mcxt);
230 75112 : my_extra->element_type = element_type;
231 : }
232 237212 : typlen = my_extra->typlen;
233 237212 : typbyval = my_extra->typbyval;
234 237212 : typalign = my_extra->typalign;
235 237212 : typdelim = my_extra->typdelim;
236 237212 : typioparam = my_extra->typioparam;
237 :
238 : /*
239 : * Initialize dim[] and lBound[] for ReadArrayStr, in case there is no
240 : * explicit dimension info. (If there is, ReadArrayDimensions will
241 : * overwrite this.)
242 : */
243 1660484 : for (int i = 0; i < MAXDIM; i++)
244 : {
245 1423272 : dim[i] = -1; /* indicates "not yet known" */
246 1423272 : lBound[i] = 1; /* default lower bound */
247 : }
248 :
249 : /*
250 : * Start processing the input string.
251 : *
252 : * If the input string starts with dimension info, read and use that.
253 : * Otherwise, we'll determine the dimensions during ReadArrayStr.
254 : */
255 237212 : p = string;
256 237212 : if (!ReadArrayDimensions(&p, &ndim, dim, lBound, string, escontext))
257 0 : return (Datum) 0;
258 :
259 237170 : if (ndim == 0)
260 : {
261 : /* No array dimensions, so next character should be a left brace */
262 237022 : if (*p != '{')
263 14 : ereturn(escontext, (Datum) 0,
264 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
265 : errmsg("malformed array literal: \"%s\"", string),
266 : errdetail("Array value must start with \"{\" or dimension information.")));
267 : }
268 : else
269 : {
270 : /* If array dimensions are given, expect '=' operator */
271 148 : if (strncmp(p, ASSGN, strlen(ASSGN)) != 0)
272 0 : ereturn(escontext, (Datum) 0,
273 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
274 : errmsg("malformed array literal: \"%s\"", string),
275 : errdetail("Missing \"%s\" after array dimensions.",
276 : ASSGN)));
277 148 : p += strlen(ASSGN);
278 : /* Allow whitespace after it */
279 148 : while (scanner_isspace(*p))
280 0 : p++;
281 :
282 148 : if (*p != '{')
283 0 : ereturn(escontext, (Datum) 0,
284 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
285 : errmsg("malformed array literal: \"%s\"", string),
286 : errdetail("Array contents must start with \"{\".")));
287 : }
288 :
289 : /* Parse the value part, in the curly braces: { ... } */
290 237156 : if (!ReadArrayStr(&p,
291 : &my_extra->proc, typioparam, typmod,
292 : typdelim,
293 : typlen, typbyval, typalign,
294 : &ndim,
295 : dim,
296 : &nitems,
297 : &values, &nulls,
298 : string,
299 : escontext))
300 42 : return (Datum) 0;
301 :
302 : /* only whitespace is allowed after the closing brace */
303 236964 : while (*p)
304 : {
305 12 : if (!scanner_isspace(*p++))
306 12 : ereturn(escontext, (Datum) 0,
307 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
308 : errmsg("malformed array literal: \"%s\"", string),
309 : errdetail("Junk after closing right brace.")));
310 : }
311 :
312 : /* Empty array? */
313 236952 : if (nitems == 0)
314 4418 : PG_RETURN_ARRAYTYPE_P(construct_empty_array(element_type));
315 :
316 : /*
317 : * Check for nulls, compute total data space needed
318 : */
319 232534 : hasnulls = false;
320 232534 : nbytes = 0;
321 1434414 : for (int i = 0; i < nitems; i++)
322 : {
323 1201880 : if (nulls[i])
324 534 : hasnulls = true;
325 : else
326 : {
327 : /* let's just make sure data is not toasted */
328 1201346 : if (typlen == -1)
329 704016 : values[i] = PointerGetDatum(PG_DETOAST_DATUM(values[i]));
330 1201346 : nbytes = att_addlength_datum(nbytes, typlen, values[i]);
331 1201346 : nbytes = att_align_nominal(nbytes, typalign);
332 : /* check for overflow of total request */
333 1201346 : if (!AllocSizeIsValid(nbytes))
334 0 : ereturn(escontext, (Datum) 0,
335 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
336 : errmsg("array size exceeds the maximum allowed (%d)",
337 : (int) MaxAllocSize)));
338 : }
339 : }
340 232534 : if (hasnulls)
341 : {
342 472 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nitems);
343 472 : nbytes += dataoffset;
344 : }
345 : else
346 : {
347 232062 : dataoffset = 0; /* marker for no null bitmap */
348 232062 : nbytes += ARR_OVERHEAD_NONULLS(ndim);
349 : }
350 :
351 : /*
352 : * Construct the final array datum
353 : */
354 232534 : retval = (ArrayType *) palloc0(nbytes);
355 232534 : SET_VARSIZE(retval, nbytes);
356 232534 : retval->ndim = ndim;
357 232534 : retval->dataoffset = dataoffset;
358 :
359 : /*
360 : * This comes from the array's pg_type.typelem (which points to the base
361 : * data type's pg_type.oid) and stores system oids in user tables. This
362 : * oid must be preserved by binary upgrades.
363 : */
364 232534 : retval->elemtype = element_type;
365 232534 : memcpy(ARR_DIMS(retval), dim, ndim * sizeof(int));
366 232534 : memcpy(ARR_LBOUND(retval), lBound, ndim * sizeof(int));
367 :
368 232534 : CopyArrayEls(retval,
369 : values, nulls, nitems,
370 : typlen, typbyval, typalign,
371 : true);
372 :
373 232534 : pfree(values);
374 232534 : pfree(nulls);
375 :
376 232534 : PG_RETURN_ARRAYTYPE_P(retval);
377 : }
378 :
379 : /*
380 : * ReadArrayDimensions
381 : * parses the array dimensions part of the input and converts the values
382 : * to internal format.
383 : *
384 : * On entry, *srcptr points to the string to parse. It is advanced to point
385 : * after whitespace (if any) and dimension info (if any).
386 : *
387 : * *ndim_p, dim[], and lBound[] are output variables. They are filled with the
388 : * number of dimensions (<= MAXDIM), the lengths of each dimension, and the
389 : * lower subscript bounds, respectively. If no dimension info appears,
390 : * *ndim_p will be set to zero, and dim[] and lBound[] are unchanged.
391 : *
392 : * 'origStr' is the original input string, used only in error messages.
393 : * If *escontext points to an ErrorSaveContext, details of any error are
394 : * reported there.
395 : *
396 : * Result:
397 : * true for success, false for failure (if escontext is provided).
398 : *
399 : * Note that dim[] and lBound[] are allocated by the caller, and must have
400 : * MAXDIM elements.
401 : */
402 : static bool
403 237212 : ReadArrayDimensions(char **srcptr, int *ndim_p, int *dim, int *lBound,
404 : const char *origStr, Node *escontext)
405 : {
406 237212 : char *p = *srcptr;
407 : int ndim;
408 :
409 : /*
410 : * Dimension info takes the form of one or more [n] or [m:n] items. This
411 : * loop iterates once per dimension item.
412 : */
413 237212 : ndim = 0;
414 : for (;;)
415 204 : {
416 : char *q;
417 : int ub;
418 : int i;
419 :
420 : /*
421 : * Note: we currently allow whitespace between, but not within,
422 : * dimension items.
423 : */
424 237428 : while (scanner_isspace(*p))
425 12 : p++;
426 237416 : if (*p != '[')
427 237170 : break; /* no more dimension items */
428 246 : p++;
429 246 : if (ndim >= MAXDIM)
430 0 : ereturn(escontext, false,
431 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
432 : errmsg("number of array dimensions exceeds the maximum allowed (%d)",
433 : MAXDIM)));
434 :
435 246 : q = p;
436 246 : if (!ReadDimensionInt(&p, &i, origStr, escontext))
437 0 : return false;
438 234 : if (p == q) /* no digits? */
439 6 : ereturn(escontext, false,
440 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
441 : errmsg("malformed array literal: \"%s\"", origStr),
442 : errdetail("\"[\" must introduce explicitly-specified array dimensions.")));
443 :
444 228 : if (*p == ':')
445 : {
446 : /* [m:n] format */
447 216 : lBound[ndim] = i;
448 216 : p++;
449 216 : q = p;
450 216 : if (!ReadDimensionInt(&p, &ub, origStr, escontext))
451 0 : return false;
452 216 : if (p == q) /* no digits? */
453 6 : ereturn(escontext, false,
454 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
455 : errmsg("malformed array literal: \"%s\"", origStr),
456 : errdetail("Missing array dimension value.")));
457 : }
458 : else
459 : {
460 : /* [n] format */
461 12 : lBound[ndim] = 1;
462 12 : ub = i;
463 : }
464 222 : if (*p != ']')
465 0 : ereturn(escontext, false,
466 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
467 : errmsg("malformed array literal: \"%s\"", origStr),
468 : errdetail("Missing \"%s\" after array dimensions.",
469 : "]")));
470 222 : p++;
471 :
472 : /*
473 : * Note: we could accept ub = lb-1 to represent a zero-length
474 : * dimension. However, that would result in an empty array, for which
475 : * we don't keep any dimension data, so that e.g. [1:0] and [101:100]
476 : * would be equivalent. Given the lack of field demand, there seems
477 : * little point in allowing such cases.
478 : */
479 222 : if (ub < lBound[ndim])
480 12 : ereturn(escontext, false,
481 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
482 : errmsg("upper bound cannot be less than lower bound")));
483 :
484 : /* Upper bound of INT_MAX must be disallowed, cf ArrayCheckBounds() */
485 210 : if (ub == INT_MAX)
486 6 : ereturn(escontext, false,
487 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
488 : errmsg("array upper bound is too large: %d", ub)));
489 :
490 : /* Compute "ub - lBound[ndim] + 1", detecting overflow */
491 408 : if (pg_sub_s32_overflow(ub, lBound[ndim], &ub) ||
492 204 : pg_add_s32_overflow(ub, 1, &ub))
493 0 : ereturn(escontext, false,
494 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
495 : errmsg("array size exceeds the maximum allowed (%d)",
496 : (int) MaxArraySize)));
497 :
498 204 : dim[ndim] = ub;
499 204 : ndim++;
500 : }
501 :
502 237170 : *srcptr = p;
503 237170 : *ndim_p = ndim;
504 237170 : return true;
505 : }
506 :
507 : /*
508 : * ReadDimensionInt
509 : * parse an integer, for the array dimensions
510 : *
511 : * On entry, *srcptr points to the string to parse. It is advanced past the
512 : * digits of the integer. If there are no digits, returns true and leaves
513 : * *srcptr unchanged.
514 : *
515 : * Result:
516 : * true for success, false for failure (if escontext is provided).
517 : * On success, the parsed integer is returned in *result.
518 : */
519 : static bool
520 462 : ReadDimensionInt(char **srcptr, int *result,
521 : const char *origStr, Node *escontext)
522 : {
523 462 : char *p = *srcptr;
524 : long l;
525 :
526 : /* don't accept leading whitespace */
527 462 : if (!isdigit((unsigned char) *p) && *p != '-' && *p != '+')
528 : {
529 12 : *result = 0;
530 12 : return true;
531 : }
532 :
533 450 : errno = 0;
534 450 : l = strtol(p, srcptr, 10);
535 :
536 450 : if (errno == ERANGE || l > PG_INT32_MAX || l < PG_INT32_MIN)
537 12 : ereturn(escontext, false,
538 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
539 : errmsg("array bound is out of integer range")));
540 :
541 438 : *result = (int) l;
542 438 : return true;
543 : }
544 :
545 : /*
546 : * ReadArrayStr :
547 : * parses the array string pointed to by *srcptr and converts the values
548 : * to internal format. Determines the array dimensions as it goes.
549 : *
550 : * On entry, *srcptr points to the string to parse (it must point to a '{').
551 : * On successful return, it is advanced to point past the closing '}'.
552 : *
553 : * If dimensions were specified explicitly, they are passed in *ndim_p and
554 : * dim[]. This function will check that the array values match the specified
555 : * dimensions. If dimensions were not given, caller must pass *ndim_p == 0
556 : * and initialize all elements of dim[] to -1. Then this function will
557 : * deduce the dimensions from the structure of the input and store them in
558 : * *ndim_p and the dim[] array.
559 : *
560 : * Element type information:
561 : * inputproc: type-specific input procedure for element datatype.
562 : * typioparam, typmod: auxiliary values to pass to inputproc.
563 : * typdelim: the value delimiter (type-specific).
564 : * typlen, typbyval, typalign: storage parameters of element datatype.
565 : *
566 : * Outputs:
567 : * *ndim_p, dim: dimensions deduced from the input structure.
568 : * *nitems_p: total number of elements.
569 : * *values_p[]: palloc'd array, filled with converted data values.
570 : * *nulls_p[]: palloc'd array, filled with is-null markers.
571 : *
572 : * 'origStr' is the original input string, used only in error messages.
573 : * If *escontext points to an ErrorSaveContext, details of any error are
574 : * reported there.
575 : *
576 : * Result:
577 : * true for success, false for failure (if escontext is provided).
578 : */
579 : static bool
580 237156 : ReadArrayStr(char **srcptr,
581 : FmgrInfo *inputproc,
582 : Oid typioparam,
583 : int32 typmod,
584 : char typdelim,
585 : int typlen,
586 : bool typbyval,
587 : char typalign,
588 : int *ndim_p,
589 : int *dim,
590 : int *nitems_p,
591 : Datum **values_p,
592 : bool **nulls_p,
593 : const char *origStr,
594 : Node *escontext)
595 : {
596 237156 : int ndim = *ndim_p;
597 237156 : bool dimensions_specified = (ndim != 0);
598 : int maxitems;
599 : Datum *values;
600 : bool *nulls;
601 : StringInfoData elembuf;
602 : int nest_level;
603 : int nitems;
604 : bool ndim_frozen;
605 : bool expect_delim;
606 : int nelems[MAXDIM];
607 :
608 : /* Allocate some starting output workspace; we'll enlarge as needed */
609 237156 : maxitems = 16;
610 237156 : values = palloc_array(Datum, maxitems);
611 237156 : nulls = palloc_array(bool, maxitems);
612 :
613 : /* Allocate workspace to hold (string representation of) one element */
614 237156 : initStringInfo(&elembuf);
615 :
616 : /* Loop below assumes first token is ATOK_LEVEL_START */
617 : Assert(**srcptr == '{');
618 :
619 : /* Parse tokens until we reach the matching right brace */
620 237156 : nest_level = 0;
621 237156 : nitems = 0;
622 237156 : ndim_frozen = dimensions_specified;
623 237156 : expect_delim = false;
624 : do
625 : {
626 : ArrayToken tok;
627 :
628 2648652 : tok = ReadArrayToken(srcptr, &elembuf, typdelim, origStr, escontext);
629 :
630 2648622 : switch (tok)
631 : {
632 238646 : case ATOK_LEVEL_START:
633 : /* Can't write left brace where delim is expected */
634 238646 : if (expect_delim)
635 6 : ereturn(escontext, false,
636 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
637 : errmsg("malformed array literal: \"%s\"", origStr),
638 : errdetail("Unexpected \"%c\" character.", '{')));
639 :
640 : /* Initialize element counting in the new level */
641 238640 : if (nest_level >= MAXDIM)
642 2 : ereturn(escontext, false,
643 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
644 : errmsg("number of array dimensions exceeds the maximum allowed (%d)",
645 : MAXDIM)));
646 :
647 238638 : nelems[nest_level] = 0;
648 238638 : nest_level++;
649 238638 : if (nest_level > ndim)
650 : {
651 : /* Can't increase ndim once it's frozen */
652 237544 : if (ndim_frozen)
653 12 : goto dimension_error;
654 237532 : ndim = nest_level;
655 : }
656 238626 : break;
657 :
658 238410 : case ATOK_LEVEL_END:
659 : /* Can't get here with nest_level == 0 */
660 : Assert(nest_level > 0);
661 :
662 : /*
663 : * We allow a right brace to terminate an empty sub-array,
664 : * otherwise it must occur where we expect a delimiter.
665 : */
666 238410 : if (nelems[nest_level - 1] > 0 && !expect_delim)
667 24 : ereturn(escontext, false,
668 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
669 : errmsg("malformed array literal: \"%s\"", origStr),
670 : errdetail("Unexpected \"%c\" character.",
671 : '}')));
672 238386 : nest_level--;
673 : /* Nested sub-arrays count as elements of outer level */
674 238386 : if (nest_level > 0)
675 1416 : nelems[nest_level - 1]++;
676 :
677 : /*
678 : * Note: if we had dimensionality info, then dim[nest_level]
679 : * is initially non-negative, and we'll check each sub-array's
680 : * length against that.
681 : */
682 238386 : if (dim[nest_level] < 0)
683 : {
684 : /* Save length of first sub-array of this level */
685 237304 : dim[nest_level] = nelems[nest_level];
686 : }
687 1082 : else if (nelems[nest_level] != dim[nest_level])
688 : {
689 : /* Subsequent sub-arrays must have same length */
690 32 : goto dimension_error;
691 : }
692 :
693 : /*
694 : * Must have a delim or another right brace following, unless
695 : * we have reached nest_level 0, where this won't matter.
696 : */
697 238354 : expect_delim = true;
698 238354 : break;
699 :
700 969478 : case ATOK_DELIM:
701 969478 : if (!expect_delim)
702 6 : ereturn(escontext, false,
703 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
704 : errmsg("malformed array literal: \"%s\"", origStr),
705 : errdetail("Unexpected \"%c\" character.",
706 : typdelim)));
707 969472 : expect_delim = false;
708 969472 : break;
709 :
710 1202082 : case ATOK_ELEM:
711 : case ATOK_ELEM_NULL:
712 : /* Can't get here with nest_level == 0 */
713 : Assert(nest_level > 0);
714 :
715 : /* Disallow consecutive ELEM tokens */
716 1202082 : if (expect_delim)
717 6 : ereturn(escontext, false,
718 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
719 : errmsg("malformed array literal: \"%s\"", origStr),
720 : errdetail("Unexpected array element.")));
721 :
722 : /* Enlarge the values/nulls arrays if needed */
723 1202076 : if (nitems >= maxitems)
724 : {
725 5808 : if (maxitems >= MaxArraySize)
726 0 : ereturn(escontext, false,
727 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
728 : errmsg("array size exceeds the maximum allowed (%d)",
729 : (int) MaxArraySize)));
730 5808 : maxitems = Min(maxitems * 2, MaxArraySize);
731 5808 : values = repalloc_array(values, Datum, maxitems);
732 5808 : nulls = repalloc_array(nulls, bool, maxitems);
733 : }
734 :
735 : /* Read the element's value, or check that NULL is allowed */
736 1202076 : if (!InputFunctionCallSafe(inputproc,
737 : (tok == ATOK_ELEM_NULL) ? NULL : elembuf.data,
738 : typioparam, typmod,
739 : escontext,
740 1202076 : &values[nitems]))
741 36 : return false;
742 1202020 : nulls[nitems] = (tok == ATOK_ELEM_NULL);
743 1202020 : nitems++;
744 :
745 : /*
746 : * Once we have found an element, the number of dimensions can
747 : * no longer increase, and subsequent elements must all be at
748 : * the same nesting depth.
749 : */
750 1202020 : ndim_frozen = true;
751 1202020 : if (nest_level != ndim)
752 12 : goto dimension_error;
753 : /* Count the new element */
754 1202008 : nelems[nest_level - 1]++;
755 :
756 : /* Must have a delim or a right brace following */
757 1202008 : expect_delim = true;
758 1202008 : break;
759 :
760 6 : case ATOK_ERROR:
761 6 : return false;
762 : }
763 2648460 : } while (nest_level > 0);
764 :
765 : /* Clean up and return results */
766 236964 : pfree(elembuf.data);
767 :
768 236964 : *ndim_p = ndim;
769 236964 : *nitems_p = nitems;
770 236964 : *values_p = values;
771 236964 : *nulls_p = nulls;
772 236964 : return true;
773 :
774 56 : dimension_error:
775 56 : if (dimensions_specified)
776 6 : ereturn(escontext, false,
777 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
778 : errmsg("malformed array literal: \"%s\"", origStr),
779 : errdetail("Specified array dimensions do not match array contents.")));
780 : else
781 50 : ereturn(escontext, false,
782 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
783 : errmsg("malformed array literal: \"%s\"", origStr),
784 : errdetail("Multidimensional arrays must have sub-arrays with matching dimensions.")));
785 : }
786 :
787 : /*
788 : * ReadArrayToken
789 : * read one token from an array value string
790 : *
791 : * Starts scanning from *srcptr. On non-error return, *srcptr is
792 : * advanced past the token.
793 : *
794 : * If the token is ATOK_ELEM, the de-escaped string is returned in elembuf.
795 : */
796 : static ArrayToken
797 2648652 : ReadArrayToken(char **srcptr, StringInfo elembuf, char typdelim,
798 : const char *origStr, Node *escontext)
799 : {
800 2648652 : char *p = *srcptr;
801 : int dstlen;
802 : bool has_escapes;
803 :
804 2648652 : resetStringInfo(elembuf);
805 :
806 : /* Identify token type. Loop advances over leading whitespace. */
807 : for (;;)
808 : {
809 2673192 : switch (*p)
810 : {
811 0 : case '\0':
812 0 : goto ending_error;
813 238646 : case '{':
814 238646 : *srcptr = p + 1;
815 238646 : return ATOK_LEVEL_START;
816 238410 : case '}':
817 238410 : *srcptr = p + 1;
818 238410 : return ATOK_LEVEL_END;
819 564936 : case '"':
820 564936 : p++;
821 564936 : goto quoted_element;
822 1631200 : default:
823 1631200 : if (*p == typdelim)
824 : {
825 969478 : *srcptr = p + 1;
826 969478 : return ATOK_DELIM;
827 : }
828 661722 : if (scanner_isspace(*p))
829 : {
830 24540 : p++;
831 24540 : continue;
832 : }
833 637182 : goto unquoted_element;
834 : }
835 : }
836 :
837 564936 : quoted_element:
838 : for (;;)
839 : {
840 4144960 : switch (*p)
841 : {
842 0 : case '\0':
843 0 : goto ending_error;
844 422 : case '\\':
845 : /* Skip backslash, copy next character as-is. */
846 422 : p++;
847 422 : if (*p == '\0')
848 0 : goto ending_error;
849 422 : appendStringInfoChar(elembuf, *p++);
850 422 : break;
851 564936 : case '"':
852 :
853 : /*
854 : * If next non-whitespace isn't typdelim or a brace, complain
855 : * about incorrect quoting. While we could leave such cases
856 : * to be detected as incorrect token sequences, the resulting
857 : * message wouldn't be as helpful. (We could also give the
858 : * incorrect-quoting error when next is '{', but treating that
859 : * as a token sequence error seems better.)
860 : */
861 564972 : while (*(++p) != '\0')
862 : {
863 564972 : if (*p == typdelim || *p == '}' || *p == '{')
864 : {
865 564918 : *srcptr = p;
866 564918 : return ATOK_ELEM;
867 : }
868 54 : if (!scanner_isspace(*p))
869 18 : ereturn(escontext, ATOK_ERROR,
870 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
871 : errmsg("malformed array literal: \"%s\"", origStr),
872 : errdetail("Incorrectly quoted array element.")));
873 : }
874 0 : goto ending_error;
875 3579602 : default:
876 3579602 : appendStringInfoChar(elembuf, *p++);
877 3579602 : break;
878 : }
879 : }
880 :
881 637182 : unquoted_element:
882 :
883 : /*
884 : * We don't include trailing whitespace in the result. dstlen tracks how
885 : * much of the output string is known to not be trailing whitespace.
886 : */
887 637182 : dstlen = 0;
888 637182 : has_escapes = false;
889 : for (;;)
890 : {
891 3552670 : switch (*p)
892 : {
893 6 : case '\0':
894 6 : goto ending_error;
895 6 : case '{':
896 6 : ereturn(escontext, ATOK_ERROR,
897 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
898 : errmsg("malformed array literal: \"%s\"", origStr),
899 : errdetail("Unexpected \"%c\" character.",
900 : '{')));
901 6 : case '"':
902 : /* Must double-quote all or none of an element. */
903 6 : ereturn(escontext, ATOK_ERROR,
904 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
905 : errmsg("malformed array literal: \"%s\"", origStr),
906 : errdetail("Incorrectly quoted array element.")));
907 18 : case '\\':
908 : /* Skip backslash, copy next character as-is. */
909 18 : p++;
910 18 : if (*p == '\0')
911 0 : goto ending_error;
912 18 : appendStringInfoChar(elembuf, *p++);
913 18 : dstlen = elembuf->len; /* treat it as non-whitespace */
914 18 : has_escapes = true;
915 18 : break;
916 3552634 : default:
917 : /* End of elem? */
918 3552634 : if (*p == typdelim || *p == '}')
919 : {
920 : /* hack: truncate the output string to dstlen */
921 637164 : elembuf->data[dstlen] = '\0';
922 637164 : elembuf->len = dstlen;
923 637164 : *srcptr = p;
924 : /* Check if it's unquoted "NULL" */
925 1274310 : if (Array_nulls && !has_escapes &&
926 637146 : pg_strcasecmp(elembuf->data, "NULL") == 0)
927 534 : return ATOK_ELEM_NULL;
928 : else
929 636630 : return ATOK_ELEM;
930 : }
931 2915470 : appendStringInfoChar(elembuf, *p);
932 2915470 : if (!scanner_isspace(*p))
933 2914478 : dstlen = elembuf->len;
934 2915470 : p++;
935 2915470 : break;
936 : }
937 : }
938 :
939 6 : ending_error:
940 6 : ereturn(escontext, ATOK_ERROR,
941 : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
942 : errmsg("malformed array literal: \"%s\"", origStr),
943 : errdetail("Unexpected end of input.")));
944 : }
945 :
946 : /*
947 : * Copy data into an array object from a temporary array of Datums.
948 : *
949 : * array: array object (with header fields already filled in)
950 : * values: array of Datums to be copied
951 : * nulls: array of is-null flags (can be NULL if no nulls)
952 : * nitems: number of Datums to be copied
953 : * typbyval, typlen, typalign: info about element datatype
954 : * freedata: if true and element type is pass-by-ref, pfree data values
955 : * referenced by Datums after copying them.
956 : *
957 : * If the input data is of varlena type, the caller must have ensured that
958 : * the values are not toasted. (Doing it here doesn't work since the
959 : * caller has already allocated space for the array...)
960 : */
961 : void
962 1793190 : CopyArrayEls(ArrayType *array,
963 : Datum *values,
964 : bool *nulls,
965 : int nitems,
966 : int typlen,
967 : bool typbyval,
968 : char typalign,
969 : bool freedata)
970 : {
971 1793190 : char *p = ARR_DATA_PTR(array);
972 1793190 : bits8 *bitmap = ARR_NULLBITMAP(array);
973 1793190 : int bitval = 0;
974 1793190 : int bitmask = 1;
975 : int i;
976 :
977 1793190 : if (typbyval)
978 1214478 : freedata = false;
979 :
980 12909266 : for (i = 0; i < nitems; i++)
981 : {
982 11116076 : if (nulls && nulls[i])
983 : {
984 33304 : if (!bitmap) /* shouldn't happen */
985 0 : elog(ERROR, "null array element where not supported");
986 : /* bitmap bit stays 0 */
987 : }
988 : else
989 : {
990 11082772 : bitval |= bitmask;
991 11082772 : p += ArrayCastAndSet(values[i], typlen, typbyval, typalign, p);
992 11082772 : if (freedata)
993 720246 : pfree(DatumGetPointer(values[i]));
994 : }
995 11116076 : if (bitmap)
996 : {
997 786266 : bitmask <<= 1;
998 786266 : if (bitmask == 0x100)
999 : {
1000 95140 : *bitmap++ = bitval;
1001 95140 : bitval = 0;
1002 95140 : bitmask = 1;
1003 : }
1004 : }
1005 : }
1006 :
1007 1793190 : if (bitmap && bitmask != 1)
1008 17910 : *bitmap = bitval;
1009 1793190 : }
1010 :
1011 : /*
1012 : * array_out :
1013 : * takes the internal representation of an array and returns a string
1014 : * containing the array in its external format.
1015 : */
1016 : Datum
1017 785724 : array_out(PG_FUNCTION_ARGS)
1018 : {
1019 785724 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1020 785724 : Oid element_type = AARR_ELEMTYPE(v);
1021 : int typlen;
1022 : bool typbyval;
1023 : char typalign;
1024 : char typdelim;
1025 : char *p,
1026 : *tmp,
1027 : *retval,
1028 : **values,
1029 : dims_str[(MAXDIM * 33) + 2];
1030 :
1031 : /*
1032 : * 33 per dim since we assume 15 digits per number + ':' +'[]'
1033 : *
1034 : * +2 allows for assignment operator + trailing null
1035 : */
1036 : bool *needquotes,
1037 785724 : needdims = false;
1038 : size_t overall_length;
1039 : int nitems,
1040 : i,
1041 : j,
1042 : k,
1043 : indx[MAXDIM];
1044 : int ndim,
1045 : *dims,
1046 : *lb;
1047 : array_iter iter;
1048 : ArrayMetaState *my_extra;
1049 :
1050 : /*
1051 : * We arrange to look up info about element type, including its output
1052 : * conversion proc, only once per series of calls, assuming the element
1053 : * type doesn't change underneath us.
1054 : */
1055 785724 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1056 785724 : if (my_extra == NULL)
1057 : {
1058 25456 : fcinfo->flinfo->fn_extra = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
1059 : sizeof(ArrayMetaState));
1060 25456 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1061 25456 : my_extra->element_type = ~element_type;
1062 : }
1063 :
1064 785724 : if (my_extra->element_type != element_type)
1065 : {
1066 : /*
1067 : * Get info about element type, including its output conversion proc
1068 : */
1069 29982 : get_type_io_data(element_type, IOFunc_output,
1070 : &my_extra->typlen, &my_extra->typbyval,
1071 : &my_extra->typalign, &my_extra->typdelim,
1072 : &my_extra->typioparam, &my_extra->typiofunc);
1073 29982 : fmgr_info_cxt(my_extra->typiofunc, &my_extra->proc,
1074 29982 : fcinfo->flinfo->fn_mcxt);
1075 29982 : my_extra->element_type = element_type;
1076 : }
1077 785724 : typlen = my_extra->typlen;
1078 785724 : typbyval = my_extra->typbyval;
1079 785724 : typalign = my_extra->typalign;
1080 785724 : typdelim = my_extra->typdelim;
1081 :
1082 785724 : ndim = AARR_NDIM(v);
1083 785724 : dims = AARR_DIMS(v);
1084 785724 : lb = AARR_LBOUND(v);
1085 785724 : nitems = ArrayGetNItems(ndim, dims);
1086 :
1087 785724 : if (nitems == 0)
1088 : {
1089 3216 : retval = pstrdup("{}");
1090 3216 : PG_RETURN_CSTRING(retval);
1091 : }
1092 :
1093 : /*
1094 : * we will need to add explicit dimensions if any dimension has a lower
1095 : * bound other than one
1096 : */
1097 1565652 : for (i = 0; i < ndim; i++)
1098 : {
1099 783452 : if (lb[i] != 1)
1100 : {
1101 308 : needdims = true;
1102 308 : break;
1103 : }
1104 : }
1105 :
1106 : /*
1107 : * Convert all values to string form, count total space needed (including
1108 : * any overhead such as escaping backslashes), and detect whether each
1109 : * item needs double quotes.
1110 : */
1111 782508 : values = (char **) palloc(nitems * sizeof(char *));
1112 782508 : needquotes = (bool *) palloc(nitems * sizeof(bool));
1113 782508 : overall_length = 0;
1114 :
1115 782508 : array_iter_setup(&iter, v);
1116 :
1117 2876678 : for (i = 0; i < nitems; i++)
1118 : {
1119 : Datum itemvalue;
1120 : bool isnull;
1121 : bool needquote;
1122 :
1123 : /* Get source element, checking for NULL */
1124 2094170 : itemvalue = array_iter_next(&iter, &isnull, i,
1125 : typlen, typbyval, typalign);
1126 :
1127 2094170 : if (isnull)
1128 : {
1129 2348 : values[i] = pstrdup("NULL");
1130 2348 : overall_length += 4;
1131 2348 : needquote = false;
1132 : }
1133 : else
1134 : {
1135 2091822 : values[i] = OutputFunctionCall(&my_extra->proc, itemvalue);
1136 :
1137 : /* count data plus backslashes; detect chars needing quotes */
1138 2091822 : if (values[i][0] == '\0')
1139 538 : needquote = true; /* force quotes for empty string */
1140 2091284 : else if (pg_strcasecmp(values[i], "NULL") == 0)
1141 20 : needquote = true; /* force quotes for literal NULL */
1142 : else
1143 2091264 : needquote = false;
1144 :
1145 31102568 : for (tmp = values[i]; *tmp != '\0'; tmp++)
1146 : {
1147 29010746 : char ch = *tmp;
1148 :
1149 29010746 : overall_length += 1;
1150 29010746 : if (ch == '"' || ch == '\\')
1151 : {
1152 4336 : needquote = true;
1153 4336 : overall_length += 1;
1154 : }
1155 57963398 : else if (ch == '{' || ch == '}' || ch == typdelim ||
1156 28956988 : scanner_isspace(ch))
1157 184866 : needquote = true;
1158 : }
1159 : }
1160 :
1161 2094170 : needquotes[i] = needquote;
1162 :
1163 : /* Count the pair of double quotes, if needed */
1164 2094170 : if (needquote)
1165 26952 : overall_length += 2;
1166 : /* and the comma (or other typdelim delimiter) */
1167 2094170 : overall_length += 1;
1168 : }
1169 :
1170 : /*
1171 : * The very last array element doesn't have a typdelim delimiter after it,
1172 : * but that's OK; that space is needed for the trailing '\0'.
1173 : *
1174 : * Now count total number of curly brace pairs in output string.
1175 : */
1176 1566026 : for (i = j = 0, k = 1; i < ndim; i++)
1177 : {
1178 783518 : j += k, k *= dims[i];
1179 : }
1180 782508 : overall_length += 2 * j;
1181 :
1182 : /* Format explicit dimensions if required */
1183 782508 : dims_str[0] = '\0';
1184 782508 : if (needdims)
1185 : {
1186 308 : char *ptr = dims_str;
1187 :
1188 684 : for (i = 0; i < ndim; i++)
1189 : {
1190 376 : sprintf(ptr, "[%d:%d]", lb[i], lb[i] + dims[i] - 1);
1191 376 : ptr += strlen(ptr);
1192 : }
1193 308 : *ptr++ = *ASSGN;
1194 308 : *ptr = '\0';
1195 308 : overall_length += ptr - dims_str;
1196 : }
1197 :
1198 : /* Now construct the output string */
1199 782508 : retval = (char *) palloc(overall_length);
1200 782508 : p = retval;
1201 :
1202 : #define APPENDSTR(str) (strcpy(p, (str)), p += strlen(p))
1203 : #define APPENDCHAR(ch) (*p++ = (ch), *p = '\0')
1204 :
1205 782508 : if (needdims)
1206 308 : APPENDSTR(dims_str);
1207 782508 : APPENDCHAR('{');
1208 1566026 : for (i = 0; i < ndim; i++)
1209 783518 : indx[i] = 0;
1210 782508 : j = 0;
1211 782508 : k = 0;
1212 : do
1213 : {
1214 2096690 : for (i = j; i < ndim - 1; i++)
1215 2520 : APPENDCHAR('{');
1216 :
1217 2094170 : if (needquotes[k])
1218 : {
1219 26952 : APPENDCHAR('"');
1220 957870 : for (tmp = values[k]; *tmp; tmp++)
1221 : {
1222 930918 : char ch = *tmp;
1223 :
1224 930918 : if (ch == '"' || ch == '\\')
1225 4336 : *p++ = '\\';
1226 930918 : *p++ = ch;
1227 : }
1228 26952 : *p = '\0';
1229 26952 : APPENDCHAR('"');
1230 : }
1231 : else
1232 2067218 : APPENDSTR(values[k]);
1233 2094170 : pfree(values[k++]);
1234 :
1235 2879198 : for (i = ndim - 1; i >= 0; i--)
1236 : {
1237 2096690 : if (++(indx[i]) < dims[i])
1238 : {
1239 1311662 : APPENDCHAR(typdelim);
1240 1311662 : break;
1241 : }
1242 : else
1243 : {
1244 785028 : indx[i] = 0;
1245 785028 : APPENDCHAR('}');
1246 : }
1247 : }
1248 2094170 : j = i;
1249 2094170 : } while (j != -1);
1250 :
1251 : #undef APPENDSTR
1252 : #undef APPENDCHAR
1253 :
1254 : /* Assert that we calculated the string length accurately */
1255 : Assert(overall_length == (p - retval + 1));
1256 :
1257 782508 : pfree(values);
1258 782508 : pfree(needquotes);
1259 :
1260 782508 : PG_RETURN_CSTRING(retval);
1261 : }
1262 :
1263 : /*
1264 : * array_recv :
1265 : * converts an array from the external binary format to
1266 : * its internal format.
1267 : *
1268 : * return value :
1269 : * the internal representation of the input array
1270 : */
1271 : Datum
1272 62 : array_recv(PG_FUNCTION_ARGS)
1273 : {
1274 62 : StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
1275 62 : Oid spec_element_type = PG_GETARG_OID(1); /* type of an array
1276 : * element */
1277 62 : int32 typmod = PG_GETARG_INT32(2); /* typmod for array elements */
1278 : Oid element_type;
1279 : int typlen;
1280 : bool typbyval;
1281 : char typalign;
1282 : Oid typioparam;
1283 : int i,
1284 : nitems;
1285 : Datum *dataPtr;
1286 : bool *nullsPtr;
1287 : bool hasnulls;
1288 : int32 nbytes;
1289 : int32 dataoffset;
1290 : ArrayType *retval;
1291 : int ndim,
1292 : flags,
1293 : dim[MAXDIM],
1294 : lBound[MAXDIM];
1295 : ArrayMetaState *my_extra;
1296 :
1297 : /* Get the array header information */
1298 62 : ndim = pq_getmsgint(buf, 4);
1299 62 : if (ndim < 0) /* we do allow zero-dimension arrays */
1300 0 : ereport(ERROR,
1301 : (errcode(ERRCODE_INVALID_BINARY_REPRESENTATION),
1302 : errmsg("invalid number of dimensions: %d", ndim)));
1303 62 : if (ndim > MAXDIM)
1304 0 : ereport(ERROR,
1305 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
1306 : errmsg("number of array dimensions (%d) exceeds the maximum allowed (%d)",
1307 : ndim, MAXDIM)));
1308 :
1309 62 : flags = pq_getmsgint(buf, 4);
1310 62 : if (flags != 0 && flags != 1)
1311 0 : ereport(ERROR,
1312 : (errcode(ERRCODE_INVALID_BINARY_REPRESENTATION),
1313 : errmsg("invalid array flags")));
1314 :
1315 : /* Check element type recorded in the data */
1316 62 : element_type = pq_getmsgint(buf, sizeof(Oid));
1317 :
1318 : /*
1319 : * From a security standpoint, it doesn't matter whether the input's
1320 : * element type matches what we expect: the element type's receive
1321 : * function has to be robust enough to cope with invalid data. However,
1322 : * from a user-friendliness standpoint, it's nicer to complain about type
1323 : * mismatches than to throw "improper binary format" errors. But there's
1324 : * a problem: only built-in types have OIDs that are stable enough to
1325 : * believe that a mismatch is a real issue. So complain only if both OIDs
1326 : * are in the built-in range. Otherwise, carry on with the element type
1327 : * we "should" be getting.
1328 : */
1329 62 : if (element_type != spec_element_type)
1330 : {
1331 0 : if (element_type < FirstGenbkiObjectId &&
1332 : spec_element_type < FirstGenbkiObjectId)
1333 0 : ereport(ERROR,
1334 : (errcode(ERRCODE_DATATYPE_MISMATCH),
1335 : errmsg("binary data has array element type %u (%s) instead of expected %u (%s)",
1336 : element_type,
1337 : format_type_extended(element_type, -1,
1338 : FORMAT_TYPE_ALLOW_INVALID),
1339 : spec_element_type,
1340 : format_type_extended(spec_element_type, -1,
1341 : FORMAT_TYPE_ALLOW_INVALID))));
1342 0 : element_type = spec_element_type;
1343 : }
1344 :
1345 124 : for (i = 0; i < ndim; i++)
1346 : {
1347 62 : dim[i] = pq_getmsgint(buf, 4);
1348 62 : lBound[i] = pq_getmsgint(buf, 4);
1349 : }
1350 :
1351 : /* This checks for overflow of array dimensions */
1352 62 : nitems = ArrayGetNItems(ndim, dim);
1353 62 : ArrayCheckBounds(ndim, dim, lBound);
1354 :
1355 : /*
1356 : * We arrange to look up info about element type, including its receive
1357 : * conversion proc, only once per series of calls, assuming the element
1358 : * type doesn't change underneath us.
1359 : */
1360 62 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1361 62 : if (my_extra == NULL)
1362 : {
1363 56 : fcinfo->flinfo->fn_extra = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
1364 : sizeof(ArrayMetaState));
1365 56 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1366 56 : my_extra->element_type = ~element_type;
1367 : }
1368 :
1369 62 : if (my_extra->element_type != element_type)
1370 : {
1371 : /* Get info about element type, including its receive proc */
1372 56 : get_type_io_data(element_type, IOFunc_receive,
1373 : &my_extra->typlen, &my_extra->typbyval,
1374 : &my_extra->typalign, &my_extra->typdelim,
1375 : &my_extra->typioparam, &my_extra->typiofunc);
1376 56 : if (!OidIsValid(my_extra->typiofunc))
1377 0 : ereport(ERROR,
1378 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
1379 : errmsg("no binary input function available for type %s",
1380 : format_type_be(element_type))));
1381 56 : fmgr_info_cxt(my_extra->typiofunc, &my_extra->proc,
1382 56 : fcinfo->flinfo->fn_mcxt);
1383 56 : my_extra->element_type = element_type;
1384 : }
1385 :
1386 62 : if (nitems == 0)
1387 : {
1388 : /* Return empty array ... but not till we've validated element_type */
1389 0 : PG_RETURN_ARRAYTYPE_P(construct_empty_array(element_type));
1390 : }
1391 :
1392 62 : typlen = my_extra->typlen;
1393 62 : typbyval = my_extra->typbyval;
1394 62 : typalign = my_extra->typalign;
1395 62 : typioparam = my_extra->typioparam;
1396 :
1397 62 : dataPtr = (Datum *) palloc(nitems * sizeof(Datum));
1398 62 : nullsPtr = (bool *) palloc(nitems * sizeof(bool));
1399 62 : ReadArrayBinary(buf, nitems,
1400 : &my_extra->proc, typioparam, typmod,
1401 : typlen, typbyval, typalign,
1402 : dataPtr, nullsPtr,
1403 : &hasnulls, &nbytes);
1404 62 : if (hasnulls)
1405 : {
1406 0 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nitems);
1407 0 : nbytes += dataoffset;
1408 : }
1409 : else
1410 : {
1411 62 : dataoffset = 0; /* marker for no null bitmap */
1412 62 : nbytes += ARR_OVERHEAD_NONULLS(ndim);
1413 : }
1414 62 : retval = (ArrayType *) palloc0(nbytes);
1415 62 : SET_VARSIZE(retval, nbytes);
1416 62 : retval->ndim = ndim;
1417 62 : retval->dataoffset = dataoffset;
1418 62 : retval->elemtype = element_type;
1419 62 : memcpy(ARR_DIMS(retval), dim, ndim * sizeof(int));
1420 62 : memcpy(ARR_LBOUND(retval), lBound, ndim * sizeof(int));
1421 :
1422 62 : CopyArrayEls(retval,
1423 : dataPtr, nullsPtr, nitems,
1424 : typlen, typbyval, typalign,
1425 : true);
1426 :
1427 62 : pfree(dataPtr);
1428 62 : pfree(nullsPtr);
1429 :
1430 62 : PG_RETURN_ARRAYTYPE_P(retval);
1431 : }
1432 :
1433 : /*
1434 : * ReadArrayBinary:
1435 : * collect the data elements of an array being read in binary style.
1436 : *
1437 : * Inputs:
1438 : * buf: the data buffer to read from.
1439 : * nitems: total number of array elements (already read).
1440 : * receiveproc: type-specific receive procedure for element datatype.
1441 : * typioparam, typmod: auxiliary values to pass to receiveproc.
1442 : * typlen, typbyval, typalign: storage parameters of element datatype.
1443 : *
1444 : * Outputs:
1445 : * values[]: filled with converted data values.
1446 : * nulls[]: filled with is-null markers.
1447 : * *hasnulls: set true iff there are any null elements.
1448 : * *nbytes: set to total size of data area needed (including alignment
1449 : * padding but not including array header overhead).
1450 : *
1451 : * Note that values[] and nulls[] are allocated by the caller, and must have
1452 : * nitems elements.
1453 : */
1454 : static void
1455 62 : ReadArrayBinary(StringInfo buf,
1456 : int nitems,
1457 : FmgrInfo *receiveproc,
1458 : Oid typioparam,
1459 : int32 typmod,
1460 : int typlen,
1461 : bool typbyval,
1462 : char typalign,
1463 : Datum *values,
1464 : bool *nulls,
1465 : bool *hasnulls,
1466 : int32 *nbytes)
1467 : {
1468 : int i;
1469 : bool hasnull;
1470 : int32 totbytes;
1471 :
1472 248 : for (i = 0; i < nitems; i++)
1473 : {
1474 : int itemlen;
1475 : StringInfoData elem_buf;
1476 :
1477 : /* Get and check the item length */
1478 186 : itemlen = pq_getmsgint(buf, 4);
1479 186 : if (itemlen < -1 || itemlen > (buf->len - buf->cursor))
1480 0 : ereport(ERROR,
1481 : (errcode(ERRCODE_INVALID_BINARY_REPRESENTATION),
1482 : errmsg("insufficient data left in message")));
1483 :
1484 186 : if (itemlen == -1)
1485 : {
1486 : /* -1 length means NULL */
1487 0 : values[i] = ReceiveFunctionCall(receiveproc, NULL,
1488 : typioparam, typmod);
1489 0 : nulls[i] = true;
1490 0 : continue;
1491 : }
1492 :
1493 : /*
1494 : * Rather than copying data around, we just initialize a StringInfo
1495 : * pointing to the correct portion of the message buffer.
1496 : */
1497 186 : initReadOnlyStringInfo(&elem_buf, &buf->data[buf->cursor], itemlen);
1498 :
1499 186 : buf->cursor += itemlen;
1500 :
1501 : /* Now call the element's receiveproc */
1502 186 : values[i] = ReceiveFunctionCall(receiveproc, &elem_buf,
1503 : typioparam, typmod);
1504 186 : nulls[i] = false;
1505 :
1506 : /* Trouble if it didn't eat the whole buffer */
1507 186 : if (elem_buf.cursor != itemlen)
1508 0 : ereport(ERROR,
1509 : (errcode(ERRCODE_INVALID_BINARY_REPRESENTATION),
1510 : errmsg("improper binary format in array element %d",
1511 : i + 1)));
1512 : }
1513 :
1514 : /*
1515 : * Check for nulls, compute total data space needed
1516 : */
1517 62 : hasnull = false;
1518 62 : totbytes = 0;
1519 248 : for (i = 0; i < nitems; i++)
1520 : {
1521 186 : if (nulls[i])
1522 0 : hasnull = true;
1523 : else
1524 : {
1525 : /* let's just make sure data is not toasted */
1526 186 : if (typlen == -1)
1527 108 : values[i] = PointerGetDatum(PG_DETOAST_DATUM(values[i]));
1528 186 : totbytes = att_addlength_datum(totbytes, typlen, values[i]);
1529 186 : totbytes = att_align_nominal(totbytes, typalign);
1530 : /* check for overflow of total request */
1531 186 : if (!AllocSizeIsValid(totbytes))
1532 0 : ereport(ERROR,
1533 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
1534 : errmsg("array size exceeds the maximum allowed (%d)",
1535 : (int) MaxAllocSize)));
1536 : }
1537 : }
1538 62 : *hasnulls = hasnull;
1539 62 : *nbytes = totbytes;
1540 62 : }
1541 :
1542 :
1543 : /*
1544 : * array_send :
1545 : * takes the internal representation of an array and returns a bytea
1546 : * containing the array in its external binary format.
1547 : */
1548 : Datum
1549 46 : array_send(PG_FUNCTION_ARGS)
1550 : {
1551 46 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1552 46 : Oid element_type = AARR_ELEMTYPE(v);
1553 : int typlen;
1554 : bool typbyval;
1555 : char typalign;
1556 : int nitems,
1557 : i;
1558 : int ndim,
1559 : *dim,
1560 : *lb;
1561 : StringInfoData buf;
1562 : array_iter iter;
1563 : ArrayMetaState *my_extra;
1564 :
1565 : /*
1566 : * We arrange to look up info about element type, including its send
1567 : * conversion proc, only once per series of calls, assuming the element
1568 : * type doesn't change underneath us.
1569 : */
1570 46 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1571 46 : if (my_extra == NULL)
1572 : {
1573 40 : fcinfo->flinfo->fn_extra = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
1574 : sizeof(ArrayMetaState));
1575 40 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
1576 40 : my_extra->element_type = ~element_type;
1577 : }
1578 :
1579 46 : if (my_extra->element_type != element_type)
1580 : {
1581 : /* Get info about element type, including its send proc */
1582 40 : get_type_io_data(element_type, IOFunc_send,
1583 : &my_extra->typlen, &my_extra->typbyval,
1584 : &my_extra->typalign, &my_extra->typdelim,
1585 : &my_extra->typioparam, &my_extra->typiofunc);
1586 40 : if (!OidIsValid(my_extra->typiofunc))
1587 0 : ereport(ERROR,
1588 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
1589 : errmsg("no binary output function available for type %s",
1590 : format_type_be(element_type))));
1591 40 : fmgr_info_cxt(my_extra->typiofunc, &my_extra->proc,
1592 40 : fcinfo->flinfo->fn_mcxt);
1593 40 : my_extra->element_type = element_type;
1594 : }
1595 46 : typlen = my_extra->typlen;
1596 46 : typbyval = my_extra->typbyval;
1597 46 : typalign = my_extra->typalign;
1598 :
1599 46 : ndim = AARR_NDIM(v);
1600 46 : dim = AARR_DIMS(v);
1601 46 : lb = AARR_LBOUND(v);
1602 46 : nitems = ArrayGetNItems(ndim, dim);
1603 :
1604 46 : pq_begintypsend(&buf);
1605 :
1606 : /* Send the array header information */
1607 46 : pq_sendint32(&buf, ndim);
1608 46 : pq_sendint32(&buf, AARR_HASNULL(v) ? 1 : 0);
1609 46 : pq_sendint32(&buf, element_type);
1610 92 : for (i = 0; i < ndim; i++)
1611 : {
1612 46 : pq_sendint32(&buf, dim[i]);
1613 46 : pq_sendint32(&buf, lb[i]);
1614 : }
1615 :
1616 : /* Send the array elements using the element's own sendproc */
1617 46 : array_iter_setup(&iter, v);
1618 :
1619 184 : for (i = 0; i < nitems; i++)
1620 : {
1621 : Datum itemvalue;
1622 : bool isnull;
1623 :
1624 : /* Get source element, checking for NULL */
1625 138 : itemvalue = array_iter_next(&iter, &isnull, i,
1626 : typlen, typbyval, typalign);
1627 :
1628 138 : if (isnull)
1629 : {
1630 : /* -1 length means a NULL */
1631 0 : pq_sendint32(&buf, -1);
1632 : }
1633 : else
1634 : {
1635 : bytea *outputbytes;
1636 :
1637 138 : outputbytes = SendFunctionCall(&my_extra->proc, itemvalue);
1638 138 : pq_sendint32(&buf, VARSIZE(outputbytes) - VARHDRSZ);
1639 138 : pq_sendbytes(&buf, VARDATA(outputbytes),
1640 138 : VARSIZE(outputbytes) - VARHDRSZ);
1641 138 : pfree(outputbytes);
1642 : }
1643 : }
1644 :
1645 46 : PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
1646 : }
1647 :
1648 : /*
1649 : * array_ndims :
1650 : * returns the number of dimensions of the array pointed to by "v"
1651 : */
1652 : Datum
1653 1938 : array_ndims(PG_FUNCTION_ARGS)
1654 : {
1655 1938 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1656 :
1657 : /* Sanity check: does it look like an array at all? */
1658 1938 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
1659 12 : PG_RETURN_NULL();
1660 :
1661 1926 : PG_RETURN_INT32(AARR_NDIM(v));
1662 : }
1663 :
1664 : /*
1665 : * array_dims :
1666 : * returns the dimensions of the array pointed to by "v", as a "text"
1667 : */
1668 : Datum
1669 8924 : array_dims(PG_FUNCTION_ARGS)
1670 : {
1671 8924 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1672 : char *p;
1673 : int i;
1674 : int *dimv,
1675 : *lb;
1676 :
1677 : /*
1678 : * 33 since we assume 15 digits per number + ':' +'[]'
1679 : *
1680 : * +1 for trailing null
1681 : */
1682 : char buf[MAXDIM * 33 + 1];
1683 :
1684 : /* Sanity check: does it look like an array at all? */
1685 8924 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
1686 68 : PG_RETURN_NULL();
1687 :
1688 8856 : dimv = AARR_DIMS(v);
1689 8856 : lb = AARR_LBOUND(v);
1690 :
1691 8856 : p = buf;
1692 17820 : for (i = 0; i < AARR_NDIM(v); i++)
1693 : {
1694 8964 : sprintf(p, "[%d:%d]", lb[i], dimv[i] + lb[i] - 1);
1695 8964 : p += strlen(p);
1696 : }
1697 :
1698 8856 : PG_RETURN_TEXT_P(cstring_to_text(buf));
1699 : }
1700 :
1701 : /*
1702 : * array_lower :
1703 : * returns the lower dimension, of the DIM requested, for
1704 : * the array pointed to by "v", as an int4
1705 : */
1706 : Datum
1707 25758 : array_lower(PG_FUNCTION_ARGS)
1708 : {
1709 25758 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1710 25758 : int reqdim = PG_GETARG_INT32(1);
1711 : int *lb;
1712 : int result;
1713 :
1714 : /* Sanity check: does it look like an array at all? */
1715 25758 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
1716 0 : PG_RETURN_NULL();
1717 :
1718 : /* Sanity check: was the requested dim valid */
1719 25758 : if (reqdim <= 0 || reqdim > AARR_NDIM(v))
1720 0 : PG_RETURN_NULL();
1721 :
1722 25758 : lb = AARR_LBOUND(v);
1723 25758 : result = lb[reqdim - 1];
1724 :
1725 25758 : PG_RETURN_INT32(result);
1726 : }
1727 :
1728 : /*
1729 : * array_upper :
1730 : * returns the upper dimension, of the DIM requested, for
1731 : * the array pointed to by "v", as an int4
1732 : */
1733 : Datum
1734 26448 : array_upper(PG_FUNCTION_ARGS)
1735 : {
1736 26448 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1737 26448 : int reqdim = PG_GETARG_INT32(1);
1738 : int *dimv,
1739 : *lb;
1740 : int result;
1741 :
1742 : /* Sanity check: does it look like an array at all? */
1743 26448 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
1744 30 : PG_RETURN_NULL();
1745 :
1746 : /* Sanity check: was the requested dim valid */
1747 26418 : if (reqdim <= 0 || reqdim > AARR_NDIM(v))
1748 0 : PG_RETURN_NULL();
1749 :
1750 26418 : lb = AARR_LBOUND(v);
1751 26418 : dimv = AARR_DIMS(v);
1752 :
1753 26418 : result = dimv[reqdim - 1] + lb[reqdim - 1] - 1;
1754 :
1755 26418 : PG_RETURN_INT32(result);
1756 : }
1757 :
1758 : /*
1759 : * array_length :
1760 : * returns the length, of the dimension requested, for
1761 : * the array pointed to by "v", as an int4
1762 : */
1763 : Datum
1764 112886 : array_length(PG_FUNCTION_ARGS)
1765 : {
1766 112886 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1767 112886 : int reqdim = PG_GETARG_INT32(1);
1768 : int *dimv;
1769 : int result;
1770 :
1771 : /* Sanity check: does it look like an array at all? */
1772 112886 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
1773 0 : PG_RETURN_NULL();
1774 :
1775 : /* Sanity check: was the requested dim valid */
1776 112886 : if (reqdim <= 0 || reqdim > AARR_NDIM(v))
1777 12 : PG_RETURN_NULL();
1778 :
1779 112874 : dimv = AARR_DIMS(v);
1780 :
1781 112874 : result = dimv[reqdim - 1];
1782 :
1783 112874 : PG_RETURN_INT32(result);
1784 : }
1785 :
1786 : /*
1787 : * array_cardinality:
1788 : * returns the total number of elements in an array
1789 : */
1790 : Datum
1791 2532 : array_cardinality(PG_FUNCTION_ARGS)
1792 : {
1793 2532 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
1794 :
1795 2532 : PG_RETURN_INT32(ArrayGetNItems(AARR_NDIM(v), AARR_DIMS(v)));
1796 : }
1797 :
1798 :
1799 : /*
1800 : * array_get_element :
1801 : * This routine takes an array datum and a subscript array and returns
1802 : * the referenced item as a Datum. Note that for a pass-by-reference
1803 : * datatype, the returned Datum is a pointer into the array object.
1804 : *
1805 : * This handles both ordinary varlena arrays and fixed-length arrays.
1806 : *
1807 : * Inputs:
1808 : * arraydatum: the array object (mustn't be NULL)
1809 : * nSubscripts: number of subscripts supplied
1810 : * indx[]: the subscript values
1811 : * arraytyplen: pg_type.typlen for the array type
1812 : * elmlen: pg_type.typlen for the array's element type
1813 : * elmbyval: pg_type.typbyval for the array's element type
1814 : * elmalign: pg_type.typalign for the array's element type
1815 : *
1816 : * Outputs:
1817 : * The return value is the element Datum.
1818 : * *isNull is set to indicate whether the element is NULL.
1819 : */
1820 : Datum
1821 856092 : array_get_element(Datum arraydatum,
1822 : int nSubscripts,
1823 : int *indx,
1824 : int arraytyplen,
1825 : int elmlen,
1826 : bool elmbyval,
1827 : char elmalign,
1828 : bool *isNull)
1829 : {
1830 : int i,
1831 : ndim,
1832 : *dim,
1833 : *lb,
1834 : offset,
1835 : fixedDim[1],
1836 : fixedLb[1];
1837 : char *arraydataptr,
1838 : *retptr;
1839 : bits8 *arraynullsptr;
1840 :
1841 856092 : if (arraytyplen > 0)
1842 : {
1843 : /*
1844 : * fixed-length arrays -- these are assumed to be 1-d, 0-based
1845 : */
1846 343438 : ndim = 1;
1847 343438 : fixedDim[0] = arraytyplen / elmlen;
1848 343438 : fixedLb[0] = 0;
1849 343438 : dim = fixedDim;
1850 343438 : lb = fixedLb;
1851 343438 : arraydataptr = (char *) DatumGetPointer(arraydatum);
1852 343438 : arraynullsptr = NULL;
1853 : }
1854 512654 : else if (VARATT_IS_EXTERNAL_EXPANDED(DatumGetPointer(arraydatum)))
1855 : {
1856 : /* expanded array: let's do this in a separate function */
1857 5014 : return array_get_element_expanded(arraydatum,
1858 : nSubscripts,
1859 : indx,
1860 : arraytyplen,
1861 : elmlen,
1862 : elmbyval,
1863 : elmalign,
1864 : isNull);
1865 : }
1866 : else
1867 : {
1868 : /* detoast array if necessary, producing normal varlena input */
1869 507640 : ArrayType *array = DatumGetArrayTypeP(arraydatum);
1870 :
1871 507640 : ndim = ARR_NDIM(array);
1872 507640 : dim = ARR_DIMS(array);
1873 507640 : lb = ARR_LBOUND(array);
1874 507640 : arraydataptr = ARR_DATA_PTR(array);
1875 507640 : arraynullsptr = ARR_NULLBITMAP(array);
1876 : }
1877 :
1878 : /*
1879 : * Return NULL for invalid subscript
1880 : */
1881 851078 : if (ndim != nSubscripts || ndim <= 0 || ndim > MAXDIM)
1882 : {
1883 96 : *isNull = true;
1884 96 : return (Datum) 0;
1885 : }
1886 1648834 : for (i = 0; i < ndim; i++)
1887 : {
1888 851078 : if (indx[i] < lb[i] || indx[i] >= (dim[i] + lb[i]))
1889 : {
1890 53226 : *isNull = true;
1891 53226 : return (Datum) 0;
1892 : }
1893 : }
1894 :
1895 : /*
1896 : * Calculate the element number
1897 : */
1898 797756 : offset = ArrayGetOffset(nSubscripts, dim, lb, indx);
1899 :
1900 : /*
1901 : * Check for NULL array element
1902 : */
1903 797756 : if (array_get_isnull(arraynullsptr, offset))
1904 : {
1905 72 : *isNull = true;
1906 72 : return (Datum) 0;
1907 : }
1908 :
1909 : /*
1910 : * OK, get the element
1911 : */
1912 797684 : *isNull = false;
1913 797684 : retptr = array_seek(arraydataptr, 0, arraynullsptr, offset,
1914 : elmlen, elmbyval, elmalign);
1915 797684 : return ArrayCast(retptr, elmbyval, elmlen);
1916 : }
1917 :
1918 : /*
1919 : * Implementation of array_get_element() for an expanded array
1920 : */
1921 : static Datum
1922 5014 : array_get_element_expanded(Datum arraydatum,
1923 : int nSubscripts, int *indx,
1924 : int arraytyplen,
1925 : int elmlen, bool elmbyval, char elmalign,
1926 : bool *isNull)
1927 : {
1928 : ExpandedArrayHeader *eah;
1929 : int i,
1930 : ndim,
1931 : *dim,
1932 : *lb,
1933 : offset;
1934 : Datum *dvalues;
1935 : bool *dnulls;
1936 :
1937 5014 : eah = (ExpandedArrayHeader *) DatumGetEOHP(arraydatum);
1938 : Assert(eah->ea_magic == EA_MAGIC);
1939 :
1940 : /* sanity-check caller's info against object */
1941 : Assert(arraytyplen == -1);
1942 : Assert(elmlen == eah->typlen);
1943 : Assert(elmbyval == eah->typbyval);
1944 : Assert(elmalign == eah->typalign);
1945 :
1946 5014 : ndim = eah->ndims;
1947 5014 : dim = eah->dims;
1948 5014 : lb = eah->lbound;
1949 :
1950 : /*
1951 : * Return NULL for invalid subscript
1952 : */
1953 5014 : if (ndim != nSubscripts || ndim <= 0 || ndim > MAXDIM)
1954 : {
1955 6 : *isNull = true;
1956 6 : return (Datum) 0;
1957 : }
1958 10016 : for (i = 0; i < ndim; i++)
1959 : {
1960 5008 : if (indx[i] < lb[i] || indx[i] >= (dim[i] + lb[i]))
1961 : {
1962 0 : *isNull = true;
1963 0 : return (Datum) 0;
1964 : }
1965 : }
1966 :
1967 : /*
1968 : * Calculate the element number
1969 : */
1970 5008 : offset = ArrayGetOffset(nSubscripts, dim, lb, indx);
1971 :
1972 : /*
1973 : * Deconstruct array if we didn't already. Note that we apply this even
1974 : * if the input is nominally read-only: it should be safe enough.
1975 : */
1976 5008 : deconstruct_expanded_array(eah);
1977 :
1978 5008 : dvalues = eah->dvalues;
1979 5008 : dnulls = eah->dnulls;
1980 :
1981 : /*
1982 : * Check for NULL array element
1983 : */
1984 5008 : if (dnulls && dnulls[offset])
1985 : {
1986 0 : *isNull = true;
1987 0 : return (Datum) 0;
1988 : }
1989 :
1990 : /*
1991 : * OK, get the element. It's OK to return a pass-by-ref value as a
1992 : * pointer into the expanded array, for the same reason that regular
1993 : * array_get_element can return a pointer into flat arrays: the value is
1994 : * assumed not to change for as long as the Datum reference can exist.
1995 : */
1996 5008 : *isNull = false;
1997 5008 : return dvalues[offset];
1998 : }
1999 :
2000 : /*
2001 : * array_get_slice :
2002 : * This routine takes an array and a range of indices (upperIndx and
2003 : * lowerIndx), creates a new array structure for the referred elements
2004 : * and returns a pointer to it.
2005 : *
2006 : * This handles both ordinary varlena arrays and fixed-length arrays.
2007 : *
2008 : * Inputs:
2009 : * arraydatum: the array object (mustn't be NULL)
2010 : * nSubscripts: number of subscripts supplied (must be same for upper/lower)
2011 : * upperIndx[]: the upper subscript values
2012 : * lowerIndx[]: the lower subscript values
2013 : * upperProvided[]: true for provided upper subscript values
2014 : * lowerProvided[]: true for provided lower subscript values
2015 : * arraytyplen: pg_type.typlen for the array type
2016 : * elmlen: pg_type.typlen for the array's element type
2017 : * elmbyval: pg_type.typbyval for the array's element type
2018 : * elmalign: pg_type.typalign for the array's element type
2019 : *
2020 : * Outputs:
2021 : * The return value is the new array Datum (it's never NULL)
2022 : *
2023 : * Omitted upper and lower subscript values are replaced by the corresponding
2024 : * array bound.
2025 : *
2026 : * NOTE: we assume it is OK to scribble on the provided subscript arrays
2027 : * lowerIndx[] and upperIndx[]; also, these arrays must be of size MAXDIM
2028 : * even when nSubscripts is less. These are generally just temporaries.
2029 : */
2030 : Datum
2031 396 : array_get_slice(Datum arraydatum,
2032 : int nSubscripts,
2033 : int *upperIndx,
2034 : int *lowerIndx,
2035 : bool *upperProvided,
2036 : bool *lowerProvided,
2037 : int arraytyplen,
2038 : int elmlen,
2039 : bool elmbyval,
2040 : char elmalign)
2041 : {
2042 : ArrayType *array;
2043 : ArrayType *newarray;
2044 : int i,
2045 : ndim,
2046 : *dim,
2047 : *lb,
2048 : *newlb;
2049 : int fixedDim[1],
2050 : fixedLb[1];
2051 : Oid elemtype;
2052 : char *arraydataptr;
2053 : bits8 *arraynullsptr;
2054 : int32 dataoffset;
2055 : int bytes,
2056 : span[MAXDIM];
2057 :
2058 396 : if (arraytyplen > 0)
2059 : {
2060 : /*
2061 : * fixed-length arrays -- currently, cannot slice these because parser
2062 : * labels output as being of the fixed-length array type! Code below
2063 : * shows how we could support it if the parser were changed to label
2064 : * output as a suitable varlena array type.
2065 : */
2066 24 : ereport(ERROR,
2067 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
2068 : errmsg("slices of fixed-length arrays not implemented")));
2069 :
2070 : /*
2071 : * fixed-length arrays -- these are assumed to be 1-d, 0-based
2072 : *
2073 : * XXX where would we get the correct ELEMTYPE from?
2074 : */
2075 : ndim = 1;
2076 : fixedDim[0] = arraytyplen / elmlen;
2077 : fixedLb[0] = 0;
2078 : dim = fixedDim;
2079 : lb = fixedLb;
2080 : elemtype = InvalidOid; /* XXX */
2081 : arraydataptr = (char *) DatumGetPointer(arraydatum);
2082 : arraynullsptr = NULL;
2083 : }
2084 : else
2085 : {
2086 : /* detoast input array if necessary */
2087 372 : array = DatumGetArrayTypeP(arraydatum);
2088 :
2089 372 : ndim = ARR_NDIM(array);
2090 372 : dim = ARR_DIMS(array);
2091 372 : lb = ARR_LBOUND(array);
2092 372 : elemtype = ARR_ELEMTYPE(array);
2093 372 : arraydataptr = ARR_DATA_PTR(array);
2094 372 : arraynullsptr = ARR_NULLBITMAP(array);
2095 : }
2096 :
2097 : /*
2098 : * Check provided subscripts. A slice exceeding the current array limits
2099 : * is silently truncated to the array limits. If we end up with an empty
2100 : * slice, return an empty array.
2101 : */
2102 372 : if (ndim < nSubscripts || ndim <= 0 || ndim > MAXDIM)
2103 96 : return PointerGetDatum(construct_empty_array(elemtype));
2104 :
2105 636 : for (i = 0; i < nSubscripts; i++)
2106 : {
2107 372 : if (!lowerProvided[i] || lowerIndx[i] < lb[i])
2108 126 : lowerIndx[i] = lb[i];
2109 372 : if (!upperProvided[i] || upperIndx[i] >= (dim[i] + lb[i]))
2110 72 : upperIndx[i] = dim[i] + lb[i] - 1;
2111 372 : if (lowerIndx[i] > upperIndx[i])
2112 12 : return PointerGetDatum(construct_empty_array(elemtype));
2113 : }
2114 : /* fill any missing subscript positions with full array range */
2115 312 : for (; i < ndim; i++)
2116 : {
2117 48 : lowerIndx[i] = lb[i];
2118 48 : upperIndx[i] = dim[i] + lb[i] - 1;
2119 48 : if (lowerIndx[i] > upperIndx[i])
2120 0 : return PointerGetDatum(construct_empty_array(elemtype));
2121 : }
2122 :
2123 264 : mda_get_range(ndim, span, lowerIndx, upperIndx);
2124 :
2125 264 : bytes = array_slice_size(arraydataptr, arraynullsptr,
2126 : ndim, dim, lb,
2127 : lowerIndx, upperIndx,
2128 : elmlen, elmbyval, elmalign);
2129 :
2130 : /*
2131 : * Currently, we put a null bitmap in the result if the source has one;
2132 : * could be smarter ...
2133 : */
2134 264 : if (arraynullsptr)
2135 : {
2136 36 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, ArrayGetNItems(ndim, span));
2137 36 : bytes += dataoffset;
2138 : }
2139 : else
2140 : {
2141 228 : dataoffset = 0; /* marker for no null bitmap */
2142 228 : bytes += ARR_OVERHEAD_NONULLS(ndim);
2143 : }
2144 :
2145 264 : newarray = (ArrayType *) palloc0(bytes);
2146 264 : SET_VARSIZE(newarray, bytes);
2147 264 : newarray->ndim = ndim;
2148 264 : newarray->dataoffset = dataoffset;
2149 264 : newarray->elemtype = elemtype;
2150 264 : memcpy(ARR_DIMS(newarray), span, ndim * sizeof(int));
2151 :
2152 : /*
2153 : * Lower bounds of the new array are set to 1. Formerly (before 7.3) we
2154 : * copied the given lowerIndx values ... but that seems confusing.
2155 : */
2156 264 : newlb = ARR_LBOUND(newarray);
2157 672 : for (i = 0; i < ndim; i++)
2158 408 : newlb[i] = 1;
2159 :
2160 264 : array_extract_slice(newarray,
2161 : ndim, dim, lb,
2162 : arraydataptr, arraynullsptr,
2163 : lowerIndx, upperIndx,
2164 : elmlen, elmbyval, elmalign);
2165 :
2166 264 : return PointerGetDatum(newarray);
2167 : }
2168 :
2169 : /*
2170 : * array_set_element :
2171 : * This routine sets the value of one array element (specified by
2172 : * a subscript array) to a new value specified by "dataValue".
2173 : *
2174 : * This handles both ordinary varlena arrays and fixed-length arrays.
2175 : *
2176 : * Inputs:
2177 : * arraydatum: the initial array object (mustn't be NULL)
2178 : * nSubscripts: number of subscripts supplied
2179 : * indx[]: the subscript values
2180 : * dataValue: the datum to be inserted at the given position
2181 : * isNull: whether dataValue is NULL
2182 : * arraytyplen: pg_type.typlen for the array type
2183 : * elmlen: pg_type.typlen for the array's element type
2184 : * elmbyval: pg_type.typbyval for the array's element type
2185 : * elmalign: pg_type.typalign for the array's element type
2186 : *
2187 : * Result:
2188 : * A new array is returned, just like the old except for the one
2189 : * modified entry. The original array object is not changed,
2190 : * unless what is passed is a read-write reference to an expanded
2191 : * array object; in that case the expanded array is updated in-place.
2192 : *
2193 : * For one-dimensional arrays only, we allow the array to be extended
2194 : * by assigning to a position outside the existing subscript range; any
2195 : * positions between the existing elements and the new one are set to NULLs.
2196 : * (XXX TODO: allow a corresponding behavior for multidimensional arrays)
2197 : *
2198 : * NOTE: For assignments, we throw an error for invalid subscripts etc,
2199 : * rather than returning a NULL as the fetch operations do.
2200 : */
2201 : Datum
2202 4150 : array_set_element(Datum arraydatum,
2203 : int nSubscripts,
2204 : int *indx,
2205 : Datum dataValue,
2206 : bool isNull,
2207 : int arraytyplen,
2208 : int elmlen,
2209 : bool elmbyval,
2210 : char elmalign)
2211 : {
2212 : ArrayType *array;
2213 : ArrayType *newarray;
2214 : int i,
2215 : ndim,
2216 : dim[MAXDIM],
2217 : lb[MAXDIM],
2218 : offset;
2219 : char *elt_ptr;
2220 : bool newhasnulls;
2221 : bits8 *oldnullbitmap;
2222 : int oldnitems,
2223 : newnitems,
2224 : olddatasize,
2225 : newsize,
2226 : olditemlen,
2227 : newitemlen,
2228 : overheadlen,
2229 : oldoverheadlen,
2230 : addedbefore,
2231 : addedafter,
2232 : lenbefore,
2233 : lenafter;
2234 :
2235 4150 : if (arraytyplen > 0)
2236 : {
2237 : /*
2238 : * fixed-length arrays -- these are assumed to be 1-d, 0-based. We
2239 : * cannot extend them, either.
2240 : */
2241 : char *resultarray;
2242 :
2243 18 : if (nSubscripts != 1)
2244 0 : ereport(ERROR,
2245 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2246 : errmsg("wrong number of array subscripts")));
2247 :
2248 18 : if (indx[0] < 0 || indx[0] >= arraytyplen / elmlen)
2249 6 : ereport(ERROR,
2250 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2251 : errmsg("array subscript out of range")));
2252 :
2253 12 : if (isNull)
2254 0 : ereport(ERROR,
2255 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
2256 : errmsg("cannot assign null value to an element of a fixed-length array")));
2257 :
2258 12 : resultarray = (char *) palloc(arraytyplen);
2259 12 : memcpy(resultarray, DatumGetPointer(arraydatum), arraytyplen);
2260 12 : elt_ptr = (char *) resultarray + indx[0] * elmlen;
2261 12 : ArrayCastAndSet(dataValue, elmlen, elmbyval, elmalign, elt_ptr);
2262 12 : return PointerGetDatum(resultarray);
2263 : }
2264 :
2265 4132 : if (nSubscripts <= 0 || nSubscripts > MAXDIM)
2266 0 : ereport(ERROR,
2267 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2268 : errmsg("wrong number of array subscripts")));
2269 :
2270 : /* make sure item to be inserted is not toasted */
2271 4132 : if (elmlen == -1 && !isNull)
2272 2598 : dataValue = PointerGetDatum(PG_DETOAST_DATUM(dataValue));
2273 :
2274 4132 : if (VARATT_IS_EXTERNAL_EXPANDED(DatumGetPointer(arraydatum)))
2275 : {
2276 : /* expanded array: let's do this in a separate function */
2277 2016 : return array_set_element_expanded(arraydatum,
2278 : nSubscripts,
2279 : indx,
2280 : dataValue,
2281 : isNull,
2282 : arraytyplen,
2283 : elmlen,
2284 : elmbyval,
2285 : elmalign);
2286 : }
2287 :
2288 : /* detoast input array if necessary */
2289 2116 : array = DatumGetArrayTypeP(arraydatum);
2290 :
2291 2116 : ndim = ARR_NDIM(array);
2292 :
2293 : /*
2294 : * if number of dims is zero, i.e. an empty array, create an array with
2295 : * nSubscripts dimensions, and set the lower bounds to the supplied
2296 : * subscripts
2297 : */
2298 2116 : if (ndim == 0)
2299 : {
2300 348 : Oid elmtype = ARR_ELEMTYPE(array);
2301 :
2302 698 : for (i = 0; i < nSubscripts; i++)
2303 : {
2304 350 : dim[i] = 1;
2305 350 : lb[i] = indx[i];
2306 : }
2307 :
2308 348 : return PointerGetDatum(construct_md_array(&dataValue, &isNull,
2309 : nSubscripts, dim, lb,
2310 : elmtype,
2311 : elmlen, elmbyval, elmalign));
2312 : }
2313 :
2314 1768 : if (ndim != nSubscripts)
2315 6 : ereport(ERROR,
2316 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2317 : errmsg("wrong number of array subscripts")));
2318 :
2319 : /* copy dim/lb since we may modify them */
2320 1762 : memcpy(dim, ARR_DIMS(array), ndim * sizeof(int));
2321 1762 : memcpy(lb, ARR_LBOUND(array), ndim * sizeof(int));
2322 :
2323 1762 : newhasnulls = (ARR_HASNULL(array) || isNull);
2324 1762 : addedbefore = addedafter = 0;
2325 :
2326 : /*
2327 : * Check subscripts. We assume the existing subscripts passed
2328 : * ArrayCheckBounds, so that dim[i] + lb[i] can be computed without
2329 : * overflow. But we must beware of other overflows in our calculations of
2330 : * new dim[] values.
2331 : */
2332 1762 : if (ndim == 1)
2333 : {
2334 1756 : if (indx[0] < lb[0])
2335 : {
2336 : /* addedbefore = lb[0] - indx[0]; */
2337 : /* dim[0] += addedbefore; */
2338 48 : if (pg_sub_s32_overflow(lb[0], indx[0], &addedbefore) ||
2339 24 : pg_add_s32_overflow(dim[0], addedbefore, &dim[0]))
2340 0 : ereport(ERROR,
2341 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2342 : errmsg("array size exceeds the maximum allowed (%d)",
2343 : (int) MaxArraySize)));
2344 24 : lb[0] = indx[0];
2345 24 : if (addedbefore > 1)
2346 12 : newhasnulls = true; /* will insert nulls */
2347 : }
2348 1756 : if (indx[0] >= (dim[0] + lb[0]))
2349 : {
2350 : /* addedafter = indx[0] - (dim[0] + lb[0]) + 1; */
2351 : /* dim[0] += addedafter; */
2352 2770 : if (pg_sub_s32_overflow(indx[0], dim[0] + lb[0], &addedafter) ||
2353 2764 : pg_add_s32_overflow(addedafter, 1, &addedafter) ||
2354 1382 : pg_add_s32_overflow(dim[0], addedafter, &dim[0]))
2355 6 : ereport(ERROR,
2356 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2357 : errmsg("array size exceeds the maximum allowed (%d)",
2358 : (int) MaxArraySize)));
2359 1382 : if (addedafter > 1)
2360 36 : newhasnulls = true; /* will insert nulls */
2361 : }
2362 : }
2363 : else
2364 : {
2365 : /*
2366 : * XXX currently we do not support extending multi-dimensional arrays
2367 : * during assignment
2368 : */
2369 18 : for (i = 0; i < ndim; i++)
2370 : {
2371 12 : if (indx[i] < lb[i] ||
2372 12 : indx[i] >= (dim[i] + lb[i]))
2373 0 : ereport(ERROR,
2374 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2375 : errmsg("array subscript out of range")));
2376 : }
2377 : }
2378 :
2379 : /* This checks for overflow of the array dimensions */
2380 1756 : newnitems = ArrayGetNItems(ndim, dim);
2381 1756 : ArrayCheckBounds(ndim, dim, lb);
2382 :
2383 : /*
2384 : * Compute sizes of items and areas to copy
2385 : */
2386 1756 : if (newhasnulls)
2387 134 : overheadlen = ARR_OVERHEAD_WITHNULLS(ndim, newnitems);
2388 : else
2389 1622 : overheadlen = ARR_OVERHEAD_NONULLS(ndim);
2390 1756 : oldnitems = ArrayGetNItems(ndim, ARR_DIMS(array));
2391 1756 : oldnullbitmap = ARR_NULLBITMAP(array);
2392 1756 : oldoverheadlen = ARR_DATA_OFFSET(array);
2393 1756 : olddatasize = ARR_SIZE(array) - oldoverheadlen;
2394 1756 : if (addedbefore)
2395 : {
2396 24 : offset = 0;
2397 24 : lenbefore = 0;
2398 24 : olditemlen = 0;
2399 24 : lenafter = olddatasize;
2400 : }
2401 1732 : else if (addedafter)
2402 : {
2403 1382 : offset = oldnitems;
2404 1382 : lenbefore = olddatasize;
2405 1382 : olditemlen = 0;
2406 1382 : lenafter = 0;
2407 : }
2408 : else
2409 : {
2410 350 : offset = ArrayGetOffset(nSubscripts, dim, lb, indx);
2411 350 : elt_ptr = array_seek(ARR_DATA_PTR(array), 0, oldnullbitmap, offset,
2412 : elmlen, elmbyval, elmalign);
2413 350 : lenbefore = (int) (elt_ptr - ARR_DATA_PTR(array));
2414 350 : if (array_get_isnull(oldnullbitmap, offset))
2415 24 : olditemlen = 0;
2416 : else
2417 : {
2418 326 : olditemlen = att_addlength_pointer(0, elmlen, elt_ptr);
2419 326 : olditemlen = att_align_nominal(olditemlen, elmalign);
2420 : }
2421 350 : lenafter = (int) (olddatasize - lenbefore - olditemlen);
2422 : }
2423 :
2424 1756 : if (isNull)
2425 20 : newitemlen = 0;
2426 : else
2427 : {
2428 1736 : newitemlen = att_addlength_datum(0, elmlen, dataValue);
2429 1736 : newitemlen = att_align_nominal(newitemlen, elmalign);
2430 : }
2431 :
2432 1756 : newsize = overheadlen + lenbefore + newitemlen + lenafter;
2433 :
2434 : /*
2435 : * OK, create the new array and fill in header/dimensions
2436 : */
2437 1756 : newarray = (ArrayType *) palloc0(newsize);
2438 1756 : SET_VARSIZE(newarray, newsize);
2439 1756 : newarray->ndim = ndim;
2440 1756 : newarray->dataoffset = newhasnulls ? overheadlen : 0;
2441 1756 : newarray->elemtype = ARR_ELEMTYPE(array);
2442 1756 : memcpy(ARR_DIMS(newarray), dim, ndim * sizeof(int));
2443 1756 : memcpy(ARR_LBOUND(newarray), lb, ndim * sizeof(int));
2444 :
2445 : /*
2446 : * Fill in data
2447 : */
2448 1756 : memcpy((char *) newarray + overheadlen,
2449 : (char *) array + oldoverheadlen,
2450 : lenbefore);
2451 1756 : if (!isNull)
2452 1736 : ArrayCastAndSet(dataValue, elmlen, elmbyval, elmalign,
2453 1736 : (char *) newarray + overheadlen + lenbefore);
2454 1756 : memcpy((char *) newarray + overheadlen + lenbefore + newitemlen,
2455 1756 : (char *) array + oldoverheadlen + lenbefore + olditemlen,
2456 : lenafter);
2457 :
2458 : /*
2459 : * Fill in nulls bitmap if needed
2460 : *
2461 : * Note: it's possible we just replaced the last NULL with a non-NULL, and
2462 : * could get rid of the bitmap. Seems not worth testing for though.
2463 : */
2464 1756 : if (newhasnulls)
2465 : {
2466 134 : bits8 *newnullbitmap = ARR_NULLBITMAP(newarray);
2467 :
2468 : /* palloc0 above already marked any inserted positions as nulls */
2469 : /* Fix the inserted value */
2470 134 : if (addedafter)
2471 56 : array_set_isnull(newnullbitmap, newnitems - 1, isNull);
2472 : else
2473 78 : array_set_isnull(newnullbitmap, offset, isNull);
2474 : /* Fix the copied range(s) */
2475 134 : if (addedbefore)
2476 24 : array_bitmap_copy(newnullbitmap, addedbefore,
2477 : oldnullbitmap, 0,
2478 : oldnitems);
2479 : else
2480 : {
2481 110 : array_bitmap_copy(newnullbitmap, 0,
2482 : oldnullbitmap, 0,
2483 : offset);
2484 110 : if (addedafter == 0)
2485 54 : array_bitmap_copy(newnullbitmap, offset + 1,
2486 : oldnullbitmap, offset + 1,
2487 54 : oldnitems - offset - 1);
2488 : }
2489 : }
2490 :
2491 1756 : return PointerGetDatum(newarray);
2492 : }
2493 :
2494 : /*
2495 : * Implementation of array_set_element() for an expanded array
2496 : *
2497 : * Note: as with any operation on a read/write expanded object, we must
2498 : * take pains not to leave the object in a corrupt state if we fail partway
2499 : * through.
2500 : */
2501 : static Datum
2502 2016 : array_set_element_expanded(Datum arraydatum,
2503 : int nSubscripts, int *indx,
2504 : Datum dataValue, bool isNull,
2505 : int arraytyplen,
2506 : int elmlen, bool elmbyval, char elmalign)
2507 : {
2508 : ExpandedArrayHeader *eah;
2509 : Datum *dvalues;
2510 : bool *dnulls;
2511 : int i,
2512 : ndim,
2513 : dim[MAXDIM],
2514 : lb[MAXDIM],
2515 : offset;
2516 : bool dimschanged,
2517 : newhasnulls;
2518 : int addedbefore,
2519 : addedafter;
2520 : char *oldValue;
2521 :
2522 : /* Convert to R/W object if not so already */
2523 2016 : eah = DatumGetExpandedArray(arraydatum);
2524 :
2525 : /* Sanity-check caller's info against object; we don't use it otherwise */
2526 : Assert(arraytyplen == -1);
2527 : Assert(elmlen == eah->typlen);
2528 : Assert(elmbyval == eah->typbyval);
2529 : Assert(elmalign == eah->typalign);
2530 :
2531 : /*
2532 : * Copy dimension info into local storage. This allows us to modify the
2533 : * dimensions if needed, while not messing up the expanded value if we
2534 : * fail partway through.
2535 : */
2536 2016 : ndim = eah->ndims;
2537 : Assert(ndim >= 0 && ndim <= MAXDIM);
2538 2016 : memcpy(dim, eah->dims, ndim * sizeof(int));
2539 2016 : memcpy(lb, eah->lbound, ndim * sizeof(int));
2540 2016 : dimschanged = false;
2541 :
2542 : /*
2543 : * if number of dims is zero, i.e. an empty array, create an array with
2544 : * nSubscripts dimensions, and set the lower bounds to the supplied
2545 : * subscripts.
2546 : */
2547 2016 : if (ndim == 0)
2548 : {
2549 : /*
2550 : * Allocate adequate space for new dimension info. This is harmless
2551 : * if we fail later.
2552 : */
2553 : Assert(nSubscripts > 0 && nSubscripts <= MAXDIM);
2554 416 : eah->dims = (int *) MemoryContextAllocZero(eah->hdr.eoh_context,
2555 : nSubscripts * sizeof(int));
2556 416 : eah->lbound = (int *) MemoryContextAllocZero(eah->hdr.eoh_context,
2557 : nSubscripts * sizeof(int));
2558 :
2559 : /* Update local copies of dimension info */
2560 416 : ndim = nSubscripts;
2561 832 : for (i = 0; i < nSubscripts; i++)
2562 : {
2563 416 : dim[i] = 0;
2564 416 : lb[i] = indx[i];
2565 : }
2566 416 : dimschanged = true;
2567 : }
2568 1600 : else if (ndim != nSubscripts)
2569 0 : ereport(ERROR,
2570 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2571 : errmsg("wrong number of array subscripts")));
2572 :
2573 : /*
2574 : * Deconstruct array if we didn't already. (Someday maybe add a special
2575 : * case path for fixed-length, no-nulls cases, where we can overwrite an
2576 : * element in place without ever deconstructing. But today is not that
2577 : * day.)
2578 : */
2579 2016 : deconstruct_expanded_array(eah);
2580 :
2581 : /*
2582 : * Copy new element into array's context, if needed (we assume it's
2583 : * already detoasted, so no junk should be created). Doing this before
2584 : * we've made any significant changes ensures that our behavior is sane
2585 : * even when the source is a reference to some element of this same array.
2586 : * If we fail further down, this memory is leaked, but that's reasonably
2587 : * harmless.
2588 : */
2589 2016 : if (!eah->typbyval && !isNull)
2590 : {
2591 898 : MemoryContext oldcxt = MemoryContextSwitchTo(eah->hdr.eoh_context);
2592 :
2593 898 : dataValue = datumCopy(dataValue, false, eah->typlen);
2594 898 : MemoryContextSwitchTo(oldcxt);
2595 : }
2596 :
2597 2016 : dvalues = eah->dvalues;
2598 2016 : dnulls = eah->dnulls;
2599 :
2600 2016 : newhasnulls = ((dnulls != NULL) || isNull);
2601 2016 : addedbefore = addedafter = 0;
2602 :
2603 : /*
2604 : * Check subscripts (this logic must match array_set_element). We assume
2605 : * the existing subscripts passed ArrayCheckBounds, so that dim[i] + lb[i]
2606 : * can be computed without overflow. But we must beware of other
2607 : * overflows in our calculations of new dim[] values.
2608 : */
2609 2016 : if (ndim == 1)
2610 : {
2611 2016 : if (indx[0] < lb[0])
2612 : {
2613 : /* addedbefore = lb[0] - indx[0]; */
2614 : /* dim[0] += addedbefore; */
2615 160 : if (pg_sub_s32_overflow(lb[0], indx[0], &addedbefore) ||
2616 80 : pg_add_s32_overflow(dim[0], addedbefore, &dim[0]))
2617 0 : ereport(ERROR,
2618 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2619 : errmsg("array size exceeds the maximum allowed (%d)",
2620 : (int) MaxArraySize)));
2621 80 : lb[0] = indx[0];
2622 80 : dimschanged = true;
2623 80 : if (addedbefore > 1)
2624 0 : newhasnulls = true; /* will insert nulls */
2625 : }
2626 2016 : if (indx[0] >= (dim[0] + lb[0]))
2627 : {
2628 : /* addedafter = indx[0] - (dim[0] + lb[0]) + 1; */
2629 : /* dim[0] += addedafter; */
2630 3766 : if (pg_sub_s32_overflow(indx[0], dim[0] + lb[0], &addedafter) ||
2631 3760 : pg_add_s32_overflow(addedafter, 1, &addedafter) ||
2632 1880 : pg_add_s32_overflow(dim[0], addedafter, &dim[0]))
2633 6 : ereport(ERROR,
2634 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2635 : errmsg("array size exceeds the maximum allowed (%d)",
2636 : (int) MaxArraySize)));
2637 1880 : dimschanged = true;
2638 1880 : if (addedafter > 1)
2639 0 : newhasnulls = true; /* will insert nulls */
2640 : }
2641 : }
2642 : else
2643 : {
2644 : /*
2645 : * XXX currently we do not support extending multi-dimensional arrays
2646 : * during assignment
2647 : */
2648 0 : for (i = 0; i < ndim; i++)
2649 : {
2650 0 : if (indx[i] < lb[i] ||
2651 0 : indx[i] >= (dim[i] + lb[i]))
2652 0 : ereport(ERROR,
2653 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2654 : errmsg("array subscript out of range")));
2655 : }
2656 : }
2657 :
2658 : /* Check for overflow of the array dimensions */
2659 2010 : if (dimschanged)
2660 : {
2661 1960 : (void) ArrayGetNItems(ndim, dim);
2662 1960 : ArrayCheckBounds(ndim, dim, lb);
2663 : }
2664 :
2665 : /* Now we can calculate linear offset of target item in array */
2666 2010 : offset = ArrayGetOffset(nSubscripts, dim, lb, indx);
2667 :
2668 : /* Physically enlarge existing dvalues/dnulls arrays if needed */
2669 2010 : if (dim[0] > eah->dvalueslen)
2670 : {
2671 : /* We want some extra space if we're enlarging */
2672 1948 : int newlen = dim[0] + dim[0] / 8;
2673 :
2674 1948 : newlen = Max(newlen, dim[0]); /* integer overflow guard */
2675 1948 : eah->dvalues = dvalues = (Datum *)
2676 1948 : repalloc(dvalues, newlen * sizeof(Datum));
2677 1948 : if (dnulls)
2678 0 : eah->dnulls = dnulls = (bool *)
2679 0 : repalloc(dnulls, newlen * sizeof(bool));
2680 1948 : eah->dvalueslen = newlen;
2681 : }
2682 :
2683 : /*
2684 : * If we need a nulls bitmap and don't already have one, create it, being
2685 : * sure to mark all existing entries as not null.
2686 : */
2687 2010 : if (newhasnulls && dnulls == NULL)
2688 0 : eah->dnulls = dnulls = (bool *)
2689 0 : MemoryContextAllocZero(eah->hdr.eoh_context,
2690 0 : eah->dvalueslen * sizeof(bool));
2691 :
2692 : /*
2693 : * We now have all the needed space allocated, so we're ready to make
2694 : * irreversible changes. Be very wary of allowing failure below here.
2695 : */
2696 :
2697 : /* Flattened value will no longer represent array accurately */
2698 2010 : eah->fvalue = NULL;
2699 : /* And we don't know the flattened size either */
2700 2010 : eah->flat_size = 0;
2701 :
2702 : /* Update dimensionality info if needed */
2703 2010 : if (dimschanged)
2704 : {
2705 1960 : eah->ndims = ndim;
2706 1960 : memcpy(eah->dims, dim, ndim * sizeof(int));
2707 1960 : memcpy(eah->lbound, lb, ndim * sizeof(int));
2708 : }
2709 :
2710 : /* Reposition items if needed, and fill addedbefore items with nulls */
2711 2010 : if (addedbefore > 0)
2712 : {
2713 80 : memmove(dvalues + addedbefore, dvalues, eah->nelems * sizeof(Datum));
2714 160 : for (i = 0; i < addedbefore; i++)
2715 80 : dvalues[i] = (Datum) 0;
2716 80 : if (dnulls)
2717 : {
2718 0 : memmove(dnulls + addedbefore, dnulls, eah->nelems * sizeof(bool));
2719 0 : for (i = 0; i < addedbefore; i++)
2720 0 : dnulls[i] = true;
2721 : }
2722 80 : eah->nelems += addedbefore;
2723 : }
2724 :
2725 : /* fill addedafter items with nulls */
2726 2010 : if (addedafter > 0)
2727 : {
2728 3760 : for (i = 0; i < addedafter; i++)
2729 1880 : dvalues[eah->nelems + i] = (Datum) 0;
2730 1880 : if (dnulls)
2731 : {
2732 0 : for (i = 0; i < addedafter; i++)
2733 0 : dnulls[eah->nelems + i] = true;
2734 : }
2735 1880 : eah->nelems += addedafter;
2736 : }
2737 :
2738 : /* Grab old element value for pfree'ing, if needed. */
2739 2010 : if (!eah->typbyval && (dnulls == NULL || !dnulls[offset]))
2740 898 : oldValue = (char *) DatumGetPointer(dvalues[offset]);
2741 : else
2742 1112 : oldValue = NULL;
2743 :
2744 : /* And finally we can insert the new element. */
2745 2010 : dvalues[offset] = dataValue;
2746 2010 : if (dnulls)
2747 0 : dnulls[offset] = isNull;
2748 :
2749 : /*
2750 : * Free old element if needed; this keeps repeated element replacements
2751 : * from bloating the array's storage. If the pfree somehow fails, it
2752 : * won't corrupt the array.
2753 : */
2754 2010 : if (oldValue)
2755 : {
2756 : /* Don't try to pfree a part of the original flat array */
2757 2 : if (oldValue < eah->fstartptr || oldValue >= eah->fendptr)
2758 0 : pfree(oldValue);
2759 : }
2760 :
2761 : /* Done, return standard TOAST pointer for object */
2762 2010 : return EOHPGetRWDatum(&eah->hdr);
2763 : }
2764 :
2765 : /*
2766 : * array_set_slice :
2767 : * This routine sets the value of a range of array locations (specified
2768 : * by upper and lower subscript values) to new values passed as
2769 : * another array.
2770 : *
2771 : * This handles both ordinary varlena arrays and fixed-length arrays.
2772 : *
2773 : * Inputs:
2774 : * arraydatum: the initial array object (mustn't be NULL)
2775 : * nSubscripts: number of subscripts supplied (must be same for upper/lower)
2776 : * upperIndx[]: the upper subscript values
2777 : * lowerIndx[]: the lower subscript values
2778 : * upperProvided[]: true for provided upper subscript values
2779 : * lowerProvided[]: true for provided lower subscript values
2780 : * srcArrayDatum: the source for the inserted values
2781 : * isNull: indicates whether srcArrayDatum is NULL
2782 : * arraytyplen: pg_type.typlen for the array type
2783 : * elmlen: pg_type.typlen for the array's element type
2784 : * elmbyval: pg_type.typbyval for the array's element type
2785 : * elmalign: pg_type.typalign for the array's element type
2786 : *
2787 : * Result:
2788 : * A new array is returned, just like the old except for the
2789 : * modified range. The original array object is not changed.
2790 : *
2791 : * Omitted upper and lower subscript values are replaced by the corresponding
2792 : * array bound.
2793 : *
2794 : * For one-dimensional arrays only, we allow the array to be extended
2795 : * by assigning to positions outside the existing subscript range; any
2796 : * positions between the existing elements and the new ones are set to NULLs.
2797 : * (XXX TODO: allow a corresponding behavior for multidimensional arrays)
2798 : *
2799 : * NOTE: we assume it is OK to scribble on the provided index arrays
2800 : * lowerIndx[] and upperIndx[]; also, these arrays must be of size MAXDIM
2801 : * even when nSubscripts is less. These are generally just temporaries.
2802 : *
2803 : * NOTE: For assignments, we throw an error for silly subscripts etc,
2804 : * rather than returning a NULL or empty array as the fetch operations do.
2805 : */
2806 : Datum
2807 262 : array_set_slice(Datum arraydatum,
2808 : int nSubscripts,
2809 : int *upperIndx,
2810 : int *lowerIndx,
2811 : bool *upperProvided,
2812 : bool *lowerProvided,
2813 : Datum srcArrayDatum,
2814 : bool isNull,
2815 : int arraytyplen,
2816 : int elmlen,
2817 : bool elmbyval,
2818 : char elmalign)
2819 : {
2820 : ArrayType *array;
2821 : ArrayType *srcArray;
2822 : ArrayType *newarray;
2823 : int i,
2824 : ndim,
2825 : dim[MAXDIM],
2826 : lb[MAXDIM],
2827 : span[MAXDIM];
2828 : bool newhasnulls;
2829 : int nitems,
2830 : nsrcitems,
2831 : olddatasize,
2832 : newsize,
2833 : olditemsize,
2834 : newitemsize,
2835 : overheadlen,
2836 : oldoverheadlen,
2837 : addedbefore,
2838 : addedafter,
2839 : lenbefore,
2840 : lenafter,
2841 : itemsbefore,
2842 : itemsafter,
2843 : nolditems;
2844 :
2845 : /* Currently, assignment from a NULL source array is a no-op */
2846 262 : if (isNull)
2847 0 : return arraydatum;
2848 :
2849 262 : if (arraytyplen > 0)
2850 : {
2851 : /*
2852 : * fixed-length arrays -- not got round to doing this...
2853 : */
2854 0 : ereport(ERROR,
2855 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
2856 : errmsg("updates on slices of fixed-length arrays not implemented")));
2857 : }
2858 :
2859 : /* detoast arrays if necessary */
2860 262 : array = DatumGetArrayTypeP(arraydatum);
2861 262 : srcArray = DatumGetArrayTypeP(srcArrayDatum);
2862 :
2863 : /* note: we assume srcArray contains no toasted elements */
2864 :
2865 262 : ndim = ARR_NDIM(array);
2866 :
2867 : /*
2868 : * if number of dims is zero, i.e. an empty array, create an array with
2869 : * nSubscripts dimensions, and set the upper and lower bounds to the
2870 : * supplied subscripts
2871 : */
2872 262 : if (ndim == 0)
2873 : {
2874 : Datum *dvalues;
2875 : bool *dnulls;
2876 : int nelems;
2877 56 : Oid elmtype = ARR_ELEMTYPE(array);
2878 :
2879 56 : deconstruct_array(srcArray, elmtype, elmlen, elmbyval, elmalign,
2880 : &dvalues, &dnulls, &nelems);
2881 :
2882 112 : for (i = 0; i < nSubscripts; i++)
2883 : {
2884 74 : if (!upperProvided[i] || !lowerProvided[i])
2885 6 : ereport(ERROR,
2886 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2887 : errmsg("array slice subscript must provide both boundaries"),
2888 : errdetail("When assigning to a slice of an empty array value,"
2889 : " slice boundaries must be fully specified.")));
2890 :
2891 : /* compute "upperIndx[i] - lowerIndx[i] + 1", detecting overflow */
2892 130 : if (pg_sub_s32_overflow(upperIndx[i], lowerIndx[i], &dim[i]) ||
2893 62 : pg_add_s32_overflow(dim[i], 1, &dim[i]))
2894 12 : ereport(ERROR,
2895 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2896 : errmsg("array size exceeds the maximum allowed (%d)",
2897 : (int) MaxArraySize)));
2898 :
2899 56 : lb[i] = lowerIndx[i];
2900 : }
2901 :
2902 : /* complain if too few source items; we ignore extras, however */
2903 38 : if (nelems < ArrayGetNItems(nSubscripts, dim))
2904 0 : ereport(ERROR,
2905 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2906 : errmsg("source array too small")));
2907 :
2908 38 : return PointerGetDatum(construct_md_array(dvalues, dnulls, nSubscripts,
2909 : dim, lb, elmtype,
2910 : elmlen, elmbyval, elmalign));
2911 : }
2912 :
2913 206 : if (ndim < nSubscripts || ndim <= 0 || ndim > MAXDIM)
2914 0 : ereport(ERROR,
2915 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2916 : errmsg("wrong number of array subscripts")));
2917 :
2918 : /* copy dim/lb since we may modify them */
2919 206 : memcpy(dim, ARR_DIMS(array), ndim * sizeof(int));
2920 206 : memcpy(lb, ARR_LBOUND(array), ndim * sizeof(int));
2921 :
2922 206 : newhasnulls = (ARR_HASNULL(array) || ARR_HASNULL(srcArray));
2923 206 : addedbefore = addedafter = 0;
2924 :
2925 : /*
2926 : * Check subscripts. We assume the existing subscripts passed
2927 : * ArrayCheckBounds, so that dim[i] + lb[i] can be computed without
2928 : * overflow. But we must beware of other overflows in our calculations of
2929 : * new dim[] values.
2930 : */
2931 206 : if (ndim == 1)
2932 : {
2933 : Assert(nSubscripts == 1);
2934 176 : if (!lowerProvided[0])
2935 36 : lowerIndx[0] = lb[0];
2936 176 : if (!upperProvided[0])
2937 42 : upperIndx[0] = dim[0] + lb[0] - 1;
2938 176 : if (lowerIndx[0] > upperIndx[0])
2939 0 : ereport(ERROR,
2940 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2941 : errmsg("upper bound cannot be less than lower bound")));
2942 176 : if (lowerIndx[0] < lb[0])
2943 : {
2944 : /* addedbefore = lb[0] - lowerIndx[0]; */
2945 : /* dim[0] += addedbefore; */
2946 96 : if (pg_sub_s32_overflow(lb[0], lowerIndx[0], &addedbefore) ||
2947 48 : pg_add_s32_overflow(dim[0], addedbefore, &dim[0]))
2948 0 : ereport(ERROR,
2949 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2950 : errmsg("array size exceeds the maximum allowed (%d)",
2951 : (int) MaxArraySize)));
2952 48 : lb[0] = lowerIndx[0];
2953 48 : if (addedbefore > 1)
2954 36 : newhasnulls = true; /* will insert nulls */
2955 : }
2956 176 : if (upperIndx[0] >= (dim[0] + lb[0]))
2957 : {
2958 : /* addedafter = upperIndx[0] - (dim[0] + lb[0]) + 1; */
2959 : /* dim[0] += addedafter; */
2960 118 : if (pg_sub_s32_overflow(upperIndx[0], dim[0] + lb[0], &addedafter) ||
2961 112 : pg_add_s32_overflow(addedafter, 1, &addedafter) ||
2962 56 : pg_add_s32_overflow(dim[0], addedafter, &dim[0]))
2963 6 : ereport(ERROR,
2964 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
2965 : errmsg("array size exceeds the maximum allowed (%d)",
2966 : (int) MaxArraySize)));
2967 56 : if (addedafter > 1)
2968 36 : newhasnulls = true; /* will insert nulls */
2969 : }
2970 : }
2971 : else
2972 : {
2973 : /*
2974 : * XXX currently we do not support extending multi-dimensional arrays
2975 : * during assignment
2976 : */
2977 102 : for (i = 0; i < nSubscripts; i++)
2978 : {
2979 72 : if (!lowerProvided[i])
2980 12 : lowerIndx[i] = lb[i];
2981 72 : if (!upperProvided[i])
2982 24 : upperIndx[i] = dim[i] + lb[i] - 1;
2983 72 : if (lowerIndx[i] > upperIndx[i])
2984 0 : ereport(ERROR,
2985 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2986 : errmsg("upper bound cannot be less than lower bound")));
2987 72 : if (lowerIndx[i] < lb[i] ||
2988 72 : upperIndx[i] >= (dim[i] + lb[i]))
2989 0 : ereport(ERROR,
2990 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
2991 : errmsg("array subscript out of range")));
2992 : }
2993 : /* fill any missing subscript positions with full array range */
2994 30 : for (; i < ndim; i++)
2995 : {
2996 0 : lowerIndx[i] = lb[i];
2997 0 : upperIndx[i] = dim[i] + lb[i] - 1;
2998 0 : if (lowerIndx[i] > upperIndx[i])
2999 0 : ereport(ERROR,
3000 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
3001 : errmsg("upper bound cannot be less than lower bound")));
3002 : }
3003 : }
3004 :
3005 : /* Do this mainly to check for overflow */
3006 200 : nitems = ArrayGetNItems(ndim, dim);
3007 200 : ArrayCheckBounds(ndim, dim, lb);
3008 :
3009 : /*
3010 : * Make sure source array has enough entries. Note we ignore the shape of
3011 : * the source array and just read entries serially.
3012 : */
3013 200 : mda_get_range(ndim, span, lowerIndx, upperIndx);
3014 200 : nsrcitems = ArrayGetNItems(ndim, span);
3015 200 : if (nsrcitems > ArrayGetNItems(ARR_NDIM(srcArray), ARR_DIMS(srcArray)))
3016 6 : ereport(ERROR,
3017 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
3018 : errmsg("source array too small")));
3019 :
3020 : /*
3021 : * Compute space occupied by new entries, space occupied by replaced
3022 : * entries, and required space for new array.
3023 : */
3024 194 : if (newhasnulls)
3025 96 : overheadlen = ARR_OVERHEAD_WITHNULLS(ndim, nitems);
3026 : else
3027 98 : overheadlen = ARR_OVERHEAD_NONULLS(ndim);
3028 194 : newitemsize = array_nelems_size(ARR_DATA_PTR(srcArray), 0,
3029 194 : ARR_NULLBITMAP(srcArray), nsrcitems,
3030 : elmlen, elmbyval, elmalign);
3031 194 : oldoverheadlen = ARR_DATA_OFFSET(array);
3032 194 : olddatasize = ARR_SIZE(array) - oldoverheadlen;
3033 194 : if (ndim > 1)
3034 : {
3035 : /*
3036 : * here we do not need to cope with extension of the array; it would
3037 : * be a lot more complicated if we had to do so...
3038 : */
3039 30 : olditemsize = array_slice_size(ARR_DATA_PTR(array),
3040 30 : ARR_NULLBITMAP(array),
3041 : ndim, dim, lb,
3042 : lowerIndx, upperIndx,
3043 : elmlen, elmbyval, elmalign);
3044 30 : lenbefore = lenafter = 0; /* keep compiler quiet */
3045 30 : itemsbefore = itemsafter = nolditems = 0;
3046 : }
3047 : else
3048 : {
3049 : /*
3050 : * here we must allow for possibility of slice larger than orig array
3051 : * and/or not adjacent to orig array subscripts
3052 : */
3053 164 : int oldlb = ARR_LBOUND(array)[0];
3054 164 : int oldub = oldlb + ARR_DIMS(array)[0] - 1;
3055 164 : int slicelb = Max(oldlb, lowerIndx[0]);
3056 164 : int sliceub = Min(oldub, upperIndx[0]);
3057 164 : char *oldarraydata = ARR_DATA_PTR(array);
3058 164 : bits8 *oldarraybitmap = ARR_NULLBITMAP(array);
3059 :
3060 : /* count/size of old array entries that will go before the slice */
3061 164 : itemsbefore = Min(slicelb, oldub + 1) - oldlb;
3062 164 : lenbefore = array_nelems_size(oldarraydata, 0, oldarraybitmap,
3063 : itemsbefore,
3064 : elmlen, elmbyval, elmalign);
3065 : /* count/size of old array entries that will be replaced by slice */
3066 164 : if (slicelb > sliceub)
3067 : {
3068 54 : nolditems = 0;
3069 54 : olditemsize = 0;
3070 : }
3071 : else
3072 : {
3073 110 : nolditems = sliceub - slicelb + 1;
3074 110 : olditemsize = array_nelems_size(oldarraydata + lenbefore,
3075 : itemsbefore, oldarraybitmap,
3076 : nolditems,
3077 : elmlen, elmbyval, elmalign);
3078 : }
3079 : /* count/size of old array entries that will go after the slice */
3080 164 : itemsafter = oldub + 1 - Max(sliceub + 1, oldlb);
3081 164 : lenafter = olddatasize - lenbefore - olditemsize;
3082 : }
3083 :
3084 194 : newsize = overheadlen + olddatasize - olditemsize + newitemsize;
3085 :
3086 194 : newarray = (ArrayType *) palloc0(newsize);
3087 194 : SET_VARSIZE(newarray, newsize);
3088 194 : newarray->ndim = ndim;
3089 194 : newarray->dataoffset = newhasnulls ? overheadlen : 0;
3090 194 : newarray->elemtype = ARR_ELEMTYPE(array);
3091 194 : memcpy(ARR_DIMS(newarray), dim, ndim * sizeof(int));
3092 194 : memcpy(ARR_LBOUND(newarray), lb, ndim * sizeof(int));
3093 :
3094 194 : if (ndim > 1)
3095 : {
3096 : /*
3097 : * here we do not need to cope with extension of the array; it would
3098 : * be a lot more complicated if we had to do so...
3099 : */
3100 30 : array_insert_slice(newarray, array, srcArray,
3101 : ndim, dim, lb,
3102 : lowerIndx, upperIndx,
3103 : elmlen, elmbyval, elmalign);
3104 : }
3105 : else
3106 : {
3107 : /* fill in data */
3108 164 : memcpy((char *) newarray + overheadlen,
3109 : (char *) array + oldoverheadlen,
3110 : lenbefore);
3111 328 : memcpy((char *) newarray + overheadlen + lenbefore,
3112 164 : ARR_DATA_PTR(srcArray),
3113 : newitemsize);
3114 164 : memcpy((char *) newarray + overheadlen + lenbefore + newitemsize,
3115 164 : (char *) array + oldoverheadlen + lenbefore + olditemsize,
3116 : lenafter);
3117 : /* fill in nulls bitmap if needed */
3118 164 : if (newhasnulls)
3119 : {
3120 96 : bits8 *newnullbitmap = ARR_NULLBITMAP(newarray);
3121 96 : bits8 *oldnullbitmap = ARR_NULLBITMAP(array);
3122 :
3123 : /* palloc0 above already marked any inserted positions as nulls */
3124 96 : array_bitmap_copy(newnullbitmap, addedbefore,
3125 : oldnullbitmap, 0,
3126 : itemsbefore);
3127 96 : array_bitmap_copy(newnullbitmap, lowerIndx[0] - lb[0],
3128 96 : ARR_NULLBITMAP(srcArray), 0,
3129 : nsrcitems);
3130 96 : array_bitmap_copy(newnullbitmap, addedbefore + itemsbefore + nolditems,
3131 : oldnullbitmap, itemsbefore + nolditems,
3132 : itemsafter);
3133 : }
3134 : }
3135 :
3136 194 : return PointerGetDatum(newarray);
3137 : }
3138 :
3139 : /*
3140 : * array_ref : backwards compatibility wrapper for array_get_element
3141 : *
3142 : * This only works for detoasted/flattened varlena arrays, since the array
3143 : * argument is declared as "ArrayType *". However there's enough code like
3144 : * that to justify preserving this API.
3145 : */
3146 : Datum
3147 44520 : array_ref(ArrayType *array, int nSubscripts, int *indx,
3148 : int arraytyplen, int elmlen, bool elmbyval, char elmalign,
3149 : bool *isNull)
3150 : {
3151 44520 : return array_get_element(PointerGetDatum(array), nSubscripts, indx,
3152 : arraytyplen, elmlen, elmbyval, elmalign,
3153 : isNull);
3154 : }
3155 :
3156 : /*
3157 : * array_set : backwards compatibility wrapper for array_set_element
3158 : *
3159 : * This only works for detoasted/flattened varlena arrays, since the array
3160 : * argument and result are declared as "ArrayType *". However there's enough
3161 : * code like that to justify preserving this API.
3162 : */
3163 : ArrayType *
3164 1118 : array_set(ArrayType *array, int nSubscripts, int *indx,
3165 : Datum dataValue, bool isNull,
3166 : int arraytyplen, int elmlen, bool elmbyval, char elmalign)
3167 : {
3168 1118 : return DatumGetArrayTypeP(array_set_element(PointerGetDatum(array),
3169 : nSubscripts, indx,
3170 : dataValue, isNull,
3171 : arraytyplen,
3172 : elmlen, elmbyval, elmalign));
3173 : }
3174 :
3175 : /*
3176 : * array_map()
3177 : *
3178 : * Map an array through an arbitrary expression. Return a new array with
3179 : * the same dimensions and each source element transformed by the given,
3180 : * already-compiled expression. Each source element is placed in the
3181 : * innermost_caseval/innermost_casenull fields of the ExprState.
3182 : *
3183 : * Parameters are:
3184 : * * arrayd: Datum representing array argument.
3185 : * * exprstate: ExprState representing the per-element transformation.
3186 : * * econtext: context for expression evaluation.
3187 : * * retType: OID of element type of output array. This must be the same as,
3188 : * or binary-compatible with, the result type of the expression. It might
3189 : * be different from the input array's element type.
3190 : * * amstate: workspace for array_map. Must be zeroed by caller before
3191 : * first call, and not touched after that.
3192 : *
3193 : * It is legitimate to pass a freshly-zeroed ArrayMapState on each call,
3194 : * but better performance can be had if the state can be preserved across
3195 : * a series of calls.
3196 : *
3197 : * NB: caller must assure that input array is not NULL. NULL elements in
3198 : * the array are OK however.
3199 : * NB: caller should be running in econtext's per-tuple memory context.
3200 : */
3201 : Datum
3202 41046 : array_map(Datum arrayd,
3203 : ExprState *exprstate, ExprContext *econtext,
3204 : Oid retType, ArrayMapState *amstate)
3205 : {
3206 41046 : AnyArrayType *v = DatumGetAnyArrayP(arrayd);
3207 : ArrayType *result;
3208 : Datum *values;
3209 : bool *nulls;
3210 : int *dim;
3211 : int ndim;
3212 : int nitems;
3213 : int i;
3214 41046 : int32 nbytes = 0;
3215 : int32 dataoffset;
3216 : bool hasnulls;
3217 : Oid inpType;
3218 : int inp_typlen;
3219 : bool inp_typbyval;
3220 : char inp_typalign;
3221 : int typlen;
3222 : bool typbyval;
3223 : char typalign;
3224 : array_iter iter;
3225 : ArrayMetaState *inp_extra;
3226 : ArrayMetaState *ret_extra;
3227 41046 : Datum *transform_source = exprstate->innermost_caseval;
3228 41046 : bool *transform_source_isnull = exprstate->innermost_casenull;
3229 :
3230 41046 : inpType = AARR_ELEMTYPE(v);
3231 41046 : ndim = AARR_NDIM(v);
3232 41046 : dim = AARR_DIMS(v);
3233 41046 : nitems = ArrayGetNItems(ndim, dim);
3234 :
3235 : /* Check for empty array */
3236 41046 : if (nitems <= 0)
3237 : {
3238 : /* Return empty array */
3239 12 : return PointerGetDatum(construct_empty_array(retType));
3240 : }
3241 :
3242 : /*
3243 : * We arrange to look up info about input and return element types only
3244 : * once per series of calls, assuming the element type doesn't change
3245 : * underneath us.
3246 : */
3247 41034 : inp_extra = &amstate->inp_extra;
3248 41034 : ret_extra = &amstate->ret_extra;
3249 :
3250 41034 : if (inp_extra->element_type != inpType)
3251 : {
3252 434 : get_typlenbyvalalign(inpType,
3253 : &inp_extra->typlen,
3254 : &inp_extra->typbyval,
3255 : &inp_extra->typalign);
3256 434 : inp_extra->element_type = inpType;
3257 : }
3258 41034 : inp_typlen = inp_extra->typlen;
3259 41034 : inp_typbyval = inp_extra->typbyval;
3260 41034 : inp_typalign = inp_extra->typalign;
3261 :
3262 41034 : if (ret_extra->element_type != retType)
3263 : {
3264 434 : get_typlenbyvalalign(retType,
3265 : &ret_extra->typlen,
3266 : &ret_extra->typbyval,
3267 : &ret_extra->typalign);
3268 434 : ret_extra->element_type = retType;
3269 : }
3270 41034 : typlen = ret_extra->typlen;
3271 41034 : typbyval = ret_extra->typbyval;
3272 41034 : typalign = ret_extra->typalign;
3273 :
3274 : /* Allocate temporary arrays for new values */
3275 41034 : values = (Datum *) palloc(nitems * sizeof(Datum));
3276 41034 : nulls = (bool *) palloc(nitems * sizeof(bool));
3277 :
3278 : /* Loop over source data */
3279 41034 : array_iter_setup(&iter, v);
3280 41034 : hasnulls = false;
3281 :
3282 462734 : for (i = 0; i < nitems; i++)
3283 : {
3284 : /* Get source element, checking for NULL */
3285 421732 : *transform_source =
3286 421732 : array_iter_next(&iter, transform_source_isnull, i,
3287 : inp_typlen, inp_typbyval, inp_typalign);
3288 :
3289 : /* Apply the given expression to source element */
3290 421732 : values[i] = ExecEvalExpr(exprstate, econtext, &nulls[i]);
3291 :
3292 421700 : if (nulls[i])
3293 12 : hasnulls = true;
3294 : else
3295 : {
3296 : /* Ensure data is not toasted */
3297 421688 : if (typlen == -1)
3298 428 : values[i] = PointerGetDatum(PG_DETOAST_DATUM(values[i]));
3299 : /* Update total result size */
3300 421688 : nbytes = att_addlength_datum(nbytes, typlen, values[i]);
3301 421688 : nbytes = att_align_nominal(nbytes, typalign);
3302 : /* check for overflow of total request */
3303 421688 : if (!AllocSizeIsValid(nbytes))
3304 0 : ereport(ERROR,
3305 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
3306 : errmsg("array size exceeds the maximum allowed (%d)",
3307 : (int) MaxAllocSize)));
3308 : }
3309 : }
3310 :
3311 : /* Allocate and fill the result array */
3312 41002 : if (hasnulls)
3313 : {
3314 6 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nitems);
3315 6 : nbytes += dataoffset;
3316 : }
3317 : else
3318 : {
3319 40996 : dataoffset = 0; /* marker for no null bitmap */
3320 40996 : nbytes += ARR_OVERHEAD_NONULLS(ndim);
3321 : }
3322 41002 : result = (ArrayType *) palloc0(nbytes);
3323 41002 : SET_VARSIZE(result, nbytes);
3324 41002 : result->ndim = ndim;
3325 41002 : result->dataoffset = dataoffset;
3326 41002 : result->elemtype = retType;
3327 41002 : memcpy(ARR_DIMS(result), AARR_DIMS(v), ndim * sizeof(int));
3328 41002 : memcpy(ARR_LBOUND(result), AARR_LBOUND(v), ndim * sizeof(int));
3329 :
3330 41002 : CopyArrayEls(result,
3331 : values, nulls, nitems,
3332 : typlen, typbyval, typalign,
3333 : false);
3334 :
3335 : /*
3336 : * Note: do not risk trying to pfree the results of the called expression
3337 : */
3338 41002 : pfree(values);
3339 41002 : pfree(nulls);
3340 :
3341 41002 : return PointerGetDatum(result);
3342 : }
3343 :
3344 : /*
3345 : * construct_array --- simple method for constructing an array object
3346 : *
3347 : * elems: array of Datum items to become the array contents
3348 : * (NULL element values are not supported).
3349 : * nelems: number of items
3350 : * elmtype, elmlen, elmbyval, elmalign: info for the datatype of the items
3351 : *
3352 : * A palloc'd 1-D array object is constructed and returned. Note that
3353 : * elem values will be copied into the object even if pass-by-ref type.
3354 : * Also note the result will be 0-D not 1-D if nelems = 0.
3355 : *
3356 : * NOTE: it would be cleaner to look up the elmlen/elmbval/elmalign info
3357 : * from the system catalogs, given the elmtype. However, the caller is
3358 : * in a better position to cache this info across multiple uses, or even
3359 : * to hard-wire values if the element type is hard-wired.
3360 : */
3361 : ArrayType *
3362 336688 : construct_array(Datum *elems, int nelems,
3363 : Oid elmtype,
3364 : int elmlen, bool elmbyval, char elmalign)
3365 : {
3366 : int dims[1];
3367 : int lbs[1];
3368 :
3369 336688 : dims[0] = nelems;
3370 336688 : lbs[0] = 1;
3371 :
3372 336688 : return construct_md_array(elems, NULL, 1, dims, lbs,
3373 : elmtype, elmlen, elmbyval, elmalign);
3374 : }
3375 :
3376 : /*
3377 : * Like construct_array(), where elmtype must be a built-in type, and
3378 : * elmlen/elmbyval/elmalign is looked up from hardcoded data. This is often
3379 : * useful when manipulating arrays from/for system catalogs.
3380 : */
3381 : ArrayType *
3382 242306 : construct_array_builtin(Datum *elems, int nelems, Oid elmtype)
3383 : {
3384 : int elmlen;
3385 : bool elmbyval;
3386 : char elmalign;
3387 :
3388 242306 : switch (elmtype)
3389 : {
3390 3454 : case CHAROID:
3391 3454 : elmlen = 1;
3392 3454 : elmbyval = true;
3393 3454 : elmalign = TYPALIGN_CHAR;
3394 3454 : break;
3395 :
3396 7878 : case CSTRINGOID:
3397 7878 : elmlen = -2;
3398 7878 : elmbyval = false;
3399 7878 : elmalign = TYPALIGN_CHAR;
3400 7878 : break;
3401 :
3402 134540 : case FLOAT4OID:
3403 134540 : elmlen = sizeof(float4);
3404 134540 : elmbyval = true;
3405 134540 : elmalign = TYPALIGN_INT;
3406 134540 : break;
3407 :
3408 48 : case FLOAT8OID:
3409 48 : elmlen = sizeof(float8);
3410 48 : elmbyval = true;
3411 48 : elmalign = TYPALIGN_DOUBLE;
3412 48 : break;
3413 :
3414 56720 : case INT2OID:
3415 56720 : elmlen = sizeof(int16);
3416 56720 : elmbyval = true;
3417 56720 : elmalign = TYPALIGN_SHORT;
3418 56720 : break;
3419 :
3420 6888 : case INT4OID:
3421 6888 : elmlen = sizeof(int32);
3422 6888 : elmbyval = true;
3423 6888 : elmalign = TYPALIGN_INT;
3424 6888 : break;
3425 :
3426 4 : case INT8OID:
3427 4 : elmlen = sizeof(int64);
3428 4 : elmbyval = true;
3429 4 : elmalign = TYPALIGN_DOUBLE;
3430 4 : break;
3431 :
3432 718 : case NAMEOID:
3433 718 : elmlen = NAMEDATALEN;
3434 718 : elmbyval = false;
3435 718 : elmalign = TYPALIGN_CHAR;
3436 718 : break;
3437 :
3438 17428 : case OIDOID:
3439 : case REGTYPEOID:
3440 17428 : elmlen = sizeof(Oid);
3441 17428 : elmbyval = true;
3442 17428 : elmalign = TYPALIGN_INT;
3443 17428 : break;
3444 :
3445 14582 : case TEXTOID:
3446 14582 : elmlen = -1;
3447 14582 : elmbyval = false;
3448 14582 : elmalign = TYPALIGN_INT;
3449 14582 : break;
3450 :
3451 42 : case TIDOID:
3452 42 : elmlen = sizeof(ItemPointerData);
3453 42 : elmbyval = false;
3454 42 : elmalign = TYPALIGN_SHORT;
3455 42 : break;
3456 :
3457 4 : case XIDOID:
3458 4 : elmlen = sizeof(TransactionId);
3459 4 : elmbyval = true;
3460 4 : elmalign = TYPALIGN_INT;
3461 4 : break;
3462 :
3463 0 : default:
3464 0 : elog(ERROR, "type %u not supported by construct_array_builtin()", elmtype);
3465 : /* keep compiler quiet */
3466 : elmlen = 0;
3467 : elmbyval = false;
3468 : elmalign = 0;
3469 : }
3470 :
3471 242306 : return construct_array(elems, nelems, elmtype, elmlen, elmbyval, elmalign);
3472 : }
3473 :
3474 : /*
3475 : * construct_md_array --- simple method for constructing an array object
3476 : * with arbitrary dimensions and possible NULLs
3477 : *
3478 : * elems: array of Datum items to become the array contents
3479 : * nulls: array of is-null flags (can be NULL if no nulls)
3480 : * ndims: number of dimensions
3481 : * dims: integer array with size of each dimension
3482 : * lbs: integer array with lower bound of each dimension
3483 : * elmtype, elmlen, elmbyval, elmalign: info for the datatype of the items
3484 : *
3485 : * A palloc'd ndims-D array object is constructed and returned. Note that
3486 : * elem values will be copied into the object even if pass-by-ref type.
3487 : * Also note the result will be 0-D not ndims-D if any dims[i] = 0.
3488 : *
3489 : * NOTE: it would be cleaner to look up the elmlen/elmbval/elmalign info
3490 : * from the system catalogs, given the elmtype. However, the caller is
3491 : * in a better position to cache this info across multiple uses, or even
3492 : * to hard-wire values if the element type is hard-wired.
3493 : */
3494 : ArrayType *
3495 1573502 : construct_md_array(Datum *elems,
3496 : bool *nulls,
3497 : int ndims,
3498 : int *dims,
3499 : int *lbs,
3500 : Oid elmtype, int elmlen, bool elmbyval, char elmalign)
3501 : {
3502 : ArrayType *result;
3503 : bool hasnulls;
3504 : int32 nbytes;
3505 : int32 dataoffset;
3506 : int i;
3507 : int nelems;
3508 :
3509 1573502 : if (ndims < 0) /* we do allow zero-dimension arrays */
3510 0 : ereport(ERROR,
3511 : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
3512 : errmsg("invalid number of dimensions: %d", ndims)));
3513 1573502 : if (ndims > MAXDIM)
3514 0 : ereport(ERROR,
3515 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
3516 : errmsg("number of array dimensions (%d) exceeds the maximum allowed (%d)",
3517 : ndims, MAXDIM)));
3518 :
3519 : /* This checks for overflow of the array dimensions */
3520 1573502 : nelems = ArrayGetNItems(ndims, dims);
3521 1573502 : ArrayCheckBounds(ndims, dims, lbs);
3522 :
3523 : /* if ndims <= 0 or any dims[i] == 0, return empty array */
3524 1573502 : if (nelems <= 0)
3525 57220 : return construct_empty_array(elmtype);
3526 :
3527 : /* compute required space */
3528 1516282 : nbytes = 0;
3529 1516282 : hasnulls = false;
3530 10999572 : for (i = 0; i < nelems; i++)
3531 : {
3532 9483290 : if (nulls && nulls[i])
3533 : {
3534 32752 : hasnulls = true;
3535 32752 : continue;
3536 : }
3537 : /* make sure data is not toasted */
3538 9450538 : if (elmlen == -1)
3539 2622716 : elems[i] = PointerGetDatum(PG_DETOAST_DATUM(elems[i]));
3540 9450538 : nbytes = att_addlength_datum(nbytes, elmlen, elems[i]);
3541 9450538 : nbytes = att_align_nominal(nbytes, elmalign);
3542 : /* check for overflow of total request */
3543 9450538 : if (!AllocSizeIsValid(nbytes))
3544 0 : ereport(ERROR,
3545 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
3546 : errmsg("array size exceeds the maximum allowed (%d)",
3547 : (int) MaxAllocSize)));
3548 : }
3549 :
3550 : /* Allocate and initialize result array */
3551 1516282 : if (hasnulls)
3552 : {
3553 17530 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndims, nelems);
3554 17530 : nbytes += dataoffset;
3555 : }
3556 : else
3557 : {
3558 1498752 : dataoffset = 0; /* marker for no null bitmap */
3559 1498752 : nbytes += ARR_OVERHEAD_NONULLS(ndims);
3560 : }
3561 1516282 : result = (ArrayType *) palloc0(nbytes);
3562 1516282 : SET_VARSIZE(result, nbytes);
3563 1516282 : result->ndim = ndims;
3564 1516282 : result->dataoffset = dataoffset;
3565 1516282 : result->elemtype = elmtype;
3566 1516282 : memcpy(ARR_DIMS(result), dims, ndims * sizeof(int));
3567 1516282 : memcpy(ARR_LBOUND(result), lbs, ndims * sizeof(int));
3568 :
3569 1516282 : CopyArrayEls(result,
3570 : elems, nulls, nelems,
3571 : elmlen, elmbyval, elmalign,
3572 : false);
3573 :
3574 1516282 : return result;
3575 : }
3576 :
3577 : /*
3578 : * construct_empty_array --- make a zero-dimensional array of given type
3579 : */
3580 : ArrayType *
3581 2846890 : construct_empty_array(Oid elmtype)
3582 : {
3583 : ArrayType *result;
3584 :
3585 2846890 : result = (ArrayType *) palloc0(sizeof(ArrayType));
3586 2846890 : SET_VARSIZE(result, sizeof(ArrayType));
3587 2846890 : result->ndim = 0;
3588 2846890 : result->dataoffset = 0;
3589 2846890 : result->elemtype = elmtype;
3590 2846890 : return result;
3591 : }
3592 :
3593 : /*
3594 : * construct_empty_expanded_array: make an empty expanded array
3595 : * given only type information. (metacache can be NULL if not needed.)
3596 : */
3597 : ExpandedArrayHeader *
3598 24 : construct_empty_expanded_array(Oid element_type,
3599 : MemoryContext parentcontext,
3600 : ArrayMetaState *metacache)
3601 : {
3602 24 : ArrayType *array = construct_empty_array(element_type);
3603 : Datum d;
3604 :
3605 24 : d = expand_array(PointerGetDatum(array), parentcontext, metacache);
3606 24 : pfree(array);
3607 24 : return (ExpandedArrayHeader *) DatumGetEOHP(d);
3608 : }
3609 :
3610 : /*
3611 : * deconstruct_array --- simple method for extracting data from an array
3612 : *
3613 : * array: array object to examine (must not be NULL)
3614 : * elmtype, elmlen, elmbyval, elmalign: info for the datatype of the items
3615 : * elemsp: return value, set to point to palloc'd array of Datum values
3616 : * nullsp: return value, set to point to palloc'd array of isnull markers
3617 : * nelemsp: return value, set to number of extracted values
3618 : *
3619 : * The caller may pass nullsp == NULL if it does not support NULLs in the
3620 : * array. Note that this produces a very uninformative error message,
3621 : * so do it only in cases where a NULL is really not expected.
3622 : *
3623 : * If array elements are pass-by-ref data type, the returned Datums will
3624 : * be pointers into the array object.
3625 : *
3626 : * NOTE: it would be cleaner to look up the elmlen/elmbval/elmalign info
3627 : * from the system catalogs, given the elmtype. However, the caller is
3628 : * in a better position to cache this info across multiple uses, or even
3629 : * to hard-wire values if the element type is hard-wired.
3630 : */
3631 : void
3632 3209094 : deconstruct_array(ArrayType *array,
3633 : Oid elmtype,
3634 : int elmlen, bool elmbyval, char elmalign,
3635 : Datum **elemsp, bool **nullsp, int *nelemsp)
3636 : {
3637 : Datum *elems;
3638 : bool *nulls;
3639 : int nelems;
3640 : char *p;
3641 : bits8 *bitmap;
3642 : int bitmask;
3643 : int i;
3644 :
3645 : Assert(ARR_ELEMTYPE(array) == elmtype);
3646 :
3647 3209094 : nelems = ArrayGetNItems(ARR_NDIM(array), ARR_DIMS(array));
3648 3209094 : *elemsp = elems = (Datum *) palloc(nelems * sizeof(Datum));
3649 3209094 : if (nullsp)
3650 1864396 : *nullsp = nulls = (bool *) palloc0(nelems * sizeof(bool));
3651 : else
3652 1344698 : nulls = NULL;
3653 3209094 : *nelemsp = nelems;
3654 :
3655 3209094 : p = ARR_DATA_PTR(array);
3656 3209094 : bitmap = ARR_NULLBITMAP(array);
3657 3209094 : bitmask = 1;
3658 :
3659 53974694 : for (i = 0; i < nelems; i++)
3660 : {
3661 : /* Get source element, checking for NULL */
3662 50765600 : if (bitmap && (*bitmap & bitmask) == 0)
3663 : {
3664 3230 : elems[i] = (Datum) 0;
3665 3230 : if (nulls)
3666 3230 : nulls[i] = true;
3667 : else
3668 0 : ereport(ERROR,
3669 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
3670 : errmsg("null array element not allowed in this context")));
3671 : }
3672 : else
3673 : {
3674 50762370 : elems[i] = fetch_att(p, elmbyval, elmlen);
3675 50762370 : p = att_addlength_pointer(p, elmlen, p);
3676 50762370 : p = (char *) att_align_nominal(p, elmalign);
3677 : }
3678 :
3679 : /* advance bitmap pointer if any */
3680 50765600 : if (bitmap)
3681 : {
3682 5064 : bitmask <<= 1;
3683 5064 : if (bitmask == 0x100)
3684 : {
3685 66 : bitmap++;
3686 66 : bitmask = 1;
3687 : }
3688 : }
3689 : }
3690 3209094 : }
3691 :
3692 : /*
3693 : * Like deconstruct_array(), where elmtype must be a built-in type, and
3694 : * elmlen/elmbyval/elmalign is looked up from hardcoded data. This is often
3695 : * useful when manipulating arrays from/for system catalogs.
3696 : */
3697 : void
3698 491324 : deconstruct_array_builtin(ArrayType *array,
3699 : Oid elmtype,
3700 : Datum **elemsp, bool **nullsp, int *nelemsp)
3701 : {
3702 : int elmlen;
3703 : bool elmbyval;
3704 : char elmalign;
3705 :
3706 491324 : switch (elmtype)
3707 : {
3708 18 : case CHAROID:
3709 18 : elmlen = 1;
3710 18 : elmbyval = true;
3711 18 : elmalign = TYPALIGN_CHAR;
3712 18 : break;
3713 :
3714 7878 : case CSTRINGOID:
3715 7878 : elmlen = -2;
3716 7878 : elmbyval = false;
3717 7878 : elmalign = TYPALIGN_CHAR;
3718 7878 : break;
3719 :
3720 30 : case FLOAT8OID:
3721 30 : elmlen = sizeof(float8);
3722 30 : elmbyval = true;
3723 30 : elmalign = TYPALIGN_DOUBLE;
3724 30 : break;
3725 :
3726 5486 : case INT2OID:
3727 5486 : elmlen = sizeof(int16);
3728 5486 : elmbyval = true;
3729 5486 : elmalign = TYPALIGN_SHORT;
3730 5486 : break;
3731 :
3732 308 : case OIDOID:
3733 308 : elmlen = sizeof(Oid);
3734 308 : elmbyval = true;
3735 308 : elmalign = TYPALIGN_INT;
3736 308 : break;
3737 :
3738 477560 : case TEXTOID:
3739 477560 : elmlen = -1;
3740 477560 : elmbyval = false;
3741 477560 : elmalign = TYPALIGN_INT;
3742 477560 : break;
3743 :
3744 44 : case TIDOID:
3745 44 : elmlen = sizeof(ItemPointerData);
3746 44 : elmbyval = false;
3747 44 : elmalign = TYPALIGN_SHORT;
3748 44 : break;
3749 :
3750 0 : default:
3751 0 : elog(ERROR, "type %u not supported by deconstruct_array_builtin()", elmtype);
3752 : /* keep compiler quiet */
3753 : elmlen = 0;
3754 : elmbyval = false;
3755 : elmalign = 0;
3756 : }
3757 :
3758 491324 : deconstruct_array(array, elmtype, elmlen, elmbyval, elmalign, elemsp, nullsp, nelemsp);
3759 491324 : }
3760 :
3761 : /*
3762 : * array_contains_nulls --- detect whether an array has any null elements
3763 : *
3764 : * This gives an accurate answer, whereas testing ARR_HASNULL only tells
3765 : * if the array *might* contain a null.
3766 : */
3767 : bool
3768 72434 : array_contains_nulls(ArrayType *array)
3769 : {
3770 : int nelems;
3771 : bits8 *bitmap;
3772 : int bitmask;
3773 :
3774 : /* Easy answer if there's no null bitmap */
3775 72434 : if (!ARR_HASNULL(array))
3776 72372 : return false;
3777 :
3778 62 : nelems = ArrayGetNItems(ARR_NDIM(array), ARR_DIMS(array));
3779 :
3780 62 : bitmap = ARR_NULLBITMAP(array);
3781 :
3782 : /* check whole bytes of the bitmap byte-at-a-time */
3783 62 : while (nelems >= 8)
3784 : {
3785 18 : if (*bitmap != 0xFF)
3786 18 : return true;
3787 0 : bitmap++;
3788 0 : nelems -= 8;
3789 : }
3790 :
3791 : /* check last partial byte */
3792 44 : bitmask = 1;
3793 98 : while (nelems > 0)
3794 : {
3795 98 : if ((*bitmap & bitmask) == 0)
3796 44 : return true;
3797 54 : bitmask <<= 1;
3798 54 : nelems--;
3799 : }
3800 :
3801 0 : return false;
3802 : }
3803 :
3804 :
3805 : /*
3806 : * array_eq :
3807 : * compares two arrays for equality
3808 : * result :
3809 : * returns true if the arrays are equal, false otherwise.
3810 : *
3811 : * Note: we do not use array_cmp here, since equality may be meaningful in
3812 : * datatypes that don't have a total ordering (and hence no btree support).
3813 : */
3814 : Datum
3815 248392 : array_eq(PG_FUNCTION_ARGS)
3816 : {
3817 248392 : LOCAL_FCINFO(locfcinfo, 2);
3818 248392 : AnyArrayType *array1 = PG_GETARG_ANY_ARRAY_P(0);
3819 248392 : AnyArrayType *array2 = PG_GETARG_ANY_ARRAY_P(1);
3820 248392 : Oid collation = PG_GET_COLLATION();
3821 248392 : int ndims1 = AARR_NDIM(array1);
3822 248392 : int ndims2 = AARR_NDIM(array2);
3823 248392 : int *dims1 = AARR_DIMS(array1);
3824 248392 : int *dims2 = AARR_DIMS(array2);
3825 248392 : int *lbs1 = AARR_LBOUND(array1);
3826 248392 : int *lbs2 = AARR_LBOUND(array2);
3827 248392 : Oid element_type = AARR_ELEMTYPE(array1);
3828 248392 : bool result = true;
3829 : int nitems;
3830 : TypeCacheEntry *typentry;
3831 : int typlen;
3832 : bool typbyval;
3833 : char typalign;
3834 : array_iter it1;
3835 : array_iter it2;
3836 : int i;
3837 :
3838 248392 : if (element_type != AARR_ELEMTYPE(array2))
3839 0 : ereport(ERROR,
3840 : (errcode(ERRCODE_DATATYPE_MISMATCH),
3841 : errmsg("cannot compare arrays of different element types")));
3842 :
3843 : /* fast path if the arrays do not have the same dimensionality */
3844 248392 : if (ndims1 != ndims2 ||
3845 233612 : memcmp(dims1, dims2, ndims1 * sizeof(int)) != 0 ||
3846 172244 : memcmp(lbs1, lbs2, ndims1 * sizeof(int)) != 0)
3847 76148 : result = false;
3848 : else
3849 : {
3850 : /*
3851 : * We arrange to look up the equality function only once per series of
3852 : * calls, assuming the element type doesn't change underneath us. The
3853 : * typcache is used so that we have no memory leakage when being used
3854 : * as an index support function.
3855 : */
3856 172244 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
3857 172244 : if (typentry == NULL ||
3858 169466 : typentry->type_id != element_type)
3859 : {
3860 2778 : typentry = lookup_type_cache(element_type,
3861 : TYPECACHE_EQ_OPR_FINFO);
3862 2778 : if (!OidIsValid(typentry->eq_opr_finfo.fn_oid))
3863 0 : ereport(ERROR,
3864 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
3865 : errmsg("could not identify an equality operator for type %s",
3866 : format_type_be(element_type))));
3867 2778 : fcinfo->flinfo->fn_extra = typentry;
3868 : }
3869 172244 : typlen = typentry->typlen;
3870 172244 : typbyval = typentry->typbyval;
3871 172244 : typalign = typentry->typalign;
3872 :
3873 : /*
3874 : * apply the operator to each pair of array elements.
3875 : */
3876 172244 : InitFunctionCallInfoData(*locfcinfo, &typentry->eq_opr_finfo, 2,
3877 : collation, NULL, NULL);
3878 :
3879 : /* Loop over source data */
3880 172244 : nitems = ArrayGetNItems(ndims1, dims1);
3881 172244 : array_iter_setup(&it1, array1);
3882 172244 : array_iter_setup(&it2, array2);
3883 :
3884 405768 : for (i = 0; i < nitems; i++)
3885 : {
3886 : Datum elt1;
3887 : Datum elt2;
3888 : bool isnull1;
3889 : bool isnull2;
3890 : bool oprresult;
3891 :
3892 : /* Get elements, checking for NULL */
3893 265082 : elt1 = array_iter_next(&it1, &isnull1, i,
3894 : typlen, typbyval, typalign);
3895 265082 : elt2 = array_iter_next(&it2, &isnull2, i,
3896 : typlen, typbyval, typalign);
3897 :
3898 : /*
3899 : * We consider two NULLs equal; NULL and not-NULL are unequal.
3900 : */
3901 265082 : if (isnull1 && isnull2)
3902 22 : continue;
3903 265060 : if (isnull1 || isnull2)
3904 : {
3905 108 : result = false;
3906 31558 : break;
3907 : }
3908 :
3909 : /*
3910 : * Apply the operator to the element pair; treat NULL as false
3911 : */
3912 264952 : locfcinfo->args[0].value = elt1;
3913 264952 : locfcinfo->args[0].isnull = false;
3914 264952 : locfcinfo->args[1].value = elt2;
3915 264952 : locfcinfo->args[1].isnull = false;
3916 264952 : locfcinfo->isnull = false;
3917 264952 : oprresult = DatumGetBool(FunctionCallInvoke(locfcinfo));
3918 264952 : if (locfcinfo->isnull || !oprresult)
3919 : {
3920 31450 : result = false;
3921 31450 : break;
3922 : }
3923 : }
3924 : }
3925 :
3926 : /* Avoid leaking memory when handed toasted input. */
3927 248392 : AARR_FREE_IF_COPY(array1, 0);
3928 248392 : AARR_FREE_IF_COPY(array2, 1);
3929 :
3930 248392 : PG_RETURN_BOOL(result);
3931 : }
3932 :
3933 :
3934 : /*-----------------------------------------------------------------------------
3935 : * array-array bool operators:
3936 : * Given two arrays, iterate comparison operators
3937 : * over the array. Uses logic similar to text comparison
3938 : * functions, except element-by-element instead of
3939 : * character-by-character.
3940 : *----------------------------------------------------------------------------
3941 : */
3942 :
3943 : Datum
3944 984 : array_ne(PG_FUNCTION_ARGS)
3945 : {
3946 984 : PG_RETURN_BOOL(!DatumGetBool(array_eq(fcinfo)));
3947 : }
3948 :
3949 : Datum
3950 7546 : array_lt(PG_FUNCTION_ARGS)
3951 : {
3952 7546 : PG_RETURN_BOOL(array_cmp(fcinfo) < 0);
3953 : }
3954 :
3955 : Datum
3956 18 : array_gt(PG_FUNCTION_ARGS)
3957 : {
3958 18 : PG_RETURN_BOOL(array_cmp(fcinfo) > 0);
3959 : }
3960 :
3961 : Datum
3962 18 : array_le(PG_FUNCTION_ARGS)
3963 : {
3964 18 : PG_RETURN_BOOL(array_cmp(fcinfo) <= 0);
3965 : }
3966 :
3967 : Datum
3968 18 : array_ge(PG_FUNCTION_ARGS)
3969 : {
3970 18 : PG_RETURN_BOOL(array_cmp(fcinfo) >= 0);
3971 : }
3972 :
3973 : Datum
3974 8264916 : btarraycmp(PG_FUNCTION_ARGS)
3975 : {
3976 8264916 : PG_RETURN_INT32(array_cmp(fcinfo));
3977 : }
3978 :
3979 : /*
3980 : * array_cmp()
3981 : * Internal comparison function for arrays.
3982 : *
3983 : * Returns -1, 0 or 1
3984 : */
3985 : static int
3986 8273062 : array_cmp(FunctionCallInfo fcinfo)
3987 : {
3988 8273062 : LOCAL_FCINFO(locfcinfo, 2);
3989 8273062 : AnyArrayType *array1 = PG_GETARG_ANY_ARRAY_P(0);
3990 8273062 : AnyArrayType *array2 = PG_GETARG_ANY_ARRAY_P(1);
3991 8273062 : Oid collation = PG_GET_COLLATION();
3992 8273062 : int ndims1 = AARR_NDIM(array1);
3993 8273062 : int ndims2 = AARR_NDIM(array2);
3994 8273062 : int *dims1 = AARR_DIMS(array1);
3995 8273062 : int *dims2 = AARR_DIMS(array2);
3996 8273062 : int nitems1 = ArrayGetNItems(ndims1, dims1);
3997 8273062 : int nitems2 = ArrayGetNItems(ndims2, dims2);
3998 8273062 : Oid element_type = AARR_ELEMTYPE(array1);
3999 8273062 : int result = 0;
4000 : TypeCacheEntry *typentry;
4001 : int typlen;
4002 : bool typbyval;
4003 : char typalign;
4004 : int min_nitems;
4005 : array_iter it1;
4006 : array_iter it2;
4007 : int i;
4008 :
4009 8273062 : if (element_type != AARR_ELEMTYPE(array2))
4010 6 : ereport(ERROR,
4011 : (errcode(ERRCODE_DATATYPE_MISMATCH),
4012 : errmsg("cannot compare arrays of different element types")));
4013 :
4014 : /*
4015 : * We arrange to look up the comparison function only once per series of
4016 : * calls, assuming the element type doesn't change underneath us. The
4017 : * typcache is used so that we have no memory leakage when being used as
4018 : * an index support function.
4019 : */
4020 8273056 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
4021 8273056 : if (typentry == NULL ||
4022 8270526 : typentry->type_id != element_type)
4023 : {
4024 2530 : typentry = lookup_type_cache(element_type,
4025 : TYPECACHE_CMP_PROC_FINFO);
4026 2530 : if (!OidIsValid(typentry->cmp_proc_finfo.fn_oid))
4027 6 : ereport(ERROR,
4028 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
4029 : errmsg("could not identify a comparison function for type %s",
4030 : format_type_be(element_type))));
4031 2524 : fcinfo->flinfo->fn_extra = typentry;
4032 : }
4033 8273050 : typlen = typentry->typlen;
4034 8273050 : typbyval = typentry->typbyval;
4035 8273050 : typalign = typentry->typalign;
4036 :
4037 : /*
4038 : * apply the operator to each pair of array elements.
4039 : */
4040 8273050 : InitFunctionCallInfoData(*locfcinfo, &typentry->cmp_proc_finfo, 2,
4041 : collation, NULL, NULL);
4042 :
4043 : /* Loop over source data */
4044 8273050 : min_nitems = Min(nitems1, nitems2);
4045 8273050 : array_iter_setup(&it1, array1);
4046 8273050 : array_iter_setup(&it2, array2);
4047 :
4048 17139508 : for (i = 0; i < min_nitems; i++)
4049 : {
4050 : Datum elt1;
4051 : Datum elt2;
4052 : bool isnull1;
4053 : bool isnull2;
4054 : int32 cmpresult;
4055 :
4056 : /* Get elements, checking for NULL */
4057 14778176 : elt1 = array_iter_next(&it1, &isnull1, i, typlen, typbyval, typalign);
4058 14778176 : elt2 = array_iter_next(&it2, &isnull2, i, typlen, typbyval, typalign);
4059 :
4060 : /*
4061 : * We consider two NULLs equal; NULL > not-NULL.
4062 : */
4063 14778176 : if (isnull1 && isnull2)
4064 8866458 : continue;
4065 14778166 : if (isnull1)
4066 : {
4067 : /* arg1 is greater than arg2 */
4068 128 : result = 1;
4069 5911718 : break;
4070 : }
4071 14778038 : if (isnull2)
4072 : {
4073 : /* arg1 is less than arg2 */
4074 240 : result = -1;
4075 240 : break;
4076 : }
4077 :
4078 : /* Compare the pair of elements */
4079 14777798 : locfcinfo->args[0].value = elt1;
4080 14777798 : locfcinfo->args[0].isnull = false;
4081 14777798 : locfcinfo->args[1].value = elt2;
4082 14777798 : locfcinfo->args[1].isnull = false;
4083 14777798 : cmpresult = DatumGetInt32(FunctionCallInvoke(locfcinfo));
4084 :
4085 : /* We don't expect comparison support functions to return null */
4086 : Assert(!locfcinfo->isnull);
4087 :
4088 14777798 : if (cmpresult == 0)
4089 8866448 : continue; /* equal */
4090 :
4091 5911350 : if (cmpresult < 0)
4092 : {
4093 : /* arg1 is less than arg2 */
4094 3366058 : result = -1;
4095 3366058 : break;
4096 : }
4097 : else
4098 : {
4099 : /* arg1 is greater than arg2 */
4100 2545292 : result = 1;
4101 2545292 : break;
4102 : }
4103 : }
4104 :
4105 : /*
4106 : * If arrays contain same data (up to end of shorter one), apply
4107 : * additional rules to sort by dimensionality. The relative significance
4108 : * of the different bits of information is historical; mainly we just care
4109 : * that we don't say "equal" for arrays of different dimensionality.
4110 : */
4111 8273050 : if (result == 0)
4112 : {
4113 2361332 : if (nitems1 != nitems2)
4114 201088 : result = (nitems1 < nitems2) ? -1 : 1;
4115 2160244 : else if (ndims1 != ndims2)
4116 0 : result = (ndims1 < ndims2) ? -1 : 1;
4117 : else
4118 : {
4119 4320436 : for (i = 0; i < ndims1; i++)
4120 : {
4121 2160192 : if (dims1[i] != dims2[i])
4122 : {
4123 0 : result = (dims1[i] < dims2[i]) ? -1 : 1;
4124 0 : break;
4125 : }
4126 : }
4127 2160244 : if (result == 0)
4128 : {
4129 2160244 : int *lbound1 = AARR_LBOUND(array1);
4130 2160244 : int *lbound2 = AARR_LBOUND(array2);
4131 :
4132 4320436 : for (i = 0; i < ndims1; i++)
4133 : {
4134 2160192 : if (lbound1[i] != lbound2[i])
4135 : {
4136 0 : result = (lbound1[i] < lbound2[i]) ? -1 : 1;
4137 0 : break;
4138 : }
4139 : }
4140 : }
4141 : }
4142 : }
4143 :
4144 : /* Avoid leaking memory when handed toasted input. */
4145 8273050 : AARR_FREE_IF_COPY(array1, 0);
4146 8273050 : AARR_FREE_IF_COPY(array2, 1);
4147 :
4148 8273050 : return result;
4149 : }
4150 :
4151 :
4152 : /*-----------------------------------------------------------------------------
4153 : * array hashing
4154 : * Hash the elements and combine the results.
4155 : *----------------------------------------------------------------------------
4156 : */
4157 :
4158 : Datum
4159 64116 : hash_array(PG_FUNCTION_ARGS)
4160 : {
4161 64116 : LOCAL_FCINFO(locfcinfo, 1);
4162 64116 : AnyArrayType *array = PG_GETARG_ANY_ARRAY_P(0);
4163 64116 : int ndims = AARR_NDIM(array);
4164 64116 : int *dims = AARR_DIMS(array);
4165 64116 : Oid element_type = AARR_ELEMTYPE(array);
4166 64116 : uint32 result = 1;
4167 : int nitems;
4168 : TypeCacheEntry *typentry;
4169 : int typlen;
4170 : bool typbyval;
4171 : char typalign;
4172 : int i;
4173 : array_iter iter;
4174 :
4175 : /*
4176 : * We arrange to look up the hash function only once per series of calls,
4177 : * assuming the element type doesn't change underneath us. The typcache
4178 : * is used so that we have no memory leakage when being used as an index
4179 : * support function.
4180 : */
4181 64116 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
4182 64116 : if (typentry == NULL ||
4183 63538 : typentry->type_id != element_type)
4184 : {
4185 578 : typentry = lookup_type_cache(element_type,
4186 : TYPECACHE_HASH_PROC_FINFO);
4187 578 : if (!OidIsValid(typentry->hash_proc_finfo.fn_oid) && element_type != RECORDOID)
4188 6 : ereport(ERROR,
4189 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
4190 : errmsg("could not identify a hash function for type %s",
4191 : format_type_be(element_type))));
4192 :
4193 : /*
4194 : * The type cache doesn't believe that record is hashable (see
4195 : * cache_record_field_properties()), but since we're here, we're
4196 : * committed to hashing, so we can assume it does. Worst case, if any
4197 : * components of the record don't support hashing, we will fail at
4198 : * execution.
4199 : */
4200 572 : if (element_type == RECORDOID)
4201 : {
4202 : MemoryContext oldcontext;
4203 : TypeCacheEntry *record_typentry;
4204 :
4205 18 : oldcontext = MemoryContextSwitchTo(fcinfo->flinfo->fn_mcxt);
4206 :
4207 : /*
4208 : * Make fake type cache entry structure. Note that we can't just
4209 : * modify typentry, since that points directly into the type
4210 : * cache.
4211 : */
4212 18 : record_typentry = palloc0(sizeof(*record_typentry));
4213 18 : record_typentry->type_id = element_type;
4214 :
4215 : /* fill in what we need below */
4216 18 : record_typentry->typlen = typentry->typlen;
4217 18 : record_typentry->typbyval = typentry->typbyval;
4218 18 : record_typentry->typalign = typentry->typalign;
4219 18 : fmgr_info(F_HASH_RECORD, &record_typentry->hash_proc_finfo);
4220 :
4221 18 : MemoryContextSwitchTo(oldcontext);
4222 :
4223 18 : typentry = record_typentry;
4224 : }
4225 :
4226 572 : fcinfo->flinfo->fn_extra = typentry;
4227 : }
4228 :
4229 64110 : typlen = typentry->typlen;
4230 64110 : typbyval = typentry->typbyval;
4231 64110 : typalign = typentry->typalign;
4232 :
4233 : /*
4234 : * apply the hash function to each array element.
4235 : */
4236 64110 : InitFunctionCallInfoData(*locfcinfo, &typentry->hash_proc_finfo, 1,
4237 : PG_GET_COLLATION(), NULL, NULL);
4238 :
4239 : /* Loop over source data */
4240 64110 : nitems = ArrayGetNItems(ndims, dims);
4241 64110 : array_iter_setup(&iter, array);
4242 :
4243 164146 : for (i = 0; i < nitems; i++)
4244 : {
4245 : Datum elt;
4246 : bool isnull;
4247 : uint32 elthash;
4248 :
4249 : /* Get element, checking for NULL */
4250 100036 : elt = array_iter_next(&iter, &isnull, i, typlen, typbyval, typalign);
4251 :
4252 100036 : if (isnull)
4253 : {
4254 : /* Treat nulls as having hashvalue 0 */
4255 0 : elthash = 0;
4256 : }
4257 : else
4258 : {
4259 : /* Apply the hash function */
4260 100036 : locfcinfo->args[0].value = elt;
4261 100036 : locfcinfo->args[0].isnull = false;
4262 100036 : elthash = DatumGetUInt32(FunctionCallInvoke(locfcinfo));
4263 : /* We don't expect hash functions to return null */
4264 : Assert(!locfcinfo->isnull);
4265 : }
4266 :
4267 : /*
4268 : * Combine hash values of successive elements by multiplying the
4269 : * current value by 31 and adding on the new element's hash value.
4270 : *
4271 : * The result is a sum in which each element's hash value is
4272 : * multiplied by a different power of 31. This is modulo 2^32
4273 : * arithmetic, and the powers of 31 modulo 2^32 form a cyclic group of
4274 : * order 2^27. So for arrays of up to 2^27 elements, each element's
4275 : * hash value is multiplied by a different (odd) number, resulting in
4276 : * a good mixing of all the elements' hash values.
4277 : */
4278 100036 : result = (result << 5) - result + elthash;
4279 : }
4280 :
4281 : /* Avoid leaking memory when handed toasted input. */
4282 64110 : AARR_FREE_IF_COPY(array, 0);
4283 :
4284 64110 : PG_RETURN_UINT32(result);
4285 : }
4286 :
4287 : /*
4288 : * Returns 64-bit value by hashing a value to a 64-bit value, with a seed.
4289 : * Otherwise, similar to hash_array.
4290 : */
4291 : Datum
4292 144 : hash_array_extended(PG_FUNCTION_ARGS)
4293 : {
4294 144 : LOCAL_FCINFO(locfcinfo, 2);
4295 144 : AnyArrayType *array = PG_GETARG_ANY_ARRAY_P(0);
4296 144 : uint64 seed = PG_GETARG_INT64(1);
4297 144 : int ndims = AARR_NDIM(array);
4298 144 : int *dims = AARR_DIMS(array);
4299 144 : Oid element_type = AARR_ELEMTYPE(array);
4300 144 : uint64 result = 1;
4301 : int nitems;
4302 : TypeCacheEntry *typentry;
4303 : int typlen;
4304 : bool typbyval;
4305 : char typalign;
4306 : int i;
4307 : array_iter iter;
4308 :
4309 144 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
4310 144 : if (typentry == NULL ||
4311 84 : typentry->type_id != element_type)
4312 : {
4313 60 : typentry = lookup_type_cache(element_type,
4314 : TYPECACHE_HASH_EXTENDED_PROC_FINFO);
4315 60 : if (!OidIsValid(typentry->hash_extended_proc_finfo.fn_oid))
4316 6 : ereport(ERROR,
4317 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
4318 : errmsg("could not identify an extended hash function for type %s",
4319 : format_type_be(element_type))));
4320 54 : fcinfo->flinfo->fn_extra = typentry;
4321 : }
4322 138 : typlen = typentry->typlen;
4323 138 : typbyval = typentry->typbyval;
4324 138 : typalign = typentry->typalign;
4325 :
4326 138 : InitFunctionCallInfoData(*locfcinfo, &typentry->hash_extended_proc_finfo, 2,
4327 : PG_GET_COLLATION(), NULL, NULL);
4328 :
4329 : /* Loop over source data */
4330 138 : nitems = ArrayGetNItems(ndims, dims);
4331 138 : array_iter_setup(&iter, array);
4332 :
4333 456 : for (i = 0; i < nitems; i++)
4334 : {
4335 : Datum elt;
4336 : bool isnull;
4337 : uint64 elthash;
4338 :
4339 : /* Get element, checking for NULL */
4340 318 : elt = array_iter_next(&iter, &isnull, i, typlen, typbyval, typalign);
4341 :
4342 318 : if (isnull)
4343 : {
4344 0 : elthash = 0;
4345 : }
4346 : else
4347 : {
4348 : /* Apply the hash function */
4349 318 : locfcinfo->args[0].value = elt;
4350 318 : locfcinfo->args[0].isnull = false;
4351 318 : locfcinfo->args[1].value = Int64GetDatum(seed);
4352 318 : locfcinfo->args[1].isnull = false;
4353 318 : elthash = DatumGetUInt64(FunctionCallInvoke(locfcinfo));
4354 : /* We don't expect hash functions to return null */
4355 : Assert(!locfcinfo->isnull);
4356 : }
4357 :
4358 318 : result = (result << 5) - result + elthash;
4359 : }
4360 :
4361 138 : AARR_FREE_IF_COPY(array, 0);
4362 :
4363 138 : PG_RETURN_UINT64(result);
4364 : }
4365 :
4366 :
4367 : /*-----------------------------------------------------------------------------
4368 : * array overlap/containment comparisons
4369 : * These use the same methods of comparing array elements as array_eq.
4370 : * We consider only the elements of the arrays, ignoring dimensionality.
4371 : *----------------------------------------------------------------------------
4372 : */
4373 :
4374 : /*
4375 : * array_contain_compare :
4376 : * compares two arrays for overlap/containment
4377 : *
4378 : * When matchall is true, return true if all members of array1 are in array2.
4379 : * When matchall is false, return true if any members of array1 are in array2.
4380 : */
4381 : static bool
4382 22458 : array_contain_compare(AnyArrayType *array1, AnyArrayType *array2, Oid collation,
4383 : bool matchall, void **fn_extra)
4384 : {
4385 22458 : LOCAL_FCINFO(locfcinfo, 2);
4386 22458 : bool result = matchall;
4387 22458 : Oid element_type = AARR_ELEMTYPE(array1);
4388 : TypeCacheEntry *typentry;
4389 : int nelems1;
4390 : Datum *values2;
4391 : bool *nulls2;
4392 : int nelems2;
4393 : int typlen;
4394 : bool typbyval;
4395 : char typalign;
4396 : int i;
4397 : int j;
4398 : array_iter it1;
4399 :
4400 22458 : if (element_type != AARR_ELEMTYPE(array2))
4401 0 : ereport(ERROR,
4402 : (errcode(ERRCODE_DATATYPE_MISMATCH),
4403 : errmsg("cannot compare arrays of different element types")));
4404 :
4405 : /*
4406 : * We arrange to look up the equality function only once per series of
4407 : * calls, assuming the element type doesn't change underneath us. The
4408 : * typcache is used so that we have no memory leakage when being used as
4409 : * an index support function.
4410 : */
4411 22458 : typentry = (TypeCacheEntry *) *fn_extra;
4412 22458 : if (typentry == NULL ||
4413 21894 : typentry->type_id != element_type)
4414 : {
4415 564 : typentry = lookup_type_cache(element_type,
4416 : TYPECACHE_EQ_OPR_FINFO);
4417 564 : if (!OidIsValid(typentry->eq_opr_finfo.fn_oid))
4418 0 : ereport(ERROR,
4419 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
4420 : errmsg("could not identify an equality operator for type %s",
4421 : format_type_be(element_type))));
4422 564 : *fn_extra = typentry;
4423 : }
4424 22458 : typlen = typentry->typlen;
4425 22458 : typbyval = typentry->typbyval;
4426 22458 : typalign = typentry->typalign;
4427 :
4428 : /*
4429 : * Since we probably will need to scan array2 multiple times, it's
4430 : * worthwhile to use deconstruct_array on it. We scan array1 the hard way
4431 : * however, since we very likely won't need to look at all of it.
4432 : */
4433 22458 : if (VARATT_IS_EXPANDED_HEADER(array2))
4434 : {
4435 : /* This should be safe even if input is read-only */
4436 5112 : deconstruct_expanded_array(&(array2->xpn));
4437 5112 : values2 = array2->xpn.dvalues;
4438 5112 : nulls2 = array2->xpn.dnulls;
4439 5112 : nelems2 = array2->xpn.nelems;
4440 : }
4441 : else
4442 17346 : deconstruct_array((ArrayType *) array2,
4443 : element_type, typlen, typbyval, typalign,
4444 : &values2, &nulls2, &nelems2);
4445 :
4446 : /*
4447 : * Apply the comparison operator to each pair of array elements.
4448 : */
4449 22458 : InitFunctionCallInfoData(*locfcinfo, &typentry->eq_opr_finfo, 2,
4450 : collation, NULL, NULL);
4451 :
4452 : /* Loop over source data */
4453 22458 : nelems1 = ArrayGetNItems(AARR_NDIM(array1), AARR_DIMS(array1));
4454 22458 : array_iter_setup(&it1, array1);
4455 :
4456 420334 : for (i = 0; i < nelems1; i++)
4457 : {
4458 : Datum elt1;
4459 : bool isnull1;
4460 :
4461 : /* Get element, checking for NULL */
4462 407322 : elt1 = array_iter_next(&it1, &isnull1, i, typlen, typbyval, typalign);
4463 :
4464 : /*
4465 : * We assume that the comparison operator is strict, so a NULL can't
4466 : * match anything. XXX this diverges from the "NULL=NULL" behavior of
4467 : * array_eq, should we act like that?
4468 : */
4469 407322 : if (isnull1)
4470 : {
4471 1320 : if (matchall)
4472 : {
4473 1260 : result = false;
4474 9446 : break;
4475 : }
4476 60 : continue;
4477 : }
4478 :
4479 18416374 : for (j = 0; j < nelems2; j++)
4480 : {
4481 18376970 : Datum elt2 = values2[j];
4482 18376970 : bool isnull2 = nulls2 ? nulls2[j] : false;
4483 : bool oprresult;
4484 :
4485 18376970 : if (isnull2)
4486 7212 : continue; /* can't match */
4487 :
4488 : /*
4489 : * Apply the operator to the element pair; treat NULL as false
4490 : */
4491 18369758 : locfcinfo->args[0].value = elt1;
4492 18369758 : locfcinfo->args[0].isnull = false;
4493 18369758 : locfcinfo->args[1].value = elt2;
4494 18369758 : locfcinfo->args[1].isnull = false;
4495 18369758 : locfcinfo->isnull = false;
4496 18369758 : oprresult = DatumGetBool(FunctionCallInvoke(locfcinfo));
4497 18369758 : if (!locfcinfo->isnull && oprresult)
4498 366598 : break;
4499 : }
4500 :
4501 406002 : if (j < nelems2)
4502 : {
4503 : /* found a match for elt1 */
4504 366598 : if (!matchall)
4505 : {
4506 228 : result = true;
4507 228 : break;
4508 : }
4509 : }
4510 : else
4511 : {
4512 : /* no match for elt1 */
4513 39404 : if (matchall)
4514 : {
4515 7958 : result = false;
4516 7958 : break;
4517 : }
4518 : }
4519 : }
4520 :
4521 22458 : return result;
4522 : }
4523 :
4524 : Datum
4525 6120 : arrayoverlap(PG_FUNCTION_ARGS)
4526 : {
4527 6120 : AnyArrayType *array1 = PG_GETARG_ANY_ARRAY_P(0);
4528 6120 : AnyArrayType *array2 = PG_GETARG_ANY_ARRAY_P(1);
4529 6120 : Oid collation = PG_GET_COLLATION();
4530 : bool result;
4531 :
4532 6120 : result = array_contain_compare(array1, array2, collation, false,
4533 6120 : &fcinfo->flinfo->fn_extra);
4534 :
4535 : /* Avoid leaking memory when handed toasted input. */
4536 6120 : AARR_FREE_IF_COPY(array1, 0);
4537 6120 : AARR_FREE_IF_COPY(array2, 1);
4538 :
4539 6120 : PG_RETURN_BOOL(result);
4540 : }
4541 :
4542 : Datum
4543 9822 : arraycontains(PG_FUNCTION_ARGS)
4544 : {
4545 9822 : AnyArrayType *array1 = PG_GETARG_ANY_ARRAY_P(0);
4546 9822 : AnyArrayType *array2 = PG_GETARG_ANY_ARRAY_P(1);
4547 9822 : Oid collation = PG_GET_COLLATION();
4548 : bool result;
4549 :
4550 9822 : result = array_contain_compare(array2, array1, collation, true,
4551 9822 : &fcinfo->flinfo->fn_extra);
4552 :
4553 : /* Avoid leaking memory when handed toasted input. */
4554 9822 : AARR_FREE_IF_COPY(array1, 0);
4555 9822 : AARR_FREE_IF_COPY(array2, 1);
4556 :
4557 9822 : PG_RETURN_BOOL(result);
4558 : }
4559 :
4560 : Datum
4561 6516 : arraycontained(PG_FUNCTION_ARGS)
4562 : {
4563 6516 : AnyArrayType *array1 = PG_GETARG_ANY_ARRAY_P(0);
4564 6516 : AnyArrayType *array2 = PG_GETARG_ANY_ARRAY_P(1);
4565 6516 : Oid collation = PG_GET_COLLATION();
4566 : bool result;
4567 :
4568 6516 : result = array_contain_compare(array1, array2, collation, true,
4569 6516 : &fcinfo->flinfo->fn_extra);
4570 :
4571 : /* Avoid leaking memory when handed toasted input. */
4572 6516 : AARR_FREE_IF_COPY(array1, 0);
4573 6516 : AARR_FREE_IF_COPY(array2, 1);
4574 :
4575 6516 : PG_RETURN_BOOL(result);
4576 : }
4577 :
4578 :
4579 : /*-----------------------------------------------------------------------------
4580 : * Array iteration functions
4581 : * These functions are used to iterate efficiently through arrays
4582 : *-----------------------------------------------------------------------------
4583 : */
4584 :
4585 : /*
4586 : * array_create_iterator --- set up to iterate through an array
4587 : *
4588 : * If slice_ndim is zero, we will iterate element-by-element; the returned
4589 : * datums are of the array's element type.
4590 : *
4591 : * If slice_ndim is 1..ARR_NDIM(arr), we will iterate by slices: the
4592 : * returned datums are of the same array type as 'arr', but of size
4593 : * equal to the rightmost N dimensions of 'arr'.
4594 : *
4595 : * The passed-in array must remain valid for the lifetime of the iterator.
4596 : */
4597 : ArrayIterator
4598 578 : array_create_iterator(ArrayType *arr, int slice_ndim, ArrayMetaState *mstate)
4599 : {
4600 578 : ArrayIterator iterator = palloc0(sizeof(ArrayIteratorData));
4601 :
4602 : /*
4603 : * Sanity-check inputs --- caller should have got this right already
4604 : */
4605 : Assert(arr);
4606 578 : if (slice_ndim < 0 || slice_ndim > ARR_NDIM(arr))
4607 0 : elog(ERROR, "invalid arguments to array_create_iterator");
4608 :
4609 : /*
4610 : * Remember basic info about the array and its element type
4611 : */
4612 578 : iterator->arr = arr;
4613 578 : iterator->nullbitmap = ARR_NULLBITMAP(arr);
4614 578 : iterator->nitems = ArrayGetNItems(ARR_NDIM(arr), ARR_DIMS(arr));
4615 :
4616 578 : if (mstate != NULL)
4617 : {
4618 : Assert(mstate->element_type == ARR_ELEMTYPE(arr));
4619 :
4620 384 : iterator->typlen = mstate->typlen;
4621 384 : iterator->typbyval = mstate->typbyval;
4622 384 : iterator->typalign = mstate->typalign;
4623 : }
4624 : else
4625 194 : get_typlenbyvalalign(ARR_ELEMTYPE(arr),
4626 : &iterator->typlen,
4627 : &iterator->typbyval,
4628 : &iterator->typalign);
4629 :
4630 : /*
4631 : * Remember the slicing parameters.
4632 : */
4633 578 : iterator->slice_ndim = slice_ndim;
4634 :
4635 578 : if (slice_ndim > 0)
4636 : {
4637 : /*
4638 : * Get pointers into the array's dims and lbound arrays to represent
4639 : * the dims/lbound arrays of a slice. These are the same as the
4640 : * rightmost N dimensions of the array.
4641 : */
4642 78 : iterator->slice_dims = ARR_DIMS(arr) + ARR_NDIM(arr) - slice_ndim;
4643 78 : iterator->slice_lbound = ARR_LBOUND(arr) + ARR_NDIM(arr) - slice_ndim;
4644 :
4645 : /*
4646 : * Compute number of elements in a slice.
4647 : */
4648 156 : iterator->slice_len = ArrayGetNItems(slice_ndim,
4649 78 : iterator->slice_dims);
4650 :
4651 : /*
4652 : * Create workspace for building sub-arrays.
4653 : */
4654 78 : iterator->slice_values = (Datum *)
4655 78 : palloc(iterator->slice_len * sizeof(Datum));
4656 78 : iterator->slice_nulls = (bool *)
4657 78 : palloc(iterator->slice_len * sizeof(bool));
4658 : }
4659 :
4660 : /*
4661 : * Initialize our data pointer and linear element number. These will
4662 : * advance through the array during array_iterate().
4663 : */
4664 578 : iterator->data_ptr = ARR_DATA_PTR(arr);
4665 578 : iterator->current_item = 0;
4666 :
4667 578 : return iterator;
4668 : }
4669 :
4670 : /*
4671 : * Iterate through the array referenced by 'iterator'.
4672 : *
4673 : * As long as there is another element (or slice), return it into
4674 : * *value / *isnull, and return true. Return false when no more data.
4675 : */
4676 : bool
4677 9424 : array_iterate(ArrayIterator iterator, Datum *value, bool *isnull)
4678 : {
4679 : /* Done if we have reached the end of the array */
4680 9424 : if (iterator->current_item >= iterator->nitems)
4681 398 : return false;
4682 :
4683 9026 : if (iterator->slice_ndim == 0)
4684 : {
4685 : /*
4686 : * Scalar case: return one element.
4687 : */
4688 8870 : if (array_get_isnull(iterator->nullbitmap, iterator->current_item++))
4689 : {
4690 66 : *isnull = true;
4691 66 : *value = (Datum) 0;
4692 : }
4693 : else
4694 : {
4695 : /* non-NULL, so fetch the individual Datum to return */
4696 8804 : char *p = iterator->data_ptr;
4697 :
4698 8804 : *isnull = false;
4699 8804 : *value = fetch_att(p, iterator->typbyval, iterator->typlen);
4700 :
4701 : /* Move our data pointer forward to the next element */
4702 8804 : p = att_addlength_pointer(p, iterator->typlen, p);
4703 8804 : p = (char *) att_align_nominal(p, iterator->typalign);
4704 8804 : iterator->data_ptr = p;
4705 : }
4706 : }
4707 : else
4708 : {
4709 : /*
4710 : * Slice case: build and return an array of the requested size.
4711 : */
4712 : ArrayType *result;
4713 156 : Datum *values = iterator->slice_values;
4714 156 : bool *nulls = iterator->slice_nulls;
4715 156 : char *p = iterator->data_ptr;
4716 : int i;
4717 :
4718 498 : for (i = 0; i < iterator->slice_len; i++)
4719 : {
4720 342 : if (array_get_isnull(iterator->nullbitmap,
4721 342 : iterator->current_item++))
4722 : {
4723 0 : nulls[i] = true;
4724 0 : values[i] = (Datum) 0;
4725 : }
4726 : else
4727 : {
4728 342 : nulls[i] = false;
4729 342 : values[i] = fetch_att(p, iterator->typbyval, iterator->typlen);
4730 :
4731 : /* Move our data pointer forward to the next element */
4732 342 : p = att_addlength_pointer(p, iterator->typlen, p);
4733 342 : p = (char *) att_align_nominal(p, iterator->typalign);
4734 : }
4735 : }
4736 :
4737 156 : iterator->data_ptr = p;
4738 :
4739 156 : result = construct_md_array(values,
4740 : nulls,
4741 : iterator->slice_ndim,
4742 : iterator->slice_dims,
4743 : iterator->slice_lbound,
4744 156 : ARR_ELEMTYPE(iterator->arr),
4745 156 : iterator->typlen,
4746 156 : iterator->typbyval,
4747 156 : iterator->typalign);
4748 :
4749 156 : *isnull = false;
4750 156 : *value = PointerGetDatum(result);
4751 : }
4752 :
4753 9026 : return true;
4754 : }
4755 :
4756 : /*
4757 : * Release an ArrayIterator data structure
4758 : */
4759 : void
4760 384 : array_free_iterator(ArrayIterator iterator)
4761 : {
4762 384 : if (iterator->slice_ndim > 0)
4763 : {
4764 42 : pfree(iterator->slice_values);
4765 42 : pfree(iterator->slice_nulls);
4766 : }
4767 384 : pfree(iterator);
4768 384 : }
4769 :
4770 :
4771 : /***************************************************************************/
4772 : /******************| Support Routines |*****************/
4773 : /***************************************************************************/
4774 :
4775 : /*
4776 : * Check whether a specific array element is NULL
4777 : *
4778 : * nullbitmap: pointer to array's null bitmap (NULL if none)
4779 : * offset: 0-based linear element number of array element
4780 : */
4781 : static bool
4782 807576 : array_get_isnull(const bits8 *nullbitmap, int offset)
4783 : {
4784 807576 : if (nullbitmap == NULL)
4785 806744 : return false; /* assume not null */
4786 832 : if (nullbitmap[offset / 8] & (1 << (offset % 8)))
4787 658 : return false; /* not null */
4788 174 : return true;
4789 : }
4790 :
4791 : /*
4792 : * Set a specific array element's null-bitmap entry
4793 : *
4794 : * nullbitmap: pointer to array's null bitmap (mustn't be NULL)
4795 : * offset: 0-based linear element number of array element
4796 : * isNull: null status to set
4797 : */
4798 : static void
4799 134 : array_set_isnull(bits8 *nullbitmap, int offset, bool isNull)
4800 : {
4801 : int bitmask;
4802 :
4803 134 : nullbitmap += offset / 8;
4804 134 : bitmask = 1 << (offset % 8);
4805 134 : if (isNull)
4806 20 : *nullbitmap &= ~bitmask;
4807 : else
4808 114 : *nullbitmap |= bitmask;
4809 134 : }
4810 :
4811 : /*
4812 : * Fetch array element at pointer, converted correctly to a Datum
4813 : *
4814 : * Caller must have handled case of NULL element
4815 : */
4816 : static Datum
4817 797684 : ArrayCast(char *value, bool byval, int len)
4818 : {
4819 797684 : return fetch_att(value, byval, len);
4820 : }
4821 :
4822 : /*
4823 : * Copy datum to *dest and return total space used (including align padding)
4824 : *
4825 : * Caller must have handled case of NULL element
4826 : */
4827 : static int
4828 14084784 : ArrayCastAndSet(Datum src,
4829 : int typlen,
4830 : bool typbyval,
4831 : char typalign,
4832 : char *dest)
4833 : {
4834 : int inc;
4835 :
4836 14084784 : if (typlen > 0)
4837 : {
4838 10741176 : if (typbyval)
4839 6838190 : store_att_byval(dest, src, typlen);
4840 : else
4841 3902986 : memmove(dest, DatumGetPointer(src), typlen);
4842 10741176 : inc = att_align_nominal(typlen, typalign);
4843 : }
4844 : else
4845 : {
4846 : Assert(!typbyval);
4847 3343608 : inc = att_addlength_datum(0, typlen, src);
4848 3343608 : memmove(dest, DatumGetPointer(src), inc);
4849 3343608 : inc = att_align_nominal(inc, typalign);
4850 : }
4851 :
4852 14084784 : return inc;
4853 : }
4854 :
4855 : /*
4856 : * Advance ptr over nitems array elements
4857 : *
4858 : * ptr: starting location in array
4859 : * offset: 0-based linear element number of first element (the one at *ptr)
4860 : * nullbitmap: start of array's null bitmap, or NULL if none
4861 : * nitems: number of array elements to advance over (>= 0)
4862 : * typlen, typbyval, typalign: storage parameters of array element datatype
4863 : *
4864 : * It is caller's responsibility to ensure that nitems is within range
4865 : */
4866 : static char *
4867 800098 : array_seek(char *ptr, int offset, bits8 *nullbitmap, int nitems,
4868 : int typlen, bool typbyval, char typalign)
4869 : {
4870 : int bitmask;
4871 : int i;
4872 :
4873 : /* easy if fixed-size elements and no NULLs */
4874 800098 : if (typlen > 0 && !nullbitmap)
4875 541066 : return ptr + nitems * ((Size) att_align_nominal(typlen, typalign));
4876 :
4877 : /* seems worth having separate loops for NULL and no-NULLs cases */
4878 259032 : if (nullbitmap)
4879 : {
4880 604 : nullbitmap += offset / 8;
4881 604 : bitmask = 1 << (offset % 8);
4882 :
4883 1852 : for (i = 0; i < nitems; i++)
4884 : {
4885 1248 : if (*nullbitmap & bitmask)
4886 : {
4887 888 : ptr = att_addlength_pointer(ptr, typlen, ptr);
4888 888 : ptr = (char *) att_align_nominal(ptr, typalign);
4889 : }
4890 1248 : bitmask <<= 1;
4891 1248 : if (bitmask == 0x100)
4892 : {
4893 48 : nullbitmap++;
4894 48 : bitmask = 1;
4895 : }
4896 : }
4897 : }
4898 : else
4899 : {
4900 1171870 : for (i = 0; i < nitems; i++)
4901 : {
4902 913442 : ptr = att_addlength_pointer(ptr, typlen, ptr);
4903 913442 : ptr = (char *) att_align_nominal(ptr, typalign);
4904 : }
4905 : }
4906 259032 : return ptr;
4907 : }
4908 :
4909 : /*
4910 : * Compute total size of the nitems array elements starting at *ptr
4911 : *
4912 : * Parameters same as for array_seek
4913 : */
4914 : static int
4915 1614 : array_nelems_size(char *ptr, int offset, bits8 *nullbitmap, int nitems,
4916 : int typlen, bool typbyval, char typalign)
4917 : {
4918 1614 : return array_seek(ptr, offset, nullbitmap, nitems,
4919 1614 : typlen, typbyval, typalign) - ptr;
4920 : }
4921 :
4922 : /*
4923 : * Copy nitems array elements from srcptr to destptr
4924 : *
4925 : * destptr: starting destination location (must be enough room!)
4926 : * nitems: number of array elements to copy (>= 0)
4927 : * srcptr: starting location in source array
4928 : * offset: 0-based linear element number of first element (the one at *srcptr)
4929 : * nullbitmap: start of source array's null bitmap, or NULL if none
4930 : * typlen, typbyval, typalign: storage parameters of array element datatype
4931 : *
4932 : * Returns number of bytes copied
4933 : *
4934 : * NB: this does not take care of setting up the destination's null bitmap!
4935 : */
4936 : static int
4937 1146 : array_copy(char *destptr, int nitems,
4938 : char *srcptr, int offset, bits8 *nullbitmap,
4939 : int typlen, bool typbyval, char typalign)
4940 : {
4941 : int numbytes;
4942 :
4943 1146 : numbytes = array_nelems_size(srcptr, offset, nullbitmap, nitems,
4944 : typlen, typbyval, typalign);
4945 1146 : memcpy(destptr, srcptr, numbytes);
4946 1146 : return numbytes;
4947 : }
4948 :
4949 : /*
4950 : * Copy nitems null-bitmap bits from source to destination
4951 : *
4952 : * destbitmap: start of destination array's null bitmap (mustn't be NULL)
4953 : * destoffset: 0-based linear element number of first dest element
4954 : * srcbitmap: start of source array's null bitmap, or NULL if none
4955 : * srcoffset: 0-based linear element number of first source element
4956 : * nitems: number of bits to copy (>= 0)
4957 : *
4958 : * If srcbitmap is NULL then we assume the source is all-non-NULL and
4959 : * fill 1's into the destination bitmap. Note that only the specified
4960 : * bits in the destination map are changed, not any before or after.
4961 : *
4962 : * Note: this could certainly be optimized using standard bitblt methods.
4963 : * However, it's not clear that the typical Postgres array has enough elements
4964 : * to make it worth worrying too much. For the moment, KISS.
4965 : */
4966 : void
4967 30866 : array_bitmap_copy(bits8 *destbitmap, int destoffset,
4968 : const bits8 *srcbitmap, int srcoffset,
4969 : int nitems)
4970 : {
4971 : int destbitmask,
4972 : destbitval,
4973 : srcbitmask,
4974 : srcbitval;
4975 :
4976 : Assert(destbitmap);
4977 30866 : if (nitems <= 0)
4978 160 : return; /* don't risk fetch off end of memory */
4979 30706 : destbitmap += destoffset / 8;
4980 30706 : destbitmask = 1 << (destoffset % 8);
4981 30706 : destbitval = *destbitmap;
4982 30706 : if (srcbitmap)
4983 : {
4984 15636 : srcbitmap += srcoffset / 8;
4985 15636 : srcbitmask = 1 << (srcoffset % 8);
4986 15636 : srcbitval = *srcbitmap;
4987 67266 : while (nitems-- > 0)
4988 : {
4989 51630 : if (srcbitval & srcbitmask)
4990 18650 : destbitval |= destbitmask;
4991 : else
4992 32980 : destbitval &= ~destbitmask;
4993 51630 : destbitmask <<= 1;
4994 51630 : if (destbitmask == 0x100)
4995 : {
4996 6014 : *destbitmap++ = destbitval;
4997 6014 : destbitmask = 1;
4998 6014 : if (nitems > 0)
4999 4406 : destbitval = *destbitmap;
5000 : }
5001 51630 : srcbitmask <<= 1;
5002 51630 : if (srcbitmask == 0x100)
5003 : {
5004 4392 : srcbitmap++;
5005 4392 : srcbitmask = 1;
5006 4392 : if (nitems > 0)
5007 4354 : srcbitval = *srcbitmap;
5008 : }
5009 : }
5010 15636 : if (destbitmask != 1)
5011 14028 : *destbitmap = destbitval;
5012 : }
5013 : else
5014 : {
5015 30264 : while (nitems-- > 0)
5016 : {
5017 15194 : destbitval |= destbitmask;
5018 15194 : destbitmask <<= 1;
5019 15194 : if (destbitmask == 0x100)
5020 : {
5021 2136 : *destbitmap++ = destbitval;
5022 2136 : destbitmask = 1;
5023 2136 : if (nitems > 0)
5024 0 : destbitval = *destbitmap;
5025 : }
5026 : }
5027 15070 : if (destbitmask != 1)
5028 12934 : *destbitmap = destbitval;
5029 : }
5030 : }
5031 :
5032 : /*
5033 : * Compute space needed for a slice of an array
5034 : *
5035 : * We assume the caller has verified that the slice coordinates are valid.
5036 : */
5037 : static int
5038 294 : array_slice_size(char *arraydataptr, bits8 *arraynullsptr,
5039 : int ndim, int *dim, int *lb,
5040 : int *st, int *endp,
5041 : int typlen, bool typbyval, char typalign)
5042 : {
5043 : int src_offset,
5044 : span[MAXDIM],
5045 : prod[MAXDIM],
5046 : dist[MAXDIM],
5047 : indx[MAXDIM];
5048 : char *ptr;
5049 : int i,
5050 : j,
5051 : inc;
5052 294 : int count = 0;
5053 :
5054 294 : mda_get_range(ndim, span, st, endp);
5055 :
5056 : /* Pretty easy for fixed element length without nulls ... */
5057 294 : if (typlen > 0 && !arraynullsptr)
5058 216 : return ArrayGetNItems(ndim, span) * att_align_nominal(typlen, typalign);
5059 :
5060 : /* Else gotta do it the hard way */
5061 78 : src_offset = ArrayGetOffset(ndim, dim, lb, st);
5062 78 : ptr = array_seek(arraydataptr, 0, arraynullsptr, src_offset,
5063 : typlen, typbyval, typalign);
5064 78 : mda_get_prod(ndim, dim, prod);
5065 78 : mda_get_offset_values(ndim, dist, prod, span);
5066 198 : for (i = 0; i < ndim; i++)
5067 120 : indx[i] = 0;
5068 78 : j = ndim - 1;
5069 : do
5070 : {
5071 258 : if (dist[j])
5072 : {
5073 0 : ptr = array_seek(ptr, src_offset, arraynullsptr, dist[j],
5074 : typlen, typbyval, typalign);
5075 0 : src_offset += dist[j];
5076 : }
5077 258 : if (!array_get_isnull(arraynullsptr, src_offset))
5078 : {
5079 246 : inc = att_addlength_pointer(0, typlen, ptr);
5080 246 : inc = att_align_nominal(inc, typalign);
5081 246 : ptr += inc;
5082 246 : count += inc;
5083 : }
5084 258 : src_offset++;
5085 258 : } while ((j = mda_next_tuple(ndim, indx, span)) != -1);
5086 78 : return count;
5087 : }
5088 :
5089 : /*
5090 : * Extract a slice of an array into consecutive elements in the destination
5091 : * array.
5092 : *
5093 : * We assume the caller has verified that the slice coordinates are valid,
5094 : * allocated enough storage for the result, and initialized the header
5095 : * of the new array.
5096 : */
5097 : static void
5098 264 : array_extract_slice(ArrayType *newarray,
5099 : int ndim,
5100 : int *dim,
5101 : int *lb,
5102 : char *arraydataptr,
5103 : bits8 *arraynullsptr,
5104 : int *st,
5105 : int *endp,
5106 : int typlen,
5107 : bool typbyval,
5108 : char typalign)
5109 : {
5110 264 : char *destdataptr = ARR_DATA_PTR(newarray);
5111 264 : bits8 *destnullsptr = ARR_NULLBITMAP(newarray);
5112 : char *srcdataptr;
5113 : int src_offset,
5114 : dest_offset,
5115 : prod[MAXDIM],
5116 : span[MAXDIM],
5117 : dist[MAXDIM],
5118 : indx[MAXDIM];
5119 : int i,
5120 : j,
5121 : inc;
5122 :
5123 264 : src_offset = ArrayGetOffset(ndim, dim, lb, st);
5124 264 : srcdataptr = array_seek(arraydataptr, 0, arraynullsptr, src_offset,
5125 : typlen, typbyval, typalign);
5126 264 : mda_get_prod(ndim, dim, prod);
5127 264 : mda_get_range(ndim, span, st, endp);
5128 264 : mda_get_offset_values(ndim, dist, prod, span);
5129 672 : for (i = 0; i < ndim; i++)
5130 408 : indx[i] = 0;
5131 264 : dest_offset = 0;
5132 264 : j = ndim - 1;
5133 : do
5134 : {
5135 990 : if (dist[j])
5136 : {
5137 : /* skip unwanted elements */
5138 30 : srcdataptr = array_seek(srcdataptr, src_offset, arraynullsptr,
5139 : dist[j],
5140 : typlen, typbyval, typalign);
5141 30 : src_offset += dist[j];
5142 : }
5143 990 : inc = array_copy(destdataptr, 1,
5144 : srcdataptr, src_offset, arraynullsptr,
5145 : typlen, typbyval, typalign);
5146 990 : if (destnullsptr)
5147 180 : array_bitmap_copy(destnullsptr, dest_offset,
5148 : arraynullsptr, src_offset,
5149 : 1);
5150 990 : destdataptr += inc;
5151 990 : srcdataptr += inc;
5152 990 : src_offset++;
5153 990 : dest_offset++;
5154 990 : } while ((j = mda_next_tuple(ndim, indx, span)) != -1);
5155 264 : }
5156 :
5157 : /*
5158 : * Insert a slice into an array.
5159 : *
5160 : * ndim/dim[]/lb[] are dimensions of the original array. A new array with
5161 : * those same dimensions is to be constructed. destArray must already
5162 : * have been allocated and its header initialized.
5163 : *
5164 : * st[]/endp[] identify the slice to be replaced. Elements within the slice
5165 : * volume are taken from consecutive elements of the srcArray; elements
5166 : * outside it are copied from origArray.
5167 : *
5168 : * We assume the caller has verified that the slice coordinates are valid.
5169 : */
5170 : static void
5171 30 : array_insert_slice(ArrayType *destArray,
5172 : ArrayType *origArray,
5173 : ArrayType *srcArray,
5174 : int ndim,
5175 : int *dim,
5176 : int *lb,
5177 : int *st,
5178 : int *endp,
5179 : int typlen,
5180 : bool typbyval,
5181 : char typalign)
5182 : {
5183 30 : char *destPtr = ARR_DATA_PTR(destArray);
5184 30 : char *origPtr = ARR_DATA_PTR(origArray);
5185 30 : char *srcPtr = ARR_DATA_PTR(srcArray);
5186 30 : bits8 *destBitmap = ARR_NULLBITMAP(destArray);
5187 30 : bits8 *origBitmap = ARR_NULLBITMAP(origArray);
5188 30 : bits8 *srcBitmap = ARR_NULLBITMAP(srcArray);
5189 30 : int orignitems = ArrayGetNItems(ARR_NDIM(origArray),
5190 : ARR_DIMS(origArray));
5191 : int dest_offset,
5192 : orig_offset,
5193 : src_offset,
5194 : prod[MAXDIM],
5195 : span[MAXDIM],
5196 : dist[MAXDIM],
5197 : indx[MAXDIM];
5198 : int i,
5199 : j,
5200 : inc;
5201 :
5202 30 : dest_offset = ArrayGetOffset(ndim, dim, lb, st);
5203 : /* copy items before the slice start */
5204 30 : inc = array_copy(destPtr, dest_offset,
5205 : origPtr, 0, origBitmap,
5206 : typlen, typbyval, typalign);
5207 30 : destPtr += inc;
5208 30 : origPtr += inc;
5209 30 : if (destBitmap)
5210 0 : array_bitmap_copy(destBitmap, 0, origBitmap, 0, dest_offset);
5211 30 : orig_offset = dest_offset;
5212 30 : mda_get_prod(ndim, dim, prod);
5213 30 : mda_get_range(ndim, span, st, endp);
5214 30 : mda_get_offset_values(ndim, dist, prod, span);
5215 102 : for (i = 0; i < ndim; i++)
5216 72 : indx[i] = 0;
5217 30 : src_offset = 0;
5218 30 : j = ndim - 1;
5219 : do
5220 : {
5221 : /* Copy/advance over elements between here and next part of slice */
5222 78 : if (dist[j])
5223 : {
5224 18 : inc = array_copy(destPtr, dist[j],
5225 : origPtr, orig_offset, origBitmap,
5226 : typlen, typbyval, typalign);
5227 18 : destPtr += inc;
5228 18 : origPtr += inc;
5229 18 : if (destBitmap)
5230 0 : array_bitmap_copy(destBitmap, dest_offset,
5231 : origBitmap, orig_offset,
5232 : dist[j]);
5233 18 : dest_offset += dist[j];
5234 18 : orig_offset += dist[j];
5235 : }
5236 : /* Copy new element at this slice position */
5237 78 : inc = array_copy(destPtr, 1,
5238 : srcPtr, src_offset, srcBitmap,
5239 : typlen, typbyval, typalign);
5240 78 : if (destBitmap)
5241 0 : array_bitmap_copy(destBitmap, dest_offset,
5242 : srcBitmap, src_offset,
5243 : 1);
5244 78 : destPtr += inc;
5245 78 : srcPtr += inc;
5246 78 : dest_offset++;
5247 78 : src_offset++;
5248 : /* Advance over old element at this slice position */
5249 78 : origPtr = array_seek(origPtr, orig_offset, origBitmap, 1,
5250 : typlen, typbyval, typalign);
5251 78 : orig_offset++;
5252 78 : } while ((j = mda_next_tuple(ndim, indx, span)) != -1);
5253 :
5254 : /* don't miss any data at the end */
5255 30 : array_copy(destPtr, orignitems - orig_offset,
5256 : origPtr, orig_offset, origBitmap,
5257 : typlen, typbyval, typalign);
5258 30 : if (destBitmap)
5259 0 : array_bitmap_copy(destBitmap, dest_offset,
5260 : origBitmap, orig_offset,
5261 : orignitems - orig_offset);
5262 30 : }
5263 :
5264 : /*
5265 : * initArrayResult - initialize an empty ArrayBuildState
5266 : *
5267 : * element_type is the array element type (must be a valid array element type)
5268 : * rcontext is where to keep working state
5269 : * subcontext is a flag determining whether to use a separate memory context
5270 : *
5271 : * Note: there are two common schemes for using accumArrayResult().
5272 : * In the older scheme, you start with a NULL ArrayBuildState pointer, and
5273 : * call accumArrayResult once per element. In this scheme you end up with
5274 : * a NULL pointer if there were no elements, which you need to special-case.
5275 : * In the newer scheme, call initArrayResult and then call accumArrayResult
5276 : * once per element. In this scheme you always end with a non-NULL pointer
5277 : * that you can pass to makeArrayResult; you get an empty array if there
5278 : * were no elements. This is preferred if an empty array is what you want.
5279 : *
5280 : * It's possible to choose whether to create a separate memory context for the
5281 : * array build state, or whether to allocate it directly within rcontext.
5282 : *
5283 : * When there are many concurrent small states (e.g. array_agg() using hash
5284 : * aggregation of many small groups), using a separate memory context for each
5285 : * one may result in severe memory bloat. In such cases, use the same memory
5286 : * context to initialize all such array build states, and pass
5287 : * subcontext=false.
5288 : *
5289 : * In cases when the array build states have different lifetimes, using a
5290 : * single memory context is impractical. Instead, pass subcontext=true so that
5291 : * the array build states can be freed individually.
5292 : */
5293 : ArrayBuildState *
5294 403962 : initArrayResult(Oid element_type, MemoryContext rcontext, bool subcontext)
5295 : {
5296 : /*
5297 : * When using a subcontext, we can afford to start with a somewhat larger
5298 : * initial array size. Without subcontexts, we'd better hope that most of
5299 : * the states stay small ...
5300 : */
5301 403962 : return initArrayResultWithSize(element_type, rcontext, subcontext,
5302 : subcontext ? 64 : 8);
5303 : }
5304 :
5305 : /*
5306 : * initArrayResultWithSize
5307 : * As initArrayResult, but allow the initial size of the allocated arrays
5308 : * to be specified.
5309 : */
5310 : ArrayBuildState *
5311 404122 : initArrayResultWithSize(Oid element_type, MemoryContext rcontext,
5312 : bool subcontext, int initsize)
5313 : {
5314 : ArrayBuildState *astate;
5315 404122 : MemoryContext arr_context = rcontext;
5316 :
5317 : /* Make a temporary context to hold all the junk */
5318 404122 : if (subcontext)
5319 262496 : arr_context = AllocSetContextCreate(rcontext,
5320 : "accumArrayResult",
5321 : ALLOCSET_DEFAULT_SIZES);
5322 :
5323 : astate = (ArrayBuildState *)
5324 404122 : MemoryContextAlloc(arr_context, sizeof(ArrayBuildState));
5325 404122 : astate->mcontext = arr_context;
5326 404122 : astate->private_cxt = subcontext;
5327 404122 : astate->alen = initsize;
5328 404122 : astate->dvalues = (Datum *)
5329 404122 : MemoryContextAlloc(arr_context, astate->alen * sizeof(Datum));
5330 404122 : astate->dnulls = (bool *)
5331 404122 : MemoryContextAlloc(arr_context, astate->alen * sizeof(bool));
5332 404122 : astate->nelems = 0;
5333 404122 : astate->element_type = element_type;
5334 404122 : get_typlenbyvalalign(element_type,
5335 : &astate->typlen,
5336 : &astate->typbyval,
5337 : &astate->typalign);
5338 :
5339 404122 : return astate;
5340 : }
5341 :
5342 : /*
5343 : * accumArrayResult - accumulate one (more) Datum for an array result
5344 : *
5345 : * astate is working state (can be NULL on first call)
5346 : * dvalue/disnull represent the new Datum to append to the array
5347 : * element_type is the Datum's type (must be a valid array element type)
5348 : * rcontext is where to keep working state
5349 : */
5350 : ArrayBuildState *
5351 3638932 : accumArrayResult(ArrayBuildState *astate,
5352 : Datum dvalue, bool disnull,
5353 : Oid element_type,
5354 : MemoryContext rcontext)
5355 : {
5356 : MemoryContext oldcontext;
5357 :
5358 3638932 : if (astate == NULL)
5359 : {
5360 : /* First time through --- initialize */
5361 204878 : astate = initArrayResult(element_type, rcontext, true);
5362 : }
5363 : else
5364 : {
5365 : Assert(astate->element_type == element_type);
5366 : }
5367 :
5368 3638932 : oldcontext = MemoryContextSwitchTo(astate->mcontext);
5369 :
5370 : /* enlarge dvalues[]/dnulls[] if needed */
5371 3638932 : if (astate->nelems >= astate->alen)
5372 : {
5373 86838 : astate->alen *= 2;
5374 : /* give an array-related error if we go past MaxAllocSize */
5375 86838 : if (!AllocSizeIsValid(astate->alen * sizeof(Datum)))
5376 0 : ereport(ERROR,
5377 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
5378 : errmsg("array size exceeds the maximum allowed (%d)",
5379 : (int) MaxAllocSize)));
5380 86838 : astate->dvalues = (Datum *)
5381 86838 : repalloc(astate->dvalues, astate->alen * sizeof(Datum));
5382 86838 : astate->dnulls = (bool *)
5383 86838 : repalloc(astate->dnulls, astate->alen * sizeof(bool));
5384 : }
5385 :
5386 : /*
5387 : * Ensure pass-by-ref stuff is copied into mcontext; and detoast it too if
5388 : * it's varlena. (You might think that detoasting is not needed here
5389 : * because construct_md_array can detoast the array elements later.
5390 : * However, we must not let construct_md_array modify the ArrayBuildState
5391 : * because that would mean array_agg_finalfn damages its input, which is
5392 : * verboten. Also, this way frequently saves one copying step.)
5393 : */
5394 3638932 : if (!disnull && !astate->typbyval)
5395 : {
5396 2483472 : if (astate->typlen == -1)
5397 1851784 : dvalue = PointerGetDatum(PG_DETOAST_DATUM_COPY(dvalue));
5398 : else
5399 631688 : dvalue = datumCopy(dvalue, astate->typbyval, astate->typlen);
5400 : }
5401 :
5402 3638932 : astate->dvalues[astate->nelems] = dvalue;
5403 3638932 : astate->dnulls[astate->nelems] = disnull;
5404 3638932 : astate->nelems++;
5405 :
5406 3638932 : MemoryContextSwitchTo(oldcontext);
5407 :
5408 3638932 : return astate;
5409 : }
5410 :
5411 : /*
5412 : * makeArrayResult - produce 1-D final result of accumArrayResult
5413 : *
5414 : * Note: only releases astate if it was initialized within a separate memory
5415 : * context (i.e. using subcontext=true when calling initArrayResult).
5416 : *
5417 : * astate is working state (must not be NULL)
5418 : * rcontext is where to construct result
5419 : */
5420 : Datum
5421 205066 : makeArrayResult(ArrayBuildState *astate,
5422 : MemoryContext rcontext)
5423 : {
5424 : int ndims;
5425 : int dims[1];
5426 : int lbs[1];
5427 :
5428 : /* If no elements were presented, we want to create an empty array */
5429 205066 : ndims = (astate->nelems > 0) ? 1 : 0;
5430 205066 : dims[0] = astate->nelems;
5431 205066 : lbs[0] = 1;
5432 :
5433 410132 : return makeMdArrayResult(astate, ndims, dims, lbs, rcontext,
5434 205066 : astate->private_cxt);
5435 : }
5436 :
5437 : /*
5438 : * makeMdArrayResult - produce multi-D final result of accumArrayResult
5439 : *
5440 : * beware: no check that specified dimensions match the number of values
5441 : * accumulated.
5442 : *
5443 : * Note: if the astate was not initialized within a separate memory context
5444 : * (that is, initArrayResult was called with subcontext=false), then using
5445 : * release=true is illegal. Instead, release astate along with the rest of its
5446 : * context when appropriate.
5447 : *
5448 : * astate is working state (must not be NULL)
5449 : * rcontext is where to construct result
5450 : * release is true if okay to release working state
5451 : */
5452 : Datum
5453 402844 : makeMdArrayResult(ArrayBuildState *astate,
5454 : int ndims,
5455 : int *dims,
5456 : int *lbs,
5457 : MemoryContext rcontext,
5458 : bool release)
5459 : {
5460 : ArrayType *result;
5461 : MemoryContext oldcontext;
5462 :
5463 : /* Build the final array result in rcontext */
5464 402844 : oldcontext = MemoryContextSwitchTo(rcontext);
5465 :
5466 402844 : result = construct_md_array(astate->dvalues,
5467 : astate->dnulls,
5468 : ndims,
5469 : dims,
5470 : lbs,
5471 : astate->element_type,
5472 402844 : astate->typlen,
5473 402844 : astate->typbyval,
5474 402844 : astate->typalign);
5475 :
5476 402844 : MemoryContextSwitchTo(oldcontext);
5477 :
5478 : /* Clean up all the junk */
5479 402844 : if (release)
5480 : {
5481 : Assert(astate->private_cxt);
5482 261892 : MemoryContextDelete(astate->mcontext);
5483 : }
5484 :
5485 402844 : return PointerGetDatum(result);
5486 : }
5487 :
5488 : /*
5489 : * The following three functions provide essentially the same API as
5490 : * initArrayResult/accumArrayResult/makeArrayResult, but instead of accepting
5491 : * inputs that are array elements, they accept inputs that are arrays and
5492 : * produce an output array having N+1 dimensions. The inputs must all have
5493 : * identical dimensionality as well as element type.
5494 : */
5495 :
5496 : /*
5497 : * initArrayResultArr - initialize an empty ArrayBuildStateArr
5498 : *
5499 : * array_type is the array type (must be a valid varlena array type)
5500 : * element_type is the type of the array's elements (lookup if InvalidOid)
5501 : * rcontext is where to keep working state
5502 : * subcontext is a flag determining whether to use a separate memory context
5503 : */
5504 : ArrayBuildStateArr *
5505 452 : initArrayResultArr(Oid array_type, Oid element_type, MemoryContext rcontext,
5506 : bool subcontext)
5507 : {
5508 : ArrayBuildStateArr *astate;
5509 452 : MemoryContext arr_context = rcontext; /* by default use the parent ctx */
5510 :
5511 : /* Lookup element type, unless element_type already provided */
5512 452 : if (!OidIsValid(element_type))
5513 : {
5514 352 : element_type = get_element_type(array_type);
5515 :
5516 352 : if (!OidIsValid(element_type))
5517 0 : ereport(ERROR,
5518 : (errcode(ERRCODE_DATATYPE_MISMATCH),
5519 : errmsg("data type %s is not an array type",
5520 : format_type_be(array_type))));
5521 : }
5522 :
5523 : /* Make a temporary context to hold all the junk */
5524 452 : if (subcontext)
5525 48 : arr_context = AllocSetContextCreate(rcontext,
5526 : "accumArrayResultArr",
5527 : ALLOCSET_DEFAULT_SIZES);
5528 :
5529 : /* Note we initialize all fields to zero */
5530 : astate = (ArrayBuildStateArr *)
5531 452 : MemoryContextAllocZero(arr_context, sizeof(ArrayBuildStateArr));
5532 452 : astate->mcontext = arr_context;
5533 452 : astate->private_cxt = subcontext;
5534 :
5535 : /* Save relevant datatype information */
5536 452 : astate->array_type = array_type;
5537 452 : astate->element_type = element_type;
5538 :
5539 452 : return astate;
5540 : }
5541 :
5542 : /*
5543 : * accumArrayResultArr - accumulate one (more) sub-array for an array result
5544 : *
5545 : * astate is working state (can be NULL on first call)
5546 : * dvalue/disnull represent the new sub-array to append to the array
5547 : * array_type is the array type (must be a valid varlena array type)
5548 : * rcontext is where to keep working state
5549 : */
5550 : ArrayBuildStateArr *
5551 60360 : accumArrayResultArr(ArrayBuildStateArr *astate,
5552 : Datum dvalue, bool disnull,
5553 : Oid array_type,
5554 : MemoryContext rcontext)
5555 : {
5556 : ArrayType *arg;
5557 : MemoryContext oldcontext;
5558 : int *dims,
5559 : *lbs,
5560 : ndims,
5561 : nitems,
5562 : ndatabytes;
5563 : char *data;
5564 : int i;
5565 :
5566 : /*
5567 : * We disallow accumulating null subarrays. Another plausible definition
5568 : * is to ignore them, but callers that want that can just skip calling
5569 : * this function.
5570 : */
5571 60360 : if (disnull)
5572 6 : ereport(ERROR,
5573 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
5574 : errmsg("cannot accumulate null arrays")));
5575 :
5576 : /* Detoast input array in caller's context */
5577 60354 : arg = DatumGetArrayTypeP(dvalue);
5578 :
5579 60354 : if (astate == NULL)
5580 0 : astate = initArrayResultArr(array_type, InvalidOid, rcontext, true);
5581 : else
5582 : Assert(astate->array_type == array_type);
5583 :
5584 60354 : oldcontext = MemoryContextSwitchTo(astate->mcontext);
5585 :
5586 : /* Collect this input's dimensions */
5587 60354 : ndims = ARR_NDIM(arg);
5588 60354 : dims = ARR_DIMS(arg);
5589 60354 : lbs = ARR_LBOUND(arg);
5590 60354 : data = ARR_DATA_PTR(arg);
5591 60354 : nitems = ArrayGetNItems(ndims, dims);
5592 60354 : ndatabytes = ARR_SIZE(arg) - ARR_DATA_OFFSET(arg);
5593 :
5594 60354 : if (astate->ndims == 0)
5595 : {
5596 : /* First input; check/save the dimensionality info */
5597 :
5598 : /* Should we allow empty inputs and just produce an empty output? */
5599 286 : if (ndims == 0)
5600 6 : ereport(ERROR,
5601 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
5602 : errmsg("cannot accumulate empty arrays")));
5603 280 : if (ndims + 1 > MAXDIM)
5604 0 : ereport(ERROR,
5605 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
5606 : errmsg("number of array dimensions (%d) exceeds the maximum allowed (%d)",
5607 : ndims + 1, MAXDIM)));
5608 :
5609 : /*
5610 : * The output array will have n+1 dimensions, with the ones after the
5611 : * first matching the input's dimensions.
5612 : */
5613 280 : astate->ndims = ndims + 1;
5614 280 : astate->dims[0] = 0;
5615 280 : memcpy(&astate->dims[1], dims, ndims * sizeof(int));
5616 280 : astate->lbs[0] = 1;
5617 280 : memcpy(&astate->lbs[1], lbs, ndims * sizeof(int));
5618 :
5619 : /* Allocate at least enough data space for this item */
5620 280 : astate->abytes = pg_nextpower2_32(Max(1024, ndatabytes + 1));
5621 280 : astate->data = (char *) palloc(astate->abytes);
5622 : }
5623 : else
5624 : {
5625 : /* Second or later input: must match first input's dimensionality */
5626 60068 : if (astate->ndims != ndims + 1)
5627 0 : ereport(ERROR,
5628 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
5629 : errmsg("cannot accumulate arrays of different dimensionality")));
5630 120130 : for (i = 0; i < ndims; i++)
5631 : {
5632 60068 : if (astate->dims[i + 1] != dims[i] || astate->lbs[i + 1] != lbs[i])
5633 6 : ereport(ERROR,
5634 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
5635 : errmsg("cannot accumulate arrays of different dimensionality")));
5636 : }
5637 :
5638 : /* Enlarge data space if needed */
5639 60062 : if (astate->nbytes + ndatabytes > astate->abytes)
5640 : {
5641 50 : astate->abytes = Max(astate->abytes * 2,
5642 : astate->nbytes + ndatabytes);
5643 50 : astate->data = (char *) repalloc(astate->data, astate->abytes);
5644 : }
5645 : }
5646 :
5647 : /*
5648 : * Copy the data portion of the sub-array. Note we assume that the
5649 : * advertised data length of the sub-array is properly aligned. We do not
5650 : * have to worry about detoasting elements since whatever's in the
5651 : * sub-array should be OK already.
5652 : */
5653 60342 : memcpy(astate->data + astate->nbytes, data, ndatabytes);
5654 60342 : astate->nbytes += ndatabytes;
5655 :
5656 : /* Deal with null bitmap if needed */
5657 60342 : if (astate->nullbitmap || ARR_HASNULL(arg))
5658 : {
5659 29978 : int newnitems = astate->nitems + nitems;
5660 :
5661 29978 : if (astate->nullbitmap == NULL)
5662 : {
5663 : /*
5664 : * First input with nulls; we must retrospectively handle any
5665 : * previous inputs by marking all their items non-null.
5666 : */
5667 86 : astate->aitems = pg_nextpower2_32(Max(256, newnitems + 1));
5668 86 : astate->nullbitmap = (bits8 *) palloc((astate->aitems + 7) / 8);
5669 86 : array_bitmap_copy(astate->nullbitmap, 0,
5670 : NULL, 0,
5671 : astate->nitems);
5672 : }
5673 29892 : else if (newnitems > astate->aitems)
5674 : {
5675 50 : astate->aitems = Max(astate->aitems * 2, newnitems);
5676 50 : astate->nullbitmap = (bits8 *)
5677 50 : repalloc(astate->nullbitmap, (astate->aitems + 7) / 8);
5678 : }
5679 29978 : array_bitmap_copy(astate->nullbitmap, astate->nitems,
5680 29978 : ARR_NULLBITMAP(arg), 0,
5681 : nitems);
5682 : }
5683 :
5684 60342 : astate->nitems += nitems;
5685 60342 : astate->dims[0] += 1;
5686 :
5687 60342 : MemoryContextSwitchTo(oldcontext);
5688 :
5689 : /* Release detoasted copy if any */
5690 60342 : if ((Pointer) arg != DatumGetPointer(dvalue))
5691 126 : pfree(arg);
5692 :
5693 60342 : return astate;
5694 : }
5695 :
5696 : /*
5697 : * makeArrayResultArr - produce N+1-D final result of accumArrayResultArr
5698 : *
5699 : * astate is working state (must not be NULL)
5700 : * rcontext is where to construct result
5701 : * release is true if okay to release working state
5702 : */
5703 : Datum
5704 234 : makeArrayResultArr(ArrayBuildStateArr *astate,
5705 : MemoryContext rcontext,
5706 : bool release)
5707 : {
5708 : ArrayType *result;
5709 : MemoryContext oldcontext;
5710 :
5711 : /* Build the final array result in rcontext */
5712 234 : oldcontext = MemoryContextSwitchTo(rcontext);
5713 :
5714 234 : if (astate->ndims == 0)
5715 : {
5716 : /* No inputs, return empty array */
5717 0 : result = construct_empty_array(astate->element_type);
5718 : }
5719 : else
5720 : {
5721 : int dataoffset,
5722 : nbytes;
5723 :
5724 : /* Check for overflow of the array dimensions */
5725 234 : (void) ArrayGetNItems(astate->ndims, astate->dims);
5726 234 : ArrayCheckBounds(astate->ndims, astate->dims, astate->lbs);
5727 :
5728 : /* Compute required space */
5729 234 : nbytes = astate->nbytes;
5730 234 : if (astate->nullbitmap != NULL)
5731 : {
5732 66 : dataoffset = ARR_OVERHEAD_WITHNULLS(astate->ndims, astate->nitems);
5733 66 : nbytes += dataoffset;
5734 : }
5735 : else
5736 : {
5737 168 : dataoffset = 0;
5738 168 : nbytes += ARR_OVERHEAD_NONULLS(astate->ndims);
5739 : }
5740 :
5741 234 : result = (ArrayType *) palloc0(nbytes);
5742 234 : SET_VARSIZE(result, nbytes);
5743 234 : result->ndim = astate->ndims;
5744 234 : result->dataoffset = dataoffset;
5745 234 : result->elemtype = astate->element_type;
5746 :
5747 234 : memcpy(ARR_DIMS(result), astate->dims, astate->ndims * sizeof(int));
5748 234 : memcpy(ARR_LBOUND(result), astate->lbs, astate->ndims * sizeof(int));
5749 234 : memcpy(ARR_DATA_PTR(result), astate->data, astate->nbytes);
5750 :
5751 234 : if (astate->nullbitmap != NULL)
5752 66 : array_bitmap_copy(ARR_NULLBITMAP(result), 0,
5753 66 : astate->nullbitmap, 0,
5754 : astate->nitems);
5755 : }
5756 :
5757 234 : MemoryContextSwitchTo(oldcontext);
5758 :
5759 : /* Clean up all the junk */
5760 234 : if (release)
5761 : {
5762 : Assert(astate->private_cxt);
5763 48 : MemoryContextDelete(astate->mcontext);
5764 : }
5765 :
5766 234 : return PointerGetDatum(result);
5767 : }
5768 :
5769 : /*
5770 : * The following three functions provide essentially the same API as
5771 : * initArrayResult/accumArrayResult/makeArrayResult, but can accept either
5772 : * scalar or array inputs, invoking the appropriate set of functions above.
5773 : */
5774 :
5775 : /*
5776 : * initArrayResultAny - initialize an empty ArrayBuildStateAny
5777 : *
5778 : * input_type is the input datatype (either element or array type)
5779 : * rcontext is where to keep working state
5780 : * subcontext is a flag determining whether to use a separate memory context
5781 : */
5782 : ArrayBuildStateAny *
5783 55232 : initArrayResultAny(Oid input_type, MemoryContext rcontext, bool subcontext)
5784 : {
5785 : ArrayBuildStateAny *astate;
5786 :
5787 : /*
5788 : * int2vector and oidvector will satisfy both get_element_type and
5789 : * get_array_type. We prefer to treat them as scalars, to be consistent
5790 : * with get_promoted_array_type. Hence, check get_array_type not
5791 : * get_element_type.
5792 : */
5793 55232 : if (!OidIsValid(get_array_type(input_type)))
5794 : {
5795 : /* Array case */
5796 : ArrayBuildStateArr *arraystate;
5797 :
5798 48 : arraystate = initArrayResultArr(input_type, InvalidOid, rcontext, subcontext);
5799 : astate = (ArrayBuildStateAny *)
5800 48 : MemoryContextAlloc(arraystate->mcontext,
5801 : sizeof(ArrayBuildStateAny));
5802 48 : astate->scalarstate = NULL;
5803 48 : astate->arraystate = arraystate;
5804 : }
5805 : else
5806 : {
5807 : /* Scalar case */
5808 : ArrayBuildState *scalarstate;
5809 :
5810 55184 : scalarstate = initArrayResult(input_type, rcontext, subcontext);
5811 : astate = (ArrayBuildStateAny *)
5812 55184 : MemoryContextAlloc(scalarstate->mcontext,
5813 : sizeof(ArrayBuildStateAny));
5814 55184 : astate->scalarstate = scalarstate;
5815 55184 : astate->arraystate = NULL;
5816 : }
5817 :
5818 55232 : return astate;
5819 : }
5820 :
5821 : /*
5822 : * accumArrayResultAny - accumulate one (more) input for an array result
5823 : *
5824 : * astate is working state (can be NULL on first call)
5825 : * dvalue/disnull represent the new input to append to the array
5826 : * input_type is the input datatype (either element or array type)
5827 : * rcontext is where to keep working state
5828 : */
5829 : ArrayBuildStateAny *
5830 17652 : accumArrayResultAny(ArrayBuildStateAny *astate,
5831 : Datum dvalue, bool disnull,
5832 : Oid input_type,
5833 : MemoryContext rcontext)
5834 : {
5835 17652 : if (astate == NULL)
5836 120 : astate = initArrayResultAny(input_type, rcontext, true);
5837 :
5838 17652 : if (astate->scalarstate)
5839 17514 : (void) accumArrayResult(astate->scalarstate,
5840 : dvalue, disnull,
5841 : input_type, rcontext);
5842 : else
5843 138 : (void) accumArrayResultArr(astate->arraystate,
5844 : dvalue, disnull,
5845 : input_type, rcontext);
5846 :
5847 17652 : return astate;
5848 : }
5849 :
5850 : /*
5851 : * makeArrayResultAny - produce final result of accumArrayResultAny
5852 : *
5853 : * astate is working state (must not be NULL)
5854 : * rcontext is where to construct result
5855 : * release is true if okay to release working state
5856 : */
5857 : Datum
5858 55232 : makeArrayResultAny(ArrayBuildStateAny *astate,
5859 : MemoryContext rcontext, bool release)
5860 : {
5861 : Datum result;
5862 :
5863 55232 : if (astate->scalarstate)
5864 : {
5865 : /* Must use makeMdArrayResult to support "release" parameter */
5866 : int ndims;
5867 : int dims[1];
5868 : int lbs[1];
5869 :
5870 : /* If no elements were presented, we want to create an empty array */
5871 55184 : ndims = (astate->scalarstate->nelems > 0) ? 1 : 0;
5872 55184 : dims[0] = astate->scalarstate->nelems;
5873 55184 : lbs[0] = 1;
5874 :
5875 55184 : result = makeMdArrayResult(astate->scalarstate, ndims, dims, lbs,
5876 : rcontext, release);
5877 : }
5878 : else
5879 : {
5880 48 : result = makeArrayResultArr(astate->arraystate,
5881 : rcontext, release);
5882 : }
5883 55232 : return result;
5884 : }
5885 :
5886 :
5887 : Datum
5888 288 : array_larger(PG_FUNCTION_ARGS)
5889 : {
5890 288 : if (array_cmp(fcinfo) > 0)
5891 144 : PG_RETURN_DATUM(PG_GETARG_DATUM(0));
5892 : else
5893 138 : PG_RETURN_DATUM(PG_GETARG_DATUM(1));
5894 : }
5895 :
5896 : Datum
5897 258 : array_smaller(PG_FUNCTION_ARGS)
5898 : {
5899 258 : if (array_cmp(fcinfo) < 0)
5900 174 : PG_RETURN_DATUM(PG_GETARG_DATUM(0));
5901 : else
5902 84 : PG_RETURN_DATUM(PG_GETARG_DATUM(1));
5903 : }
5904 :
5905 :
5906 : typedef struct generate_subscripts_fctx
5907 : {
5908 : int32 lower;
5909 : int32 upper;
5910 : bool reverse;
5911 : } generate_subscripts_fctx;
5912 :
5913 : /*
5914 : * generate_subscripts(array anyarray, dim int [, reverse bool])
5915 : * Returns all subscripts of the array for any dimension
5916 : */
5917 : Datum
5918 4906 : generate_subscripts(PG_FUNCTION_ARGS)
5919 : {
5920 : FuncCallContext *funcctx;
5921 : MemoryContext oldcontext;
5922 : generate_subscripts_fctx *fctx;
5923 :
5924 : /* stuff done only on the first call of the function */
5925 4906 : if (SRF_IS_FIRSTCALL())
5926 : {
5927 2288 : AnyArrayType *v = PG_GETARG_ANY_ARRAY_P(0);
5928 2288 : int reqdim = PG_GETARG_INT32(1);
5929 : int *lb,
5930 : *dimv;
5931 :
5932 : /* create a function context for cross-call persistence */
5933 2288 : funcctx = SRF_FIRSTCALL_INIT();
5934 :
5935 : /* Sanity check: does it look like an array at all? */
5936 2288 : if (AARR_NDIM(v) <= 0 || AARR_NDIM(v) > MAXDIM)
5937 6 : SRF_RETURN_DONE(funcctx);
5938 :
5939 : /* Sanity check: was the requested dim valid */
5940 2282 : if (reqdim <= 0 || reqdim > AARR_NDIM(v))
5941 0 : SRF_RETURN_DONE(funcctx);
5942 :
5943 : /*
5944 : * switch to memory context appropriate for multiple function calls
5945 : */
5946 2282 : oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);
5947 2282 : fctx = (generate_subscripts_fctx *) palloc(sizeof(generate_subscripts_fctx));
5948 :
5949 2282 : lb = AARR_LBOUND(v);
5950 2282 : dimv = AARR_DIMS(v);
5951 :
5952 2282 : fctx->lower = lb[reqdim - 1];
5953 2282 : fctx->upper = dimv[reqdim - 1] + lb[reqdim - 1] - 1;
5954 2282 : fctx->reverse = (PG_NARGS() < 3) ? false : PG_GETARG_BOOL(2);
5955 :
5956 2282 : funcctx->user_fctx = fctx;
5957 :
5958 2282 : MemoryContextSwitchTo(oldcontext);
5959 : }
5960 :
5961 4900 : funcctx = SRF_PERCALL_SETUP();
5962 :
5963 4900 : fctx = funcctx->user_fctx;
5964 :
5965 4900 : if (fctx->lower <= fctx->upper)
5966 : {
5967 2618 : if (!fctx->reverse)
5968 2618 : SRF_RETURN_NEXT(funcctx, Int32GetDatum(fctx->lower++));
5969 : else
5970 0 : SRF_RETURN_NEXT(funcctx, Int32GetDatum(fctx->upper--));
5971 : }
5972 : else
5973 : /* done when there are no more elements left */
5974 2282 : SRF_RETURN_DONE(funcctx);
5975 : }
5976 :
5977 : /*
5978 : * generate_subscripts_nodir
5979 : * Implements the 2-argument version of generate_subscripts
5980 : */
5981 : Datum
5982 4906 : generate_subscripts_nodir(PG_FUNCTION_ARGS)
5983 : {
5984 : /* just call the other one -- it can handle both cases */
5985 4906 : return generate_subscripts(fcinfo);
5986 : }
5987 :
5988 : /*
5989 : * array_fill_with_lower_bounds
5990 : * Create and fill array with defined lower bounds.
5991 : */
5992 : Datum
5993 66 : array_fill_with_lower_bounds(PG_FUNCTION_ARGS)
5994 : {
5995 : ArrayType *dims;
5996 : ArrayType *lbs;
5997 : ArrayType *result;
5998 : Oid elmtype;
5999 : Datum value;
6000 : bool isnull;
6001 :
6002 66 : if (PG_ARGISNULL(1) || PG_ARGISNULL(2))
6003 12 : ereport(ERROR,
6004 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
6005 : errmsg("dimension array or low bound array cannot be null")));
6006 :
6007 54 : dims = PG_GETARG_ARRAYTYPE_P(1);
6008 54 : lbs = PG_GETARG_ARRAYTYPE_P(2);
6009 :
6010 54 : if (!PG_ARGISNULL(0))
6011 : {
6012 42 : value = PG_GETARG_DATUM(0);
6013 42 : isnull = false;
6014 : }
6015 : else
6016 : {
6017 12 : value = 0;
6018 12 : isnull = true;
6019 : }
6020 :
6021 54 : elmtype = get_fn_expr_argtype(fcinfo->flinfo, 0);
6022 54 : if (!OidIsValid(elmtype))
6023 0 : elog(ERROR, "could not determine data type of input");
6024 :
6025 54 : result = array_fill_internal(dims, lbs, value, isnull, elmtype, fcinfo);
6026 42 : PG_RETURN_ARRAYTYPE_P(result);
6027 : }
6028 :
6029 : /*
6030 : * array_fill
6031 : * Create and fill array with default lower bounds.
6032 : */
6033 : Datum
6034 90 : array_fill(PG_FUNCTION_ARGS)
6035 : {
6036 : ArrayType *dims;
6037 : ArrayType *result;
6038 : Oid elmtype;
6039 : Datum value;
6040 : bool isnull;
6041 :
6042 90 : if (PG_ARGISNULL(1))
6043 0 : ereport(ERROR,
6044 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
6045 : errmsg("dimension array or low bound array cannot be null")));
6046 :
6047 90 : dims = PG_GETARG_ARRAYTYPE_P(1);
6048 :
6049 90 : if (!PG_ARGISNULL(0))
6050 : {
6051 78 : value = PG_GETARG_DATUM(0);
6052 78 : isnull = false;
6053 : }
6054 : else
6055 : {
6056 12 : value = 0;
6057 12 : isnull = true;
6058 : }
6059 :
6060 90 : elmtype = get_fn_expr_argtype(fcinfo->flinfo, 0);
6061 90 : if (!OidIsValid(elmtype))
6062 0 : elog(ERROR, "could not determine data type of input");
6063 :
6064 90 : result = array_fill_internal(dims, NULL, value, isnull, elmtype, fcinfo);
6065 78 : PG_RETURN_ARRAYTYPE_P(result);
6066 : }
6067 :
6068 : static ArrayType *
6069 66 : create_array_envelope(int ndims, int *dimv, int *lbsv, int nbytes,
6070 : Oid elmtype, int dataoffset)
6071 : {
6072 : ArrayType *result;
6073 :
6074 66 : result = (ArrayType *) palloc0(nbytes);
6075 66 : SET_VARSIZE(result, nbytes);
6076 66 : result->ndim = ndims;
6077 66 : result->dataoffset = dataoffset;
6078 66 : result->elemtype = elmtype;
6079 66 : memcpy(ARR_DIMS(result), dimv, ndims * sizeof(int));
6080 66 : memcpy(ARR_LBOUND(result), lbsv, ndims * sizeof(int));
6081 :
6082 66 : return result;
6083 : }
6084 :
6085 : static ArrayType *
6086 144 : array_fill_internal(ArrayType *dims, ArrayType *lbs,
6087 : Datum value, bool isnull, Oid elmtype,
6088 : FunctionCallInfo fcinfo)
6089 : {
6090 : ArrayType *result;
6091 : int *dimv;
6092 : int *lbsv;
6093 : int ndims;
6094 : int nitems;
6095 : int deflbs[MAXDIM];
6096 : int16 elmlen;
6097 : bool elmbyval;
6098 : char elmalign;
6099 : ArrayMetaState *my_extra;
6100 :
6101 : /*
6102 : * Params checks
6103 : */
6104 144 : if (ARR_NDIM(dims) > 1)
6105 6 : ereport(ERROR,
6106 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
6107 : errmsg("wrong number of array subscripts"),
6108 : errdetail("Dimension array must be one dimensional.")));
6109 :
6110 138 : if (array_contains_nulls(dims))
6111 6 : ereport(ERROR,
6112 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
6113 : errmsg("dimension values cannot be null")));
6114 :
6115 132 : dimv = (int *) ARR_DATA_PTR(dims);
6116 132 : ndims = (ARR_NDIM(dims) > 0) ? ARR_DIMS(dims)[0] : 0;
6117 :
6118 132 : if (ndims < 0) /* we do allow zero-dimension arrays */
6119 0 : ereport(ERROR,
6120 : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
6121 : errmsg("invalid number of dimensions: %d", ndims)));
6122 132 : if (ndims > MAXDIM)
6123 0 : ereport(ERROR,
6124 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
6125 : errmsg("number of array dimensions (%d) exceeds the maximum allowed (%d)",
6126 : ndims, MAXDIM)));
6127 :
6128 132 : if (lbs != NULL)
6129 : {
6130 54 : if (ARR_NDIM(lbs) > 1)
6131 0 : ereport(ERROR,
6132 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
6133 : errmsg("wrong number of array subscripts"),
6134 : errdetail("Dimension array must be one dimensional.")));
6135 :
6136 54 : if (array_contains_nulls(lbs))
6137 0 : ereport(ERROR,
6138 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
6139 : errmsg("dimension values cannot be null")));
6140 :
6141 54 : if (ndims != ((ARR_NDIM(lbs) > 0) ? ARR_DIMS(lbs)[0] : 0))
6142 12 : ereport(ERROR,
6143 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
6144 : errmsg("wrong number of array subscripts"),
6145 : errdetail("Low bound array has different size than dimensions array.")));
6146 :
6147 42 : lbsv = (int *) ARR_DATA_PTR(lbs);
6148 : }
6149 : else
6150 : {
6151 : int i;
6152 :
6153 546 : for (i = 0; i < MAXDIM; i++)
6154 468 : deflbs[i] = 1;
6155 :
6156 78 : lbsv = deflbs;
6157 : }
6158 :
6159 : /* This checks for overflow of the array dimensions */
6160 120 : nitems = ArrayGetNItems(ndims, dimv);
6161 120 : ArrayCheckBounds(ndims, dimv, lbsv);
6162 :
6163 : /* fast track for empty array */
6164 120 : if (nitems <= 0)
6165 54 : return construct_empty_array(elmtype);
6166 :
6167 : /*
6168 : * We arrange to look up info about element type only once per series of
6169 : * calls, assuming the element type doesn't change underneath us.
6170 : */
6171 66 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
6172 66 : if (my_extra == NULL)
6173 : {
6174 66 : fcinfo->flinfo->fn_extra = MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
6175 : sizeof(ArrayMetaState));
6176 66 : my_extra = (ArrayMetaState *) fcinfo->flinfo->fn_extra;
6177 66 : my_extra->element_type = InvalidOid;
6178 : }
6179 :
6180 66 : if (my_extra->element_type != elmtype)
6181 : {
6182 : /* Get info about element type */
6183 66 : get_typlenbyvalalign(elmtype,
6184 : &my_extra->typlen,
6185 : &my_extra->typbyval,
6186 : &my_extra->typalign);
6187 66 : my_extra->element_type = elmtype;
6188 : }
6189 :
6190 66 : elmlen = my_extra->typlen;
6191 66 : elmbyval = my_extra->typbyval;
6192 66 : elmalign = my_extra->typalign;
6193 :
6194 : /* compute required space */
6195 66 : if (!isnull)
6196 : {
6197 : int i;
6198 : char *p;
6199 : int nbytes;
6200 : int totbytes;
6201 :
6202 : /* make sure data is not toasted */
6203 42 : if (elmlen == -1)
6204 12 : value = PointerGetDatum(PG_DETOAST_DATUM(value));
6205 :
6206 42 : nbytes = att_addlength_datum(0, elmlen, value);
6207 42 : nbytes = att_align_nominal(nbytes, elmalign);
6208 : Assert(nbytes > 0);
6209 :
6210 42 : totbytes = nbytes * nitems;
6211 :
6212 : /* check for overflow of multiplication or total request */
6213 42 : if (totbytes / nbytes != nitems ||
6214 42 : !AllocSizeIsValid(totbytes))
6215 0 : ereport(ERROR,
6216 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
6217 : errmsg("array size exceeds the maximum allowed (%d)",
6218 : (int) MaxAllocSize)));
6219 :
6220 : /*
6221 : * This addition can't overflow, but it might cause us to go past
6222 : * MaxAllocSize. We leave it to palloc to complain in that case.
6223 : */
6224 42 : totbytes += ARR_OVERHEAD_NONULLS(ndims);
6225 :
6226 42 : result = create_array_envelope(ndims, dimv, lbsv, totbytes,
6227 : elmtype, 0);
6228 :
6229 42 : p = ARR_DATA_PTR(result);
6230 3000306 : for (i = 0; i < nitems; i++)
6231 3000264 : p += ArrayCastAndSet(value, elmlen, elmbyval, elmalign, p);
6232 : }
6233 : else
6234 : {
6235 : int nbytes;
6236 : int dataoffset;
6237 :
6238 24 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndims, nitems);
6239 24 : nbytes = dataoffset;
6240 :
6241 24 : result = create_array_envelope(ndims, dimv, lbsv, nbytes,
6242 : elmtype, dataoffset);
6243 :
6244 : /* create_array_envelope already zeroed the bitmap, so we're done */
6245 : }
6246 :
6247 66 : return result;
6248 : }
6249 :
6250 :
6251 : /*
6252 : * UNNEST
6253 : */
6254 : Datum
6255 537582 : array_unnest(PG_FUNCTION_ARGS)
6256 : {
6257 : typedef struct
6258 : {
6259 : array_iter iter;
6260 : int nextelem;
6261 : int numelems;
6262 : int16 elmlen;
6263 : bool elmbyval;
6264 : char elmalign;
6265 : } array_unnest_fctx;
6266 :
6267 : FuncCallContext *funcctx;
6268 : array_unnest_fctx *fctx;
6269 : MemoryContext oldcontext;
6270 :
6271 : /* stuff done only on the first call of the function */
6272 537582 : if (SRF_IS_FIRSTCALL())
6273 : {
6274 : AnyArrayType *arr;
6275 :
6276 : /* create a function context for cross-call persistence */
6277 87148 : funcctx = SRF_FIRSTCALL_INIT();
6278 :
6279 : /*
6280 : * switch to memory context appropriate for multiple function calls
6281 : */
6282 87148 : oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);
6283 :
6284 : /*
6285 : * Get the array value and detoast if needed. We can't do this
6286 : * earlier because if we have to detoast, we want the detoasted copy
6287 : * to be in multi_call_memory_ctx, so it will go away when we're done
6288 : * and not before. (If no detoast happens, we assume the originally
6289 : * passed array will stick around till then.)
6290 : */
6291 87148 : arr = PG_GETARG_ANY_ARRAY_P(0);
6292 :
6293 : /* allocate memory for user context */
6294 87148 : fctx = (array_unnest_fctx *) palloc(sizeof(array_unnest_fctx));
6295 :
6296 : /* initialize state */
6297 87148 : array_iter_setup(&fctx->iter, arr);
6298 87148 : fctx->nextelem = 0;
6299 87148 : fctx->numelems = ArrayGetNItems(AARR_NDIM(arr), AARR_DIMS(arr));
6300 :
6301 87148 : if (VARATT_IS_EXPANDED_HEADER(arr))
6302 : {
6303 : /* we can just grab the type data from expanded array */
6304 2 : fctx->elmlen = arr->xpn.typlen;
6305 2 : fctx->elmbyval = arr->xpn.typbyval;
6306 2 : fctx->elmalign = arr->xpn.typalign;
6307 : }
6308 : else
6309 87146 : get_typlenbyvalalign(AARR_ELEMTYPE(arr),
6310 : &fctx->elmlen,
6311 : &fctx->elmbyval,
6312 : &fctx->elmalign);
6313 :
6314 87148 : funcctx->user_fctx = fctx;
6315 87148 : MemoryContextSwitchTo(oldcontext);
6316 : }
6317 :
6318 : /* stuff done on every call of the function */
6319 537582 : funcctx = SRF_PERCALL_SETUP();
6320 537582 : fctx = funcctx->user_fctx;
6321 :
6322 537582 : if (fctx->nextelem < fctx->numelems)
6323 : {
6324 450434 : int offset = fctx->nextelem++;
6325 : Datum elem;
6326 :
6327 450434 : elem = array_iter_next(&fctx->iter, &fcinfo->isnull, offset,
6328 450434 : fctx->elmlen, fctx->elmbyval, fctx->elmalign);
6329 :
6330 450434 : SRF_RETURN_NEXT(funcctx, elem);
6331 : }
6332 : else
6333 : {
6334 : /* do when there is no more left */
6335 87148 : SRF_RETURN_DONE(funcctx);
6336 : }
6337 : }
6338 :
6339 : /*
6340 : * Planner support function for array_unnest(anyarray)
6341 : *
6342 : * Note: this is now also used for information_schema._pg_expandarray(),
6343 : * which is simply a wrapper around array_unnest().
6344 : */
6345 : Datum
6346 19200 : array_unnest_support(PG_FUNCTION_ARGS)
6347 : {
6348 19200 : Node *rawreq = (Node *) PG_GETARG_POINTER(0);
6349 19200 : Node *ret = NULL;
6350 :
6351 19200 : if (IsA(rawreq, SupportRequestRows))
6352 : {
6353 : /* Try to estimate the number of rows returned */
6354 6356 : SupportRequestRows *req = (SupportRequestRows *) rawreq;
6355 :
6356 6356 : if (is_funcclause(req->node)) /* be paranoid */
6357 : {
6358 6350 : List *args = ((FuncExpr *) req->node)->args;
6359 : Node *arg1;
6360 :
6361 : /* We can use estimated argument values here */
6362 6350 : arg1 = estimate_expression_value(req->root, linitial(args));
6363 :
6364 6350 : req->rows = estimate_array_length(req->root, arg1);
6365 6350 : ret = (Node *) req;
6366 : }
6367 : }
6368 :
6369 19200 : PG_RETURN_POINTER(ret);
6370 : }
6371 :
6372 :
6373 : /*
6374 : * array_replace/array_remove support
6375 : *
6376 : * Find all array entries matching (not distinct from) search/search_isnull,
6377 : * and delete them if remove is true, else replace them with
6378 : * replace/replace_isnull. Comparisons are done using the specified
6379 : * collation. fcinfo is passed only for caching purposes.
6380 : */
6381 : static ArrayType *
6382 3406 : array_replace_internal(ArrayType *array,
6383 : Datum search, bool search_isnull,
6384 : Datum replace, bool replace_isnull,
6385 : bool remove, Oid collation,
6386 : FunctionCallInfo fcinfo)
6387 : {
6388 3406 : LOCAL_FCINFO(locfcinfo, 2);
6389 : ArrayType *result;
6390 : Oid element_type;
6391 : Datum *values;
6392 : bool *nulls;
6393 : int *dim;
6394 : int ndim;
6395 : int nitems,
6396 : nresult;
6397 : int i;
6398 3406 : int32 nbytes = 0;
6399 : int32 dataoffset;
6400 : bool hasnulls;
6401 : int typlen;
6402 : bool typbyval;
6403 : char typalign;
6404 : char *arraydataptr;
6405 : bits8 *bitmap;
6406 : int bitmask;
6407 3406 : bool changed = false;
6408 : TypeCacheEntry *typentry;
6409 :
6410 3406 : element_type = ARR_ELEMTYPE(array);
6411 3406 : ndim = ARR_NDIM(array);
6412 3406 : dim = ARR_DIMS(array);
6413 3406 : nitems = ArrayGetNItems(ndim, dim);
6414 :
6415 : /* Return input array unmodified if it is empty */
6416 3406 : if (nitems <= 0)
6417 0 : return array;
6418 :
6419 : /*
6420 : * We can't remove elements from multi-dimensional arrays, since the
6421 : * result might not be rectangular.
6422 : */
6423 3406 : if (remove && ndim > 1)
6424 6 : ereport(ERROR,
6425 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
6426 : errmsg("removing elements from multidimensional arrays is not supported")));
6427 :
6428 : /*
6429 : * We arrange to look up the equality function only once per series of
6430 : * calls, assuming the element type doesn't change underneath us.
6431 : */
6432 3400 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
6433 3400 : if (typentry == NULL ||
6434 2584 : typentry->type_id != element_type)
6435 : {
6436 816 : typentry = lookup_type_cache(element_type,
6437 : TYPECACHE_EQ_OPR_FINFO);
6438 816 : if (!OidIsValid(typentry->eq_opr_finfo.fn_oid))
6439 0 : ereport(ERROR,
6440 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
6441 : errmsg("could not identify an equality operator for type %s",
6442 : format_type_be(element_type))));
6443 816 : fcinfo->flinfo->fn_extra = typentry;
6444 : }
6445 3400 : typlen = typentry->typlen;
6446 3400 : typbyval = typentry->typbyval;
6447 3400 : typalign = typentry->typalign;
6448 :
6449 : /*
6450 : * Detoast values if they are toasted. The replacement value must be
6451 : * detoasted for insertion into the result array, while detoasting the
6452 : * search value only once saves cycles.
6453 : */
6454 3400 : if (typlen == -1)
6455 : {
6456 3358 : if (!search_isnull)
6457 3352 : search = PointerGetDatum(PG_DETOAST_DATUM(search));
6458 3358 : if (!replace_isnull)
6459 12 : replace = PointerGetDatum(PG_DETOAST_DATUM(replace));
6460 : }
6461 :
6462 : /* Prepare to apply the comparison operator */
6463 3400 : InitFunctionCallInfoData(*locfcinfo, &typentry->eq_opr_finfo, 2,
6464 : collation, NULL, NULL);
6465 :
6466 : /* Allocate temporary arrays for new values */
6467 3400 : values = (Datum *) palloc(nitems * sizeof(Datum));
6468 3400 : nulls = (bool *) palloc(nitems * sizeof(bool));
6469 :
6470 : /* Loop over source data */
6471 3400 : arraydataptr = ARR_DATA_PTR(array);
6472 3400 : bitmap = ARR_NULLBITMAP(array);
6473 3400 : bitmask = 1;
6474 3400 : hasnulls = false;
6475 3400 : nresult = 0;
6476 :
6477 7420 : for (i = 0; i < nitems; i++)
6478 : {
6479 : Datum elt;
6480 : bool isNull;
6481 : bool oprresult;
6482 4020 : bool skip = false;
6483 :
6484 : /* Get source element, checking for NULL */
6485 4020 : if (bitmap && (*bitmap & bitmask) == 0)
6486 : {
6487 36 : isNull = true;
6488 : /* If searching for NULL, we have a match */
6489 36 : if (search_isnull)
6490 : {
6491 36 : if (remove)
6492 : {
6493 12 : skip = true;
6494 12 : changed = true;
6495 : }
6496 24 : else if (!replace_isnull)
6497 : {
6498 18 : values[nresult] = replace;
6499 18 : isNull = false;
6500 18 : changed = true;
6501 : }
6502 : }
6503 : }
6504 : else
6505 : {
6506 3984 : isNull = false;
6507 3984 : elt = fetch_att(arraydataptr, typbyval, typlen);
6508 3984 : arraydataptr = att_addlength_datum(arraydataptr, typlen, elt);
6509 3984 : arraydataptr = (char *) att_align_nominal(arraydataptr, typalign);
6510 :
6511 3984 : if (search_isnull)
6512 : {
6513 : /* no match possible, keep element */
6514 54 : values[nresult] = elt;
6515 : }
6516 : else
6517 : {
6518 : /*
6519 : * Apply the operator to the element pair; treat NULL as false
6520 : */
6521 3930 : locfcinfo->args[0].value = elt;
6522 3930 : locfcinfo->args[0].isnull = false;
6523 3930 : locfcinfo->args[1].value = search;
6524 3930 : locfcinfo->args[1].isnull = false;
6525 3930 : locfcinfo->isnull = false;
6526 3930 : oprresult = DatumGetBool(FunctionCallInvoke(locfcinfo));
6527 3930 : if (locfcinfo->isnull || !oprresult)
6528 : {
6529 : /* no match, keep element */
6530 3772 : values[nresult] = elt;
6531 : }
6532 : else
6533 : {
6534 : /* match, so replace or delete */
6535 158 : changed = true;
6536 158 : if (remove)
6537 134 : skip = true;
6538 : else
6539 : {
6540 24 : values[nresult] = replace;
6541 24 : isNull = replace_isnull;
6542 : }
6543 : }
6544 : }
6545 : }
6546 :
6547 4020 : if (!skip)
6548 : {
6549 3874 : nulls[nresult] = isNull;
6550 3874 : if (isNull)
6551 12 : hasnulls = true;
6552 : else
6553 : {
6554 : /* Update total result size */
6555 3862 : nbytes = att_addlength_datum(nbytes, typlen, values[nresult]);
6556 3862 : nbytes = att_align_nominal(nbytes, typalign);
6557 : /* check for overflow of total request */
6558 3862 : if (!AllocSizeIsValid(nbytes))
6559 0 : ereport(ERROR,
6560 : (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
6561 : errmsg("array size exceeds the maximum allowed (%d)",
6562 : (int) MaxAllocSize)));
6563 : }
6564 3874 : nresult++;
6565 : }
6566 :
6567 : /* advance bitmap pointer if any */
6568 4020 : if (bitmap)
6569 : {
6570 90 : bitmask <<= 1;
6571 90 : if (bitmask == 0x100)
6572 : {
6573 0 : bitmap++;
6574 0 : bitmask = 1;
6575 : }
6576 : }
6577 : }
6578 :
6579 : /*
6580 : * If not changed just return the original array
6581 : */
6582 3400 : if (!changed)
6583 : {
6584 3248 : pfree(values);
6585 3248 : pfree(nulls);
6586 3248 : return array;
6587 : }
6588 :
6589 : /* If all elements were removed return an empty array */
6590 152 : if (nresult == 0)
6591 : {
6592 6 : pfree(values);
6593 6 : pfree(nulls);
6594 6 : return construct_empty_array(element_type);
6595 : }
6596 :
6597 : /* Allocate and initialize the result array */
6598 146 : if (hasnulls)
6599 : {
6600 6 : dataoffset = ARR_OVERHEAD_WITHNULLS(ndim, nresult);
6601 6 : nbytes += dataoffset;
6602 : }
6603 : else
6604 : {
6605 140 : dataoffset = 0; /* marker for no null bitmap */
6606 140 : nbytes += ARR_OVERHEAD_NONULLS(ndim);
6607 : }
6608 146 : result = (ArrayType *) palloc0(nbytes);
6609 146 : SET_VARSIZE(result, nbytes);
6610 146 : result->ndim = ndim;
6611 146 : result->dataoffset = dataoffset;
6612 146 : result->elemtype = element_type;
6613 146 : memcpy(ARR_DIMS(result), ARR_DIMS(array), ndim * sizeof(int));
6614 146 : memcpy(ARR_LBOUND(result), ARR_LBOUND(array), ndim * sizeof(int));
6615 :
6616 146 : if (remove)
6617 : {
6618 : /* Adjust the result length */
6619 116 : ARR_DIMS(result)[0] = nresult;
6620 : }
6621 :
6622 : /* Insert data into result array */
6623 146 : CopyArrayEls(result,
6624 : values, nulls, nresult,
6625 : typlen, typbyval, typalign,
6626 : false);
6627 :
6628 146 : pfree(values);
6629 146 : pfree(nulls);
6630 :
6631 146 : return result;
6632 : }
6633 :
6634 : /*
6635 : * Remove any occurrences of an element from an array
6636 : *
6637 : * If used on a multi-dimensional array this will raise an error.
6638 : */
6639 : Datum
6640 196402 : array_remove(PG_FUNCTION_ARGS)
6641 : {
6642 : ArrayType *array;
6643 196402 : Datum search = PG_GETARG_DATUM(1);
6644 196402 : bool search_isnull = PG_ARGISNULL(1);
6645 :
6646 196402 : if (PG_ARGISNULL(0))
6647 193032 : PG_RETURN_NULL();
6648 3370 : array = PG_GETARG_ARRAYTYPE_P(0);
6649 :
6650 3370 : array = array_replace_internal(array,
6651 : search, search_isnull,
6652 : (Datum) 0, true,
6653 : true, PG_GET_COLLATION(),
6654 : fcinfo);
6655 3364 : PG_RETURN_ARRAYTYPE_P(array);
6656 : }
6657 :
6658 : /*
6659 : * Replace any occurrences of an element in an array
6660 : */
6661 : Datum
6662 36 : array_replace(PG_FUNCTION_ARGS)
6663 : {
6664 : ArrayType *array;
6665 36 : Datum search = PG_GETARG_DATUM(1);
6666 36 : bool search_isnull = PG_ARGISNULL(1);
6667 36 : Datum replace = PG_GETARG_DATUM(2);
6668 36 : bool replace_isnull = PG_ARGISNULL(2);
6669 :
6670 36 : if (PG_ARGISNULL(0))
6671 0 : PG_RETURN_NULL();
6672 36 : array = PG_GETARG_ARRAYTYPE_P(0);
6673 :
6674 36 : array = array_replace_internal(array,
6675 : search, search_isnull,
6676 : replace, replace_isnull,
6677 : false, PG_GET_COLLATION(),
6678 : fcinfo);
6679 36 : PG_RETURN_ARRAYTYPE_P(array);
6680 : }
6681 :
6682 : /*
6683 : * Implements width_bucket(anyelement, anyarray).
6684 : *
6685 : * 'thresholds' is an array containing lower bound values for each bucket;
6686 : * these must be sorted from smallest to largest, or bogus results will be
6687 : * produced. If N thresholds are supplied, the output is from 0 to N:
6688 : * 0 is for inputs < first threshold, N is for inputs >= last threshold.
6689 : */
6690 : Datum
6691 810 : width_bucket_array(PG_FUNCTION_ARGS)
6692 : {
6693 810 : Datum operand = PG_GETARG_DATUM(0);
6694 810 : ArrayType *thresholds = PG_GETARG_ARRAYTYPE_P(1);
6695 810 : Oid collation = PG_GET_COLLATION();
6696 810 : Oid element_type = ARR_ELEMTYPE(thresholds);
6697 : int result;
6698 :
6699 : /* Check input */
6700 810 : if (ARR_NDIM(thresholds) > 1)
6701 6 : ereport(ERROR,
6702 : (errcode(ERRCODE_ARRAY_SUBSCRIPT_ERROR),
6703 : errmsg("thresholds must be one-dimensional array")));
6704 :
6705 804 : if (array_contains_nulls(thresholds))
6706 6 : ereport(ERROR,
6707 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
6708 : errmsg("thresholds array must not contain NULLs")));
6709 :
6710 : /* We have a dedicated implementation for float8 data */
6711 798 : if (element_type == FLOAT8OID)
6712 366 : result = width_bucket_array_float8(operand, thresholds);
6713 : else
6714 : {
6715 : TypeCacheEntry *typentry;
6716 :
6717 : /* Cache information about the input type */
6718 432 : typentry = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
6719 432 : if (typentry == NULL ||
6720 390 : typentry->type_id != element_type)
6721 : {
6722 42 : typentry = lookup_type_cache(element_type,
6723 : TYPECACHE_CMP_PROC_FINFO);
6724 42 : if (!OidIsValid(typentry->cmp_proc_finfo.fn_oid))
6725 0 : ereport(ERROR,
6726 : (errcode(ERRCODE_UNDEFINED_FUNCTION),
6727 : errmsg("could not identify a comparison function for type %s",
6728 : format_type_be(element_type))));
6729 42 : fcinfo->flinfo->fn_extra = typentry;
6730 : }
6731 :
6732 : /*
6733 : * We have separate implementation paths for fixed- and variable-width
6734 : * types, since indexing the array is a lot cheaper in the first case.
6735 : */
6736 432 : if (typentry->typlen > 0)
6737 90 : result = width_bucket_array_fixed(operand, thresholds,
6738 : collation, typentry);
6739 : else
6740 342 : result = width_bucket_array_variable(operand, thresholds,
6741 : collation, typentry);
6742 : }
6743 :
6744 : /* Avoid leaking memory when handed toasted input. */
6745 798 : PG_FREE_IF_COPY(thresholds, 1);
6746 :
6747 798 : PG_RETURN_INT32(result);
6748 : }
6749 :
6750 : /*
6751 : * width_bucket_array for float8 data.
6752 : */
6753 : static int
6754 366 : width_bucket_array_float8(Datum operand, ArrayType *thresholds)
6755 : {
6756 366 : float8 op = DatumGetFloat8(operand);
6757 : float8 *thresholds_data;
6758 : int left;
6759 : int right;
6760 :
6761 : /*
6762 : * Since we know the array contains no NULLs, we can just index it
6763 : * directly.
6764 : */
6765 366 : thresholds_data = (float8 *) ARR_DATA_PTR(thresholds);
6766 :
6767 366 : left = 0;
6768 366 : right = ArrayGetNItems(ARR_NDIM(thresholds), ARR_DIMS(thresholds));
6769 :
6770 : /*
6771 : * If the probe value is a NaN, it's greater than or equal to all possible
6772 : * threshold values (including other NaNs), so we need not search. Note
6773 : * that this would give the same result as searching even if the array
6774 : * contains multiple NaNs (as long as they're correctly sorted), since the
6775 : * loop logic will find the rightmost of multiple equal threshold values.
6776 : */
6777 366 : if (isnan(op))
6778 6 : return right;
6779 :
6780 : /* Find the bucket */
6781 1134 : while (left < right)
6782 : {
6783 774 : int mid = (left + right) / 2;
6784 :
6785 774 : if (isnan(thresholds_data[mid]) || op < thresholds_data[mid])
6786 336 : right = mid;
6787 : else
6788 438 : left = mid + 1;
6789 : }
6790 :
6791 360 : return left;
6792 : }
6793 :
6794 : /*
6795 : * width_bucket_array for generic fixed-width data types.
6796 : */
6797 : static int
6798 90 : width_bucket_array_fixed(Datum operand,
6799 : ArrayType *thresholds,
6800 : Oid collation,
6801 : TypeCacheEntry *typentry)
6802 : {
6803 90 : LOCAL_FCINFO(locfcinfo, 2);
6804 : char *thresholds_data;
6805 90 : int typlen = typentry->typlen;
6806 90 : bool typbyval = typentry->typbyval;
6807 : int left;
6808 : int right;
6809 :
6810 : /*
6811 : * Since we know the array contains no NULLs, we can just index it
6812 : * directly.
6813 : */
6814 90 : thresholds_data = (char *) ARR_DATA_PTR(thresholds);
6815 :
6816 90 : InitFunctionCallInfoData(*locfcinfo, &typentry->cmp_proc_finfo, 2,
6817 : collation, NULL, NULL);
6818 :
6819 : /* Find the bucket */
6820 90 : left = 0;
6821 90 : right = ArrayGetNItems(ARR_NDIM(thresholds), ARR_DIMS(thresholds));
6822 270 : while (left < right)
6823 : {
6824 180 : int mid = (left + right) / 2;
6825 : char *ptr;
6826 : int32 cmpresult;
6827 :
6828 180 : ptr = thresholds_data + mid * typlen;
6829 :
6830 180 : locfcinfo->args[0].value = operand;
6831 180 : locfcinfo->args[0].isnull = false;
6832 180 : locfcinfo->args[1].value = fetch_att(ptr, typbyval, typlen);
6833 180 : locfcinfo->args[1].isnull = false;
6834 :
6835 180 : cmpresult = DatumGetInt32(FunctionCallInvoke(locfcinfo));
6836 :
6837 : /* We don't expect comparison support functions to return null */
6838 : Assert(!locfcinfo->isnull);
6839 :
6840 180 : if (cmpresult < 0)
6841 90 : right = mid;
6842 : else
6843 90 : left = mid + 1;
6844 : }
6845 :
6846 90 : return left;
6847 : }
6848 :
6849 : /*
6850 : * width_bucket_array for generic variable-width data types.
6851 : */
6852 : static int
6853 342 : width_bucket_array_variable(Datum operand,
6854 : ArrayType *thresholds,
6855 : Oid collation,
6856 : TypeCacheEntry *typentry)
6857 : {
6858 342 : LOCAL_FCINFO(locfcinfo, 2);
6859 : char *thresholds_data;
6860 342 : int typlen = typentry->typlen;
6861 342 : bool typbyval = typentry->typbyval;
6862 342 : char typalign = typentry->typalign;
6863 : int left;
6864 : int right;
6865 :
6866 342 : thresholds_data = (char *) ARR_DATA_PTR(thresholds);
6867 :
6868 342 : InitFunctionCallInfoData(*locfcinfo, &typentry->cmp_proc_finfo, 2,
6869 : collation, NULL, NULL);
6870 :
6871 : /* Find the bucket */
6872 342 : left = 0;
6873 342 : right = ArrayGetNItems(ARR_NDIM(thresholds), ARR_DIMS(thresholds));
6874 1068 : while (left < right)
6875 : {
6876 726 : int mid = (left + right) / 2;
6877 : char *ptr;
6878 : int i;
6879 : int32 cmpresult;
6880 :
6881 : /* Locate mid'th array element by advancing from left element */
6882 726 : ptr = thresholds_data;
6883 1242 : for (i = left; i < mid; i++)
6884 : {
6885 516 : ptr = att_addlength_pointer(ptr, typlen, ptr);
6886 516 : ptr = (char *) att_align_nominal(ptr, typalign);
6887 : }
6888 :
6889 726 : locfcinfo->args[0].value = operand;
6890 726 : locfcinfo->args[0].isnull = false;
6891 726 : locfcinfo->args[1].value = fetch_att(ptr, typbyval, typlen);
6892 726 : locfcinfo->args[1].isnull = false;
6893 :
6894 726 : cmpresult = DatumGetInt32(FunctionCallInvoke(locfcinfo));
6895 :
6896 : /* We don't expect comparison support functions to return null */
6897 : Assert(!locfcinfo->isnull);
6898 :
6899 726 : if (cmpresult < 0)
6900 300 : right = mid;
6901 : else
6902 : {
6903 426 : left = mid + 1;
6904 :
6905 : /*
6906 : * Move the thresholds pointer to match new "left" index, so we
6907 : * don't have to seek over those elements again. This trick
6908 : * ensures we do only O(N) array indexing work, not O(N^2).
6909 : */
6910 426 : ptr = att_addlength_pointer(ptr, typlen, ptr);
6911 426 : thresholds_data = (char *) att_align_nominal(ptr, typalign);
6912 : }
6913 : }
6914 :
6915 342 : return left;
6916 : }
6917 :
6918 : /*
6919 : * Trim the last N elements from an array by building an appropriate slice.
6920 : * Only the first dimension is trimmed.
6921 : */
6922 : Datum
6923 48 : trim_array(PG_FUNCTION_ARGS)
6924 : {
6925 48 : ArrayType *v = PG_GETARG_ARRAYTYPE_P(0);
6926 48 : int n = PG_GETARG_INT32(1);
6927 48 : int array_length = (ARR_NDIM(v) > 0) ? ARR_DIMS(v)[0] : 0;
6928 : int16 elmlen;
6929 : bool elmbyval;
6930 : char elmalign;
6931 : int lower[MAXDIM];
6932 : int upper[MAXDIM];
6933 : bool lowerProvided[MAXDIM];
6934 : bool upperProvided[MAXDIM];
6935 : Datum result;
6936 :
6937 : /* Per spec, throw an error if out of bounds */
6938 48 : if (n < 0 || n > array_length)
6939 18 : ereport(ERROR,
6940 : (errcode(ERRCODE_ARRAY_ELEMENT_ERROR),
6941 : errmsg("number of elements to trim must be between 0 and %d",
6942 : array_length)));
6943 :
6944 : /* Set all the bounds as unprovided except the first upper bound */
6945 30 : memset(lowerProvided, false, sizeof(lowerProvided));
6946 30 : memset(upperProvided, false, sizeof(upperProvided));
6947 30 : if (ARR_NDIM(v) > 0)
6948 : {
6949 30 : upper[0] = ARR_LBOUND(v)[0] + array_length - n - 1;
6950 30 : upperProvided[0] = true;
6951 : }
6952 :
6953 : /* Fetch the needed information about the element type */
6954 30 : get_typlenbyvalalign(ARR_ELEMTYPE(v), &elmlen, &elmbyval, &elmalign);
6955 :
6956 : /* Get the slice */
6957 30 : result = array_get_slice(PointerGetDatum(v), 1,
6958 : upper, lower, upperProvided, lowerProvided,
6959 : -1, elmlen, elmbyval, elmalign);
6960 :
6961 30 : PG_RETURN_DATUM(result);
6962 : }
|