Age Owner Branch data TLA Line data Source code
1 : : /*-------------------------------------------------------------------------
2 : : *
3 : : * rangetypes.c
4 : : * I/O functions, operators, and support functions for range types.
5 : : *
6 : : * The stored (serialized) format of a range value is:
7 : : *
8 : : * 4 bytes: varlena header
9 : : * 4 bytes: range type's OID
10 : : * Lower boundary value, if any, aligned according to subtype's typalign
11 : : * Upper boundary value, if any, aligned according to subtype's typalign
12 : : * 1 byte for flags
13 : : *
14 : : * This representation is chosen to avoid needing any padding before the
15 : : * lower boundary value, even when it requires double alignment. We can
16 : : * expect that the varlena header is presented to us on a suitably aligned
17 : : * boundary (possibly after detoasting), and then the lower boundary is too.
18 : : * Note that this means we can't work with a packed (short varlena header)
19 : : * value; we must detoast it first.
20 : : *
21 : : *
22 : : * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
23 : : * Portions Copyright (c) 1994, Regents of the University of California
24 : : *
25 : : *
26 : : * IDENTIFICATION
27 : : * src/backend/utils/adt/rangetypes.c
28 : : *
29 : : *-------------------------------------------------------------------------
30 : : */
31 : : #include "postgres.h"
32 : :
33 : : #include "common/hashfn.h"
34 : : #include "libpq/pqformat.h"
35 : : #include "miscadmin.h"
36 : : #include "nodes/makefuncs.h"
37 : : #include "nodes/miscnodes.h"
38 : : #include "nodes/supportnodes.h"
39 : : #include "optimizer/clauses.h"
40 : : #include "optimizer/cost.h"
41 : : #include "optimizer/optimizer.h"
42 : : #include "utils/builtins.h"
43 : : #include "utils/date.h"
44 : : #include "utils/lsyscache.h"
45 : : #include "utils/rangetypes.h"
46 : : #include "utils/timestamp.h"
47 : :
48 : :
49 : : /* fn_extra cache entry for one of the range I/O functions */
50 : : typedef struct RangeIOData
51 : : {
52 : : TypeCacheEntry *typcache; /* range type's typcache entry */
53 : : FmgrInfo typioproc; /* element type's I/O function */
54 : : Oid typioparam; /* element type's I/O parameter */
55 : : } RangeIOData;
56 : :
57 : :
58 : : static RangeIOData *get_range_io_data(FunctionCallInfo fcinfo, Oid rngtypid,
59 : : IOFuncSelector func);
60 : : static char range_parse_flags(const char *flags_str);
61 : : static bool range_parse(const char *string, char *flags, char **lbound_str,
62 : : char **ubound_str, Node *escontext);
63 : : static const char *range_parse_bound(const char *string, const char *ptr,
64 : : char **bound_str, bool *infinite,
65 : : Node *escontext);
66 : : static char *range_deparse(char flags, const char *lbound_str,
67 : : const char *ubound_str);
68 : : static char *range_bound_escape(const char *value);
69 : : static Size datum_compute_size(Size data_length, Datum val, bool typbyval,
70 : : char typalign, int16 typlen, char typstorage);
71 : : static Pointer datum_write(Pointer ptr, Datum datum, bool typbyval,
72 : : char typalign, int16 typlen, char typstorage);
73 : : static Node *find_simplified_clause(PlannerInfo *root,
74 : : Expr *rangeExpr, Expr *elemExpr);
75 : : static Expr *build_bound_expr(Expr *elemExpr, Datum val,
76 : : bool isLowerBound, bool isInclusive,
77 : : TypeCacheEntry *typeCache,
78 : : Oid opfamily, Oid rng_collation);
79 : :
80 : :
81 : : /*
82 : : *----------------------------------------------------------
83 : : * I/O FUNCTIONS
84 : : *----------------------------------------------------------
85 : : */
86 : :
87 : : Datum
4546 heikki.linnakangas@i 88 :CBC 2229 : range_in(PG_FUNCTION_ARGS)
89 : : {
4535 bruce@momjian.us 90 : 2229 : char *input_str = PG_GETARG_CSTRING(0);
91 : 2229 : Oid rngtypoid = PG_GETARG_OID(1);
92 : 2229 : Oid typmod = PG_GETARG_INT32(2);
486 tgl@sss.pgh.pa.us 93 : 2229 : Node *escontext = fcinfo->context;
94 : : RangeType *range;
95 : : RangeIOData *cache;
96 : : char flags;
97 : : char *lbound_str;
98 : : char *ubound_str;
99 : : RangeBound lower;
100 : : RangeBound upper;
101 : :
3114 noah@leadboat.com 102 : 2229 : check_stack_depth(); /* recurses when subtype is a range type */
103 : :
4534 tgl@sss.pgh.pa.us 104 : 2229 : cache = get_range_io_data(fcinfo, rngtypoid, IOFunc_input);
105 : :
106 : : /* parse */
486 107 [ + + ]: 2229 : if (!range_parse(input_str, &flags, &lbound_str, &ubound_str, escontext))
108 : 9 : PG_RETURN_NULL();
109 : :
110 : : /* call element type's input function */
4546 heikki.linnakangas@i 111 [ + + ]: 2181 : if (RANGE_HAS_LBOUND(flags))
486 tgl@sss.pgh.pa.us 112 [ - + ]: 1845 : if (!InputFunctionCallSafe(&cache->typioproc, lbound_str,
113 : : cache->typioparam, typmod,
114 : : escontext, &lower.val))
486 tgl@sss.pgh.pa.us 115 :UBC 0 : PG_RETURN_NULL();
4546 heikki.linnakangas@i 116 [ + + ]:CBC 2181 : if (RANGE_HAS_UBOUND(flags))
486 tgl@sss.pgh.pa.us 117 [ + + ]: 1797 : if (!InputFunctionCallSafe(&cache->typioproc, ubound_str,
118 : : cache->typioparam, typmod,
119 : : escontext, &upper.val))
120 : 12 : PG_RETURN_NULL();
121 : :
4534 122 : 2169 : lower.infinite = (flags & RANGE_LB_INF) != 0;
123 : 2169 : lower.inclusive = (flags & RANGE_LB_INC) != 0;
124 : 2169 : lower.lower = true;
125 : 2169 : upper.infinite = (flags & RANGE_UB_INF) != 0;
126 : 2169 : upper.inclusive = (flags & RANGE_UB_INC) != 0;
127 : 2169 : upper.lower = false;
128 : :
129 : : /* serialize and canonicalize */
486 130 : 2169 : range = make_range(cache->typcache, &lower, &upper,
131 : 2169 : flags & RANGE_EMPTY, escontext);
132 : :
2400 133 : 2160 : PG_RETURN_RANGE_P(range);
134 : : }
135 : :
136 : : Datum
4546 heikki.linnakangas@i 137 : 52522 : range_out(PG_FUNCTION_ARGS)
138 : : {
2400 tgl@sss.pgh.pa.us 139 : 52522 : RangeType *range = PG_GETARG_RANGE_P(0);
140 : : char *output_str;
141 : : RangeIOData *cache;
142 : : char flags;
4546 heikki.linnakangas@i 143 : 52522 : char *lbound_str = NULL;
144 : 52522 : char *ubound_str = NULL;
145 : : RangeBound lower;
146 : : RangeBound upper;
147 : : bool empty;
148 : :
3114 noah@leadboat.com 149 : 52522 : check_stack_depth(); /* recurses when subtype is a range type */
150 : :
4534 tgl@sss.pgh.pa.us 151 : 52522 : cache = get_range_io_data(fcinfo, RangeTypeGetOid(range), IOFunc_output);
152 : :
153 : : /* deserialize */
154 : 52522 : range_deserialize(cache->typcache, range, &lower, &upper, &empty);
155 : 52522 : flags = range_get_flags(range);
156 : :
157 : : /* call element type's output function */
4546 heikki.linnakangas@i 158 [ + + ]: 52522 : if (RANGE_HAS_LBOUND(flags))
1501 alvherre@alvh.no-ip. 159 : 42839 : lbound_str = OutputFunctionCall(&cache->typioproc, lower.val);
4546 heikki.linnakangas@i 160 [ + + ]: 52522 : if (RANGE_HAS_UBOUND(flags))
1501 alvherre@alvh.no-ip. 161 : 42803 : ubound_str = OutputFunctionCall(&cache->typioproc, upper.val);
162 : :
163 : : /* construct result string */
4546 heikki.linnakangas@i 164 : 52522 : output_str = range_deparse(flags, lbound_str, ubound_str);
165 : :
166 : 52522 : PG_RETURN_CSTRING(output_str);
167 : : }
168 : :
169 : : /*
170 : : * Binary representation: The first byte is the flags, then the lower bound
171 : : * (if present), then the upper bound (if present). Each bound is represented
172 : : * by a 4-byte length header and the binary representation of that bound (as
173 : : * returned by a call to the send function for the subtype).
174 : : */
175 : :
176 : : Datum
4546 heikki.linnakangas@i 177 :UBC 0 : range_recv(PG_FUNCTION_ARGS)
178 : : {
4535 bruce@momjian.us 179 : 0 : StringInfo buf = (StringInfo) PG_GETARG_POINTER(0);
4534 tgl@sss.pgh.pa.us 180 : 0 : Oid rngtypoid = PG_GETARG_OID(1);
4535 bruce@momjian.us 181 : 0 : int32 typmod = PG_GETARG_INT32(2);
182 : : RangeType *range;
183 : : RangeIOData *cache;
184 : : char flags;
185 : : RangeBound lower;
186 : : RangeBound upper;
187 : :
3114 noah@leadboat.com 188 : 0 : check_stack_depth(); /* recurses when subtype is a range type */
189 : :
4534 tgl@sss.pgh.pa.us 190 : 0 : cache = get_range_io_data(fcinfo, rngtypoid, IOFunc_receive);
191 : :
192 : : /* receive the flags... */
193 : 0 : flags = (unsigned char) pq_getmsgbyte(buf);
194 : :
195 : : /*
196 : : * Mask out any unsupported flags, particularly RANGE_xB_NULL which would
197 : : * confuse following tests. Note that range_serialize will take care of
198 : : * cleaning up any inconsistencies in the remaining flags.
199 : : */
200 : 0 : flags &= (RANGE_EMPTY |
201 : : RANGE_LB_INC |
202 : : RANGE_LB_INF |
203 : : RANGE_UB_INC |
204 : : RANGE_UB_INF);
205 : :
206 : : /* receive the bounds ... */
4546 heikki.linnakangas@i 207 [ # # ]: 0 : if (RANGE_HAS_LBOUND(flags))
208 : : {
4535 bruce@momjian.us 209 : 0 : uint32 bound_len = pq_getmsgint(buf, 4);
210 : 0 : const char *bound_data = pq_getmsgbytes(buf, bound_len);
211 : : StringInfoData bound_buf;
212 : :
4546 heikki.linnakangas@i 213 : 0 : initStringInfo(&bound_buf);
214 : 0 : appendBinaryStringInfo(&bound_buf, bound_data, bound_len);
215 : :
1501 alvherre@alvh.no-ip. 216 : 0 : lower.val = ReceiveFunctionCall(&cache->typioproc,
217 : : &bound_buf,
218 : : cache->typioparam,
219 : : typmod);
4546 heikki.linnakangas@i 220 : 0 : pfree(bound_buf.data);
221 : : }
222 : : else
223 : 0 : lower.val = (Datum) 0;
224 : :
225 [ # # ]: 0 : if (RANGE_HAS_UBOUND(flags))
226 : : {
4535 bruce@momjian.us 227 : 0 : uint32 bound_len = pq_getmsgint(buf, 4);
228 : 0 : const char *bound_data = pq_getmsgbytes(buf, bound_len);
229 : : StringInfoData bound_buf;
230 : :
4546 heikki.linnakangas@i 231 : 0 : initStringInfo(&bound_buf);
232 : 0 : appendBinaryStringInfo(&bound_buf, bound_data, bound_len);
233 : :
1501 alvherre@alvh.no-ip. 234 : 0 : upper.val = ReceiveFunctionCall(&cache->typioproc,
235 : : &bound_buf,
236 : : cache->typioparam,
237 : : typmod);
4546 heikki.linnakangas@i 238 : 0 : pfree(bound_buf.data);
239 : : }
240 : : else
241 : 0 : upper.val = (Datum) 0;
242 : :
243 : 0 : pq_getmsgend(buf);
244 : :
245 : : /* finish constructing RangeBound representation */
4535 bruce@momjian.us 246 : 0 : lower.infinite = (flags & RANGE_LB_INF) != 0;
4546 heikki.linnakangas@i 247 : 0 : lower.inclusive = (flags & RANGE_LB_INC) != 0;
4535 bruce@momjian.us 248 : 0 : lower.lower = true;
249 : 0 : upper.infinite = (flags & RANGE_UB_INF) != 0;
4546 heikki.linnakangas@i 250 : 0 : upper.inclusive = (flags & RANGE_UB_INC) != 0;
4535 bruce@momjian.us 251 : 0 : upper.lower = false;
252 : :
253 : : /* serialize and canonicalize */
486 tgl@sss.pgh.pa.us 254 : 0 : range = make_range(cache->typcache, &lower, &upper,
255 : 0 : flags & RANGE_EMPTY, NULL);
256 : :
2400 257 : 0 : PG_RETURN_RANGE_P(range);
258 : : }
259 : :
260 : : Datum
4546 heikki.linnakangas@i 261 : 0 : range_send(PG_FUNCTION_ARGS)
262 : : {
2400 tgl@sss.pgh.pa.us 263 : 0 : RangeType *range = PG_GETARG_RANGE_P(0);
4535 bruce@momjian.us 264 : 0 : StringInfo buf = makeStringInfo();
265 : : RangeIOData *cache;
266 : : char flags;
267 : : RangeBound lower;
268 : : RangeBound upper;
269 : : bool empty;
270 : :
3114 noah@leadboat.com 271 : 0 : check_stack_depth(); /* recurses when subtype is a range type */
272 : :
4534 tgl@sss.pgh.pa.us 273 : 0 : cache = get_range_io_data(fcinfo, RangeTypeGetOid(range), IOFunc_send);
274 : :
275 : : /* deserialize */
276 : 0 : range_deserialize(cache->typcache, range, &lower, &upper, &empty);
277 : 0 : flags = range_get_flags(range);
278 : :
279 : : /* construct output */
4546 heikki.linnakangas@i 280 : 0 : pq_begintypsend(buf);
281 : :
282 : 0 : pq_sendbyte(buf, flags);
283 : :
284 [ # # ]: 0 : if (RANGE_HAS_LBOUND(flags))
285 : : {
1501 alvherre@alvh.no-ip. 286 : 0 : Datum bound = PointerGetDatum(SendFunctionCall(&cache->typioproc,
287 : : lower.val));
4535 bruce@momjian.us 288 : 0 : uint32 bound_len = VARSIZE(bound) - VARHDRSZ;
289 : 0 : char *bound_data = VARDATA(bound);
290 : :
2377 andres@anarazel.de 291 : 0 : pq_sendint32(buf, bound_len);
4546 heikki.linnakangas@i 292 : 0 : pq_sendbytes(buf, bound_data, bound_len);
293 : : }
294 : :
295 [ # # ]: 0 : if (RANGE_HAS_UBOUND(flags))
296 : : {
1501 alvherre@alvh.no-ip. 297 : 0 : Datum bound = PointerGetDatum(SendFunctionCall(&cache->typioproc,
298 : : upper.val));
4535 bruce@momjian.us 299 : 0 : uint32 bound_len = VARSIZE(bound) - VARHDRSZ;
300 : 0 : char *bound_data = VARDATA(bound);
301 : :
2377 andres@anarazel.de 302 : 0 : pq_sendint32(buf, bound_len);
4546 heikki.linnakangas@i 303 : 0 : pq_sendbytes(buf, bound_data, bound_len);
304 : : }
305 : :
306 : 0 : PG_RETURN_BYTEA_P(pq_endtypsend(buf));
307 : : }
308 : :
309 : : /*
310 : : * get_range_io_data: get cached information needed for range type I/O
311 : : *
312 : : * The range I/O functions need a bit more cached info than other range
313 : : * functions, so they store a RangeIOData struct in fn_extra, not just a
314 : : * pointer to a type cache entry.
315 : : */
316 : : static RangeIOData *
4534 tgl@sss.pgh.pa.us 317 :CBC 54751 : get_range_io_data(FunctionCallInfo fcinfo, Oid rngtypid, IOFuncSelector func)
318 : : {
319 : 54751 : RangeIOData *cache = (RangeIOData *) fcinfo->flinfo->fn_extra;
320 : :
321 [ + + - + ]: 54751 : if (cache == NULL || cache->typcache->type_id != rngtypid)
322 : : {
323 : : int16 typlen;
324 : : bool typbyval;
325 : : char typalign;
326 : : char typdelim;
327 : : Oid typiofunc;
328 : :
329 : 3397 : cache = (RangeIOData *) MemoryContextAlloc(fcinfo->flinfo->fn_mcxt,
330 : : sizeof(RangeIOData));
331 : 3397 : cache->typcache = lookup_type_cache(rngtypid, TYPECACHE_RANGE_INFO);
332 [ - + ]: 3397 : if (cache->typcache->rngelemtype == NULL)
4534 tgl@sss.pgh.pa.us 333 [ # # ]:UBC 0 : elog(ERROR, "type %u is not a range type", rngtypid);
334 : :
335 : : /* get_type_io_data does more than we need, but is convenient */
4534 tgl@sss.pgh.pa.us 336 :CBC 3397 : get_type_io_data(cache->typcache->rngelemtype->type_id,
337 : : func,
338 : : &typlen,
339 : : &typbyval,
340 : : &typalign,
341 : : &typdelim,
342 : : &cache->typioparam,
343 : : &typiofunc);
344 : :
1501 alvherre@alvh.no-ip. 345 [ - + ]: 3397 : if (!OidIsValid(typiofunc))
346 : : {
347 : : /* this could only happen for receive or send */
4534 tgl@sss.pgh.pa.us 348 [ # # ]:UBC 0 : if (func == IOFunc_receive)
349 [ # # ]: 0 : ereport(ERROR,
350 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
351 : : errmsg("no binary input function available for type %s",
352 : : format_type_be(cache->typcache->rngelemtype->type_id))));
353 : : else
354 [ # # ]: 0 : ereport(ERROR,
355 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
356 : : errmsg("no binary output function available for type %s",
357 : : format_type_be(cache->typcache->rngelemtype->type_id))));
358 : : }
1501 alvherre@alvh.no-ip. 359 :CBC 3397 : fmgr_info_cxt(typiofunc, &cache->typioproc,
4534 tgl@sss.pgh.pa.us 360 : 3397 : fcinfo->flinfo->fn_mcxt);
361 : :
362 : 3397 : fcinfo->flinfo->fn_extra = (void *) cache;
363 : : }
364 : :
365 : 54751 : return cache;
366 : : }
367 : :
368 : :
369 : : /*
370 : : *----------------------------------------------------------
371 : : * GENERIC FUNCTIONS
372 : : *----------------------------------------------------------
373 : : */
374 : :
375 : : /* Construct standard-form range value from two arguments */
376 : : Datum
4546 heikki.linnakangas@i 377 : 54480 : range_constructor2(PG_FUNCTION_ARGS)
378 : : {
4535 bruce@momjian.us 379 : 54480 : Datum arg1 = PG_GETARG_DATUM(0);
380 : 54480 : Datum arg2 = PG_GETARG_DATUM(1);
381 : 54480 : Oid rngtypid = get_fn_expr_rettype(fcinfo->flinfo);
382 : : RangeType *range;
383 : : TypeCacheEntry *typcache;
384 : : RangeBound lower;
385 : : RangeBound upper;
386 : :
4534 tgl@sss.pgh.pa.us 387 : 54480 : typcache = range_get_typcache(fcinfo, rngtypid);
388 : :
4535 bruce@momjian.us 389 [ + + ]: 54480 : lower.val = PG_ARGISNULL(0) ? (Datum) 0 : arg1;
390 : 54480 : lower.infinite = PG_ARGISNULL(0);
4534 tgl@sss.pgh.pa.us 391 : 54480 : lower.inclusive = true;
4535 bruce@momjian.us 392 : 54480 : lower.lower = true;
393 : :
394 [ + + ]: 54480 : upper.val = PG_ARGISNULL(1) ? (Datum) 0 : arg2;
395 : 54480 : upper.infinite = PG_ARGISNULL(1);
4534 tgl@sss.pgh.pa.us 396 : 54480 : upper.inclusive = false;
4535 bruce@momjian.us 397 : 54480 : upper.lower = false;
398 : :
486 tgl@sss.pgh.pa.us 399 : 54480 : range = make_range(typcache, &lower, &upper, false, NULL);
400 : :
2400 401 : 54462 : PG_RETURN_RANGE_P(range);
402 : : }
403 : :
404 : : /* Construct general range value from three arguments */
405 : : Datum
4546 heikki.linnakangas@i 406 : 2583 : range_constructor3(PG_FUNCTION_ARGS)
407 : : {
4535 bruce@momjian.us 408 : 2583 : Datum arg1 = PG_GETARG_DATUM(0);
409 : 2583 : Datum arg2 = PG_GETARG_DATUM(1);
410 : 2583 : Oid rngtypid = get_fn_expr_rettype(fcinfo->flinfo);
411 : : RangeType *range;
412 : : TypeCacheEntry *typcache;
413 : : RangeBound lower;
414 : : RangeBound upper;
415 : : char flags;
416 : :
4534 tgl@sss.pgh.pa.us 417 : 2583 : typcache = range_get_typcache(fcinfo, rngtypid);
418 : :
4546 heikki.linnakangas@i 419 [ - + ]: 2583 : if (PG_ARGISNULL(2))
4546 heikki.linnakangas@i 420 [ # # ]:UBC 0 : ereport(ERROR,
421 : : (errcode(ERRCODE_DATA_EXCEPTION),
422 : : errmsg("range constructor flags argument must not be null")));
423 : :
2590 noah@leadboat.com 424 :CBC 2583 : flags = range_parse_flags(text_to_cstring(PG_GETARG_TEXT_PP(2)));
425 : :
4535 bruce@momjian.us 426 [ + + ]: 2583 : lower.val = PG_ARGISNULL(0) ? (Datum) 0 : arg1;
427 : 2583 : lower.infinite = PG_ARGISNULL(0);
4534 tgl@sss.pgh.pa.us 428 : 2583 : lower.inclusive = (flags & RANGE_LB_INC) != 0;
4535 bruce@momjian.us 429 : 2583 : lower.lower = true;
430 : :
431 [ + + ]: 2583 : upper.val = PG_ARGISNULL(1) ? (Datum) 0 : arg2;
432 : 2583 : upper.infinite = PG_ARGISNULL(1);
4534 tgl@sss.pgh.pa.us 433 : 2583 : upper.inclusive = (flags & RANGE_UB_INC) != 0;
4535 bruce@momjian.us 434 : 2583 : upper.lower = false;
435 : :
486 tgl@sss.pgh.pa.us 436 : 2583 : range = make_range(typcache, &lower, &upper, false, NULL);
437 : :
2400 438 : 2583 : PG_RETURN_RANGE_P(range);
439 : : }
440 : :
441 : :
442 : : /* range -> subtype functions */
443 : :
444 : : /* extract lower bound value */
445 : : Datum
4546 heikki.linnakangas@i 446 : 120 : range_lower(PG_FUNCTION_ARGS)
447 : : {
2400 tgl@sss.pgh.pa.us 448 : 120 : RangeType *r1 = PG_GETARG_RANGE_P(0);
449 : : TypeCacheEntry *typcache;
450 : : RangeBound lower;
451 : : RangeBound upper;
452 : : bool empty;
453 : :
4534 454 : 120 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
455 : :
456 : 120 : range_deserialize(typcache, r1, &lower, &upper, &empty);
457 : :
458 : : /* Return NULL if there's no finite lower bound */
4535 459 [ + + + + ]: 120 : if (empty || lower.infinite)
460 : 18 : PG_RETURN_NULL();
461 : :
4546 heikki.linnakangas@i 462 : 102 : PG_RETURN_DATUM(lower.val);
463 : : }
464 : :
465 : : /* extract upper bound value */
466 : : Datum
467 : 114 : range_upper(PG_FUNCTION_ARGS)
468 : : {
2400 tgl@sss.pgh.pa.us 469 : 114 : RangeType *r1 = PG_GETARG_RANGE_P(0);
470 : : TypeCacheEntry *typcache;
471 : : RangeBound lower;
472 : : RangeBound upper;
473 : : bool empty;
474 : :
4534 475 : 114 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
476 : :
477 : 114 : range_deserialize(typcache, r1, &lower, &upper, &empty);
478 : :
479 : : /* Return NULL if there's no finite upper bound */
4535 480 [ + + + + ]: 114 : if (empty || upper.infinite)
481 : 18 : PG_RETURN_NULL();
482 : :
4546 heikki.linnakangas@i 483 : 96 : PG_RETURN_DATUM(upper.val);
484 : : }
485 : :
486 : :
487 : : /* range -> bool functions */
488 : :
489 : : /* is range empty? */
490 : : Datum
491 : 1098 : range_empty(PG_FUNCTION_ARGS)
492 : : {
2400 tgl@sss.pgh.pa.us 493 : 1098 : RangeType *r1 = PG_GETARG_RANGE_P(0);
4534 494 : 1098 : char flags = range_get_flags(r1);
495 : :
496 : 1098 : PG_RETURN_BOOL(flags & RANGE_EMPTY);
497 : : }
498 : :
499 : : /* is lower bound inclusive? */
500 : : Datum
4546 heikki.linnakangas@i 501 : 36 : range_lower_inc(PG_FUNCTION_ARGS)
502 : : {
2400 tgl@sss.pgh.pa.us 503 : 36 : RangeType *r1 = PG_GETARG_RANGE_P(0);
4534 504 : 36 : char flags = range_get_flags(r1);
505 : :
506 : 36 : PG_RETURN_BOOL(flags & RANGE_LB_INC);
507 : : }
508 : :
509 : : /* is upper bound inclusive? */
510 : : Datum
4546 heikki.linnakangas@i 511 : 36 : range_upper_inc(PG_FUNCTION_ARGS)
512 : : {
2400 tgl@sss.pgh.pa.us 513 : 36 : RangeType *r1 = PG_GETARG_RANGE_P(0);
4534 514 : 36 : char flags = range_get_flags(r1);
515 : :
516 : 36 : PG_RETURN_BOOL(flags & RANGE_UB_INC);
517 : : }
518 : :
519 : : /* is lower bound infinite? */
520 : : Datum
4546 heikki.linnakangas@i 521 : 36 : range_lower_inf(PG_FUNCTION_ARGS)
522 : : {
2400 tgl@sss.pgh.pa.us 523 : 36 : RangeType *r1 = PG_GETARG_RANGE_P(0);
4534 524 : 36 : char flags = range_get_flags(r1);
525 : :
526 : 36 : PG_RETURN_BOOL(flags & RANGE_LB_INF);
527 : : }
528 : :
529 : : /* is upper bound infinite? */
530 : : Datum
4546 heikki.linnakangas@i 531 : 36 : range_upper_inf(PG_FUNCTION_ARGS)
532 : : {
2400 tgl@sss.pgh.pa.us 533 : 36 : RangeType *r1 = PG_GETARG_RANGE_P(0);
4534 534 : 36 : char flags = range_get_flags(r1);
535 : :
536 : 36 : PG_RETURN_BOOL(flags & RANGE_UB_INF);
537 : : }
538 : :
539 : :
540 : : /* range, element -> bool functions */
541 : :
542 : : /* contains? */
543 : : Datum
4527 544 : 38013 : range_contains_elem(PG_FUNCTION_ARGS)
545 : : {
2400 546 : 38013 : RangeType *r = PG_GETARG_RANGE_P(0);
4527 547 : 38013 : Datum val = PG_GETARG_DATUM(1);
548 : : TypeCacheEntry *typcache;
549 : :
550 : 38013 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
551 : :
552 : 38013 : PG_RETURN_BOOL(range_contains_elem_internal(typcache, r, val));
553 : : }
554 : :
555 : : /* contained by? */
556 : : Datum
557 : 42 : elem_contained_by_range(PG_FUNCTION_ARGS)
558 : : {
559 : 42 : Datum val = PG_GETARG_DATUM(0);
2400 560 : 42 : RangeType *r = PG_GETARG_RANGE_P(1);
561 : : TypeCacheEntry *typcache;
562 : :
4527 563 : 42 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
564 : :
565 : 42 : PG_RETURN_BOOL(range_contains_elem_internal(typcache, r, val));
566 : : }
567 : :
568 : :
569 : : /* range, range -> bool functions */
570 : :
571 : : /* equality (internal version) */
572 : : bool
1627 peter@eisentraut.org 573 : 92770 : range_eq_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
574 : : {
575 : : RangeBound lower1,
576 : : lower2;
577 : : RangeBound upper1,
578 : : upper2;
579 : : bool empty1,
580 : : empty2;
581 : :
582 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 583 [ - + ]: 92770 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 584 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
585 : :
4534 tgl@sss.pgh.pa.us 586 :CBC 92770 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
587 : 92770 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
588 : :
4546 heikki.linnakangas@i 589 [ + + + + ]: 92770 : if (empty1 && empty2)
4288 590 : 6231 : return true;
4546 591 [ + + ]: 86539 : if (empty1 != empty2)
4288 592 : 6762 : return false;
593 : :
4534 tgl@sss.pgh.pa.us 594 [ + + ]: 79777 : if (range_cmp_bounds(typcache, &lower1, &lower2) != 0)
4288 heikki.linnakangas@i 595 : 37099 : return false;
596 : :
4534 tgl@sss.pgh.pa.us 597 [ + + ]: 42678 : if (range_cmp_bounds(typcache, &upper1, &upper2) != 0)
4288 heikki.linnakangas@i 598 : 25157 : return false;
599 : :
600 : 17521 : return true;
601 : : }
602 : :
603 : : /* equality */
604 : : Datum
605 : 38973 : range_eq(PG_FUNCTION_ARGS)
606 : : {
2400 tgl@sss.pgh.pa.us 607 : 38973 : RangeType *r1 = PG_GETARG_RANGE_P(0);
608 : 38973 : RangeType *r2 = PG_GETARG_RANGE_P(1);
609 : : TypeCacheEntry *typcache;
610 : :
4288 heikki.linnakangas@i 611 : 38973 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
612 : :
613 : 38973 : PG_RETURN_BOOL(range_eq_internal(typcache, r1, r2));
614 : : }
615 : :
616 : : /* inequality (internal version) */
617 : : bool
1627 peter@eisentraut.org 618 :UBC 0 : range_ne_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
619 : : {
4288 heikki.linnakangas@i 620 : 0 : return (!range_eq_internal(typcache, r1, r2));
621 : : }
622 : :
623 : : /* inequality */
624 : : Datum
4546 625 : 0 : range_ne(PG_FUNCTION_ARGS)
626 : : {
2400 tgl@sss.pgh.pa.us 627 : 0 : RangeType *r1 = PG_GETARG_RANGE_P(0);
628 : 0 : RangeType *r2 = PG_GETARG_RANGE_P(1);
629 : : TypeCacheEntry *typcache;
630 : :
4288 heikki.linnakangas@i 631 : 0 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
632 : :
633 : 0 : PG_RETURN_BOOL(range_ne_internal(typcache, r1, r2));
634 : : }
635 : :
636 : : /* contains? */
637 : : Datum
4546 heikki.linnakangas@i 638 :CBC 77235 : range_contains(PG_FUNCTION_ARGS)
639 : : {
2400 tgl@sss.pgh.pa.us 640 : 77235 : RangeType *r1 = PG_GETARG_RANGE_P(0);
641 : 77235 : RangeType *r2 = PG_GETARG_RANGE_P(1);
642 : : TypeCacheEntry *typcache;
643 : :
4534 644 : 77235 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
645 : :
646 : 77235 : PG_RETURN_BOOL(range_contains_internal(typcache, r1, r2));
647 : : }
648 : :
649 : : /* contained by? */
650 : : Datum
4546 heikki.linnakangas@i 651 : 38466 : range_contained_by(PG_FUNCTION_ARGS)
652 : : {
2400 tgl@sss.pgh.pa.us 653 : 38466 : RangeType *r1 = PG_GETARG_RANGE_P(0);
654 : 38466 : RangeType *r2 = PG_GETARG_RANGE_P(1);
655 : : TypeCacheEntry *typcache;
656 : :
4534 657 : 38466 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
658 : :
4288 heikki.linnakangas@i 659 : 38466 : PG_RETURN_BOOL(range_contained_by_internal(typcache, r1, r2));
660 : : }
661 : :
662 : : /* strictly left of? (internal version) */
663 : : bool
1627 peter@eisentraut.org 664 : 62358 : range_before_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
665 : : {
666 : : RangeBound lower1,
667 : : lower2;
668 : : RangeBound upper1,
669 : : upper2;
670 : : bool empty1,
671 : : empty2;
672 : :
673 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 674 [ - + ]: 62358 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 675 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
676 : :
4534 tgl@sss.pgh.pa.us 677 :CBC 62358 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
678 : 62358 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
679 : :
680 : : /* An empty range is neither before nor after any other range */
4546 heikki.linnakangas@i 681 [ + + + + ]: 62358 : if (empty1 || empty2)
4288 682 : 7455 : return false;
683 : :
684 : 54903 : return (range_cmp_bounds(typcache, &upper1, &lower2) < 0);
685 : : }
686 : :
687 : : /* strictly left of? */
688 : : Datum
689 : 39459 : range_before(PG_FUNCTION_ARGS)
690 : : {
2400 tgl@sss.pgh.pa.us 691 : 39459 : RangeType *r1 = PG_GETARG_RANGE_P(0);
692 : 39459 : RangeType *r2 = PG_GETARG_RANGE_P(1);
693 : : TypeCacheEntry *typcache;
694 : :
4288 heikki.linnakangas@i 695 : 39459 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
696 : :
697 : 39459 : PG_RETURN_BOOL(range_before_internal(typcache, r1, r2));
698 : : }
699 : :
700 : : /* strictly right of? (internal version) */
701 : : bool
1627 peter@eisentraut.org 702 : 97443 : range_after_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
703 : : {
704 : : RangeBound lower1,
705 : : lower2;
706 : : RangeBound upper1,
707 : : upper2;
708 : : bool empty1,
709 : : empty2;
710 : :
711 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 712 [ - + ]: 97443 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 713 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
714 : :
4534 tgl@sss.pgh.pa.us 715 :CBC 97443 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
716 : 97443 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
717 : :
718 : : /* An empty range is neither before nor after any other range */
4546 heikki.linnakangas@i 719 [ + + + + ]: 97443 : if (empty1 || empty2)
4288 720 : 7155 : return false;
721 : :
722 : 90288 : return (range_cmp_bounds(typcache, &lower1, &upper2) > 0);
723 : : }
724 : :
725 : : /* strictly right of? */
726 : : Datum
727 : 39153 : range_after(PG_FUNCTION_ARGS)
728 : : {
2400 tgl@sss.pgh.pa.us 729 : 39153 : RangeType *r1 = PG_GETARG_RANGE_P(0);
730 : 39153 : RangeType *r2 = PG_GETARG_RANGE_P(1);
731 : : TypeCacheEntry *typcache;
732 : :
4288 heikki.linnakangas@i 733 : 39153 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
734 : :
735 : 39153 : PG_RETURN_BOOL(range_after_internal(typcache, r1, r2));
736 : : }
737 : :
738 : : /*
739 : : * Check if two bounds A and B are "adjacent", where A is an upper bound and B
740 : : * is a lower bound. For the bounds to be adjacent, each subtype value must
741 : : * satisfy strictly one of the bounds: there are no values which satisfy both
742 : : * bounds (i.e. less than A and greater than B); and there are no values which
743 : : * satisfy neither bound (i.e. greater than A and less than B).
744 : : *
745 : : * For discrete ranges, we rely on the canonicalization function to see if A..B
746 : : * normalizes to empty. (If there is no canonicalization function, it's
747 : : * impossible for such a range to normalize to empty, so we needn't bother to
748 : : * try.)
749 : : *
750 : : * If A == B, the ranges are adjacent only if the bounds have different
751 : : * inclusive flags (i.e., exactly one of the ranges includes the common
752 : : * boundary point).
753 : : *
754 : : * And if A > B then the ranges are not adjacent in this order.
755 : : */
756 : : bool
4055 757 : 237603 : bounds_adjacent(TypeCacheEntry *typcache, RangeBound boundA, RangeBound boundB)
758 : : {
759 : : int cmp;
760 : :
761 [ + - - + ]: 237603 : Assert(!boundA.lower && boundB.lower);
762 : :
763 : 237603 : cmp = range_cmp_bound_values(typcache, &boundA, &boundB);
764 [ + + ]: 237603 : if (cmp < 0)
765 : : {
766 : : RangeType *r;
767 : :
768 : : /*
769 : : * Bounds do not overlap; see if there are points in between.
770 : : */
771 : :
772 : : /* in a continuous subtype, there are assumed to be points between */
773 [ + + ]: 73144 : if (!OidIsValid(typcache->rng_canonical_finfo.fn_oid))
774 : 408 : return false;
775 : :
776 : : /*
777 : : * The bounds are of a discrete range type; so make a range A..B and
778 : : * see if it's empty.
779 : : */
780 : :
781 : : /* flip the inclusion flags */
782 : 72736 : boundA.inclusive = !boundA.inclusive;
783 : 72736 : boundB.inclusive = !boundB.inclusive;
784 : : /* change upper/lower labels to avoid Assert failures */
785 : 72736 : boundA.lower = true;
786 : 72736 : boundB.lower = false;
486 tgl@sss.pgh.pa.us 787 : 72736 : r = make_range(typcache, &boundA, &boundB, false, NULL);
4055 heikki.linnakangas@i 788 : 72736 : return RangeIsEmpty(r);
789 : : }
790 [ + + ]: 164459 : else if (cmp == 0)
791 : 916 : return boundA.inclusive != boundB.inclusive;
792 : : else
3973 bruce@momjian.us 793 : 163543 : return false; /* bounds overlap */
794 : : }
795 : :
796 : : /* adjacent to (but not overlapping)? (internal version) */
797 : : bool
1627 peter@eisentraut.org 798 : 71762 : range_adjacent_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
799 : : {
800 : : RangeBound lower1,
801 : : lower2;
802 : : RangeBound upper1,
803 : : upper2;
804 : : bool empty1,
805 : : empty2;
806 : :
807 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 808 [ - + ]: 71762 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 809 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
810 : :
4534 tgl@sss.pgh.pa.us 811 :CBC 71762 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
812 : 71762 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
813 : :
814 : : /* An empty range is not adjacent to any other range */
4546 heikki.linnakangas@i 815 [ + + - + ]: 71762 : if (empty1 || empty2)
4288 816 : 6000 : return false;
817 : :
818 : : /*
819 : : * Given two ranges A..B and C..D, the ranges are adjacent if and only if
820 : : * B is adjacent to C, or D is adjacent to A.
821 : : */
4055 822 [ + + + + ]: 130800 : return (bounds_adjacent(typcache, upper1, lower2) ||
823 : 65038 : bounds_adjacent(typcache, upper2, lower1));
824 : : }
825 : :
826 : : /* adjacent to (but not overlapping)? */
827 : : Datum
4288 828 : 37218 : range_adjacent(PG_FUNCTION_ARGS)
829 : : {
2400 tgl@sss.pgh.pa.us 830 : 37218 : RangeType *r1 = PG_GETARG_RANGE_P(0);
831 : 37218 : RangeType *r2 = PG_GETARG_RANGE_P(1);
832 : : TypeCacheEntry *typcache;
833 : :
4288 heikki.linnakangas@i 834 : 37218 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
835 : :
836 : 37218 : PG_RETURN_BOOL(range_adjacent_internal(typcache, r1, r2));
837 : : }
838 : :
839 : : /* overlaps? (internal version) */
840 : : bool
1627 peter@eisentraut.org 841 : 49009 : range_overlaps_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
842 : : {
843 : : RangeBound lower1,
844 : : lower2;
845 : : RangeBound upper1,
846 : : upper2;
847 : : bool empty1,
848 : : empty2;
849 : :
850 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 851 [ - + ]: 49009 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 852 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
853 : :
4534 tgl@sss.pgh.pa.us 854 :CBC 49009 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
855 : 49009 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
856 : :
857 : : /* An empty range does not overlap any other range */
4546 heikki.linnakangas@i 858 [ + + + + ]: 49009 : if (empty1 || empty2)
4288 859 : 7046 : return false;
860 : :
4534 tgl@sss.pgh.pa.us 861 [ + + + + ]: 80668 : if (range_cmp_bounds(typcache, &lower1, &lower2) >= 0 &&
862 : 38705 : range_cmp_bounds(typcache, &lower1, &upper2) <= 0)
4288 heikki.linnakangas@i 863 : 1999 : return true;
864 : :
4534 tgl@sss.pgh.pa.us 865 [ + + + + ]: 43222 : if (range_cmp_bounds(typcache, &lower2, &lower1) >= 0 &&
866 : 3258 : range_cmp_bounds(typcache, &lower2, &upper1) <= 0)
4288 heikki.linnakangas@i 867 : 2949 : return true;
868 : :
869 : 37015 : return false;
870 : : }
871 : :
872 : : /* overlaps? */
873 : : Datum
874 : 38705 : range_overlaps(PG_FUNCTION_ARGS)
875 : : {
2400 tgl@sss.pgh.pa.us 876 : 38705 : RangeType *r1 = PG_GETARG_RANGE_P(0);
877 : 38705 : RangeType *r2 = PG_GETARG_RANGE_P(1);
878 : : TypeCacheEntry *typcache;
879 : :
4288 heikki.linnakangas@i 880 : 38705 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
881 : :
882 : 38705 : PG_RETURN_BOOL(range_overlaps_internal(typcache, r1, r2));
883 : : }
884 : :
885 : : /* does not extend to right of? (internal version) */
886 : : bool
1627 peter@eisentraut.org 887 : 66009 : range_overleft_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
888 : : {
889 : : RangeBound lower1,
890 : : lower2;
891 : : RangeBound upper1,
892 : : upper2;
893 : : bool empty1,
894 : : empty2;
895 : :
896 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 897 [ - + ]: 66009 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 898 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
899 : :
4534 tgl@sss.pgh.pa.us 900 :CBC 66009 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
901 : 66009 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
902 : :
903 : : /* An empty range is neither before nor after any other range */
4546 heikki.linnakangas@i 904 [ + + - + ]: 66009 : if (empty1 || empty2)
4288 905 : 6573 : return false;
906 : :
4534 tgl@sss.pgh.pa.us 907 [ + + ]: 59436 : if (range_cmp_bounds(typcache, &upper1, &upper2) <= 0)
4288 heikki.linnakangas@i 908 : 20315 : return true;
909 : :
910 : 39121 : return false;
911 : : }
912 : :
913 : : /* does not extend to right of? */
914 : : Datum
915 : 38253 : range_overleft(PG_FUNCTION_ARGS)
916 : : {
2400 tgl@sss.pgh.pa.us 917 : 38253 : RangeType *r1 = PG_GETARG_RANGE_P(0);
918 : 38253 : RangeType *r2 = PG_GETARG_RANGE_P(1);
919 : : TypeCacheEntry *typcache;
920 : :
4288 heikki.linnakangas@i 921 : 38253 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
922 : :
923 : 38253 : PG_RETURN_BOOL(range_overleft_internal(typcache, r1, r2));
924 : : }
925 : :
926 : : /* does not extend to left of? (internal version) */
927 : : bool
1627 peter@eisentraut.org 928 : 109024 : range_overright_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
929 : : {
930 : : RangeBound lower1,
931 : : lower2;
932 : : RangeBound upper1,
933 : : upper2;
934 : : bool empty1,
935 : : empty2;
936 : :
937 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 938 [ - + ]: 109024 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 939 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
940 : :
4534 tgl@sss.pgh.pa.us 941 :CBC 109024 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
942 : 109024 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
943 : :
944 : : /* An empty range is neither before nor after any other range */
4546 heikki.linnakangas@i 945 [ + + - + ]: 109024 : if (empty1 || empty2)
3529 tgl@sss.pgh.pa.us 946 : 6573 : return false;
947 : :
4534 948 [ + + ]: 102451 : if (range_cmp_bounds(typcache, &lower1, &lower2) >= 0)
3529 949 : 95490 : return true;
950 : :
951 : 6961 : return false;
952 : : }
953 : :
954 : : /* does not extend to left of? */
955 : : Datum
4288 heikki.linnakangas@i 956 : 38250 : range_overright(PG_FUNCTION_ARGS)
957 : : {
2400 tgl@sss.pgh.pa.us 958 : 38250 : RangeType *r1 = PG_GETARG_RANGE_P(0);
959 : 38250 : RangeType *r2 = PG_GETARG_RANGE_P(1);
960 : : TypeCacheEntry *typcache;
961 : :
4288 heikki.linnakangas@i 962 : 38250 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
963 : :
964 : 38250 : PG_RETURN_BOOL(range_overright_internal(typcache, r1, r2));
965 : : }
966 : :
967 : :
968 : : /* range, range -> range functions */
969 : :
970 : : /* set difference */
971 : : Datum
4546 972 : 15 : range_minus(PG_FUNCTION_ARGS)
973 : : {
2400 tgl@sss.pgh.pa.us 974 : 15 : RangeType *r1 = PG_GETARG_RANGE_P(0);
975 : 15 : RangeType *r2 = PG_GETARG_RANGE_P(1);
976 : : RangeType *ret;
977 : : TypeCacheEntry *typcache;
978 : :
979 : : /* Different types should be prevented by ANYRANGE matching rules */
1211 akorotkov@postgresql 980 [ - + ]: 15 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
1211 akorotkov@postgresql 981 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
982 : :
1211 akorotkov@postgresql 983 :CBC 15 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
984 : :
985 : 15 : ret = range_minus_internal(typcache, r1, r2);
986 [ + - ]: 15 : if (ret)
987 : 15 : PG_RETURN_RANGE_P(ret);
988 : : else
1211 akorotkov@postgresql 989 :UBC 0 : PG_RETURN_NULL();
990 : : }
991 : :
992 : : RangeType *
1211 akorotkov@postgresql 993 :CBC 48 : range_minus_internal(TypeCacheEntry *typcache, RangeType *r1, RangeType *r2)
994 : : {
995 : : RangeBound lower1,
996 : : lower2;
997 : : RangeBound upper1,
998 : : upper2;
999 : : bool empty1,
1000 : : empty2;
1001 : : int cmp_l1l2,
1002 : : cmp_l1u2,
1003 : : cmp_u1l2,
1004 : : cmp_u1u2;
1005 : :
4534 tgl@sss.pgh.pa.us 1006 : 48 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
1007 : 48 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
1008 : :
1009 : : /* if either is empty, r1 is the correct answer */
4546 heikki.linnakangas@i 1010 [ + - - + ]: 48 : if (empty1 || empty2)
1211 akorotkov@postgresql 1011 :UBC 0 : return r1;
1012 : :
4534 tgl@sss.pgh.pa.us 1013 :CBC 48 : cmp_l1l2 = range_cmp_bounds(typcache, &lower1, &lower2);
1014 : 48 : cmp_l1u2 = range_cmp_bounds(typcache, &lower1, &upper2);
1015 : 48 : cmp_u1l2 = range_cmp_bounds(typcache, &upper1, &lower2);
1016 : 48 : cmp_u1u2 = range_cmp_bounds(typcache, &upper1, &upper2);
1017 : :
4546 heikki.linnakangas@i 1018 [ + + - + ]: 48 : if (cmp_l1l2 < 0 && cmp_u1u2 > 0)
4546 heikki.linnakangas@i 1019 [ # # ]:UBC 0 : ereport(ERROR,
1020 : : (errcode(ERRCODE_DATA_EXCEPTION),
1021 : : errmsg("result of range difference would not be contiguous")));
1022 : :
4546 heikki.linnakangas@i 1023 [ + - + + ]:CBC 48 : if (cmp_l1u2 > 0 || cmp_u1l2 < 0)
1211 akorotkov@postgresql 1024 : 6 : return r1;
1025 : :
4546 heikki.linnakangas@i 1026 [ + + + + ]: 42 : if (cmp_l1l2 >= 0 && cmp_u1u2 <= 0)
1211 akorotkov@postgresql 1027 : 21 : return make_empty_range(typcache);
1028 : :
4546 heikki.linnakangas@i 1029 [ + + + - : 21 : if (cmp_l1l2 <= 0 && cmp_u1l2 >= 0 && cmp_u1u2 <= 0)
+ + ]
1030 : : {
1031 : 12 : lower2.inclusive = !lower2.inclusive;
4535 bruce@momjian.us 1032 : 12 : lower2.lower = false; /* it will become the upper bound */
486 tgl@sss.pgh.pa.us 1033 : 12 : return make_range(typcache, &lower1, &lower2, false, NULL);
1034 : : }
1035 : :
4546 heikki.linnakangas@i 1036 [ + - + - : 9 : if (cmp_l1l2 >= 0 && cmp_u1u2 >= 0 && cmp_l1u2 <= 0)
+ - ]
1037 : : {
1038 : 9 : upper2.inclusive = !upper2.inclusive;
4535 bruce@momjian.us 1039 : 9 : upper2.lower = true; /* it will become the lower bound */
486 tgl@sss.pgh.pa.us 1040 : 9 : return make_range(typcache, &upper2, &upper1, false, NULL);
1041 : : }
1042 : :
4535 tgl@sss.pgh.pa.us 1043 [ # # ]:UBC 0 : elog(ERROR, "unexpected case in range_minus");
1044 : : return NULL;
1045 : : }
1046 : :
1047 : : /*
1048 : : * Set union. If strict is true, it is an error that the two input ranges
1049 : : * are not adjacent or overlapping.
1050 : : */
1051 : : RangeType *
3267 alvherre@alvh.no-ip. 1052 :CBC 757 : range_union_internal(TypeCacheEntry *typcache, RangeType *r1, RangeType *r2,
1053 : : bool strict)
1054 : : {
1055 : : RangeBound lower1,
1056 : : lower2;
1057 : : RangeBound upper1,
1058 : : upper2;
1059 : : bool empty1,
1060 : : empty2;
1061 : : RangeBound *result_lower;
1062 : : RangeBound *result_upper;
1063 : :
1064 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 1065 [ - + ]: 757 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4535 tgl@sss.pgh.pa.us 1066 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
1067 : :
4534 tgl@sss.pgh.pa.us 1068 :CBC 757 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
1069 : 757 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
1070 : :
1071 : : /* if either is empty, the other is the correct answer */
4546 heikki.linnakangas@i 1072 [ + + ]: 757 : if (empty1)
3267 alvherre@alvh.no-ip. 1073 : 3 : return r2;
4546 heikki.linnakangas@i 1074 [ - + ]: 754 : if (empty2)
3267 alvherre@alvh.no-ip. 1075 :UBC 0 : return r1;
1076 : :
3267 alvherre@alvh.no-ip. 1077 [ + + ]:CBC 754 : if (strict &&
1078 [ + + ]: 69 : !DatumGetBool(range_overlaps_internal(typcache, r1, r2)) &&
1079 [ + + ]: 6 : !DatumGetBool(range_adjacent_internal(typcache, r1, r2)))
4546 heikki.linnakangas@i 1080 [ + - ]: 3 : ereport(ERROR,
1081 : : (errcode(ERRCODE_DATA_EXCEPTION),
1082 : : errmsg("result of range union would not be contiguous")));
1083 : :
4534 tgl@sss.pgh.pa.us 1084 [ + + ]: 751 : if (range_cmp_bounds(typcache, &lower1, &lower2) < 0)
4546 heikki.linnakangas@i 1085 : 733 : result_lower = &lower1;
1086 : : else
1087 : 18 : result_lower = &lower2;
1088 : :
4534 tgl@sss.pgh.pa.us 1089 [ + + ]: 751 : if (range_cmp_bounds(typcache, &upper1, &upper2) > 0)
4546 heikki.linnakangas@i 1090 : 24 : result_upper = &upper1;
1091 : : else
1092 : 727 : result_upper = &upper2;
1093 : :
486 tgl@sss.pgh.pa.us 1094 : 751 : return make_range(typcache, result_lower, result_upper, false, NULL);
1095 : : }
1096 : :
1097 : : Datum
3267 alvherre@alvh.no-ip. 1098 : 9 : range_union(PG_FUNCTION_ARGS)
1099 : : {
2400 tgl@sss.pgh.pa.us 1100 : 9 : RangeType *r1 = PG_GETARG_RANGE_P(0);
1101 : 9 : RangeType *r2 = PG_GETARG_RANGE_P(1);
1102 : : TypeCacheEntry *typcache;
1103 : :
3267 alvherre@alvh.no-ip. 1104 : 9 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
1105 : :
2400 tgl@sss.pgh.pa.us 1106 : 9 : PG_RETURN_RANGE_P(range_union_internal(typcache, r1, r2, true));
1107 : : }
1108 : :
1109 : : /*
1110 : : * range merge: like set union, except also allow and account for non-adjacent
1111 : : * input ranges.
1112 : : */
1113 : : Datum
3267 alvherre@alvh.no-ip. 1114 : 15 : range_merge(PG_FUNCTION_ARGS)
1115 : : {
2400 tgl@sss.pgh.pa.us 1116 : 15 : RangeType *r1 = PG_GETARG_RANGE_P(0);
1117 : 15 : RangeType *r2 = PG_GETARG_RANGE_P(1);
1118 : : TypeCacheEntry *typcache;
1119 : :
3267 alvherre@alvh.no-ip. 1120 : 15 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
1121 : :
2400 tgl@sss.pgh.pa.us 1122 : 15 : PG_RETURN_RANGE_P(range_union_internal(typcache, r1, r2, false));
1123 : : }
1124 : :
1125 : : /* set intersection */
1126 : : Datum
4546 heikki.linnakangas@i 1127 : 11 : range_intersect(PG_FUNCTION_ARGS)
1128 : : {
2400 tgl@sss.pgh.pa.us 1129 : 11 : RangeType *r1 = PG_GETARG_RANGE_P(0);
1130 : 11 : RangeType *r2 = PG_GETARG_RANGE_P(1);
1131 : : TypeCacheEntry *typcache;
1132 : :
1133 : : /* Different types should be prevented by ANYRANGE matching rules */
1211 akorotkov@postgresql 1134 [ - + ]: 11 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
1211 akorotkov@postgresql 1135 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
1136 : :
1211 akorotkov@postgresql 1137 :CBC 11 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
1138 : :
1139 : 11 : PG_RETURN_RANGE_P(range_intersect_internal(typcache, r1, r2));
1140 : : }
1141 : :
1142 : : RangeType *
1143 : 110 : range_intersect_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
1144 : : {
1145 : : RangeBound lower1,
1146 : : lower2;
1147 : : RangeBound upper1,
1148 : : upper2;
1149 : : bool empty1,
1150 : : empty2;
1151 : : RangeBound *result_lower;
1152 : : RangeBound *result_upper;
1153 : :
4534 tgl@sss.pgh.pa.us 1154 : 110 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
1155 : 110 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
1156 : :
1211 akorotkov@postgresql 1157 [ + + + - : 110 : if (empty1 || empty2 || !range_overlaps_internal(typcache, r1, r2))
+ + ]
1158 : 15 : return make_empty_range(typcache);
1159 : :
4534 tgl@sss.pgh.pa.us 1160 [ + + ]: 95 : if (range_cmp_bounds(typcache, &lower1, &lower2) >= 0)
4546 heikki.linnakangas@i 1161 : 46 : result_lower = &lower1;
1162 : : else
1163 : 49 : result_lower = &lower2;
1164 : :
4534 tgl@sss.pgh.pa.us 1165 [ + + ]: 95 : if (range_cmp_bounds(typcache, &upper1, &upper2) <= 0)
4546 heikki.linnakangas@i 1166 : 64 : result_upper = &upper1;
1167 : : else
1168 : 31 : result_upper = &upper2;
1169 : :
486 tgl@sss.pgh.pa.us 1170 : 95 : return make_range(typcache, result_lower, result_upper, false, NULL);
1171 : : }
1172 : :
1173 : : /* range, range -> range, range functions */
1174 : :
1175 : : /*
1176 : : * range_split_internal - if r2 intersects the middle of r1, leaving non-empty
1177 : : * ranges on both sides, then return true and set output1 and output2 to the
1178 : : * results of r1 - r2 (in order). Otherwise return false and don't set output1
1179 : : * or output2. Neither input range should be empty.
1180 : : */
1181 : : bool
1211 akorotkov@postgresql 1182 : 66 : range_split_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2,
1183 : : RangeType **output1, RangeType **output2)
1184 : : {
1185 : : RangeBound lower1,
1186 : : lower2;
1187 : : RangeBound upper1,
1188 : : upper2;
1189 : : bool empty1,
1190 : : empty2;
1191 : :
1192 : 66 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
1193 : 66 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
1194 : :
1195 [ + + + + ]: 105 : if (range_cmp_bounds(typcache, &lower1, &lower2) < 0 &&
1196 : 39 : range_cmp_bounds(typcache, &upper1, &upper2) > 0)
1197 : : {
1198 : : /*
1199 : : * Need to invert inclusive/exclusive for the lower2 and upper2
1200 : : * points. They can't be infinite though. We're allowed to overwrite
1201 : : * these RangeBounds since they only exist locally.
1202 : : */
1203 : 9 : lower2.inclusive = !lower2.inclusive;
1204 : 9 : lower2.lower = false;
1205 : 9 : upper2.inclusive = !upper2.inclusive;
1206 : 9 : upper2.lower = true;
1207 : :
486 tgl@sss.pgh.pa.us 1208 : 9 : *output1 = make_range(typcache, &lower1, &lower2, false, NULL);
1209 : 9 : *output2 = make_range(typcache, &upper2, &upper1, false, NULL);
1211 akorotkov@postgresql 1210 : 9 : return true;
1211 : : }
1212 : :
1213 : 57 : return false;
1214 : : }
1215 : :
1216 : : /* range -> range aggregate functions */
1217 : :
1218 : : Datum
1219 : 21 : range_intersect_agg_transfn(PG_FUNCTION_ARGS)
1220 : : {
1221 : : MemoryContext aggContext;
1222 : : Oid rngtypoid;
1223 : : TypeCacheEntry *typcache;
1224 : : RangeType *result;
1225 : : RangeType *current;
1226 : :
1227 [ - + ]: 21 : if (!AggCheckCallContext(fcinfo, &aggContext))
1211 akorotkov@postgresql 1228 [ # # ]:UBC 0 : elog(ERROR, "range_intersect_agg_transfn called in non-aggregate context");
1229 : :
1211 akorotkov@postgresql 1230 :CBC 21 : rngtypoid = get_fn_expr_argtype(fcinfo->flinfo, 1);
1231 [ - + ]: 21 : if (!type_is_range(rngtypoid))
746 peter@eisentraut.org 1232 [ # # ]:UBC 0 : elog(ERROR, "range_intersect_agg must be called with a range");
1233 : :
1211 akorotkov@postgresql 1234 :CBC 21 : typcache = range_get_typcache(fcinfo, rngtypoid);
1235 : :
1236 : : /* strictness ensures these are non-null */
1237 : 21 : result = PG_GETARG_RANGE_P(0);
1238 : 21 : current = PG_GETARG_RANGE_P(1);
1239 : :
1240 : 21 : result = range_intersect_internal(typcache, result, current);
1241 : 21 : PG_RETURN_RANGE_P(result);
1242 : : }
1243 : :
1244 : :
1245 : : /* Btree support */
1246 : :
1247 : : /* btree comparator */
1248 : : Datum
4546 heikki.linnakangas@i 1249 : 7634 : range_cmp(PG_FUNCTION_ARGS)
1250 : : {
2400 tgl@sss.pgh.pa.us 1251 : 7634 : RangeType *r1 = PG_GETARG_RANGE_P(0);
1252 : 7634 : RangeType *r2 = PG_GETARG_RANGE_P(1);
1253 : : TypeCacheEntry *typcache;
1254 : : RangeBound lower1,
1255 : : lower2;
1256 : : RangeBound upper1,
1257 : : upper2;
1258 : : bool empty1,
1259 : : empty2;
1260 : : int cmp;
1261 : :
3114 noah@leadboat.com 1262 : 7634 : check_stack_depth(); /* recurses when subtype is a range type */
1263 : :
1264 : : /* Different types should be prevented by ANYRANGE matching rules */
4534 tgl@sss.pgh.pa.us 1265 [ - + ]: 7634 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4546 heikki.linnakangas@i 1266 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
1267 : :
4534 tgl@sss.pgh.pa.us 1268 :CBC 7634 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r1));
1269 : :
1270 : 7634 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
1271 : 7634 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
1272 : :
1273 : : /* For b-tree use, empty ranges sort before all else */
4546 heikki.linnakangas@i 1274 [ + + + + ]: 7634 : if (empty1 && empty2)
3854 1275 : 1320 : cmp = 0;
4546 1276 [ + + ]: 6314 : else if (empty1)
3854 1277 : 1740 : cmp = -1;
4546 1278 [ + + ]: 4574 : else if (empty2)
3854 1279 : 1068 : cmp = 1;
1280 : : else
1281 : : {
1282 : 3506 : cmp = range_cmp_bounds(typcache, &lower1, &lower2);
1283 [ + + ]: 3506 : if (cmp == 0)
1284 : 240 : cmp = range_cmp_bounds(typcache, &upper1, &upper2);
1285 : : }
1286 : :
1287 [ + + ]: 7634 : PG_FREE_IF_COPY(r1, 0);
1288 [ + + ]: 7634 : PG_FREE_IF_COPY(r2, 1);
1289 : :
1290 : 7634 : PG_RETURN_INT32(cmp);
1291 : : }
1292 : :
1293 : : /* inequality operators using the range_cmp function */
1294 : : Datum
4546 1295 : 702 : range_lt(PG_FUNCTION_ARGS)
1296 : : {
4535 bruce@momjian.us 1297 : 702 : int cmp = range_cmp(fcinfo);
1298 : :
4546 heikki.linnakangas@i 1299 : 702 : PG_RETURN_BOOL(cmp < 0);
1300 : : }
1301 : :
1302 : : Datum
1303 : 1509 : range_le(PG_FUNCTION_ARGS)
1304 : : {
4535 bruce@momjian.us 1305 : 1509 : int cmp = range_cmp(fcinfo);
1306 : :
4546 heikki.linnakangas@i 1307 : 1509 : PG_RETURN_BOOL(cmp <= 0);
1308 : : }
1309 : :
1310 : : Datum
1311 : 1518 : range_ge(PG_FUNCTION_ARGS)
1312 : : {
4535 bruce@momjian.us 1313 : 1518 : int cmp = range_cmp(fcinfo);
1314 : :
4546 heikki.linnakangas@i 1315 : 1518 : PG_RETURN_BOOL(cmp >= 0);
1316 : : }
1317 : :
1318 : : Datum
1319 : 1536 : range_gt(PG_FUNCTION_ARGS)
1320 : : {
4535 bruce@momjian.us 1321 : 1536 : int cmp = range_cmp(fcinfo);
1322 : :
4546 heikki.linnakangas@i 1323 : 1536 : PG_RETURN_BOOL(cmp > 0);
1324 : : }
1325 : :
1326 : : /* Hash support */
1327 : :
1328 : : /* hash a range value */
1329 : : Datum
1330 : 105 : hash_range(PG_FUNCTION_ARGS)
1331 : : {
2400 tgl@sss.pgh.pa.us 1332 : 105 : RangeType *r = PG_GETARG_RANGE_P(0);
1333 : : uint32 result;
1334 : : TypeCacheEntry *typcache;
1335 : : TypeCacheEntry *scache;
1336 : : RangeBound lower;
1337 : : RangeBound upper;
1338 : : bool empty;
1339 : : char flags;
1340 : : uint32 lower_hash;
1341 : : uint32 upper_hash;
1342 : :
3114 noah@leadboat.com 1343 : 105 : check_stack_depth(); /* recurses when subtype is a range type */
1344 : :
4534 tgl@sss.pgh.pa.us 1345 : 105 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
1346 : :
1347 : : /* deserialize */
1348 : 105 : range_deserialize(typcache, r, &lower, &upper, &empty);
1349 : 105 : flags = range_get_flags(r);
1350 : :
1351 : : /*
1352 : : * Look up the element type's hash function, if not done already.
1353 : : */
1354 : 105 : scache = typcache->rngelemtype;
1355 [ + + ]: 105 : if (!OidIsValid(scache->hash_proc_finfo.fn_oid))
1356 : : {
1357 : 3 : scache = lookup_type_cache(scache->type_id, TYPECACHE_HASH_PROC_FINFO);
1358 [ - + ]: 3 : if (!OidIsValid(scache->hash_proc_finfo.fn_oid))
4546 heikki.linnakangas@i 1359 [ # # ]:UBC 0 : ereport(ERROR,
1360 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
1361 : : errmsg("could not identify a hash function for type %s",
1362 : : format_type_be(scache->type_id))));
1363 : : }
1364 : :
1365 : : /*
1366 : : * Apply the hash function to each bound.
1367 : : */
4546 heikki.linnakangas@i 1368 [ + + ]:CBC 105 : if (RANGE_HAS_LBOUND(flags))
4534 tgl@sss.pgh.pa.us 1369 : 72 : lower_hash = DatumGetUInt32(FunctionCall1Coll(&scache->hash_proc_finfo,
1370 : : typcache->rng_collation,
1371 : : lower.val));
1372 : : else
1373 : 33 : lower_hash = 0;
1374 : :
4546 heikki.linnakangas@i 1375 [ + + ]: 105 : if (RANGE_HAS_UBOUND(flags))
4534 tgl@sss.pgh.pa.us 1376 : 78 : upper_hash = DatumGetUInt32(FunctionCall1Coll(&scache->hash_proc_finfo,
1377 : : typcache->rng_collation,
1378 : : upper.val));
1379 : : else
1380 : 27 : upper_hash = 0;
1381 : :
1382 : : /* Merge hashes of flags and bounds */
4535 bruce@momjian.us 1383 : 105 : result = hash_uint32((uint32) flags);
4546 heikki.linnakangas@i 1384 : 105 : result ^= lower_hash;
784 john.naylor@postgres 1385 : 105 : result = pg_rotate_left32(result, 1);
4546 heikki.linnakangas@i 1386 : 105 : result ^= upper_hash;
1387 : :
1388 : 105 : PG_RETURN_INT32(result);
1389 : : }
1390 : :
1391 : : /*
1392 : : * Returns 64-bit value by hashing a value to a 64-bit value, with a seed.
1393 : : * Otherwise, similar to hash_range.
1394 : : */
1395 : : Datum
2418 rhaas@postgresql.org 1396 : 30 : hash_range_extended(PG_FUNCTION_ARGS)
1397 : : {
2400 tgl@sss.pgh.pa.us 1398 : 30 : RangeType *r = PG_GETARG_RANGE_P(0);
2417 rhaas@postgresql.org 1399 : 30 : Datum seed = PG_GETARG_DATUM(1);
1400 : : uint64 result;
1401 : : TypeCacheEntry *typcache;
1402 : : TypeCacheEntry *scache;
1403 : : RangeBound lower;
1404 : : RangeBound upper;
1405 : : bool empty;
1406 : : char flags;
1407 : : uint64 lower_hash;
1408 : : uint64 upper_hash;
1409 : :
2418 1410 : 30 : check_stack_depth();
1411 : :
1412 : 30 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
1413 : :
1414 : 30 : range_deserialize(typcache, r, &lower, &upper, &empty);
1415 : 30 : flags = range_get_flags(r);
1416 : :
1417 : 30 : scache = typcache->rngelemtype;
1418 [ - + ]: 30 : if (!OidIsValid(scache->hash_extended_proc_finfo.fn_oid))
1419 : : {
2418 rhaas@postgresql.org 1420 :UBC 0 : scache = lookup_type_cache(scache->type_id,
1421 : : TYPECACHE_HASH_EXTENDED_PROC_FINFO);
1422 [ # # ]: 0 : if (!OidIsValid(scache->hash_extended_proc_finfo.fn_oid))
1423 [ # # ]: 0 : ereport(ERROR,
1424 : : (errcode(ERRCODE_UNDEFINED_FUNCTION),
1425 : : errmsg("could not identify a hash function for type %s",
1426 : : format_type_be(scache->type_id))));
1427 : : }
1428 : :
2418 rhaas@postgresql.org 1429 [ + - ]:CBC 30 : if (RANGE_HAS_LBOUND(flags))
1430 : 30 : lower_hash = DatumGetUInt64(FunctionCall2Coll(&scache->hash_extended_proc_finfo,
1431 : : typcache->rng_collation,
1432 : : lower.val,
1433 : : seed));
1434 : : else
2418 rhaas@postgresql.org 1435 :UBC 0 : lower_hash = 0;
1436 : :
2418 rhaas@postgresql.org 1437 [ + - ]:CBC 30 : if (RANGE_HAS_UBOUND(flags))
1438 : 30 : upper_hash = DatumGetUInt64(FunctionCall2Coll(&scache->hash_extended_proc_finfo,
1439 : : typcache->rng_collation,
1440 : : upper.val,
1441 : : seed));
1442 : : else
2418 rhaas@postgresql.org 1443 :UBC 0 : upper_hash = 0;
1444 : :
1445 : : /* Merge hashes of flags and bounds */
2417 rhaas@postgresql.org 1446 :CBC 30 : result = DatumGetUInt64(hash_uint32_extended((uint32) flags,
1447 : 30 : DatumGetInt64(seed)));
2418 1448 : 30 : result ^= lower_hash;
1449 : 30 : result = ROTATE_HIGH_AND_LOW_32BITS(result);
1450 : 30 : result ^= upper_hash;
1451 : :
1452 : 30 : PG_RETURN_UINT64(result);
1453 : : }
1454 : :
1455 : : /*
1456 : : *----------------------------------------------------------
1457 : : * CANONICAL FUNCTIONS
1458 : : *
1459 : : * Functions for specific built-in range types.
1460 : : *----------------------------------------------------------
1461 : : */
1462 : :
1463 : : Datum
4546 heikki.linnakangas@i 1464 : 233397 : int4range_canonical(PG_FUNCTION_ARGS)
1465 : : {
2400 tgl@sss.pgh.pa.us 1466 : 233397 : RangeType *r = PG_GETARG_RANGE_P(0);
486 1467 : 233397 : Node *escontext = fcinfo->context;
1468 : : TypeCacheEntry *typcache;
1469 : : RangeBound lower;
1470 : : RangeBound upper;
1471 : : bool empty;
1472 : :
4534 1473 : 233397 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
1474 : :
1475 : 233397 : range_deserialize(typcache, r, &lower, &upper, &empty);
1476 : :
4546 heikki.linnakangas@i 1477 [ - + ]: 233397 : if (empty)
2400 tgl@sss.pgh.pa.us 1478 :UBC 0 : PG_RETURN_RANGE_P(r);
1479 : :
4546 heikki.linnakangas@i 1480 [ + + + + ]:CBC 233397 : if (!lower.infinite && !lower.inclusive)
1481 : : {
486 tgl@sss.pgh.pa.us 1482 : 1624 : int32 bnd = DatumGetInt32(lower.val);
1483 : :
1484 : : /* Handle possible overflow manually */
1485 [ - + ]: 1624 : if (unlikely(bnd == PG_INT32_MAX))
486 tgl@sss.pgh.pa.us 1486 [ # # ]:UBC 0 : ereturn(escontext, (Datum) 0,
1487 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1488 : : errmsg("integer out of range")));
486 tgl@sss.pgh.pa.us 1489 :CBC 1624 : lower.val = Int32GetDatum(bnd + 1);
4546 heikki.linnakangas@i 1490 : 1624 : lower.inclusive = true;
1491 : : }
1492 : :
1493 [ + + + + ]: 233397 : if (!upper.infinite && upper.inclusive)
1494 : : {
486 tgl@sss.pgh.pa.us 1495 : 1621 : int32 bnd = DatumGetInt32(upper.val);
1496 : :
1497 : : /* Handle possible overflow manually */
1498 [ + + ]: 1621 : if (unlikely(bnd == PG_INT32_MAX))
1499 [ + + ]: 6 : ereturn(escontext, (Datum) 0,
1500 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1501 : : errmsg("integer out of range")));
1502 : 1615 : upper.val = Int32GetDatum(bnd + 1);
4546 heikki.linnakangas@i 1503 : 1615 : upper.inclusive = false;
1504 : : }
1505 : :
486 tgl@sss.pgh.pa.us 1506 : 233391 : PG_RETURN_RANGE_P(range_serialize(typcache, &lower, &upper,
1507 : : false, escontext));
1508 : : }
1509 : :
1510 : : Datum
4546 heikki.linnakangas@i 1511 : 49 : int8range_canonical(PG_FUNCTION_ARGS)
1512 : : {
2400 tgl@sss.pgh.pa.us 1513 : 49 : RangeType *r = PG_GETARG_RANGE_P(0);
486 1514 : 49 : Node *escontext = fcinfo->context;
1515 : : TypeCacheEntry *typcache;
1516 : : RangeBound lower;
1517 : : RangeBound upper;
1518 : : bool empty;
1519 : :
4534 1520 : 49 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
1521 : :
1522 : 49 : range_deserialize(typcache, r, &lower, &upper, &empty);
1523 : :
4546 heikki.linnakangas@i 1524 [ - + ]: 49 : if (empty)
2400 tgl@sss.pgh.pa.us 1525 :UBC 0 : PG_RETURN_RANGE_P(r);
1526 : :
4546 heikki.linnakangas@i 1527 [ + - + + ]:CBC 49 : if (!lower.infinite && !lower.inclusive)
1528 : : {
486 tgl@sss.pgh.pa.us 1529 : 9 : int64 bnd = DatumGetInt64(lower.val);
1530 : :
1531 : : /* Handle possible overflow manually */
1532 [ - + ]: 9 : if (unlikely(bnd == PG_INT64_MAX))
486 tgl@sss.pgh.pa.us 1533 [ # # ]:UBC 0 : ereturn(escontext, (Datum) 0,
1534 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1535 : : errmsg("bigint out of range")));
486 tgl@sss.pgh.pa.us 1536 :CBC 9 : lower.val = Int64GetDatum(bnd + 1);
4546 heikki.linnakangas@i 1537 : 9 : lower.inclusive = true;
1538 : : }
1539 : :
1540 [ + - + + ]: 49 : if (!upper.infinite && upper.inclusive)
1541 : : {
486 tgl@sss.pgh.pa.us 1542 : 12 : int64 bnd = DatumGetInt64(upper.val);
1543 : :
1544 : : /* Handle possible overflow manually */
1545 [ - + ]: 12 : if (unlikely(bnd == PG_INT64_MAX))
486 tgl@sss.pgh.pa.us 1546 [ # # ]:UBC 0 : ereturn(escontext, (Datum) 0,
1547 : : (errcode(ERRCODE_NUMERIC_VALUE_OUT_OF_RANGE),
1548 : : errmsg("bigint out of range")));
486 tgl@sss.pgh.pa.us 1549 :CBC 12 : upper.val = Int64GetDatum(bnd + 1);
4546 heikki.linnakangas@i 1550 : 12 : upper.inclusive = false;
1551 : : }
1552 : :
486 tgl@sss.pgh.pa.us 1553 : 49 : PG_RETURN_RANGE_P(range_serialize(typcache, &lower, &upper,
1554 : : false, escontext));
1555 : : }
1556 : :
1557 : : Datum
4546 heikki.linnakangas@i 1558 : 800 : daterange_canonical(PG_FUNCTION_ARGS)
1559 : : {
2400 tgl@sss.pgh.pa.us 1560 : 800 : RangeType *r = PG_GETARG_RANGE_P(0);
486 1561 : 800 : Node *escontext = fcinfo->context;
1562 : : TypeCacheEntry *typcache;
1563 : : RangeBound lower;
1564 : : RangeBound upper;
1565 : : bool empty;
1566 : :
4534 1567 : 800 : typcache = range_get_typcache(fcinfo, RangeTypeGetOid(r));
1568 : :
1569 : 800 : range_deserialize(typcache, r, &lower, &upper, &empty);
1570 : :
4546 heikki.linnakangas@i 1571 [ - + ]: 800 : if (empty)
2400 tgl@sss.pgh.pa.us 1572 :UBC 0 : PG_RETURN_RANGE_P(r);
1573 : :
1732 jdavis@postgresql.or 1574 [ + + + + :CBC 800 : if (!lower.infinite && !DATE_NOT_FINITE(DatumGetDateADT(lower.val)) &&
+ - ]
1575 [ + + ]: 776 : !lower.inclusive)
1576 : : {
486 tgl@sss.pgh.pa.us 1577 : 18 : DateADT bnd = DatumGetDateADT(lower.val);
1578 : :
1579 : : /* Check for overflow -- note we already eliminated PG_INT32_MAX */
1580 : 18 : bnd++;
1581 [ + - - + : 18 : if (unlikely(!IS_VALID_DATE(bnd)))
- + ]
486 tgl@sss.pgh.pa.us 1582 [ # # ]:UBC 0 : ereturn(escontext, (Datum) 0,
1583 : : (errcode(ERRCODE_DATETIME_VALUE_OUT_OF_RANGE),
1584 : : errmsg("date out of range")));
486 tgl@sss.pgh.pa.us 1585 :CBC 18 : lower.val = DateADTGetDatum(bnd);
4546 heikki.linnakangas@i 1586 : 18 : lower.inclusive = true;
1587 : : }
1588 : :
1732 jdavis@postgresql.or 1589 [ + + + - : 800 : if (!upper.infinite && !DATE_NOT_FINITE(DatumGetDateADT(upper.val)) &&
+ + ]
1590 [ + + ]: 764 : upper.inclusive)
1591 : : {
486 tgl@sss.pgh.pa.us 1592 : 18 : DateADT bnd = DatumGetDateADT(upper.val);
1593 : :
1594 : : /* Check for overflow -- note we already eliminated PG_INT32_MAX */
1595 : 18 : bnd++;
1596 [ + - + + : 18 : if (unlikely(!IS_VALID_DATE(bnd)))
+ + ]
1597 [ + + ]: 6 : ereturn(escontext, (Datum) 0,
1598 : : (errcode(ERRCODE_DATETIME_VALUE_OUT_OF_RANGE),
1599 : : errmsg("date out of range")));
1600 : 12 : upper.val = DateADTGetDatum(bnd);
4546 heikki.linnakangas@i 1601 : 12 : upper.inclusive = false;
1602 : : }
1603 : :
486 tgl@sss.pgh.pa.us 1604 : 794 : PG_RETURN_RANGE_P(range_serialize(typcache, &lower, &upper,
1605 : : false, escontext));
1606 : : }
1607 : :
1608 : : /*
1609 : : *----------------------------------------------------------
1610 : : * SUBTYPE_DIFF FUNCTIONS
1611 : : *
1612 : : * Functions for specific built-in range types.
1613 : : *
1614 : : * Note that subtype_diff does return the difference, not the absolute value
1615 : : * of the difference, and it must take care to avoid overflow.
1616 : : * (numrange_subdiff is at some risk there ...)
1617 : : *----------------------------------------------------------
1618 : : */
1619 : :
1620 : : Datum
4546 heikki.linnakangas@i 1621 : 620193 : int4range_subdiff(PG_FUNCTION_ARGS)
1622 : : {
4535 bruce@momjian.us 1623 : 620193 : int32 v1 = PG_GETARG_INT32(0);
1624 : 620193 : int32 v2 = PG_GETARG_INT32(1);
1625 : :
tgl@sss.pgh.pa.us 1626 : 620193 : PG_RETURN_FLOAT8((float8) v1 - (float8) v2);
1627 : : }
1628 : :
1629 : : Datum
4546 heikki.linnakangas@i 1630 :UBC 0 : int8range_subdiff(PG_FUNCTION_ARGS)
1631 : : {
4535 bruce@momjian.us 1632 : 0 : int64 v1 = PG_GETARG_INT64(0);
1633 : 0 : int64 v2 = PG_GETARG_INT64(1);
1634 : :
tgl@sss.pgh.pa.us 1635 : 0 : PG_RETURN_FLOAT8((float8) v1 - (float8) v2);
1636 : : }
1637 : :
1638 : : Datum
4546 heikki.linnakangas@i 1639 :CBC 123 : numrange_subdiff(PG_FUNCTION_ARGS)
1640 : : {
1641 : 123 : Datum v1 = PG_GETARG_DATUM(0);
1642 : 123 : Datum v2 = PG_GETARG_DATUM(1);
1643 : : Datum numresult;
1644 : : float8 floatresult;
1645 : :
1646 : 123 : numresult = DirectFunctionCall2(numeric_sub, v1, v2);
1647 : :
4535 tgl@sss.pgh.pa.us 1648 : 123 : floatresult = DatumGetFloat8(DirectFunctionCall1(numeric_float8,
1649 : : numresult));
1650 : :
4546 heikki.linnakangas@i 1651 : 123 : PG_RETURN_FLOAT8(floatresult);
1652 : : }
1653 : :
1654 : : Datum
4535 tgl@sss.pgh.pa.us 1655 :UBC 0 : daterange_subdiff(PG_FUNCTION_ARGS)
1656 : : {
1657 : 0 : int32 v1 = PG_GETARG_INT32(0);
1658 : 0 : int32 v2 = PG_GETARG_INT32(1);
1659 : :
1660 : 0 : PG_RETURN_FLOAT8((float8) v1 - (float8) v2);
1661 : : }
1662 : :
1663 : : Datum
4546 heikki.linnakangas@i 1664 : 0 : tsrange_subdiff(PG_FUNCTION_ARGS)
1665 : : {
1666 : 0 : Timestamp v1 = PG_GETARG_TIMESTAMP(0);
1667 : 0 : Timestamp v2 = PG_GETARG_TIMESTAMP(1);
1668 : : float8 result;
1669 : :
4535 tgl@sss.pgh.pa.us 1670 : 0 : result = ((float8) v1 - (float8) v2) / USECS_PER_SEC;
4546 heikki.linnakangas@i 1671 : 0 : PG_RETURN_FLOAT8(result);
1672 : : }
1673 : :
1674 : : Datum
1675 : 0 : tstzrange_subdiff(PG_FUNCTION_ARGS)
1676 : : {
1677 : 0 : Timestamp v1 = PG_GETARG_TIMESTAMP(0);
1678 : 0 : Timestamp v2 = PG_GETARG_TIMESTAMP(1);
1679 : : float8 result;
1680 : :
4535 tgl@sss.pgh.pa.us 1681 : 0 : result = ((float8) v1 - (float8) v2) / USECS_PER_SEC;
4546 heikki.linnakangas@i 1682 : 0 : PG_RETURN_FLOAT8(result);
1683 : : }
1684 : :
1685 : : /*
1686 : : *----------------------------------------------------------
1687 : : * SUPPORT FUNCTIONS
1688 : : *
1689 : : * These functions aren't in pg_proc, but are useful for
1690 : : * defining new generic range functions in C.
1691 : : *----------------------------------------------------------
1692 : : */
1693 : :
1694 : : /*
1695 : : * range_get_typcache: get cached information about a range type
1696 : : *
1697 : : * This is for use by range-related functions that follow the convention
1698 : : * of using the fn_extra field as a pointer to the type cache entry for
1699 : : * the range type. Functions that need to cache more information than
1700 : : * that must fend for themselves.
1701 : : */
1702 : : TypeCacheEntry *
4534 tgl@sss.pgh.pa.us 1703 :CBC 2362491 : range_get_typcache(FunctionCallInfo fcinfo, Oid rngtypid)
1704 : : {
1705 : 2362491 : TypeCacheEntry *typcache = (TypeCacheEntry *) fcinfo->flinfo->fn_extra;
1706 : :
1707 [ + + ]: 2362491 : if (typcache == NULL ||
1708 [ - + ]: 2355334 : typcache->type_id != rngtypid)
1709 : : {
1710 : 7157 : typcache = lookup_type_cache(rngtypid, TYPECACHE_RANGE_INFO);
1711 [ - + ]: 7157 : if (typcache->rngelemtype == NULL)
4534 tgl@sss.pgh.pa.us 1712 [ # # ]:UBC 0 : elog(ERROR, "type %u is not a range type", rngtypid);
4534 tgl@sss.pgh.pa.us 1713 :CBC 7157 : fcinfo->flinfo->fn_extra = (void *) typcache;
1714 : : }
1715 : :
1716 : 2362491 : return typcache;
1717 : : }
1718 : :
1719 : : /*
1720 : : * range_serialize: construct a range value from bounds and empty-flag
1721 : : *
1722 : : * This does not force canonicalization of the range value. In most cases,
1723 : : * external callers should only be canonicalization functions. Note that
1724 : : * we perform some datatype-independent canonicalization checks anyway.
1725 : : */
1726 : : RangeType *
1727 : 474641 : range_serialize(TypeCacheEntry *typcache, RangeBound *lower, RangeBound *upper,
1728 : : bool empty, struct Node *escontext)
1729 : : {
1730 : : RangeType *range;
1731 : : int cmp;
1732 : : Size msize;
1733 : : Pointer ptr;
1734 : : int16 typlen;
1735 : : bool typbyval;
1736 : : char typalign;
1737 : : char typstorage;
4535 bruce@momjian.us 1738 : 474641 : char flags = 0;
1739 : :
1740 : : /*
1741 : : * Verify range is not invalid on its face, and construct flags value,
1742 : : * preventing any non-canonical combinations such as infinite+inclusive.
1743 : : */
4527 tgl@sss.pgh.pa.us 1744 [ - + ]: 474641 : Assert(lower->lower);
1745 [ - + ]: 474641 : Assert(!upper->lower);
1746 : :
4546 heikki.linnakangas@i 1747 [ + + ]: 474641 : if (empty)
1748 : 1833 : flags |= RANGE_EMPTY;
1749 : : else
1750 : : {
4527 tgl@sss.pgh.pa.us 1751 : 472808 : cmp = range_cmp_bound_values(typcache, lower, upper);
1752 : :
1753 : : /* error check: if lower bound value is above upper, it's wrong */
1754 [ + + ]: 472808 : if (cmp > 0)
486 1755 [ + + ]: 33 : ereturn(escontext, NULL,
1756 : : (errcode(ERRCODE_DATA_EXCEPTION),
1757 : : errmsg("range lower bound must be less than or equal to range upper bound")));
1758 : :
1759 : : /* if bounds are equal, and not both inclusive, range is empty */
4527 1760 [ + + + + : 472775 : if (cmp == 0 && !(lower->inclusive && upper->inclusive))
+ + ]
1761 : 192 : flags |= RANGE_EMPTY;
1762 : : else
1763 : : {
1764 : : /* infinite boundaries are never inclusive */
1765 [ + + ]: 472583 : if (lower->infinite)
1766 : 5107 : flags |= RANGE_LB_INF;
1767 [ + + ]: 467476 : else if (lower->inclusive)
1768 : 465660 : flags |= RANGE_LB_INC;
1769 [ + + ]: 472583 : if (upper->infinite)
1770 : 3416 : flags |= RANGE_UB_INF;
1771 [ + + ]: 469167 : else if (upper->inclusive)
1772 : 1996 : flags |= RANGE_UB_INC;
1773 : : }
1774 : : }
1775 : :
1776 : : /* Fetch information about range's element type */
1777 : 474608 : typlen = typcache->rngelemtype->typlen;
1778 : 474608 : typbyval = typcache->rngelemtype->typbyval;
1779 : 474608 : typalign = typcache->rngelemtype->typalign;
1780 : 474608 : typstorage = typcache->rngelemtype->typstorage;
1781 : :
1782 : : /* Count space for varlena header and range type's OID */
4534 1783 : 474608 : msize = sizeof(RangeType);
1784 [ - + ]: 474608 : Assert(msize == MAXALIGN(msize));
1785 : :
1786 : : /* Count space for bounds */
4546 heikki.linnakangas@i 1787 [ + + ]: 474608 : if (RANGE_HAS_LBOUND(flags))
1788 : : {
1789 : : /*
1790 : : * Make sure item to be inserted is not toasted. It is essential that
1791 : : * we not insert an out-of-line toast value pointer into a range
1792 : : * object, for the same reasons that arrays and records can't contain
1793 : : * them. It would work to store a compressed-in-line value, but we
1794 : : * prefer to decompress and then let compression be applied to the
1795 : : * whole range object if necessary. But, unlike arrays, we do allow
1796 : : * short-header varlena objects to stay as-is.
1797 : : */
4535 tgl@sss.pgh.pa.us 1798 [ + + ]: 467476 : if (typlen == -1)
1799 : 2373 : lower->val = PointerGetDatum(PG_DETOAST_DATUM_PACKED(lower->val));
1800 : :
4546 heikki.linnakangas@i 1801 : 467476 : msize = datum_compute_size(msize, lower->val, typbyval, typalign,
1802 : : typlen, typstorage);
1803 : : }
1804 : :
1805 [ + + ]: 474608 : if (RANGE_HAS_UBOUND(flags))
1806 : : {
1807 : : /* Make sure item to be inserted is not toasted */
4535 tgl@sss.pgh.pa.us 1808 [ + + ]: 469167 : if (typlen == -1)
1809 : 2355 : upper->val = PointerGetDatum(PG_DETOAST_DATUM_PACKED(upper->val));
1810 : :
4546 heikki.linnakangas@i 1811 : 469167 : msize = datum_compute_size(msize, upper->val, typbyval, typalign,
1812 : : typlen, typstorage);
1813 : : }
1814 : :
1815 : : /* Add space for flag byte */
1816 : 474608 : msize += sizeof(char);
1817 : :
1818 : : /* Note: zero-fill is required here, just as in heap tuples */
4534 tgl@sss.pgh.pa.us 1819 : 474608 : range = (RangeType *) palloc0(msize);
1820 : 474608 : SET_VARSIZE(range, msize);
1821 : :
1822 : : /* Now fill in the datum */
1823 : 474608 : range->rangetypid = typcache->type_id;
1824 : :
1825 : 474608 : ptr = (char *) (range + 1);
1826 : :
4546 heikki.linnakangas@i 1827 [ + + ]: 474608 : if (RANGE_HAS_LBOUND(flags))
1828 : : {
1829 [ - + ]: 467476 : Assert(lower->lower);
1830 : 467476 : ptr = datum_write(ptr, lower->val, typbyval, typalign, typlen,
1831 : : typstorage);
1832 : : }
1833 : :
1834 [ + + ]: 474608 : if (RANGE_HAS_UBOUND(flags))
1835 : : {
1836 [ - + ]: 469167 : Assert(!upper->lower);
1837 : 469167 : ptr = datum_write(ptr, upper->val, typbyval, typalign, typlen,
1838 : : typstorage);
1839 : : }
1840 : :
4534 tgl@sss.pgh.pa.us 1841 : 474608 : *((char *) ptr) = flags;
1842 : :
1843 : 474608 : return range;
1844 : : }
1845 : :
1846 : : /*
1847 : : * range_deserialize: deconstruct a range value
1848 : : *
1849 : : * NB: the given range object must be fully detoasted; it cannot have a
1850 : : * short varlena header.
1851 : : *
1852 : : * Note that if the element type is pass-by-reference, the datums in the
1853 : : * RangeBound structs will be pointers into the given range object.
1854 : : */
1855 : : void
1627 peter@eisentraut.org 1856 : 4836887 : range_deserialize(TypeCacheEntry *typcache, const RangeType *range,
1857 : : RangeBound *lower, RangeBound *upper, bool *empty)
1858 : : {
1859 : : char flags;
1860 : : int16 typlen;
1861 : : bool typbyval;
1862 : : char typalign;
1863 : : Pointer ptr;
1864 : : Datum lbound;
1865 : : Datum ubound;
1866 : :
1867 : : /* assert caller passed the right typcache entry */
4534 tgl@sss.pgh.pa.us 1868 [ - + ]: 4836887 : Assert(RangeTypeGetOid(range) == typcache->type_id);
1869 : :
1870 : : /* fetch the flag byte from datum's last byte */
1627 peter@eisentraut.org 1871 : 4836887 : flags = *((const char *) range + VARSIZE(range) - 1);
1872 : :
1873 : : /* fetch information about range's element type */
4534 tgl@sss.pgh.pa.us 1874 : 4836887 : typlen = typcache->rngelemtype->typlen;
1875 : 4836887 : typbyval = typcache->rngelemtype->typbyval;
1876 : 4836887 : typalign = typcache->rngelemtype->typalign;
1877 : :
1878 : : /* initialize data pointer just after the range OID */
1879 : 4836887 : ptr = (Pointer) (range + 1);
1880 : :
1881 : : /* fetch lower bound, if any */
4546 heikki.linnakangas@i 1882 [ + + ]: 4836887 : if (RANGE_HAS_LBOUND(flags))
1883 : : {
1884 : : /* att_align_pointer cannot be necessary here */
1885 : 4278893 : lbound = fetch_att(ptr, typbyval, typlen);
4535 tgl@sss.pgh.pa.us 1886 [ + + + - : 4278893 : ptr = (Pointer) att_addlength_pointer(ptr, typlen, ptr);
- + - - -
- - - + -
- - ]
1887 : : }
1888 : : else
4546 heikki.linnakangas@i 1889 : 557994 : lbound = (Datum) 0;
1890 : :
1891 : : /* fetch upper bound, if any */
1892 [ + + ]: 4836887 : if (RANGE_HAS_UBOUND(flags))
1893 : : {
1894 [ + + + + : 4285636 : ptr = (Pointer) att_align_pointer(ptr, typalign, typlen, ptr);
+ + + - +
- - - ]
1895 : 4285636 : ubound = fetch_att(ptr, typbyval, typlen);
1896 : : /* no need for att_addlength_pointer */
1897 : : }
1898 : : else
1899 : 551251 : ubound = (Datum) 0;
1900 : :
1901 : : /* emit results */
1902 : :
4534 tgl@sss.pgh.pa.us 1903 : 4836887 : *empty = (flags & RANGE_EMPTY) != 0;
1904 : :
4535 bruce@momjian.us 1905 : 4836887 : lower->val = lbound;
tgl@sss.pgh.pa.us 1906 : 4836887 : lower->infinite = (flags & RANGE_LB_INF) != 0;
4534 1907 : 4836887 : lower->inclusive = (flags & RANGE_LB_INC) != 0;
4535 bruce@momjian.us 1908 : 4836887 : lower->lower = true;
1909 : :
1910 : 4836887 : upper->val = ubound;
tgl@sss.pgh.pa.us 1911 : 4836887 : upper->infinite = (flags & RANGE_UB_INF) != 0;
4534 1912 : 4836887 : upper->inclusive = (flags & RANGE_UB_INC) != 0;
4535 bruce@momjian.us 1913 : 4836887 : upper->lower = false;
4546 heikki.linnakangas@i 1914 : 4836887 : }
1915 : :
1916 : : /*
1917 : : * range_get_flags: just get the flags from a RangeType value.
1918 : : *
1919 : : * This is frequently useful in places that only need the flags and not
1920 : : * the full results of range_deserialize.
1921 : : */
1922 : : char
1627 peter@eisentraut.org 1923 : 1481193 : range_get_flags(const RangeType *range)
1924 : : {
1925 : : /* fetch the flag byte from datum's last byte */
4534 tgl@sss.pgh.pa.us 1926 : 1481193 : return *((char *) range + VARSIZE(range) - 1);
1927 : : }
1928 : :
1929 : : /*
1930 : : * range_set_contain_empty: set the RANGE_CONTAIN_EMPTY bit in the value.
1931 : : *
1932 : : * This is only needed in GiST operations, so we don't include a provision
1933 : : * for setting it in range_serialize; rather, this function must be applied
1934 : : * afterwards.
1935 : : */
1936 : : void
4522 1937 : 9 : range_set_contain_empty(RangeType *range)
1938 : : {
1939 : : char *flagsp;
1940 : :
1941 : : /* flag byte is datum's last byte */
1942 : 9 : flagsp = (char *) range + VARSIZE(range) - 1;
1943 : :
1944 : 9 : *flagsp |= RANGE_CONTAIN_EMPTY;
1945 : 9 : }
1946 : :
1947 : : /*
1948 : : * This both serializes and canonicalizes (if applicable) the range.
1949 : : * This should be used by most callers.
1950 : : */
1951 : : RangeType *
4534 1952 : 238900 : make_range(TypeCacheEntry *typcache, RangeBound *lower, RangeBound *upper,
1953 : : bool empty, struct Node *escontext)
1954 : : {
1955 : : RangeType *range;
1956 : :
486 1957 : 238900 : range = range_serialize(typcache, lower, upper, empty, escontext);
1958 : :
1959 [ + + + - : 238873 : if (SOFT_ERROR_OCCURRED(escontext))
+ + ]
1960 : 6 : return NULL;
1961 : :
1962 : : /* no need to call canonical on empty ranges ... */
4527 1963 [ + + ]: 238867 : if (OidIsValid(typcache->rng_canonical_finfo.fn_oid) &&
1964 [ + + ]: 236064 : !RangeIsEmpty(range))
1965 : : {
1966 : : /* Do this the hard way so that we can pass escontext */
486 1967 : 234246 : LOCAL_FCINFO(fcinfo, 1);
1968 : : Datum result;
1969 : :
1970 : 234246 : InitFunctionCallInfoData(*fcinfo, &typcache->rng_canonical_finfo, 1,
1971 : : InvalidOid, escontext, NULL);
1972 : :
1973 : 234246 : fcinfo->args[0].value = RangeTypePGetDatum(range);
1974 : 234246 : fcinfo->args[0].isnull = false;
1975 : :
1976 : 234246 : result = FunctionCallInvoke(fcinfo);
1977 : :
1978 [ + + + - : 234246 : if (SOFT_ERROR_OCCURRED(escontext))
+ + ]
1979 : 12 : return NULL;
1980 : :
1981 : : /* Should not get a null result if there was no error */
1982 [ - + ]: 234234 : if (fcinfo->isnull)
486 tgl@sss.pgh.pa.us 1983 [ # # ]:UBC 0 : elog(ERROR, "function %u returned NULL",
1984 : : typcache->rng_canonical_finfo.fn_oid);
1985 : :
486 tgl@sss.pgh.pa.us 1986 :CBC 234234 : range = DatumGetRangeTypeP(result);
1987 : : }
1988 : :
4534 1989 : 238855 : return range;
1990 : : }
1991 : :
1992 : : /*
1993 : : * Compare two range boundary points, returning <0, 0, or >0 according to
1994 : : * whether b1 is less than, equal to, or greater than b2.
1995 : : *
1996 : : * The boundaries can be any combination of upper and lower; so it's useful
1997 : : * for a variety of operators.
1998 : : *
1999 : : * The simple case is when b1 and b2 are both finite and inclusive, in which
2000 : : * case the result is just a comparison of the values held in b1 and b2.
2001 : : *
2002 : : * If a bound is exclusive, then we need to know whether it's a lower bound,
2003 : : * in which case we treat the boundary point as "just greater than" the held
2004 : : * value; or an upper bound, in which case we treat the boundary point as
2005 : : * "just less than" the held value.
2006 : : *
2007 : : * If a bound is infinite, it represents minus infinity (less than every other
2008 : : * point) if it's a lower bound; or plus infinity (greater than every other
2009 : : * point) if it's an upper bound.
2010 : : *
2011 : : * There is only one case where two boundaries compare equal but are not
2012 : : * identical: when both bounds are inclusive and hold the same finite value,
2013 : : * but one is an upper bound and the other a lower bound.
2014 : : */
2015 : : int
1627 peter@eisentraut.org 2016 : 6334051 : range_cmp_bounds(TypeCacheEntry *typcache, const RangeBound *b1, const RangeBound *b2)
2017 : : {
2018 : : int32 result;
2019 : :
2020 : : /*
2021 : : * First, handle cases involving infinity, which don't require invoking
2022 : : * the comparison proc.
2023 : : */
4546 heikki.linnakangas@i 2024 [ + + + + ]: 6334051 : if (b1->infinite && b2->infinite)
2025 : : {
2026 : : /*
2027 : : * Both are infinity, so they are equal unless one is lower and the
2028 : : * other not.
2029 : : */
2030 [ + + ]: 8982 : if (b1->lower == b2->lower)
2031 : 8937 : return 0;
2032 : : else
4532 tgl@sss.pgh.pa.us 2033 [ + + ]: 45 : return b1->lower ? -1 : 1;
2034 : : }
2035 [ + + ]: 6325069 : else if (b1->infinite)
2036 [ + + ]: 49809 : return b1->lower ? -1 : 1;
2037 [ + + ]: 6275260 : else if (b2->infinite)
2038 [ + + ]: 14170 : return b2->lower ? 1 : -1;
2039 : :
2040 : : /*
2041 : : * Both boundaries are finite, so compare the held values.
2042 : : */
4534 2043 : 6261090 : result = DatumGetInt32(FunctionCall2Coll(&typcache->rng_cmp_proc_finfo,
2044 : : typcache->rng_collation,
4546 heikki.linnakangas@i 2045 : 6261090 : b1->val, b2->val));
2046 : :
2047 : : /*
2048 : : * If the comparison is anything other than equal, we're done. If they
2049 : : * compare equal though, we still have to consider whether the boundaries
2050 : : * are inclusive or exclusive.
2051 : : */
2052 [ + + ]: 6261090 : if (result == 0)
2053 : : {
4532 tgl@sss.pgh.pa.us 2054 [ + + + + ]: 365427 : if (!b1->inclusive && !b2->inclusive)
2055 : : {
2056 : : /* both are exclusive */
2057 [ + + ]: 161601 : if (b1->lower == b2->lower)
2058 : 161598 : return 0;
2059 : : else
2060 [ - + ]: 3 : return b1->lower ? 1 : -1;
2061 : : }
2062 [ + + ]: 203826 : else if (!b1->inclusive)
2063 [ - + ]: 363 : return b1->lower ? 1 : -1;
2064 [ + + ]: 203463 : else if (!b2->inclusive)
2065 [ + + ]: 593 : return b2->lower ? -1 : 1;
2066 : : else
2067 : : {
2068 : : /*
2069 : : * Both are inclusive and the values held are equal, so they are
2070 : : * equal regardless of whether they are upper or lower boundaries,
2071 : : * or a mix.
2072 : : */
2073 : 202870 : return 0;
2074 : : }
2075 : : }
2076 : :
4546 heikki.linnakangas@i 2077 : 5895663 : return result;
2078 : : }
2079 : :
2080 : : /*
2081 : : * Compare two range boundary point values, returning <0, 0, or >0 according
2082 : : * to whether b1 is less than, equal to, or greater than b2.
2083 : : *
2084 : : * This is similar to but simpler than range_cmp_bounds(). We just compare
2085 : : * the values held in b1 and b2, ignoring inclusive/exclusive flags. The
2086 : : * lower/upper flags only matter for infinities, where they tell us if the
2087 : : * infinity is plus or minus.
2088 : : */
2089 : : int
1627 peter@eisentraut.org 2090 : 710411 : range_cmp_bound_values(TypeCacheEntry *typcache, const RangeBound *b1,
2091 : : const RangeBound *b2)
2092 : : {
2093 : : /*
2094 : : * First, handle cases involving infinity, which don't require invoking
2095 : : * the comparison proc.
2096 : : */
4527 tgl@sss.pgh.pa.us 2097 [ + + + + ]: 710411 : if (b1->infinite && b2->infinite)
2098 : : {
2099 : : /*
2100 : : * Both are infinity, so they are equal unless one is lower and the
2101 : : * other not.
2102 : : */
2103 [ - + ]: 153 : if (b1->lower == b2->lower)
4527 tgl@sss.pgh.pa.us 2104 :UBC 0 : return 0;
2105 : : else
4527 tgl@sss.pgh.pa.us 2106 [ + + ]:CBC 153 : return b1->lower ? -1 : 1;
2107 : : }
2108 [ + + ]: 710258 : else if (b1->infinite)
2109 [ + + ]: 8345 : return b1->lower ? -1 : 1;
2110 [ + + ]: 701913 : else if (b2->infinite)
2111 [ + + ]: 6947 : return b2->lower ? 1 : -1;
2112 : :
2113 : : /*
2114 : : * Both boundaries are finite, so compare the held values.
2115 : : */
2116 : 694966 : return DatumGetInt32(FunctionCall2Coll(&typcache->rng_cmp_proc_finfo,
2117 : : typcache->rng_collation,
2118 : 694966 : b1->val, b2->val));
2119 : : }
2120 : :
2121 : : /*
2122 : : * qsort callback for sorting ranges.
2123 : : *
2124 : : * Two empty ranges compare equal; an empty range sorts to the left of any
2125 : : * non-empty range. Two non-empty ranges are sorted by lower bound first
2126 : : * and by upper bound next.
2127 : : */
2128 : : int
1211 akorotkov@postgresql 2129 : 13426 : range_compare(const void *key1, const void *key2, void *arg)
2130 : : {
2131 : 13426 : RangeType *r1 = *(RangeType **) key1;
2132 : 13426 : RangeType *r2 = *(RangeType **) key2;
2133 : 13426 : TypeCacheEntry *typcache = (TypeCacheEntry *) arg;
2134 : : RangeBound lower1;
2135 : : RangeBound upper1;
2136 : : RangeBound lower2;
2137 : : RangeBound upper2;
2138 : : bool empty1;
2139 : : bool empty2;
2140 : : int cmp;
2141 : :
2142 : 13426 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
2143 : 13426 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
2144 : :
2145 [ + + + + ]: 13426 : if (empty1 && empty2)
2146 : 24 : cmp = 0;
2147 [ + + ]: 13402 : else if (empty1)
2148 : 21 : cmp = -1;
2149 [ + + ]: 13381 : else if (empty2)
2150 : 6 : cmp = 1;
2151 : : else
2152 : : {
2153 : 13375 : cmp = range_cmp_bounds(typcache, &lower1, &lower2);
2154 [ + + ]: 13375 : if (cmp == 0)
2155 : 24 : cmp = range_cmp_bounds(typcache, &upper1, &upper2);
2156 : : }
2157 : :
2158 : 13426 : return cmp;
2159 : : }
2160 : :
2161 : : /*
2162 : : * Build an empty range value of the type indicated by the typcache entry.
2163 : : */
2164 : : RangeType *
4534 tgl@sss.pgh.pa.us 2165 : 1560 : make_empty_range(TypeCacheEntry *typcache)
2166 : : {
2167 : : RangeBound lower;
2168 : : RangeBound upper;
2169 : :
2170 : 1560 : lower.val = (Datum) 0;
2171 : 1560 : lower.infinite = false;
2172 : 1560 : lower.inclusive = false;
4546 heikki.linnakangas@i 2173 : 1560 : lower.lower = true;
2174 : :
4534 tgl@sss.pgh.pa.us 2175 : 1560 : upper.val = (Datum) 0;
2176 : 1560 : upper.infinite = false;
2177 : 1560 : upper.inclusive = false;
4546 heikki.linnakangas@i 2178 : 1560 : upper.lower = false;
2179 : :
486 tgl@sss.pgh.pa.us 2180 : 1560 : return make_range(typcache, &lower, &upper, true, NULL);
2181 : : }
2182 : :
2183 : : /*
2184 : : * Planner support function for elem_contained_by_range (<@ operator).
2185 : : */
2186 : : Datum
85 tgl@sss.pgh.pa.us 2187 :GNC 63 : elem_contained_by_range_support(PG_FUNCTION_ARGS)
2188 : : {
2189 : 63 : Node *rawreq = (Node *) PG_GETARG_POINTER(0);
2190 : 63 : Node *ret = NULL;
2191 : :
2192 [ + + ]: 63 : if (IsA(rawreq, SupportRequestSimplify))
2193 : : {
2194 : 48 : SupportRequestSimplify *req = (SupportRequestSimplify *) rawreq;
2195 : 48 : FuncExpr *fexpr = req->fcall;
2196 : : Expr *leftop,
2197 : : *rightop;
2198 : :
2199 [ - + ]: 48 : Assert(list_length(fexpr->args) == 2);
2200 : 48 : leftop = linitial(fexpr->args);
2201 : 48 : rightop = lsecond(fexpr->args);
2202 : :
2203 : 48 : ret = find_simplified_clause(req->root, rightop, leftop);
2204 : : }
2205 : :
2206 : 63 : PG_RETURN_POINTER(ret);
2207 : : }
2208 : :
2209 : : /*
2210 : : * Planner support function for range_contains_elem (@> operator).
2211 : : */
2212 : : Datum
2213 : 87 : range_contains_elem_support(PG_FUNCTION_ARGS)
2214 : : {
2215 : 87 : Node *rawreq = (Node *) PG_GETARG_POINTER(0);
2216 : 87 : Node *ret = NULL;
2217 : :
2218 [ + + ]: 87 : if (IsA(rawreq, SupportRequestSimplify))
2219 : : {
2220 : 45 : SupportRequestSimplify *req = (SupportRequestSimplify *) rawreq;
2221 : 45 : FuncExpr *fexpr = req->fcall;
2222 : : Expr *leftop,
2223 : : *rightop;
2224 : :
2225 [ - + ]: 45 : Assert(list_length(fexpr->args) == 2);
2226 : 45 : leftop = linitial(fexpr->args);
2227 : 45 : rightop = lsecond(fexpr->args);
2228 : :
2229 : 45 : ret = find_simplified_clause(req->root, leftop, rightop);
2230 : : }
2231 : :
2232 : 87 : PG_RETURN_POINTER(ret);
2233 : : }
2234 : :
2235 : :
2236 : : /*
2237 : : *----------------------------------------------------------
2238 : : * STATIC FUNCTIONS
2239 : : *----------------------------------------------------------
2240 : : */
2241 : :
2242 : : /*
2243 : : * Given a string representing the flags for the range type, return the flags
2244 : : * represented as a char.
2245 : : */
2246 : : static char
4535 tgl@sss.pgh.pa.us 2247 :CBC 2583 : range_parse_flags(const char *flags_str)
2248 : : {
bruce@momjian.us 2249 : 2583 : char flags = 0;
2250 : :
4546 heikki.linnakangas@i 2251 [ + - ]: 2583 : if (flags_str[0] == '\0' ||
2252 [ + - ]: 2583 : flags_str[1] == '\0' ||
2253 [ - + ]: 2583 : flags_str[2] != '\0')
4546 heikki.linnakangas@i 2254 [ # # ]:UBC 0 : ereport(ERROR,
2255 : : (errcode(ERRCODE_SYNTAX_ERROR),
2256 : : errmsg("invalid range bound flags"),
2257 : : errhint("Valid values are \"[]\", \"[)\", \"(]\", and \"()\".")));
2258 : :
4546 heikki.linnakangas@i 2259 [ + + - ]:CBC 2583 : switch (flags_str[0])
2260 : : {
2261 : 120 : case '[':
2262 : 120 : flags |= RANGE_LB_INC;
2263 : 120 : break;
2264 : 2463 : case '(':
2265 : 2463 : break;
4546 heikki.linnakangas@i 2266 :UBC 0 : default:
2267 [ # # ]: 0 : ereport(ERROR,
2268 : : (errcode(ERRCODE_SYNTAX_ERROR),
2269 : : errmsg("invalid range bound flags"),
2270 : : errhint("Valid values are \"[]\", \"[)\", \"(]\", and \"()\".")));
2271 : : }
2272 : :
4546 heikki.linnakangas@i 2273 [ + + - ]:CBC 2583 : switch (flags_str[1])
2274 : : {
2275 : 2526 : case ']':
2276 : 2526 : flags |= RANGE_UB_INC;
2277 : 2526 : break;
2278 : 57 : case ')':
2279 : 57 : break;
4546 heikki.linnakangas@i 2280 :UBC 0 : default:
2281 [ # # ]: 0 : ereport(ERROR,
2282 : : (errcode(ERRCODE_SYNTAX_ERROR),
2283 : : errmsg("invalid range bound flags"),
2284 : : errhint("Valid values are \"[]\", \"[)\", \"(]\", and \"()\".")));
2285 : : }
2286 : :
4546 heikki.linnakangas@i 2287 :CBC 2583 : return flags;
2288 : : }
2289 : :
2290 : : /*
2291 : : * Parse range input.
2292 : : *
2293 : : * Input parameters:
2294 : : * string: input string to be parsed
2295 : : * Output parameters:
2296 : : * *flags: receives flags bitmask
2297 : : * *lbound_str: receives palloc'd lower bound string, or NULL if none
2298 : : * *ubound_str: receives palloc'd upper bound string, or NULL if none
2299 : : *
2300 : : * This is modeled somewhat after record_in in rowtypes.c.
2301 : : * The input syntax is:
2302 : : * <range> := EMPTY
2303 : : * | <lb-inc> <string>, <string> <ub-inc>
2304 : : * <lb-inc> := '[' | '('
2305 : : * <ub-inc> := ']' | ')'
2306 : : *
2307 : : * Whitespace before or after <range> is ignored. Whitespace within a <string>
2308 : : * is taken literally and becomes part of the input string for that bound.
2309 : : *
2310 : : * A <string> of length zero is taken as "infinite" (i.e. no bound), unless it
2311 : : * is surrounded by double-quotes, in which case it is the literal empty
2312 : : * string.
2313 : : *
2314 : : * Within a <string>, special characters (such as comma, parenthesis, or
2315 : : * brackets) can be enclosed in double-quotes or escaped with backslash. Within
2316 : : * double-quotes, a double-quote can be escaped with double-quote or backslash.
2317 : : *
2318 : : * Returns true on success, false on failure (but failures will return only if
2319 : : * escontext is an ErrorSaveContext).
2320 : : */
2321 : : static bool
4527 tgl@sss.pgh.pa.us 2322 : 2229 : range_parse(const char *string, char *flags, char **lbound_str,
2323 : : char **ubound_str, Node *escontext)
2324 : : {
2325 : 2229 : const char *ptr = string;
2326 : : bool infinite;
2327 : :
4546 heikki.linnakangas@i 2328 : 2229 : *flags = 0;
2329 : :
2330 : : /* consume whitespace */
4534 tgl@sss.pgh.pa.us 2331 [ + + + + ]: 2241 : while (*ptr != '\0' && isspace((unsigned char) *ptr))
4546 heikki.linnakangas@i 2332 : 12 : ptr++;
2333 : :
2334 : : /* check for empty range */
2335 [ + + ]: 2229 : if (pg_strncasecmp(ptr, RANGE_EMPTY_LITERAL,
2336 : : strlen(RANGE_EMPTY_LITERAL)) == 0)
2337 : : {
2338 : 273 : *flags = RANGE_EMPTY;
4527 tgl@sss.pgh.pa.us 2339 : 273 : *lbound_str = NULL;
2340 : 273 : *ubound_str = NULL;
2341 : :
4546 heikki.linnakangas@i 2342 : 273 : ptr += strlen(RANGE_EMPTY_LITERAL);
2343 : :
2344 : : /* the rest should be whitespace */
4534 tgl@sss.pgh.pa.us 2345 [ + + + - ]: 279 : while (*ptr != '\0' && isspace((unsigned char) *ptr))
4546 heikki.linnakangas@i 2346 : 6 : ptr++;
2347 : :
2348 : : /* should have consumed everything */
2349 [ - + ]: 273 : if (*ptr != '\0')
486 tgl@sss.pgh.pa.us 2350 [ # # ]:UBC 0 : ereturn(escontext, false,
2351 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2352 : : errmsg("malformed range literal: \"%s\"",
2353 : : string),
2354 : : errdetail("Junk after \"empty\" key word.")));
2355 : :
486 tgl@sss.pgh.pa.us 2356 :CBC 273 : return true;
2357 : : }
2358 : :
4527 2359 [ + + ]: 1956 : if (*ptr == '[')
2360 : : {
2361 : 1613 : *flags |= RANGE_LB_INC;
4546 heikki.linnakangas@i 2362 : 1613 : ptr++;
2363 : : }
4527 tgl@sss.pgh.pa.us 2364 [ + + ]: 343 : else if (*ptr == '(')
2365 : 331 : ptr++;
2366 : : else
486 2367 [ + + ]: 12 : ereturn(escontext, false,
2368 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2369 : : errmsg("malformed range literal: \"%s\"",
2370 : : string),
2371 : : errdetail("Missing left parenthesis or bracket.")));
2372 : :
2373 : 1944 : ptr = range_parse_bound(string, ptr, lbound_str, &infinite, escontext);
2374 [ - + ]: 1941 : if (ptr == NULL)
486 tgl@sss.pgh.pa.us 2375 :UBC 0 : return false;
4546 heikki.linnakangas@i 2376 [ + + ]:CBC 1941 : if (infinite)
2377 : 81 : *flags |= RANGE_LB_INF;
2378 : :
4527 tgl@sss.pgh.pa.us 2379 [ + + ]: 1941 : if (*ptr == ',')
2380 : 1929 : ptr++;
2381 : : else
486 2382 [ + - ]: 12 : ereturn(escontext, false,
2383 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2384 : : errmsg("malformed range literal: \"%s\"",
2385 : : string),
2386 : : errdetail("Missing comma after lower bound.")));
2387 : :
2388 : 1929 : ptr = range_parse_bound(string, ptr, ubound_str, &infinite, escontext);
2389 [ + + ]: 1929 : if (ptr == NULL)
2390 : 6 : return false;
4527 2391 [ + + ]: 1923 : if (infinite)
2392 : 123 : *flags |= RANGE_UB_INF;
2393 : :
2394 [ + + ]: 1923 : if (*ptr == ']')
2395 : : {
2396 : 325 : *flags |= RANGE_UB_INC;
2397 : 325 : ptr++;
2398 : : }
2399 [ + + ]: 1598 : else if (*ptr == ')')
2400 : 1592 : ptr++;
2401 : : else /* must be a comma */
486 2402 [ + - ]: 6 : ereturn(escontext, false,
2403 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2404 : : errmsg("malformed range literal: \"%s\"",
2405 : : string),
2406 : : errdetail("Too many commas.")));
2407 : :
2408 : : /* consume whitespace */
4534 2409 [ + + + + ]: 1932 : while (*ptr != '\0' && isspace((unsigned char) *ptr))
4546 heikki.linnakangas@i 2410 : 15 : ptr++;
2411 : :
2412 [ + + ]: 1917 : if (*ptr != '\0')
486 tgl@sss.pgh.pa.us 2413 [ + - ]: 9 : ereturn(escontext, false,
2414 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2415 : : errmsg("malformed range literal: \"%s\"",
2416 : : string),
2417 : : errdetail("Junk after right parenthesis or bracket.")));
2418 : :
2419 : 1908 : return true;
2420 : : }
2421 : :
2422 : : /*
2423 : : * Helper for range_parse: parse and de-quote one bound string.
2424 : : *
2425 : : * We scan until finding comma, right parenthesis, or right bracket.
2426 : : *
2427 : : * Input parameters:
2428 : : * string: entire input string (used only for error reports)
2429 : : * ptr: where to start parsing bound
2430 : : * Output parameters:
2431 : : * *bound_str: receives palloc'd bound string, or NULL if none
2432 : : * *infinite: set true if no bound, else false
2433 : : *
2434 : : * The return value is the scan ptr, advanced past the bound string.
2435 : : * However, if escontext is an ErrorSaveContext, we return NULL on failure.
2436 : : */
2437 : : static const char *
4527 2438 : 3873 : range_parse_bound(const char *string, const char *ptr,
2439 : : char **bound_str, bool *infinite, Node *escontext)
2440 : : {
2441 : : StringInfoData buf;
2442 : :
2443 : : /* Check for null: completely empty input means null */
4546 heikki.linnakangas@i 2444 [ + + + + : 3873 : if (*ptr == ',' || *ptr == ')' || *ptr == ']')
+ + ]
2445 : : {
2446 : 204 : *bound_str = NULL;
4535 bruce@momjian.us 2447 : 204 : *infinite = true;
2448 : : }
2449 : : else
2450 : : {
2451 : : /* Extract string for this bound */
4546 heikki.linnakangas@i 2452 : 3669 : bool inquote = false;
2453 : :
2454 : 3669 : initStringInfo(&buf);
2455 [ + + + + : 11282 : while (inquote || !(*ptr == ',' || *ptr == ')' || *ptr == ']'))
+ + + + ]
2456 : : {
2457 : 7622 : char ch = *ptr++;
2458 : :
2459 [ + + ]: 7622 : if (ch == '\0')
486 tgl@sss.pgh.pa.us 2460 [ + + ]: 9 : ereturn(escontext, NULL,
2461 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2462 : : errmsg("malformed range literal: \"%s\"",
2463 : : string),
2464 : : errdetail("Unexpected end of input.")));
4546 heikki.linnakangas@i 2465 [ + + ]: 7613 : if (ch == '\\')
2466 : : {
2467 [ - + ]: 21 : if (*ptr == '\0')
486 tgl@sss.pgh.pa.us 2468 [ # # ]:UBC 0 : ereturn(escontext, NULL,
2469 : : (errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
2470 : : errmsg("malformed range literal: \"%s\"",
2471 : : string),
2472 : : errdetail("Unexpected end of input.")));
4546 heikki.linnakangas@i 2473 :CBC 21 : appendStringInfoChar(&buf, *ptr++);
2474 : : }
3036 peter_e@gmx.net 2475 [ + + ]: 7592 : else if (ch == '"')
2476 : : {
4546 heikki.linnakangas@i 2477 [ + + ]: 200 : if (!inquote)
2478 : 100 : inquote = true;
3036 peter_e@gmx.net 2479 [ + + ]: 100 : else if (*ptr == '"')
2480 : : {
2481 : : /* doubled quote within quote sequence */
4546 heikki.linnakangas@i 2482 : 3 : appendStringInfoChar(&buf, *ptr++);
2483 : : }
2484 : : else
2485 : 97 : inquote = false;
2486 : : }
2487 : : else
2488 : 7392 : appendStringInfoChar(&buf, ch);
2489 : : }
2490 : :
2491 : 3660 : *bound_str = buf.data;
4535 bruce@momjian.us 2492 : 3660 : *infinite = false;
2493 : : }
2494 : :
4546 heikki.linnakangas@i 2495 : 3864 : return ptr;
2496 : : }
2497 : :
2498 : : /*
2499 : : * Convert a deserialized range value to text form
2500 : : *
2501 : : * Inputs are the flags byte, and the two bound values already converted to
2502 : : * text (but not yet quoted). If no bound value, pass NULL.
2503 : : *
2504 : : * Result is a palloc'd string
2505 : : */
2506 : : static char *
4527 tgl@sss.pgh.pa.us 2507 : 52522 : range_deparse(char flags, const char *lbound_str, const char *ubound_str)
2508 : : {
2509 : : StringInfoData buf;
2510 : :
4546 heikki.linnakangas@i 2511 [ + + ]: 52522 : if (flags & RANGE_EMPTY)
2512 : 8452 : return pstrdup(RANGE_EMPTY_LITERAL);
2513 : :
4534 tgl@sss.pgh.pa.us 2514 : 44070 : initStringInfo(&buf);
2515 : :
4527 2516 [ + + ]: 44070 : appendStringInfoChar(&buf, (flags & RANGE_LB_INC) ? '[' : '(');
2517 : :
4546 heikki.linnakangas@i 2518 [ + + ]: 44070 : if (RANGE_HAS_LBOUND(flags))
2519 : 42839 : appendStringInfoString(&buf, range_bound_escape(lbound_str));
2520 : :
4527 tgl@sss.pgh.pa.us 2521 : 44070 : appendStringInfoChar(&buf, ',');
2522 : :
4546 heikki.linnakangas@i 2523 [ + + ]: 44070 : if (RANGE_HAS_UBOUND(flags))
2524 : 42803 : appendStringInfoString(&buf, range_bound_escape(ubound_str));
2525 : :
4527 tgl@sss.pgh.pa.us 2526 [ + + ]: 44070 : appendStringInfoChar(&buf, (flags & RANGE_UB_INC) ? ']' : ')');
2527 : :
4546 heikki.linnakangas@i 2528 : 44070 : return buf.data;
2529 : : }
2530 : :
2531 : : /*
2532 : : * Helper for range_deparse: quote a bound value as needed
2533 : : *
2534 : : * Result is a palloc'd string
2535 : : */
2536 : : static char *
4527 tgl@sss.pgh.pa.us 2537 : 85642 : range_bound_escape(const char *value)
2538 : : {
2539 : : bool nq;
2540 : : const char *ptr;
2541 : : StringInfoData buf;
2542 : :
4546 heikki.linnakangas@i 2543 : 85642 : initStringInfo(&buf);
2544 : :
2545 : : /* Detect whether we need double quotes for this value */
2546 : 85642 : nq = (value[0] == '\0'); /* force quotes for empty string */
4527 tgl@sss.pgh.pa.us 2547 [ + + ]: 389927 : for (ptr = value; *ptr; ptr++)
2548 : : {
2549 : 304540 : char ch = *ptr;
2550 : :
4546 heikki.linnakangas@i 2551 [ + + + + : 304540 : if (ch == '"' || ch == '\\' ||
+ + ]
2552 [ + - + + ]: 304471 : ch == '(' || ch == ')' ||
2553 [ + - + + ]: 304459 : ch == '[' || ch == ']' ||
2554 : 304435 : ch == ',' ||
2555 [ + + ]: 304435 : isspace((unsigned char) ch))
2556 : : {
2557 : 255 : nq = true;
2558 : 255 : break;
2559 : : }
2560 : : }
2561 : :
2562 : : /* And emit the string */
2563 [ + + ]: 85642 : if (nq)
2564 : 267 : appendStringInfoChar(&buf, '"');
4527 tgl@sss.pgh.pa.us 2565 [ + + ]: 392024 : for (ptr = value; *ptr; ptr++)
2566 : : {
2567 : 306382 : char ch = *ptr;
2568 : :
4546 heikki.linnakangas@i 2569 [ + + + + ]: 306382 : if (ch == '"' || ch == '\\')
2570 : 60 : appendStringInfoChar(&buf, ch);
2571 : 306382 : appendStringInfoChar(&buf, ch);
2572 : : }
2573 [ + + ]: 85642 : if (nq)
2574 : 267 : appendStringInfoChar(&buf, '"');
2575 : :
2576 : 85642 : return buf.data;
2577 : : }
2578 : :
2579 : : /*
2580 : : * Test whether range r1 contains range r2.
2581 : : *
2582 : : * Caller has already checked that they are the same range type, and looked up
2583 : : * the necessary typcache entry.
2584 : : */
2585 : : bool
1627 peter@eisentraut.org 2586 : 243211 : range_contains_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
2587 : : {
2588 : : RangeBound lower1;
2589 : : RangeBound upper1;
2590 : : bool empty1;
2591 : : RangeBound lower2;
2592 : : RangeBound upper2;
2593 : : bool empty2;
2594 : :
2595 : : /* Different types should be prevented by ANYRANGE matching rules */
4288 heikki.linnakangas@i 2596 [ - + ]: 243211 : if (RangeTypeGetOid(r1) != RangeTypeGetOid(r2))
4288 heikki.linnakangas@i 2597 [ # # ]:UBC 0 : elog(ERROR, "range types do not match");
2598 : :
4534 tgl@sss.pgh.pa.us 2599 :CBC 243211 : range_deserialize(typcache, r1, &lower1, &upper1, &empty1);
2600 : 243211 : range_deserialize(typcache, r2, &lower2, &upper2, &empty2);
2601 : :
2602 : : /* If either range is empty, the answer is easy */
4546 heikki.linnakangas@i 2603 [ + + ]: 243211 : if (empty2)
2604 : 157375 : return true;
2605 [ + + ]: 85836 : else if (empty1)
2606 : 6783 : return false;
2607 : :
2608 : : /* Else we must have lower1 <= lower2 and upper1 >= upper2 */
4534 tgl@sss.pgh.pa.us 2609 [ + + ]: 79053 : if (range_cmp_bounds(typcache, &lower1, &lower2) > 0)
4546 heikki.linnakangas@i 2610 : 37727 : return false;
4534 tgl@sss.pgh.pa.us 2611 [ + + ]: 41326 : if (range_cmp_bounds(typcache, &upper1, &upper2) < 0)
4546 heikki.linnakangas@i 2612 : 37192 : return false;
2613 : :
4527 tgl@sss.pgh.pa.us 2614 : 4134 : return true;
2615 : : }
2616 : :
2617 : : bool
1627 peter@eisentraut.org 2618 : 61894 : range_contained_by_internal(TypeCacheEntry *typcache, const RangeType *r1, const RangeType *r2)
2619 : : {
4288 heikki.linnakangas@i 2620 : 61894 : return range_contains_internal(typcache, r2, r1);
2621 : : }
2622 : :
2623 : : /*
2624 : : * Test whether range r contains a specific element value.
2625 : : */
2626 : : bool
1627 peter@eisentraut.org 2627 : 44979 : range_contains_elem_internal(TypeCacheEntry *typcache, const RangeType *r, Datum val)
2628 : : {
2629 : : RangeBound lower;
2630 : : RangeBound upper;
2631 : : bool empty;
2632 : : int32 cmp;
2633 : :
4527 tgl@sss.pgh.pa.us 2634 : 44979 : range_deserialize(typcache, r, &lower, &upper, &empty);
2635 : :
2636 [ + + ]: 44979 : if (empty)
2637 : 6432 : return false;
2638 : :
2639 [ + + ]: 38547 : if (!lower.infinite)
2640 : : {
2641 : 36420 : cmp = DatumGetInt32(FunctionCall2Coll(&typcache->rng_cmp_proc_finfo,
2642 : : typcache->rng_collation,
2643 : : lower.val, val));
2644 [ + + ]: 36420 : if (cmp > 0)
2645 : 35372 : return false;
2646 [ + + - + ]: 1048 : if (cmp == 0 && !lower.inclusive)
4527 tgl@sss.pgh.pa.us 2647 :UBC 0 : return false;
2648 : : }
2649 : :
4527 tgl@sss.pgh.pa.us 2650 [ + + ]:CBC 3175 : if (!upper.infinite)
2651 : : {
2652 : 3156 : cmp = DatumGetInt32(FunctionCall2Coll(&typcache->rng_cmp_proc_finfo,
2653 : : typcache->rng_collation,
2654 : : upper.val, val));
2655 [ + + ]: 3156 : if (cmp < 0)
2656 : 204 : return false;
2657 [ - + - - ]: 2952 : if (cmp == 0 && !upper.inclusive)
4527 tgl@sss.pgh.pa.us 2658 :LBC (3) : return false;
2659 : : }
2660 : :
4546 heikki.linnakangas@i 2661 :CBC 2971 : return true;
2662 : : }
2663 : :
2664 : :
2665 : : /*
2666 : : * datum_compute_size() and datum_write() are used to insert the bound
2667 : : * values into a range object. They are modeled after heaptuple.c's
2668 : : * heap_compute_data_size() and heap_fill_tuple(), but we need not handle
2669 : : * null values here. TYPE_IS_PACKABLE must test the same conditions as
2670 : : * heaptuple.c's ATT_IS_PACKABLE macro. See the comments there for more
2671 : : * details.
2672 : : */
2673 : :
2674 : : /* Does datatype allow packing into the 1-byte-header varlena format? */
2675 : : #define TYPE_IS_PACKABLE(typlen, typstorage) \
2676 : : ((typlen) == -1 && (typstorage) != TYPSTORAGE_PLAIN)
2677 : :
2678 : : /*
2679 : : * Increment data_length by the space needed by the datum, including any
2680 : : * preceding alignment padding.
2681 : : */
2682 : : static Size
2683 : 936643 : datum_compute_size(Size data_length, Datum val, bool typbyval, char typalign,
2684 : : int16 typlen, char typstorage)
2685 : : {
2686 [ + + + - ]: 936643 : if (TYPE_IS_PACKABLE(typlen, typstorage) &&
2687 [ + + + + ]: 4728 : VARATT_CAN_MAKE_SHORT(DatumGetPointer(val)))
2688 : : {
2689 : : /*
2690 : : * we're anticipating converting to a short varlena header, so adjust
2691 : : * length and don't count any alignment
2692 : : */
2693 : 4269 : data_length += VARATT_CONVERTED_SHORT_SIZE(DatumGetPointer(val));
2694 : : }
2695 : : else
2696 : : {
2697 [ + + + + : 932374 : data_length = att_align_datum(data_length, typalign, typlen, val);
+ + + - +
- - - ]
2698 [ + + + - : 932374 : data_length = att_addlength_datum(data_length, typlen, val);
- + - - -
- - - + +
- - ]
2699 : : }
2700 : :
2701 : 936643 : return data_length;
2702 : : }
2703 : :
2704 : : /*
2705 : : * Write the given datum beginning at ptr (after advancing to correct
2706 : : * alignment, if needed). Return the pointer incremented by space used.
2707 : : */
2708 : : static Pointer
2709 : 936643 : datum_write(Pointer ptr, Datum datum, bool typbyval, char typalign,
2710 : : int16 typlen, char typstorage)
2711 : : {
2712 : : Size data_length;
2713 : :
2714 [ + + ]: 936643 : if (typbyval)
2715 : : {
2716 : : /* pass-by-value */
2717 [ + + + - : 931915 : ptr = (char *) att_align_nominal(ptr, typalign);
+ - - - ]
2718 : 931915 : store_att_byval(ptr, datum, typlen);
2719 : 931915 : data_length = typlen;
2720 : : }
2721 [ + - ]: 4728 : else if (typlen == -1)
2722 : : {
2723 : : /* varlena */
2724 : 4728 : Pointer val = DatumGetPointer(datum);
2725 : :
2726 [ - + ]: 4728 : if (VARATT_IS_EXTERNAL(val))
2727 : : {
2728 : : /*
2729 : : * Throw error, because we must never put a toast pointer inside a
2730 : : * range object. Caller should have detoasted it.
2731 : : */
4535 tgl@sss.pgh.pa.us 2732 [ # # ]:UBC 0 : elog(ERROR, "cannot store a toast pointer inside a range");
2733 : : data_length = 0; /* keep compiler quiet */
2734 : : }
4546 heikki.linnakangas@i 2735 [ + + ]:CBC 4728 : else if (VARATT_IS_SHORT(val))
2736 : : {
2737 : : /* no alignment for short varlenas */
2738 : 441 : data_length = VARSIZE_SHORT(val);
2739 : 441 : memcpy(ptr, val, data_length);
2740 : : }
2741 [ + - + - ]: 4287 : else if (TYPE_IS_PACKABLE(typlen, typstorage) &&
2742 [ + - + + ]: 4287 : VARATT_CAN_MAKE_SHORT(val))
2743 : : {
2744 : : /* convert to short varlena -- no alignment */
2745 : 4269 : data_length = VARATT_CONVERTED_SHORT_SIZE(val);
2746 : 4269 : SET_VARSIZE_SHORT(ptr, data_length);
2747 : 4269 : memcpy(ptr + 1, VARDATA(val), data_length - 1);
2748 : : }
2749 : : else
2750 : : {
2751 : : /* full 4-byte header varlena */
2752 [ + - - - : 18 : ptr = (char *) att_align_nominal(ptr, typalign);
- - - - ]
2753 : 18 : data_length = VARSIZE(val);
2754 : 18 : memcpy(ptr, val, data_length);
2755 : : }
2756 : : }
4546 heikki.linnakangas@i 2757 [ # # ]:UBC 0 : else if (typlen == -2)
2758 : : {
2759 : : /* cstring ... never needs alignment */
1502 tgl@sss.pgh.pa.us 2760 [ # # ]: 0 : Assert(typalign == TYPALIGN_CHAR);
4546 heikki.linnakangas@i 2761 : 0 : data_length = strlen(DatumGetCString(datum)) + 1;
2762 : 0 : memcpy(ptr, DatumGetPointer(datum), data_length);
2763 : : }
2764 : : else
2765 : : {
2766 : : /* fixed-length pass-by-reference */
2767 [ # # # # : 0 : ptr = (char *) att_align_nominal(ptr, typalign);
# # # # ]
2768 [ # # ]: 0 : Assert(typlen > 0);
2769 : 0 : data_length = typlen;
2770 : 0 : memcpy(ptr, DatumGetPointer(datum), data_length);
2771 : : }
2772 : :
4546 heikki.linnakangas@i 2773 :CBC 936643 : ptr += data_length;
2774 : :
2775 : 936643 : return ptr;
2776 : : }
2777 : :
2778 : : /*
2779 : : * Common code for the elem_contained_by_range and range_contains_elem
2780 : : * support functions. The caller has extracted the function argument
2781 : : * expressions, and swapped them if necessary to pass the range first.
2782 : : *
2783 : : * Returns a simplified replacement expression, or NULL if we can't simplify.
2784 : : */
2785 : : static Node *
85 tgl@sss.pgh.pa.us 2786 :GNC 93 : find_simplified_clause(PlannerInfo *root, Expr *rangeExpr, Expr *elemExpr)
2787 : : {
2788 : : RangeType *range;
2789 : : TypeCacheEntry *rangetypcache;
2790 : : RangeBound lower;
2791 : : RangeBound upper;
2792 : : bool empty;
2793 : :
2794 : : /* can't do anything unless the range is a non-null constant */
2795 [ + + - + ]: 93 : if (!IsA(rangeExpr, Const) || ((Const *) rangeExpr)->constisnull)
2796 : 45 : return NULL;
2797 : 48 : range = DatumGetRangeTypeP(((Const *) rangeExpr)->constvalue);
2798 : :
2799 : 48 : rangetypcache = lookup_type_cache(RangeTypeGetOid(range),
2800 : : TYPECACHE_RANGE_INFO);
2801 [ - + ]: 48 : if (rangetypcache->rngelemtype == NULL)
85 tgl@sss.pgh.pa.us 2802 [ # # ]:UNC 0 : elog(ERROR, "type %u is not a range type", RangeTypeGetOid(range));
2803 : :
85 tgl@sss.pgh.pa.us 2804 :GNC 48 : range_deserialize(rangetypcache, range, &lower, &upper, &empty);
2805 : :
2806 [ + + ]: 48 : if (empty)
2807 : : {
2808 : : /* if the range is empty, then there can be no matches */
2809 : 3 : return makeBoolConst(false, false);
2810 : : }
2811 [ + + + + ]: 45 : else if (lower.infinite && upper.infinite)
2812 : : {
2813 : : /* the range has infinite bounds, so it matches everything */
2814 : 3 : return makeBoolConst(true, false);
2815 : : }
2816 : : else
2817 : : {
2818 : : /* at least one bound is available, we have something to work with */
2819 : 42 : TypeCacheEntry *elemTypcache = rangetypcache->rngelemtype;
2820 : 42 : Oid opfamily = rangetypcache->rng_opfamily;
2821 : 42 : Oid rng_collation = rangetypcache->rng_collation;
2822 : 42 : Expr *lowerExpr = NULL;
2823 : 42 : Expr *upperExpr = NULL;
2824 : :
2825 [ + + + + ]: 42 : if (!lower.infinite && !upper.infinite)
2826 : : {
2827 : : /*
2828 : : * When both bounds are present, we have a problem: the
2829 : : * "simplified" clause would need to evaluate the elemExpr twice.
2830 : : * That's definitely not okay if the elemExpr is volatile, and
2831 : : * it's also unattractive if the elemExpr is expensive.
2832 : : */
2833 : : QualCost eval_cost;
2834 : :
2835 [ + + ]: 33 : if (contain_volatile_functions((Node *) elemExpr))
2836 : 3 : return NULL;
2837 : :
2838 : : /*
2839 : : * We define "expensive" as "contains any subplan or more than 10
2840 : : * operators". Note that the subplan search has to be done
2841 : : * explicitly, since cost_qual_eval() will barf on unplanned
2842 : : * subselects.
2843 : : */
2844 [ - + ]: 30 : if (contain_subplans((Node *) elemExpr))
85 tgl@sss.pgh.pa.us 2845 :UNC 0 : return NULL;
85 tgl@sss.pgh.pa.us 2846 :GNC 30 : cost_qual_eval_node(&eval_cost, (Node *) elemExpr, root);
2847 : 30 : if (eval_cost.startup + eval_cost.per_tuple >
2848 [ - + ]: 30 : 10 * cpu_operator_cost)
85 tgl@sss.pgh.pa.us 2849 :UNC 0 : return NULL;
2850 : : }
2851 : :
2852 : : /* Okay, try to build boundary comparison expressions */
85 tgl@sss.pgh.pa.us 2853 [ + + ]:GNC 39 : if (!lower.infinite)
2854 : : {
2855 : 36 : lowerExpr = build_bound_expr(elemExpr,
2856 : : lower.val,
2857 : : true,
2858 : 36 : lower.inclusive,
2859 : : elemTypcache,
2860 : : opfamily,
2861 : : rng_collation);
2862 [ - + ]: 36 : if (lowerExpr == NULL)
85 tgl@sss.pgh.pa.us 2863 :UNC 0 : return NULL;
2864 : : }
2865 : :
85 tgl@sss.pgh.pa.us 2866 [ + + ]:GNC 39 : if (!upper.infinite)
2867 : : {
2868 : : /* Copy the elemExpr if we need two copies */
2869 [ + + ]: 33 : if (!lower.infinite)
2870 : 30 : elemExpr = copyObject(elemExpr);
2871 : 33 : upperExpr = build_bound_expr(elemExpr,
2872 : : upper.val,
2873 : : false,
2874 : 33 : upper.inclusive,
2875 : : elemTypcache,
2876 : : opfamily,
2877 : : rng_collation);
2878 [ - + ]: 33 : if (upperExpr == NULL)
85 tgl@sss.pgh.pa.us 2879 :UNC 0 : return NULL;
2880 : : }
2881 : :
85 tgl@sss.pgh.pa.us 2882 [ + + + + ]:GNC 39 : if (lowerExpr != NULL && upperExpr != NULL)
2883 : 30 : return (Node *) make_andclause(list_make2(lowerExpr, upperExpr));
2884 [ + + ]: 9 : else if (lowerExpr != NULL)
2885 : 6 : return (Node *) lowerExpr;
2886 [ + - ]: 3 : else if (upperExpr != NULL)
2887 : 3 : return (Node *) upperExpr;
2888 : : else
2889 : : {
85 tgl@sss.pgh.pa.us 2890 :UNC 0 : Assert(false);
2891 : : return NULL;
2892 : : }
2893 : : }
2894 : : }
2895 : :
2896 : : /*
2897 : : * Helper function for find_simplified_clause().
2898 : : *
2899 : : * Build the expression (elemExpr Operator val), where the operator is
2900 : : * the appropriate member of the given opfamily depending on
2901 : : * isLowerBound and isInclusive. typeCache is the typcache entry for
2902 : : * the "val" value (presently, this will be the same type as elemExpr).
2903 : : * rng_collation is the collation to use in the comparison.
2904 : : *
2905 : : * Return NULL on failure (if, for some reason, we can't find the operator).
2906 : : */
2907 : : static Expr *
85 tgl@sss.pgh.pa.us 2908 :GNC 69 : build_bound_expr(Expr *elemExpr, Datum val,
2909 : : bool isLowerBound, bool isInclusive,
2910 : : TypeCacheEntry *typeCache,
2911 : : Oid opfamily, Oid rng_collation)
2912 : : {
2913 : 69 : Oid elemType = typeCache->type_id;
2914 : 69 : int16 elemTypeLen = typeCache->typlen;
2915 : 69 : bool elemByValue = typeCache->typbyval;
2916 : 69 : Oid elemCollation = typeCache->typcollation;
2917 : : int16 strategy;
2918 : : Oid oproid;
2919 : : Expr *constExpr;
2920 : :
2921 : : /* Identify the comparison operator to use */
2922 [ + + ]: 69 : if (isLowerBound)
2923 [ + + ]: 36 : strategy = isInclusive ? BTGreaterEqualStrategyNumber : BTGreaterStrategyNumber;
2924 : : else
2925 [ + + ]: 33 : strategy = isInclusive ? BTLessEqualStrategyNumber : BTLessStrategyNumber;
2926 : :
2927 : : /*
2928 : : * We could use exprType(elemExpr) here, if it ever becomes possible that
2929 : : * elemExpr is not the exact same type as the range elements.
2930 : : */
2931 : 69 : oproid = get_opfamily_member(opfamily, elemType, elemType, strategy);
2932 : :
2933 : : /* We don't really expect failure here, but just in case ... */
2934 [ - + ]: 69 : if (!OidIsValid(oproid))
85 tgl@sss.pgh.pa.us 2935 :UNC 0 : return NULL;
2936 : :
2937 : : /* OK, convert "val" to a full-fledged Const node, and make the OpExpr */
85 tgl@sss.pgh.pa.us 2938 :GNC 69 : constExpr = (Expr *) makeConst(elemType,
2939 : : -1,
2940 : : elemCollation,
2941 : : elemTypeLen,
2942 : : val,
2943 : : false,
2944 : : elemByValue);
2945 : :
2946 : 69 : return make_opclause(oproid,
2947 : : BOOLOID,
2948 : : false,
2949 : : elemExpr,
2950 : : constExpr,
2951 : : InvalidOid,
2952 : : rng_collation);
2953 : : }
|