PostgreSQL Source Code git master
lsyscache.c
Go to the documentation of this file.
1/*-------------------------------------------------------------------------
2 *
3 * lsyscache.c
4 * Convenience routines for common queries in the system catalog cache.
5 *
6 * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
7 * Portions Copyright (c) 1994, Regents of the University of California
8 *
9 * IDENTIFICATION
10 * src/backend/utils/cache/lsyscache.c
11 *
12 * NOTES
13 * Eventually, the index information should go through here, too.
14 *-------------------------------------------------------------------------
15 */
16#include "postgres.h"
17
18#include "access/hash.h"
19#include "access/htup_details.h"
20#include "bootstrap/bootstrap.h"
21#include "catalog/namespace.h"
22#include "catalog/pg_am.h"
23#include "catalog/pg_amop.h"
24#include "catalog/pg_amproc.h"
25#include "catalog/pg_cast.h"
26#include "catalog/pg_class.h"
29#include "catalog/pg_database.h"
30#include "catalog/pg_index.h"
31#include "catalog/pg_language.h"
33#include "catalog/pg_opclass.h"
34#include "catalog/pg_opfamily.h"
35#include "catalog/pg_operator.h"
36#include "catalog/pg_proc.h"
38#include "catalog/pg_range.h"
42#include "catalog/pg_type.h"
43#include "miscadmin.h"
44#include "nodes/makefuncs.h"
45#include "utils/array.h"
46#include "utils/builtins.h"
47#include "utils/catcache.h"
48#include "utils/datum.h"
49#include "utils/fmgroids.h"
50#include "utils/lsyscache.h"
51#include "utils/syscache.h"
52#include "utils/typcache.h"
53
54/* Hook for plugins to get control in get_attavgwidth() */
56
57
58/* ---------- AMOP CACHES ---------- */
59
60/*
61 * op_in_opfamily
62 *
63 * Return t iff operator 'opno' is in operator family 'opfamily'.
64 *
65 * This function only considers search operators, not ordering operators.
66 */
67bool
68op_in_opfamily(Oid opno, Oid opfamily)
69{
70 return SearchSysCacheExists3(AMOPOPID,
71 ObjectIdGetDatum(opno),
72 CharGetDatum(AMOP_SEARCH),
73 ObjectIdGetDatum(opfamily));
74}
75
76/*
77 * get_op_opfamily_strategy
78 *
79 * Get the operator's strategy number within the specified opfamily,
80 * or 0 if it's not a member of the opfamily.
81 *
82 * This function only considers search operators, not ordering operators.
83 */
84int
86{
87 HeapTuple tp;
88 Form_pg_amop amop_tup;
89 int result;
90
91 tp = SearchSysCache3(AMOPOPID,
92 ObjectIdGetDatum(opno),
93 CharGetDatum(AMOP_SEARCH),
94 ObjectIdGetDatum(opfamily));
95 if (!HeapTupleIsValid(tp))
96 return 0;
97 amop_tup = (Form_pg_amop) GETSTRUCT(tp);
98 result = amop_tup->amopstrategy;
100 return result;
101}
102
103/*
104 * get_op_opfamily_sortfamily
105 *
106 * If the operator is an ordering operator within the specified opfamily,
107 * return its amopsortfamily OID; else return InvalidOid.
108 */
109Oid
111{
112 HeapTuple tp;
113 Form_pg_amop amop_tup;
114 Oid result;
115
116 tp = SearchSysCache3(AMOPOPID,
117 ObjectIdGetDatum(opno),
118 CharGetDatum(AMOP_ORDER),
119 ObjectIdGetDatum(opfamily));
120 if (!HeapTupleIsValid(tp))
121 return InvalidOid;
122 amop_tup = (Form_pg_amop) GETSTRUCT(tp);
123 result = amop_tup->amopsortfamily;
124 ReleaseSysCache(tp);
125 return result;
126}
127
128/*
129 * get_op_opfamily_properties
130 *
131 * Get the operator's strategy number and declared input data types
132 * within the specified opfamily.
133 *
134 * Caller should already have verified that opno is a member of opfamily,
135 * therefore we raise an error if the tuple is not found.
136 */
137void
138get_op_opfamily_properties(Oid opno, Oid opfamily, bool ordering_op,
139 int *strategy,
140 Oid *lefttype,
141 Oid *righttype)
142{
143 HeapTuple tp;
144 Form_pg_amop amop_tup;
145
146 tp = SearchSysCache3(AMOPOPID,
147 ObjectIdGetDatum(opno),
148 CharGetDatum(ordering_op ? AMOP_ORDER : AMOP_SEARCH),
149 ObjectIdGetDatum(opfamily));
150 if (!HeapTupleIsValid(tp))
151 elog(ERROR, "operator %u is not a member of opfamily %u",
152 opno, opfamily);
153 amop_tup = (Form_pg_amop) GETSTRUCT(tp);
154 *strategy = amop_tup->amopstrategy;
155 *lefttype = amop_tup->amoplefttype;
156 *righttype = amop_tup->amoprighttype;
157 ReleaseSysCache(tp);
158}
159
160/*
161 * get_opfamily_member
162 * Get the OID of the operator that implements the specified strategy
163 * with the specified datatypes for the specified opfamily.
164 *
165 * Returns InvalidOid if there is no pg_amop entry for the given keys.
166 */
167Oid
168get_opfamily_member(Oid opfamily, Oid lefttype, Oid righttype,
169 int16 strategy)
170{
171 HeapTuple tp;
172 Form_pg_amop amop_tup;
173 Oid result;
174
175 tp = SearchSysCache4(AMOPSTRATEGY,
176 ObjectIdGetDatum(opfamily),
177 ObjectIdGetDatum(lefttype),
178 ObjectIdGetDatum(righttype),
179 Int16GetDatum(strategy));
180 if (!HeapTupleIsValid(tp))
181 return InvalidOid;
182 amop_tup = (Form_pg_amop) GETSTRUCT(tp);
183 result = amop_tup->amopopr;
184 ReleaseSysCache(tp);
185 return result;
186}
187
188/*
189 * get_opfamily_member_for_cmptype
190 * Get the OID of the operator that implements the specified comparison
191 * type with the specified datatypes for the specified opfamily.
192 *
193 * Returns InvalidOid if there is no mapping for the comparison type or no
194 * pg_amop entry for the given keys.
195 */
196Oid
197get_opfamily_member_for_cmptype(Oid opfamily, Oid lefttype, Oid righttype,
198 CompareType cmptype)
199{
200 Oid opmethod;
201 StrategyNumber strategy;
202
203 opmethod = get_opfamily_method(opfamily);
204 strategy = IndexAmTranslateCompareType(cmptype, opmethod, opfamily, true);
205 if (!strategy)
206 return InvalidOid;
207 return get_opfamily_member(opfamily, lefttype, righttype, strategy);
208}
209
210/*
211 * get_opmethod_canorder
212 * Return amcanorder field for given index AM.
213 *
214 * To speed things up in the common cases, we're hardcoding the results from
215 * the built-in index types. Note that we also need to hardcode the negative
216 * results from the built-in non-btree index types, since you'll usually get a
217 * few hits for those as well. It would be nice to organize and cache this a
218 * bit differently to avoid the hardcoding.
219 */
220static bool
222{
223 switch (amoid)
224 {
225 case BTREE_AM_OID:
226 return true;
227 case HASH_AM_OID:
228 case GIST_AM_OID:
229 case GIN_AM_OID:
230 case SPGIST_AM_OID:
231 case BRIN_AM_OID:
232 return false;
233 default:
234 return GetIndexAmRoutineByAmId(amoid, false)->amcanorder;
235 }
236}
237
238/*
239 * get_ordering_op_properties
240 * Given the OID of an ordering operator (a "<" or ">" operator),
241 * determine its opfamily, its declared input datatype, and its
242 * comparison type.
243 *
244 * Returns true if successful, false if no matching pg_amop entry exists.
245 * (This indicates that the operator is not a valid ordering operator.)
246 *
247 * Note: the operator could be registered in multiple families, for example
248 * if someone were to build a "reverse sort" opfamily. This would result in
249 * uncertainty as to whether "ORDER BY USING op" would default to NULLS FIRST
250 * or NULLS LAST, as well as inefficient planning due to failure to match up
251 * pathkeys that should be the same. So we want a determinate result here.
252 * Because of the way the syscache search works, we'll use the interpretation
253 * associated with the opfamily with smallest OID, which is probably
254 * determinate enough. Since there is no longer any particularly good reason
255 * to build reverse-sort opfamilies, it doesn't seem worth expending any
256 * additional effort on ensuring consistency.
257 */
258bool
260 Oid *opfamily, Oid *opcintype, CompareType *cmptype)
261{
262 bool result = false;
263 CatCList *catlist;
264 int i;
265
266 /* ensure outputs are initialized on failure */
267 *opfamily = InvalidOid;
268 *opcintype = InvalidOid;
269 *cmptype = COMPARE_INVALID;
270
271 /*
272 * Search pg_amop to see if the target operator is registered as the "<"
273 * or ">" operator of any btree opfamily.
274 */
275 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
276
277 for (i = 0; i < catlist->n_members; i++)
278 {
279 HeapTuple tuple = &catlist->members[i]->tuple;
280 Form_pg_amop aform = (Form_pg_amop) GETSTRUCT(tuple);
281 CompareType am_cmptype;
282
283 /* must be ordering index */
284 if (!get_opmethod_canorder(aform->amopmethod))
285 continue;
286
287 am_cmptype = IndexAmTranslateStrategy(aform->amopstrategy,
288 aform->amopmethod,
289 aform->amopfamily,
290 true);
291
292 if (am_cmptype == COMPARE_LT || am_cmptype == COMPARE_GT)
293 {
294 /* Found it ... should have consistent input types */
295 if (aform->amoplefttype == aform->amoprighttype)
296 {
297 /* Found a suitable opfamily, return info */
298 *opfamily = aform->amopfamily;
299 *opcintype = aform->amoplefttype;
300 *cmptype = am_cmptype;
301 result = true;
302 break;
303 }
304 }
305 }
306
307 ReleaseSysCacheList(catlist);
308
309 return result;
310}
311
312/*
313 * get_equality_op_for_ordering_op
314 * Get the OID of the datatype-specific equality operator
315 * associated with an ordering operator (a "<" or ">" operator).
316 *
317 * If "reverse" isn't NULL, also set *reverse to false if the operator is "<",
318 * true if it's ">"
319 *
320 * Returns InvalidOid if no matching equality operator can be found.
321 * (This indicates that the operator is not a valid ordering operator.)
322 */
323Oid
325{
326 Oid result = InvalidOid;
327 Oid opfamily;
328 Oid opcintype;
329 CompareType cmptype;
330
331 /* Find the operator in pg_amop */
333 &opfamily, &opcintype, &cmptype))
334 {
335 /* Found a suitable opfamily, get matching equality operator */
336 result = get_opfamily_member_for_cmptype(opfamily,
337 opcintype,
338 opcintype,
339 COMPARE_EQ);
340 if (reverse)
341 *reverse = (cmptype == COMPARE_GT);
342 }
343
344 return result;
345}
346
347/*
348 * get_ordering_op_for_equality_op
349 * Get the OID of a datatype-specific "less than" ordering operator
350 * associated with an equality operator. (If there are multiple
351 * possibilities, assume any one will do.)
352 *
353 * This function is used when we have to sort data before unique-ifying,
354 * and don't much care which sorting op is used as long as it's compatible
355 * with the intended equality operator. Since we need a sorting operator,
356 * it should be single-data-type even if the given operator is cross-type.
357 * The caller specifies whether to find an op for the LHS or RHS data type.
358 *
359 * Returns InvalidOid if no matching ordering operator can be found.
360 */
361Oid
362get_ordering_op_for_equality_op(Oid opno, bool use_lhs_type)
363{
364 Oid result = InvalidOid;
365 CatCList *catlist;
366 int i;
367
368 /*
369 * Search pg_amop to see if the target operator is registered as the "="
370 * operator of any btree opfamily.
371 */
372 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
373
374 for (i = 0; i < catlist->n_members; i++)
375 {
376 HeapTuple tuple = &catlist->members[i]->tuple;
377 Form_pg_amop aform = (Form_pg_amop) GETSTRUCT(tuple);
378 CompareType cmptype;
379
380 /* must be ordering index */
381 if (!get_opmethod_canorder(aform->amopmethod))
382 continue;
383
384 cmptype = IndexAmTranslateStrategy(aform->amopstrategy,
385 aform->amopmethod,
386 aform->amopfamily,
387 true);
388 if (cmptype == COMPARE_EQ)
389 {
390 /* Found a suitable opfamily, get matching ordering operator */
391 Oid typid;
392
393 typid = use_lhs_type ? aform->amoplefttype : aform->amoprighttype;
394 result = get_opfamily_member_for_cmptype(aform->amopfamily,
395 typid, typid,
396 COMPARE_LT);
397 if (OidIsValid(result))
398 break;
399 /* failure probably shouldn't happen, but keep looking if so */
400 }
401 }
402
403 ReleaseSysCacheList(catlist);
404
405 return result;
406}
407
408/*
409 * get_mergejoin_opfamilies
410 * Given a putatively mergejoinable operator, return a list of the OIDs
411 * of the amcanorder opfamilies in which it represents equality.
412 *
413 * It is possible (though at present unusual) for an operator to be equality
414 * in more than one opfamily, hence the result is a list. This also lets us
415 * return NIL if the operator is not found in any opfamilies.
416 *
417 * The planner currently uses simple equal() tests to compare the lists
418 * returned by this function, which makes the list order relevant, though
419 * strictly speaking it should not be. Because of the way syscache list
420 * searches are handled, in normal operation the result will be sorted by OID
421 * so everything works fine. If running with system index usage disabled,
422 * the result ordering is unspecified and hence the planner might fail to
423 * recognize optimization opportunities ... but that's hardly a scenario in
424 * which performance is good anyway, so there's no point in expending code
425 * or cycles here to guarantee the ordering in that case.
426 */
427List *
429{
430 List *result = NIL;
431 CatCList *catlist;
432 int i;
433
434 /*
435 * Search pg_amop to see if the target operator is registered as the "="
436 * operator of any opfamily of an ordering index type.
437 */
438 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
439
440 for (i = 0; i < catlist->n_members; i++)
441 {
442 HeapTuple tuple = &catlist->members[i]->tuple;
443 Form_pg_amop aform = (Form_pg_amop) GETSTRUCT(tuple);
444
445 /* must be ordering index equality */
446 if (get_opmethod_canorder(aform->amopmethod) &&
447 IndexAmTranslateStrategy(aform->amopstrategy,
448 aform->amopmethod,
449 aform->amopfamily,
450 true) == COMPARE_EQ)
451 result = lappend_oid(result, aform->amopfamily);
452 }
453
454 ReleaseSysCacheList(catlist);
455
456 return result;
457}
458
459/*
460 * get_compatible_hash_operators
461 * Get the OID(s) of hash equality operator(s) compatible with the given
462 * operator, but operating on its LHS and/or RHS datatype.
463 *
464 * An operator for the LHS type is sought and returned into *lhs_opno if
465 * lhs_opno isn't NULL. Similarly, an operator for the RHS type is sought
466 * and returned into *rhs_opno if rhs_opno isn't NULL.
467 *
468 * If the given operator is not cross-type, the results should be the same
469 * operator, but in cross-type situations they will be different.
470 *
471 * Returns true if able to find the requested operator(s), false if not.
472 * (This indicates that the operator should not have been marked oprcanhash.)
473 */
474bool
476 Oid *lhs_opno, Oid *rhs_opno)
477{
478 bool result = false;
479 CatCList *catlist;
480 int i;
481
482 /* Ensure output args are initialized on failure */
483 if (lhs_opno)
484 *lhs_opno = InvalidOid;
485 if (rhs_opno)
486 *rhs_opno = InvalidOid;
487
488 /*
489 * Search pg_amop to see if the target operator is registered as the "="
490 * operator of any hash opfamily. If the operator is registered in
491 * multiple opfamilies, assume we can use any one.
492 */
493 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
494
495 for (i = 0; i < catlist->n_members; i++)
496 {
497 HeapTuple tuple = &catlist->members[i]->tuple;
498 Form_pg_amop aform = (Form_pg_amop) GETSTRUCT(tuple);
499
500 if (aform->amopmethod == HASH_AM_OID &&
501 aform->amopstrategy == HTEqualStrategyNumber)
502 {
503 /* No extra lookup needed if given operator is single-type */
504 if (aform->amoplefttype == aform->amoprighttype)
505 {
506 if (lhs_opno)
507 *lhs_opno = opno;
508 if (rhs_opno)
509 *rhs_opno = opno;
510 result = true;
511 break;
512 }
513
514 /*
515 * Get the matching single-type operator(s). Failure probably
516 * shouldn't happen --- it implies a bogus opfamily --- but
517 * continue looking if so.
518 */
519 if (lhs_opno)
520 {
521 *lhs_opno = get_opfamily_member(aform->amopfamily,
522 aform->amoplefttype,
523 aform->amoplefttype,
525 if (!OidIsValid(*lhs_opno))
526 continue;
527 /* Matching LHS found, done if caller doesn't want RHS */
528 if (!rhs_opno)
529 {
530 result = true;
531 break;
532 }
533 }
534 if (rhs_opno)
535 {
536 *rhs_opno = get_opfamily_member(aform->amopfamily,
537 aform->amoprighttype,
538 aform->amoprighttype,
540 if (!OidIsValid(*rhs_opno))
541 {
542 /* Forget any LHS operator from this opfamily */
543 if (lhs_opno)
544 *lhs_opno = InvalidOid;
545 continue;
546 }
547 /* Matching RHS found, so done */
548 result = true;
549 break;
550 }
551 }
552 }
553
554 ReleaseSysCacheList(catlist);
555
556 return result;
557}
558
559/*
560 * get_op_hash_functions
561 * Get the OID(s) of the standard hash support function(s) compatible with
562 * the given operator, operating on its LHS and/or RHS datatype as required.
563 *
564 * A function for the LHS type is sought and returned into *lhs_procno if
565 * lhs_procno isn't NULL. Similarly, a function for the RHS type is sought
566 * and returned into *rhs_procno if rhs_procno isn't NULL.
567 *
568 * If the given operator is not cross-type, the results should be the same
569 * function, but in cross-type situations they will be different.
570 *
571 * Returns true if able to find the requested function(s), false if not.
572 * (This indicates that the operator should not have been marked oprcanhash.)
573 */
574bool
576 RegProcedure *lhs_procno, RegProcedure *rhs_procno)
577{
578 bool result = false;
579 CatCList *catlist;
580 int i;
581
582 /* Ensure output args are initialized on failure */
583 if (lhs_procno)
584 *lhs_procno = InvalidOid;
585 if (rhs_procno)
586 *rhs_procno = InvalidOid;
587
588 /*
589 * Search pg_amop to see if the target operator is registered as the "="
590 * operator of any hash opfamily. If the operator is registered in
591 * multiple opfamilies, assume we can use any one.
592 */
593 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
594
595 for (i = 0; i < catlist->n_members; i++)
596 {
597 HeapTuple tuple = &catlist->members[i]->tuple;
598 Form_pg_amop aform = (Form_pg_amop) GETSTRUCT(tuple);
599
600 if (aform->amopmethod == HASH_AM_OID &&
601 aform->amopstrategy == HTEqualStrategyNumber)
602 {
603 /*
604 * Get the matching support function(s). Failure probably
605 * shouldn't happen --- it implies a bogus opfamily --- but
606 * continue looking if so.
607 */
608 if (lhs_procno)
609 {
610 *lhs_procno = get_opfamily_proc(aform->amopfamily,
611 aform->amoplefttype,
612 aform->amoplefttype,
614 if (!OidIsValid(*lhs_procno))
615 continue;
616 /* Matching LHS found, done if caller doesn't want RHS */
617 if (!rhs_procno)
618 {
619 result = true;
620 break;
621 }
622 /* Only one lookup needed if given operator is single-type */
623 if (aform->amoplefttype == aform->amoprighttype)
624 {
625 *rhs_procno = *lhs_procno;
626 result = true;
627 break;
628 }
629 }
630 if (rhs_procno)
631 {
632 *rhs_procno = get_opfamily_proc(aform->amopfamily,
633 aform->amoprighttype,
634 aform->amoprighttype,
636 if (!OidIsValid(*rhs_procno))
637 {
638 /* Forget any LHS function from this opfamily */
639 if (lhs_procno)
640 *lhs_procno = InvalidOid;
641 continue;
642 }
643 /* Matching RHS found, so done */
644 result = true;
645 break;
646 }
647 }
648 }
649
650 ReleaseSysCacheList(catlist);
651
652 return result;
653}
654
655/*
656 * get_op_index_interpretation
657 * Given an operator's OID, find out which amcanorder opfamilies it belongs to,
658 * and what properties it has within each one. The results are returned
659 * as a palloc'd list of OpIndexInterpretation structs.
660 *
661 * In addition to the normal btree operators, we consider a <> operator to be
662 * a "member" of an opfamily if its negator is an equality operator of the
663 * opfamily. COMPARE_NE is returned as the strategy number for this case.
664 */
665List *
667{
668 List *result = NIL;
669 OpIndexInterpretation *thisresult;
670 CatCList *catlist;
671 int i;
672
673 /*
674 * Find all the pg_amop entries containing the operator.
675 */
676 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno));
677
678 for (i = 0; i < catlist->n_members; i++)
679 {
680 HeapTuple op_tuple = &catlist->members[i]->tuple;
681 Form_pg_amop op_form = (Form_pg_amop) GETSTRUCT(op_tuple);
682 CompareType cmptype;
683
684 /* must be ordering index */
685 if (!get_opmethod_canorder(op_form->amopmethod))
686 continue;
687
688 /* Get the operator's comparision type */
689 cmptype = IndexAmTranslateStrategy(op_form->amopstrategy,
690 op_form->amopmethod,
691 op_form->amopfamily,
692 true);
693
694 /* should not happen */
695 if (cmptype == COMPARE_INVALID)
696 continue;
697
699 thisresult->opfamily_id = op_form->amopfamily;
700 thisresult->cmptype = cmptype;
701 thisresult->oplefttype = op_form->amoplefttype;
702 thisresult->oprighttype = op_form->amoprighttype;
703 result = lappend(result, thisresult);
704 }
705
706 ReleaseSysCacheList(catlist);
707
708 /*
709 * If we didn't find any btree opfamily containing the operator, perhaps
710 * it is a <> operator. See if it has a negator that is in an opfamily.
711 */
712 if (result == NIL)
713 {
714 Oid op_negator = get_negator(opno);
715
716 if (OidIsValid(op_negator))
717 {
718 catlist = SearchSysCacheList1(AMOPOPID,
719 ObjectIdGetDatum(op_negator));
720
721 for (i = 0; i < catlist->n_members; i++)
722 {
723 HeapTuple op_tuple = &catlist->members[i]->tuple;
724 Form_pg_amop op_form = (Form_pg_amop) GETSTRUCT(op_tuple);
725 const IndexAmRoutine *amroutine = GetIndexAmRoutineByAmId(op_form->amopmethod, false);
726 CompareType cmptype;
727
728 /* must be ordering index */
729 if (!amroutine->amcanorder)
730 continue;
731
732 /* Get the operator's comparision type */
733 cmptype = IndexAmTranslateStrategy(op_form->amopstrategy,
734 op_form->amopmethod,
735 op_form->amopfamily,
736 true);
737
738 /* Only consider negators that are = */
739 if (cmptype != COMPARE_EQ)
740 continue;
741
742 /* OK, report it as COMPARE_NE */
744 thisresult->opfamily_id = op_form->amopfamily;
745 thisresult->cmptype = COMPARE_NE;
746 thisresult->oplefttype = op_form->amoplefttype;
747 thisresult->oprighttype = op_form->amoprighttype;
748 result = lappend(result, thisresult);
749 }
750
751 ReleaseSysCacheList(catlist);
752 }
753 }
754
755 return result;
756}
757
758/*
759 * equality_ops_are_compatible
760 * Return true if the two given equality operators have compatible
761 * semantics.
762 *
763 * This is trivially true if they are the same operator. Otherwise,
764 * we look to see if they both belong to an opfamily that guarantees
765 * compatible semantics for equality. Either finding allows us to assume
766 * that they have compatible notions of equality. (The reason we need
767 * to do these pushups is that one might be a cross-type operator; for
768 * instance int24eq vs int4eq.)
769 */
770bool
772{
773 bool result;
774 CatCList *catlist;
775 int i;
776
777 /* Easy if they're the same operator */
778 if (opno1 == opno2)
779 return true;
780
781 /*
782 * We search through all the pg_amop entries for opno1.
783 */
784 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno1));
785
786 result = false;
787 for (i = 0; i < catlist->n_members; i++)
788 {
789 HeapTuple op_tuple = &catlist->members[i]->tuple;
790 Form_pg_amop op_form = (Form_pg_amop) GETSTRUCT(op_tuple);
791
792 /*
793 * op_in_opfamily() is cheaper than GetIndexAmRoutineByAmId(), so
794 * check it first
795 */
796 if (op_in_opfamily(opno2, op_form->amopfamily) &&
797 GetIndexAmRoutineByAmId(op_form->amopmethod, false)->amconsistentequality)
798 {
799 result = true;
800 break;
801 }
802 }
803
804 ReleaseSysCacheList(catlist);
805
806 return result;
807}
808
809/*
810 * comparison_ops_are_compatible
811 * Return true if the two given comparison operators have compatible
812 * semantics.
813 *
814 * This is trivially true if they are the same operator. Otherwise, we look
815 * to see if they both belong to an opfamily that guarantees compatible
816 * semantics for ordering. (For example, for btree, '<' and '>=' ops match if
817 * they belong to the same family.)
818 *
819 * (This is identical to equality_ops_are_compatible(), except that we check
820 * amconsistentordering instead of amconsistentequality.)
821 */
822bool
824{
825 bool result;
826 CatCList *catlist;
827 int i;
828
829 /* Easy if they're the same operator */
830 if (opno1 == opno2)
831 return true;
832
833 /*
834 * We search through all the pg_amop entries for opno1.
835 */
836 catlist = SearchSysCacheList1(AMOPOPID, ObjectIdGetDatum(opno1));
837
838 result = false;
839 for (i = 0; i < catlist->n_members; i++)
840 {
841 HeapTuple op_tuple = &catlist->members[i]->tuple;
842 Form_pg_amop op_form = (Form_pg_amop) GETSTRUCT(op_tuple);
843
844 /*
845 * op_in_opfamily() is cheaper than GetIndexAmRoutineByAmId(), so
846 * check it first
847 */
848 if (op_in_opfamily(opno2, op_form->amopfamily) &&
849 GetIndexAmRoutineByAmId(op_form->amopmethod, false)->amconsistentordering)
850 {
851 result = true;
852 break;
853 }
854 }
855
856 ReleaseSysCacheList(catlist);
857
858 return result;
859}
860
861
862/* ---------- AMPROC CACHES ---------- */
863
864/*
865 * get_opfamily_proc
866 * Get the OID of the specified support function
867 * for the specified opfamily and datatypes.
868 *
869 * Returns InvalidOid if there is no pg_amproc entry for the given keys.
870 */
871Oid
872get_opfamily_proc(Oid opfamily, Oid lefttype, Oid righttype, int16 procnum)
873{
874 HeapTuple tp;
875 Form_pg_amproc amproc_tup;
876 RegProcedure result;
877
878 tp = SearchSysCache4(AMPROCNUM,
879 ObjectIdGetDatum(opfamily),
880 ObjectIdGetDatum(lefttype),
881 ObjectIdGetDatum(righttype),
882 Int16GetDatum(procnum));
883 if (!HeapTupleIsValid(tp))
884 return InvalidOid;
885 amproc_tup = (Form_pg_amproc) GETSTRUCT(tp);
886 result = amproc_tup->amproc;
887 ReleaseSysCache(tp);
888 return result;
889}
890
891
892/* ---------- ATTRIBUTE CACHES ---------- */
893
894/*
895 * get_attname
896 * Given the relation id and the attribute number, return the "attname"
897 * field from the attribute relation as a palloc'ed string.
898 *
899 * If no such attribute exists and missing_ok is true, NULL is returned;
900 * otherwise a not-intended-for-user-consumption error is thrown.
901 */
902char *
903get_attname(Oid relid, AttrNumber attnum, bool missing_ok)
904{
905 HeapTuple tp;
906
907 tp = SearchSysCache2(ATTNUM,
909 if (HeapTupleIsValid(tp))
910 {
912 char *result;
913
914 result = pstrdup(NameStr(att_tup->attname));
915 ReleaseSysCache(tp);
916 return result;
917 }
918
919 if (!missing_ok)
920 elog(ERROR, "cache lookup failed for attribute %d of relation %u",
921 attnum, relid);
922 return NULL;
923}
924
925/*
926 * get_attnum
927 *
928 * Given the relation id and the attribute name,
929 * return the "attnum" field from the attribute relation.
930 *
931 * Returns InvalidAttrNumber if the attr doesn't exist (or is dropped).
932 */
934get_attnum(Oid relid, const char *attname)
935{
936 HeapTuple tp;
937
938 tp = SearchSysCacheAttName(relid, attname);
939 if (HeapTupleIsValid(tp))
940 {
942 AttrNumber result;
943
944 result = att_tup->attnum;
945 ReleaseSysCache(tp);
946 return result;
947 }
948 else
949 return InvalidAttrNumber;
950}
951
952/*
953 * get_attgenerated
954 *
955 * Given the relation id and the attribute number,
956 * return the "attgenerated" field from the attribute relation.
957 *
958 * Errors if not found.
959 *
960 * Since not generated is represented by '\0', this can also be used as a
961 * Boolean test.
962 */
963char
965{
966 HeapTuple tp;
967 Form_pg_attribute att_tup;
968 char result;
969
970 tp = SearchSysCache2(ATTNUM,
971 ObjectIdGetDatum(relid),
973 if (!HeapTupleIsValid(tp))
974 elog(ERROR, "cache lookup failed for attribute %d of relation %u",
975 attnum, relid);
976 att_tup = (Form_pg_attribute) GETSTRUCT(tp);
977 result = att_tup->attgenerated;
978 ReleaseSysCache(tp);
979 return result;
980}
981
982/*
983 * get_atttype
984 *
985 * Given the relation OID and the attribute number with the relation,
986 * return the attribute type OID.
987 */
988Oid
990{
991 HeapTuple tp;
992
993 tp = SearchSysCache2(ATTNUM,
994 ObjectIdGetDatum(relid),
996 if (HeapTupleIsValid(tp))
997 {
999 Oid result;
1000
1001 result = att_tup->atttypid;
1002 ReleaseSysCache(tp);
1003 return result;
1004 }
1005 else
1006 return InvalidOid;
1007}
1008
1009/*
1010 * get_atttypetypmodcoll
1011 *
1012 * A three-fer: given the relation id and the attribute number,
1013 * fetch atttypid, atttypmod, and attcollation in a single cache lookup.
1014 *
1015 * Unlike the otherwise-similar get_atttype, this routine
1016 * raises an error if it can't obtain the information.
1017 */
1018void
1020 Oid *typid, int32 *typmod, Oid *collid)
1021{
1022 HeapTuple tp;
1023 Form_pg_attribute att_tup;
1024
1025 tp = SearchSysCache2(ATTNUM,
1026 ObjectIdGetDatum(relid),
1028 if (!HeapTupleIsValid(tp))
1029 elog(ERROR, "cache lookup failed for attribute %d of relation %u",
1030 attnum, relid);
1031 att_tup = (Form_pg_attribute) GETSTRUCT(tp);
1032
1033 *typid = att_tup->atttypid;
1034 *typmod = att_tup->atttypmod;
1035 *collid = att_tup->attcollation;
1036 ReleaseSysCache(tp);
1037}
1038
1039/*
1040 * get_attoptions
1041 *
1042 * Given the relation id and the attribute number,
1043 * return the attribute options text[] datum, if any.
1044 */
1045Datum
1047{
1048 HeapTuple tuple;
1049 Datum attopts;
1050 Datum result;
1051 bool isnull;
1052
1053 tuple = SearchSysCache2(ATTNUM,
1054 ObjectIdGetDatum(relid),
1056
1057 if (!HeapTupleIsValid(tuple))
1058 elog(ERROR, "cache lookup failed for attribute %d of relation %u",
1059 attnum, relid);
1060
1061 attopts = SysCacheGetAttr(ATTNAME, tuple, Anum_pg_attribute_attoptions,
1062 &isnull);
1063
1064 if (isnull)
1065 result = (Datum) 0;
1066 else
1067 result = datumCopy(attopts, false, -1); /* text[] */
1068
1069 ReleaseSysCache(tuple);
1070
1071 return result;
1072}
1073
1074/* ---------- PG_CAST CACHE ---------- */
1075
1076/*
1077 * get_cast_oid - given two type OIDs, look up a cast OID
1078 *
1079 * If missing_ok is false, throw an error if the cast is not found. If
1080 * true, just return InvalidOid.
1081 */
1082Oid
1083get_cast_oid(Oid sourcetypeid, Oid targettypeid, bool missing_ok)
1084{
1085 Oid oid;
1086
1087 oid = GetSysCacheOid2(CASTSOURCETARGET, Anum_pg_cast_oid,
1088 ObjectIdGetDatum(sourcetypeid),
1089 ObjectIdGetDatum(targettypeid));
1090 if (!OidIsValid(oid) && !missing_ok)
1091 ereport(ERROR,
1092 (errcode(ERRCODE_UNDEFINED_OBJECT),
1093 errmsg("cast from type %s to type %s does not exist",
1094 format_type_be(sourcetypeid),
1095 format_type_be(targettypeid))));
1096 return oid;
1097}
1098
1099/* ---------- COLLATION CACHE ---------- */
1100
1101/*
1102 * get_collation_name
1103 * Returns the name of a given pg_collation entry.
1104 *
1105 * Returns a palloc'd copy of the string, or NULL if no such collation.
1106 *
1107 * NOTE: since collation name is not unique, be wary of code that uses this
1108 * for anything except preparing error messages.
1109 */
1110char *
1112{
1113 HeapTuple tp;
1114
1115 tp = SearchSysCache1(COLLOID, ObjectIdGetDatum(colloid));
1116 if (HeapTupleIsValid(tp))
1117 {
1119 char *result;
1120
1121 result = pstrdup(NameStr(colltup->collname));
1122 ReleaseSysCache(tp);
1123 return result;
1124 }
1125 else
1126 return NULL;
1127}
1128
1129bool
1131{
1132 HeapTuple tp;
1133 Form_pg_collation colltup;
1134 bool result;
1135
1136 tp = SearchSysCache1(COLLOID, ObjectIdGetDatum(colloid));
1137 if (!HeapTupleIsValid(tp))
1138 elog(ERROR, "cache lookup failed for collation %u", colloid);
1139 colltup = (Form_pg_collation) GETSTRUCT(tp);
1140 result = colltup->collisdeterministic;
1141 ReleaseSysCache(tp);
1142 return result;
1143}
1144
1145/* ---------- CONSTRAINT CACHE ---------- */
1146
1147/*
1148 * get_constraint_name
1149 * Returns the name of a given pg_constraint entry.
1150 *
1151 * Returns a palloc'd copy of the string, or NULL if no such constraint.
1152 *
1153 * NOTE: since constraint name is not unique, be wary of code that uses this
1154 * for anything except preparing error messages.
1155 */
1156char *
1158{
1159 HeapTuple tp;
1160
1161 tp = SearchSysCache1(CONSTROID, ObjectIdGetDatum(conoid));
1162 if (HeapTupleIsValid(tp))
1163 {
1165 char *result;
1166
1167 result = pstrdup(NameStr(contup->conname));
1168 ReleaseSysCache(tp);
1169 return result;
1170 }
1171 else
1172 return NULL;
1173}
1174
1175/*
1176 * get_constraint_index
1177 * Given the OID of a unique, primary-key, or exclusion constraint,
1178 * return the OID of the underlying index.
1179 *
1180 * Returns InvalidOid if the constraint could not be found or is of
1181 * the wrong type.
1182 *
1183 * The intent of this function is to return the index "owned" by the
1184 * specified constraint. Therefore we must check contype, since some
1185 * pg_constraint entries (e.g. for foreign-key constraints) store the
1186 * OID of an index that is referenced but not owned by the constraint.
1187 */
1188Oid
1190{
1191 HeapTuple tp;
1192
1193 tp = SearchSysCache1(CONSTROID, ObjectIdGetDatum(conoid));
1194 if (HeapTupleIsValid(tp))
1195 {
1197 Oid result;
1198
1199 if (contup->contype == CONSTRAINT_UNIQUE ||
1200 contup->contype == CONSTRAINT_PRIMARY ||
1201 contup->contype == CONSTRAINT_EXCLUSION)
1202 result = contup->conindid;
1203 else
1204 result = InvalidOid;
1205 ReleaseSysCache(tp);
1206 return result;
1207 }
1208 else
1209 return InvalidOid;
1210}
1211
1212/*
1213 * get_constraint_type
1214 * Return the pg_constraint.contype value for the given constraint.
1215 *
1216 * No frills.
1217 */
1218char
1220{
1221 HeapTuple tp;
1222 char contype;
1223
1224 tp = SearchSysCache1(CONSTROID, ObjectIdGetDatum(conoid));
1225 if (!HeapTupleIsValid(tp))
1226 elog(ERROR, "cache lookup failed for constraint %u", conoid);
1227
1228 contype = ((Form_pg_constraint) GETSTRUCT(tp))->contype;
1229 ReleaseSysCache(tp);
1230
1231 return contype;
1232}
1233
1234/* ---------- DATABASE CACHE ---------- */
1235
1236/*
1237 * get_database_name - given a database OID, look up the name
1238 *
1239 * Returns a palloc'd string, or NULL if no such database.
1240 */
1241char *
1243{
1244 HeapTuple dbtuple;
1245 char *result;
1246
1247 dbtuple = SearchSysCache1(DATABASEOID, ObjectIdGetDatum(dbid));
1248 if (HeapTupleIsValid(dbtuple))
1249 {
1250 result = pstrdup(NameStr(((Form_pg_database) GETSTRUCT(dbtuple))->datname));
1251 ReleaseSysCache(dbtuple);
1252 }
1253 else
1254 result = NULL;
1255
1256 return result;
1257}
1258
1259
1260/* ---------- LANGUAGE CACHE ---------- */
1261
1262char *
1263get_language_name(Oid langoid, bool missing_ok)
1264{
1265 HeapTuple tp;
1266
1267 tp = SearchSysCache1(LANGOID, ObjectIdGetDatum(langoid));
1268 if (HeapTupleIsValid(tp))
1269 {
1271 char *result;
1272
1273 result = pstrdup(NameStr(lantup->lanname));
1274 ReleaseSysCache(tp);
1275 return result;
1276 }
1277
1278 if (!missing_ok)
1279 elog(ERROR, "cache lookup failed for language %u",
1280 langoid);
1281 return NULL;
1282}
1283
1284/* ---------- OPCLASS CACHE ---------- */
1285
1286/*
1287 * get_opclass_family
1288 *
1289 * Returns the OID of the operator family the opclass belongs to.
1290 */
1291Oid
1293{
1294 HeapTuple tp;
1295 Form_pg_opclass cla_tup;
1296 Oid result;
1297
1298 tp = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclass));
1299 if (!HeapTupleIsValid(tp))
1300 elog(ERROR, "cache lookup failed for opclass %u", opclass);
1301 cla_tup = (Form_pg_opclass) GETSTRUCT(tp);
1302
1303 result = cla_tup->opcfamily;
1304 ReleaseSysCache(tp);
1305 return result;
1306}
1307
1308/*
1309 * get_opclass_input_type
1310 *
1311 * Returns the OID of the datatype the opclass indexes.
1312 */
1313Oid
1315{
1316 HeapTuple tp;
1317 Form_pg_opclass cla_tup;
1318 Oid result;
1319
1320 tp = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclass));
1321 if (!HeapTupleIsValid(tp))
1322 elog(ERROR, "cache lookup failed for opclass %u", opclass);
1323 cla_tup = (Form_pg_opclass) GETSTRUCT(tp);
1324
1325 result = cla_tup->opcintype;
1326 ReleaseSysCache(tp);
1327 return result;
1328}
1329
1330/*
1331 * get_opclass_opfamily_and_input_type
1332 *
1333 * Returns the OID of the operator family the opclass belongs to,
1334 * the OID of the datatype the opclass indexes
1335 */
1336bool
1337get_opclass_opfamily_and_input_type(Oid opclass, Oid *opfamily, Oid *opcintype)
1338{
1339 HeapTuple tp;
1340 Form_pg_opclass cla_tup;
1341
1342 tp = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclass));
1343 if (!HeapTupleIsValid(tp))
1344 return false;
1345
1346 cla_tup = (Form_pg_opclass) GETSTRUCT(tp);
1347
1348 *opfamily = cla_tup->opcfamily;
1349 *opcintype = cla_tup->opcintype;
1350
1351 ReleaseSysCache(tp);
1352
1353 return true;
1354}
1355
1356/*
1357 * get_opclass_method
1358 *
1359 * Returns the OID of the index access method the opclass belongs to.
1360 */
1361Oid
1363{
1364 HeapTuple tp;
1365 Form_pg_opclass cla_tup;
1366 Oid result;
1367
1368 tp = SearchSysCache1(CLAOID, ObjectIdGetDatum(opclass));
1369 if (!HeapTupleIsValid(tp))
1370 elog(ERROR, "cache lookup failed for opclass %u", opclass);
1371 cla_tup = (Form_pg_opclass) GETSTRUCT(tp);
1372
1373 result = cla_tup->opcmethod;
1374 ReleaseSysCache(tp);
1375 return result;
1376}
1377
1378/* ---------- OPFAMILY CACHE ---------- */
1379
1380/*
1381 * get_opfamily_method
1382 *
1383 * Returns the OID of the index access method the opfamily is for.
1384 */
1385Oid
1387{
1388 HeapTuple tp;
1389 Form_pg_opfamily opfform;
1390 Oid result;
1391
1392 tp = SearchSysCache1(OPFAMILYOID, ObjectIdGetDatum(opfid));
1393 if (!HeapTupleIsValid(tp))
1394 elog(ERROR, "cache lookup failed for operator family %u", opfid);
1395 opfform = (Form_pg_opfamily) GETSTRUCT(tp);
1396
1397 result = opfform->opfmethod;
1398 ReleaseSysCache(tp);
1399 return result;
1400}
1401
1402char *
1403get_opfamily_name(Oid opfid, bool missing_ok)
1404{
1405 HeapTuple tup;
1406 char *opfname;
1407 Form_pg_opfamily opfform;
1408
1409 tup = SearchSysCache1(OPFAMILYOID, ObjectIdGetDatum(opfid));
1410
1411 if (!HeapTupleIsValid(tup))
1412 {
1413 if (!missing_ok)
1414 elog(ERROR, "cache lookup failed for operator family %u", opfid);
1415 return NULL;
1416 }
1417
1418 opfform = (Form_pg_opfamily) GETSTRUCT(tup);
1419 opfname = pstrdup(NameStr(opfform->opfname));
1420
1421 ReleaseSysCache(tup);
1422
1423 return opfname;
1424}
1425
1426/* ---------- OPERATOR CACHE ---------- */
1427
1428/*
1429 * get_opcode
1430 *
1431 * Returns the regproc id of the routine used to implement an
1432 * operator given the operator oid.
1433 */
1436{
1437 HeapTuple tp;
1438
1439 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1440 if (HeapTupleIsValid(tp))
1441 {
1443 RegProcedure result;
1444
1445 result = optup->oprcode;
1446 ReleaseSysCache(tp);
1447 return result;
1448 }
1449 else
1450 return (RegProcedure) InvalidOid;
1451}
1452
1453/*
1454 * get_opname
1455 * returns the name of the operator with the given opno
1456 *
1457 * Note: returns a palloc'd copy of the string, or NULL if no such operator.
1458 */
1459char *
1461{
1462 HeapTuple tp;
1463
1464 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1465 if (HeapTupleIsValid(tp))
1466 {
1468 char *result;
1469
1470 result = pstrdup(NameStr(optup->oprname));
1471 ReleaseSysCache(tp);
1472 return result;
1473 }
1474 else
1475 return NULL;
1476}
1477
1478/*
1479 * get_op_rettype
1480 * Given operator oid, return the operator's result type.
1481 */
1482Oid
1484{
1485 HeapTuple tp;
1486
1487 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1488 if (HeapTupleIsValid(tp))
1489 {
1491 Oid result;
1492
1493 result = optup->oprresult;
1494 ReleaseSysCache(tp);
1495 return result;
1496 }
1497 else
1498 return InvalidOid;
1499}
1500
1501/*
1502 * op_input_types
1503 *
1504 * Returns the left and right input datatypes for an operator
1505 * (InvalidOid if not relevant).
1506 */
1507void
1508op_input_types(Oid opno, Oid *lefttype, Oid *righttype)
1509{
1510 HeapTuple tp;
1511 Form_pg_operator optup;
1512
1513 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1514 if (!HeapTupleIsValid(tp)) /* shouldn't happen */
1515 elog(ERROR, "cache lookup failed for operator %u", opno);
1516 optup = (Form_pg_operator) GETSTRUCT(tp);
1517 *lefttype = optup->oprleft;
1518 *righttype = optup->oprright;
1519 ReleaseSysCache(tp);
1520}
1521
1522/*
1523 * op_mergejoinable
1524 *
1525 * Returns true if the operator is potentially mergejoinable. (The planner
1526 * will fail to find any mergejoin plans unless there are suitable btree
1527 * opfamily entries for this operator and associated sortops. The pg_operator
1528 * flag is just a hint to tell the planner whether to bother looking.)
1529 *
1530 * In some cases (currently only array_eq and record_eq), mergejoinability
1531 * depends on the specific input data type the operator is invoked for, so
1532 * that must be passed as well. We currently assume that only one input's type
1533 * is needed to check this --- by convention, pass the left input's data type.
1534 */
1535bool
1536op_mergejoinable(Oid opno, Oid inputtype)
1537{
1538 bool result = false;
1539 HeapTuple tp;
1540 TypeCacheEntry *typentry;
1541
1542 /*
1543 * For array_eq or record_eq, we can sort if the element or field types
1544 * are all sortable. We could implement all the checks for that here, but
1545 * the typcache already does that and caches the results too, so let's
1546 * rely on the typcache.
1547 */
1548 if (opno == ARRAY_EQ_OP)
1549 {
1550 typentry = lookup_type_cache(inputtype, TYPECACHE_CMP_PROC);
1551 if (typentry->cmp_proc == F_BTARRAYCMP)
1552 result = true;
1553 }
1554 else if (opno == RECORD_EQ_OP)
1555 {
1556 typentry = lookup_type_cache(inputtype, TYPECACHE_CMP_PROC);
1557 if (typentry->cmp_proc == F_BTRECORDCMP)
1558 result = true;
1559 }
1560 else
1561 {
1562 /* For all other operators, rely on pg_operator.oprcanmerge */
1563 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1564 if (HeapTupleIsValid(tp))
1565 {
1567
1568 result = optup->oprcanmerge;
1569 ReleaseSysCache(tp);
1570 }
1571 }
1572 return result;
1573}
1574
1575/*
1576 * op_hashjoinable
1577 *
1578 * Returns true if the operator is hashjoinable. (There must be a suitable
1579 * hash opfamily entry for this operator if it is so marked.)
1580 *
1581 * In some cases (currently only array_eq), hashjoinability depends on the
1582 * specific input data type the operator is invoked for, so that must be
1583 * passed as well. We currently assume that only one input's type is needed
1584 * to check this --- by convention, pass the left input's data type.
1585 */
1586bool
1587op_hashjoinable(Oid opno, Oid inputtype)
1588{
1589 bool result = false;
1590 HeapTuple tp;
1591 TypeCacheEntry *typentry;
1592
1593 /* As in op_mergejoinable, let the typcache handle the hard cases */
1594 if (opno == ARRAY_EQ_OP)
1595 {
1596 typentry = lookup_type_cache(inputtype, TYPECACHE_HASH_PROC);
1597 if (typentry->hash_proc == F_HASH_ARRAY)
1598 result = true;
1599 }
1600 else if (opno == RECORD_EQ_OP)
1601 {
1602 typentry = lookup_type_cache(inputtype, TYPECACHE_HASH_PROC);
1603 if (typentry->hash_proc == F_HASH_RECORD)
1604 result = true;
1605 }
1606 else
1607 {
1608 /* For all other operators, rely on pg_operator.oprcanhash */
1609 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1610 if (HeapTupleIsValid(tp))
1611 {
1613
1614 result = optup->oprcanhash;
1615 ReleaseSysCache(tp);
1616 }
1617 }
1618 return result;
1619}
1620
1621/*
1622 * op_strict
1623 *
1624 * Get the proisstrict flag for the operator's underlying function.
1625 */
1626bool
1628{
1629 RegProcedure funcid = get_opcode(opno);
1630
1631 if (funcid == (RegProcedure) InvalidOid)
1632 elog(ERROR, "operator %u does not exist", opno);
1633
1634 return func_strict((Oid) funcid);
1635}
1636
1637/*
1638 * op_volatile
1639 *
1640 * Get the provolatile flag for the operator's underlying function.
1641 */
1642char
1644{
1645 RegProcedure funcid = get_opcode(opno);
1646
1647 if (funcid == (RegProcedure) InvalidOid)
1648 elog(ERROR, "operator %u does not exist", opno);
1649
1650 return func_volatile((Oid) funcid);
1651}
1652
1653/*
1654 * get_commutator
1655 *
1656 * Returns the corresponding commutator of an operator.
1657 */
1658Oid
1660{
1661 HeapTuple tp;
1662
1663 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1664 if (HeapTupleIsValid(tp))
1665 {
1667 Oid result;
1668
1669 result = optup->oprcom;
1670 ReleaseSysCache(tp);
1671 return result;
1672 }
1673 else
1674 return InvalidOid;
1675}
1676
1677/*
1678 * get_negator
1679 *
1680 * Returns the corresponding negator of an operator.
1681 */
1682Oid
1684{
1685 HeapTuple tp;
1686
1687 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1688 if (HeapTupleIsValid(tp))
1689 {
1691 Oid result;
1692
1693 result = optup->oprnegate;
1694 ReleaseSysCache(tp);
1695 return result;
1696 }
1697 else
1698 return InvalidOid;
1699}
1700
1701/*
1702 * get_oprrest
1703 *
1704 * Returns procedure id for computing selectivity of an operator.
1705 */
1708{
1709 HeapTuple tp;
1710
1711 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1712 if (HeapTupleIsValid(tp))
1713 {
1715 RegProcedure result;
1716
1717 result = optup->oprrest;
1718 ReleaseSysCache(tp);
1719 return result;
1720 }
1721 else
1722 return (RegProcedure) InvalidOid;
1723}
1724
1725/*
1726 * get_oprjoin
1727 *
1728 * Returns procedure id for computing selectivity of a join.
1729 */
1732{
1733 HeapTuple tp;
1734
1735 tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
1736 if (HeapTupleIsValid(tp))
1737 {
1739 RegProcedure result;
1740
1741 result = optup->oprjoin;
1742 ReleaseSysCache(tp);
1743 return result;
1744 }
1745 else
1746 return (RegProcedure) InvalidOid;
1747}
1748
1749/* ---------- FUNCTION CACHE ---------- */
1750
1751/*
1752 * get_func_name
1753 * returns the name of the function with the given funcid
1754 *
1755 * Note: returns a palloc'd copy of the string, or NULL if no such function.
1756 */
1757char *
1759{
1760 HeapTuple tp;
1761
1762 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1763 if (HeapTupleIsValid(tp))
1764 {
1765 Form_pg_proc functup = (Form_pg_proc) GETSTRUCT(tp);
1766 char *result;
1767
1768 result = pstrdup(NameStr(functup->proname));
1769 ReleaseSysCache(tp);
1770 return result;
1771 }
1772 else
1773 return NULL;
1774}
1775
1776/*
1777 * get_func_namespace
1778 *
1779 * Returns the pg_namespace OID associated with a given function.
1780 */
1781Oid
1783{
1784 HeapTuple tp;
1785
1786 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1787 if (HeapTupleIsValid(tp))
1788 {
1789 Form_pg_proc functup = (Form_pg_proc) GETSTRUCT(tp);
1790 Oid result;
1791
1792 result = functup->pronamespace;
1793 ReleaseSysCache(tp);
1794 return result;
1795 }
1796 else
1797 return InvalidOid;
1798}
1799
1800/*
1801 * get_func_rettype
1802 * Given procedure id, return the function's result type.
1803 */
1804Oid
1806{
1807 HeapTuple tp;
1808 Oid result;
1809
1810 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1811 if (!HeapTupleIsValid(tp))
1812 elog(ERROR, "cache lookup failed for function %u", funcid);
1813
1814 result = ((Form_pg_proc) GETSTRUCT(tp))->prorettype;
1815 ReleaseSysCache(tp);
1816 return result;
1817}
1818
1819/*
1820 * get_func_nargs
1821 * Given procedure id, return the number of arguments.
1822 */
1823int
1825{
1826 HeapTuple tp;
1827 int result;
1828
1829 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1830 if (!HeapTupleIsValid(tp))
1831 elog(ERROR, "cache lookup failed for function %u", funcid);
1832
1833 result = ((Form_pg_proc) GETSTRUCT(tp))->pronargs;
1834 ReleaseSysCache(tp);
1835 return result;
1836}
1837
1838/*
1839 * get_func_signature
1840 * Given procedure id, return the function's argument and result types.
1841 * (The return value is the result type.)
1842 *
1843 * The arguments are returned as a palloc'd array.
1844 */
1845Oid
1846get_func_signature(Oid funcid, Oid **argtypes, int *nargs)
1847{
1848 HeapTuple tp;
1849 Form_pg_proc procstruct;
1850 Oid result;
1851
1852 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1853 if (!HeapTupleIsValid(tp))
1854 elog(ERROR, "cache lookup failed for function %u", funcid);
1855
1856 procstruct = (Form_pg_proc) GETSTRUCT(tp);
1857
1858 result = procstruct->prorettype;
1859 *nargs = (int) procstruct->pronargs;
1860 Assert(*nargs == procstruct->proargtypes.dim1);
1861 *argtypes = (Oid *) palloc(*nargs * sizeof(Oid));
1862 memcpy(*argtypes, procstruct->proargtypes.values, *nargs * sizeof(Oid));
1863
1864 ReleaseSysCache(tp);
1865 return result;
1866}
1867
1868/*
1869 * get_func_variadictype
1870 * Given procedure id, return the function's provariadic field.
1871 */
1872Oid
1874{
1875 HeapTuple tp;
1876 Oid result;
1877
1878 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1879 if (!HeapTupleIsValid(tp))
1880 elog(ERROR, "cache lookup failed for function %u", funcid);
1881
1882 result = ((Form_pg_proc) GETSTRUCT(tp))->provariadic;
1883 ReleaseSysCache(tp);
1884 return result;
1885}
1886
1887/*
1888 * get_func_retset
1889 * Given procedure id, return the function's proretset flag.
1890 */
1891bool
1893{
1894 HeapTuple tp;
1895 bool result;
1896
1897 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1898 if (!HeapTupleIsValid(tp))
1899 elog(ERROR, "cache lookup failed for function %u", funcid);
1900
1901 result = ((Form_pg_proc) GETSTRUCT(tp))->proretset;
1902 ReleaseSysCache(tp);
1903 return result;
1904}
1905
1906/*
1907 * func_strict
1908 * Given procedure id, return the function's proisstrict flag.
1909 */
1910bool
1912{
1913 HeapTuple tp;
1914 bool result;
1915
1916 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1917 if (!HeapTupleIsValid(tp))
1918 elog(ERROR, "cache lookup failed for function %u", funcid);
1919
1920 result = ((Form_pg_proc) GETSTRUCT(tp))->proisstrict;
1921 ReleaseSysCache(tp);
1922 return result;
1923}
1924
1925/*
1926 * func_volatile
1927 * Given procedure id, return the function's provolatile flag.
1928 */
1929char
1931{
1932 HeapTuple tp;
1933 char result;
1934
1935 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1936 if (!HeapTupleIsValid(tp))
1937 elog(ERROR, "cache lookup failed for function %u", funcid);
1938
1939 result = ((Form_pg_proc) GETSTRUCT(tp))->provolatile;
1940 ReleaseSysCache(tp);
1941 return result;
1942}
1943
1944/*
1945 * func_parallel
1946 * Given procedure id, return the function's proparallel flag.
1947 */
1948char
1950{
1951 HeapTuple tp;
1952 char result;
1953
1954 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1955 if (!HeapTupleIsValid(tp))
1956 elog(ERROR, "cache lookup failed for function %u", funcid);
1957
1958 result = ((Form_pg_proc) GETSTRUCT(tp))->proparallel;
1959 ReleaseSysCache(tp);
1960 return result;
1961}
1962
1963/*
1964 * get_func_prokind
1965 * Given procedure id, return the routine kind.
1966 */
1967char
1969{
1970 HeapTuple tp;
1971 char result;
1972
1973 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1974 if (!HeapTupleIsValid(tp))
1975 elog(ERROR, "cache lookup failed for function %u", funcid);
1976
1977 result = ((Form_pg_proc) GETSTRUCT(tp))->prokind;
1978 ReleaseSysCache(tp);
1979 return result;
1980}
1981
1982/*
1983 * get_func_leakproof
1984 * Given procedure id, return the function's leakproof field.
1985 */
1986bool
1988{
1989 HeapTuple tp;
1990 bool result;
1991
1992 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
1993 if (!HeapTupleIsValid(tp))
1994 elog(ERROR, "cache lookup failed for function %u", funcid);
1995
1996 result = ((Form_pg_proc) GETSTRUCT(tp))->proleakproof;
1997 ReleaseSysCache(tp);
1998 return result;
1999}
2000
2001/*
2002 * get_func_support
2003 *
2004 * Returns the support function OID associated with a given function,
2005 * or InvalidOid if there is none.
2006 */
2009{
2010 HeapTuple tp;
2011
2012 tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
2013 if (HeapTupleIsValid(tp))
2014 {
2015 Form_pg_proc functup = (Form_pg_proc) GETSTRUCT(tp);
2016 RegProcedure result;
2017
2018 result = functup->prosupport;
2019 ReleaseSysCache(tp);
2020 return result;
2021 }
2022 else
2023 return (RegProcedure) InvalidOid;
2024}
2025
2026/* ---------- RELATION CACHE ---------- */
2027
2028/*
2029 * get_relname_relid
2030 * Given name and namespace of a relation, look up the OID.
2031 *
2032 * Returns InvalidOid if there is no such relation.
2033 */
2034Oid
2035get_relname_relid(const char *relname, Oid relnamespace)
2036{
2037 return GetSysCacheOid2(RELNAMENSP, Anum_pg_class_oid,
2039 ObjectIdGetDatum(relnamespace));
2040}
2041
2042#ifdef NOT_USED
2043/*
2044 * get_relnatts
2045 *
2046 * Returns the number of attributes for a given relation.
2047 */
2048int
2049get_relnatts(Oid relid)
2050{
2051 HeapTuple tp;
2052
2053 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2054 if (HeapTupleIsValid(tp))
2055 {
2056 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2057 int result;
2058
2059 result = reltup->relnatts;
2060 ReleaseSysCache(tp);
2061 return result;
2062 }
2063 else
2064 return InvalidAttrNumber;
2065}
2066#endif
2067
2068/*
2069 * get_rel_name
2070 * Returns the name of a given relation.
2071 *
2072 * Returns a palloc'd copy of the string, or NULL if no such relation.
2073 *
2074 * NOTE: since relation name is not unique, be wary of code that uses this
2075 * for anything except preparing error messages.
2076 */
2077char *
2079{
2080 HeapTuple tp;
2081
2082 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2083 if (HeapTupleIsValid(tp))
2084 {
2085 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2086 char *result;
2087
2088 result = pstrdup(NameStr(reltup->relname));
2089 ReleaseSysCache(tp);
2090 return result;
2091 }
2092 else
2093 return NULL;
2094}
2095
2096/*
2097 * get_rel_namespace
2098 *
2099 * Returns the pg_namespace OID associated with a given relation.
2100 */
2101Oid
2103{
2104 HeapTuple tp;
2105
2106 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2107 if (HeapTupleIsValid(tp))
2108 {
2109 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2110 Oid result;
2111
2112 result = reltup->relnamespace;
2113 ReleaseSysCache(tp);
2114 return result;
2115 }
2116 else
2117 return InvalidOid;
2118}
2119
2120/*
2121 * get_rel_type_id
2122 *
2123 * Returns the pg_type OID associated with a given relation.
2124 *
2125 * Note: not all pg_class entries have associated pg_type OIDs; so be
2126 * careful to check for InvalidOid result.
2127 */
2128Oid
2130{
2131 HeapTuple tp;
2132
2133 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2134 if (HeapTupleIsValid(tp))
2135 {
2136 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2137 Oid result;
2138
2139 result = reltup->reltype;
2140 ReleaseSysCache(tp);
2141 return result;
2142 }
2143 else
2144 return InvalidOid;
2145}
2146
2147/*
2148 * get_rel_relkind
2149 *
2150 * Returns the relkind associated with a given relation.
2151 */
2152char
2154{
2155 HeapTuple tp;
2156
2157 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2158 if (HeapTupleIsValid(tp))
2159 {
2160 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2161 char result;
2162
2163 result = reltup->relkind;
2164 ReleaseSysCache(tp);
2165 return result;
2166 }
2167 else
2168 return '\0';
2169}
2170
2171/*
2172 * get_rel_relispartition
2173 *
2174 * Returns the relispartition flag associated with a given relation.
2175 */
2176bool
2178{
2179 HeapTuple tp;
2180
2181 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2182 if (HeapTupleIsValid(tp))
2183 {
2184 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2185 bool result;
2186
2187 result = reltup->relispartition;
2188 ReleaseSysCache(tp);
2189 return result;
2190 }
2191 else
2192 return false;
2193}
2194
2195/*
2196 * get_rel_tablespace
2197 *
2198 * Returns the pg_tablespace OID associated with a given relation.
2199 *
2200 * Note: InvalidOid might mean either that we couldn't find the relation,
2201 * or that it is in the database's default tablespace.
2202 */
2203Oid
2205{
2206 HeapTuple tp;
2207
2208 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2209 if (HeapTupleIsValid(tp))
2210 {
2211 Form_pg_class reltup = (Form_pg_class) GETSTRUCT(tp);
2212 Oid result;
2213
2214 result = reltup->reltablespace;
2215 ReleaseSysCache(tp);
2216 return result;
2217 }
2218 else
2219 return InvalidOid;
2220}
2221
2222/*
2223 * get_rel_persistence
2224 *
2225 * Returns the relpersistence associated with a given relation.
2226 */
2227char
2229{
2230 HeapTuple tp;
2231 Form_pg_class reltup;
2232 char result;
2233
2234 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2235 if (!HeapTupleIsValid(tp))
2236 elog(ERROR, "cache lookup failed for relation %u", relid);
2237 reltup = (Form_pg_class) GETSTRUCT(tp);
2238 result = reltup->relpersistence;
2239 ReleaseSysCache(tp);
2240
2241 return result;
2242}
2243
2244/*
2245 * get_rel_relam
2246 *
2247 * Returns the relam associated with a given relation.
2248 */
2249Oid
2251{
2252 HeapTuple tp;
2253 Form_pg_class reltup;
2254 Oid result;
2255
2256 tp = SearchSysCache1(RELOID, ObjectIdGetDatum(relid));
2257 if (!HeapTupleIsValid(tp))
2258 elog(ERROR, "cache lookup failed for relation %u", relid);
2259 reltup = (Form_pg_class) GETSTRUCT(tp);
2260 result = reltup->relam;
2261 ReleaseSysCache(tp);
2262
2263 return result;
2264}
2265
2266
2267/* ---------- TRANSFORM CACHE ---------- */
2268
2269Oid
2270get_transform_fromsql(Oid typid, Oid langid, List *trftypes)
2271{
2272 HeapTuple tup;
2273
2274 if (!list_member_oid(trftypes, typid))
2275 return InvalidOid;
2276
2277 tup = SearchSysCache2(TRFTYPELANG, ObjectIdGetDatum(typid),
2278 ObjectIdGetDatum(langid));
2279 if (HeapTupleIsValid(tup))
2280 {
2281 Oid funcid;
2282
2283 funcid = ((Form_pg_transform) GETSTRUCT(tup))->trffromsql;
2284 ReleaseSysCache(tup);
2285 return funcid;
2286 }
2287 else
2288 return InvalidOid;
2289}
2290
2291Oid
2292get_transform_tosql(Oid typid, Oid langid, List *trftypes)
2293{
2294 HeapTuple tup;
2295
2296 if (!list_member_oid(trftypes, typid))
2297 return InvalidOid;
2298
2299 tup = SearchSysCache2(TRFTYPELANG, ObjectIdGetDatum(typid),
2300 ObjectIdGetDatum(langid));
2301 if (HeapTupleIsValid(tup))
2302 {
2303 Oid funcid;
2304
2305 funcid = ((Form_pg_transform) GETSTRUCT(tup))->trftosql;
2306 ReleaseSysCache(tup);
2307 return funcid;
2308 }
2309 else
2310 return InvalidOid;
2311}
2312
2313
2314/* ---------- TYPE CACHE ---------- */
2315
2316/*
2317 * get_typisdefined
2318 *
2319 * Given the type OID, determine whether the type is defined
2320 * (if not, it's only a shell).
2321 */
2322bool
2324{
2325 HeapTuple tp;
2326
2327 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2328 if (HeapTupleIsValid(tp))
2329 {
2330 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2331 bool result;
2332
2333 result = typtup->typisdefined;
2334 ReleaseSysCache(tp);
2335 return result;
2336 }
2337 else
2338 return false;
2339}
2340
2341/*
2342 * get_typlen
2343 *
2344 * Given the type OID, return the length of the type.
2345 */
2346int16
2348{
2349 HeapTuple tp;
2350
2351 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2352 if (HeapTupleIsValid(tp))
2353 {
2354 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2355 int16 result;
2356
2357 result = typtup->typlen;
2358 ReleaseSysCache(tp);
2359 return result;
2360 }
2361 else
2362 return 0;
2363}
2364
2365/*
2366 * get_typbyval
2367 *
2368 * Given the type OID, determine whether the type is returned by value or
2369 * not. Returns true if by value, false if by reference.
2370 */
2371bool
2373{
2374 HeapTuple tp;
2375
2376 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2377 if (HeapTupleIsValid(tp))
2378 {
2379 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2380 bool result;
2381
2382 result = typtup->typbyval;
2383 ReleaseSysCache(tp);
2384 return result;
2385 }
2386 else
2387 return false;
2388}
2389
2390/*
2391 * get_typlenbyval
2392 *
2393 * A two-fer: given the type OID, return both typlen and typbyval.
2394 *
2395 * Since both pieces of info are needed to know how to copy a Datum,
2396 * many places need both. Might as well get them with one cache lookup
2397 * instead of two. Also, this routine raises an error instead of
2398 * returning a bogus value when given a bad type OID.
2399 */
2400void
2401get_typlenbyval(Oid typid, int16 *typlen, bool *typbyval)
2402{
2403 HeapTuple tp;
2404 Form_pg_type typtup;
2405
2406 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2407 if (!HeapTupleIsValid(tp))
2408 elog(ERROR, "cache lookup failed for type %u", typid);
2409 typtup = (Form_pg_type) GETSTRUCT(tp);
2410 *typlen = typtup->typlen;
2411 *typbyval = typtup->typbyval;
2412 ReleaseSysCache(tp);
2413}
2414
2415/*
2416 * get_typlenbyvalalign
2417 *
2418 * A three-fer: given the type OID, return typlen, typbyval, typalign.
2419 */
2420void
2421get_typlenbyvalalign(Oid typid, int16 *typlen, bool *typbyval,
2422 char *typalign)
2423{
2424 HeapTuple tp;
2425 Form_pg_type typtup;
2426
2427 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2428 if (!HeapTupleIsValid(tp))
2429 elog(ERROR, "cache lookup failed for type %u", typid);
2430 typtup = (Form_pg_type) GETSTRUCT(tp);
2431 *typlen = typtup->typlen;
2432 *typbyval = typtup->typbyval;
2433 *typalign = typtup->typalign;
2434 ReleaseSysCache(tp);
2435}
2436
2437/*
2438 * getTypeIOParam
2439 * Given a pg_type row, select the type OID to pass to I/O functions
2440 *
2441 * Formerly, all I/O functions were passed pg_type.typelem as their second
2442 * parameter, but we now have a more complex rule about what to pass.
2443 * This knowledge is intended to be centralized here --- direct references
2444 * to typelem elsewhere in the code are wrong, if they are associated with
2445 * I/O calls and not with actual subscripting operations! (But see
2446 * bootstrap.c's boot_get_type_io_data() if you need to change this.)
2447 *
2448 * As of PostgreSQL 8.1, output functions receive only the value itself
2449 * and not any auxiliary parameters, so the name of this routine is now
2450 * a bit of a misnomer ... it should be getTypeInputParam.
2451 */
2452Oid
2454{
2455 Form_pg_type typeStruct = (Form_pg_type) GETSTRUCT(typeTuple);
2456
2457 /*
2458 * Array types get their typelem as parameter; everybody else gets their
2459 * own type OID as parameter.
2460 */
2461 if (OidIsValid(typeStruct->typelem))
2462 return typeStruct->typelem;
2463 else
2464 return typeStruct->oid;
2465}
2466
2467/*
2468 * get_type_io_data
2469 *
2470 * A six-fer: given the type OID, return typlen, typbyval, typalign,
2471 * typdelim, typioparam, and IO function OID. The IO function
2472 * returned is controlled by IOFuncSelector
2473 */
2474void
2476 IOFuncSelector which_func,
2477 int16 *typlen,
2478 bool *typbyval,
2479 char *typalign,
2480 char *typdelim,
2481 Oid *typioparam,
2482 Oid *func)
2483{
2484 HeapTuple typeTuple;
2485 Form_pg_type typeStruct;
2486
2487 /*
2488 * In bootstrap mode, pass it off to bootstrap.c. This hack allows us to
2489 * use array_in and array_out during bootstrap.
2490 */
2492 {
2493 Oid typinput;
2494 Oid typoutput;
2495
2497 typlen,
2498 typbyval,
2499 typalign,
2500 typdelim,
2501 typioparam,
2502 &typinput,
2503 &typoutput);
2504 switch (which_func)
2505 {
2506 case IOFunc_input:
2507 *func = typinput;
2508 break;
2509 case IOFunc_output:
2510 *func = typoutput;
2511 break;
2512 default:
2513 elog(ERROR, "binary I/O not supported during bootstrap");
2514 break;
2515 }
2516 return;
2517 }
2518
2519 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2520 if (!HeapTupleIsValid(typeTuple))
2521 elog(ERROR, "cache lookup failed for type %u", typid);
2522 typeStruct = (Form_pg_type) GETSTRUCT(typeTuple);
2523
2524 *typlen = typeStruct->typlen;
2525 *typbyval = typeStruct->typbyval;
2526 *typalign = typeStruct->typalign;
2527 *typdelim = typeStruct->typdelim;
2528 *typioparam = getTypeIOParam(typeTuple);
2529 switch (which_func)
2530 {
2531 case IOFunc_input:
2532 *func = typeStruct->typinput;
2533 break;
2534 case IOFunc_output:
2535 *func = typeStruct->typoutput;
2536 break;
2537 case IOFunc_receive:
2538 *func = typeStruct->typreceive;
2539 break;
2540 case IOFunc_send:
2541 *func = typeStruct->typsend;
2542 break;
2543 }
2544 ReleaseSysCache(typeTuple);
2545}
2546
2547#ifdef NOT_USED
2548char
2549get_typalign(Oid typid)
2550{
2551 HeapTuple tp;
2552
2553 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2554 if (HeapTupleIsValid(tp))
2555 {
2556 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2557 char result;
2558
2559 result = typtup->typalign;
2560 ReleaseSysCache(tp);
2561 return result;
2562 }
2563 else
2564 return TYPALIGN_INT;
2565}
2566#endif
2567
2568char
2570{
2571 HeapTuple tp;
2572
2573 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2574 if (HeapTupleIsValid(tp))
2575 {
2576 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2577 char result;
2578
2579 result = typtup->typstorage;
2580 ReleaseSysCache(tp);
2581 return result;
2582 }
2583 else
2584 return TYPSTORAGE_PLAIN;
2585}
2586
2587/*
2588 * get_typdefault
2589 * Given a type OID, return the type's default value, if any.
2590 *
2591 * The result is a palloc'd expression node tree, or NULL if there
2592 * is no defined default for the datatype.
2593 *
2594 * NB: caller should be prepared to coerce result to correct datatype;
2595 * the returned expression tree might produce something of the wrong type.
2596 */
2597Node *
2599{
2600 HeapTuple typeTuple;
2602 Datum datum;
2603 bool isNull;
2604 Node *expr;
2605
2606 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2607 if (!HeapTupleIsValid(typeTuple))
2608 elog(ERROR, "cache lookup failed for type %u", typid);
2609 type = (Form_pg_type) GETSTRUCT(typeTuple);
2610
2611 /*
2612 * typdefault and typdefaultbin are potentially null, so don't try to
2613 * access 'em as struct fields. Must do it the hard way with
2614 * SysCacheGetAttr.
2615 */
2616 datum = SysCacheGetAttr(TYPEOID,
2617 typeTuple,
2618 Anum_pg_type_typdefaultbin,
2619 &isNull);
2620
2621 if (!isNull)
2622 {
2623 /* We have an expression default */
2624 expr = stringToNode(TextDatumGetCString(datum));
2625 }
2626 else
2627 {
2628 /* Perhaps we have a plain literal default */
2629 datum = SysCacheGetAttr(TYPEOID,
2630 typeTuple,
2631 Anum_pg_type_typdefault,
2632 &isNull);
2633
2634 if (!isNull)
2635 {
2636 char *strDefaultVal;
2637
2638 /* Convert text datum to C string */
2639 strDefaultVal = TextDatumGetCString(datum);
2640 /* Convert C string to a value of the given type */
2641 datum = OidInputFunctionCall(type->typinput, strDefaultVal,
2642 getTypeIOParam(typeTuple), -1);
2643 /* Build a Const node containing the value */
2644 expr = (Node *) makeConst(typid,
2645 -1,
2646 type->typcollation,
2647 type->typlen,
2648 datum,
2649 false,
2650 type->typbyval);
2651 pfree(strDefaultVal);
2652 }
2653 else
2654 {
2655 /* No default */
2656 expr = NULL;
2657 }
2658 }
2659
2660 ReleaseSysCache(typeTuple);
2661
2662 return expr;
2663}
2664
2665/*
2666 * getBaseType
2667 * If the given type is a domain, return its base type;
2668 * otherwise return the type's own OID.
2669 */
2670Oid
2672{
2673 int32 typmod = -1;
2674
2675 return getBaseTypeAndTypmod(typid, &typmod);
2676}
2677
2678/*
2679 * getBaseTypeAndTypmod
2680 * If the given type is a domain, return its base type and typmod;
2681 * otherwise return the type's own OID, and leave *typmod unchanged.
2682 *
2683 * Note that the "applied typmod" should be -1 for every domain level
2684 * above the bottommost; therefore, if the passed-in typid is indeed
2685 * a domain, *typmod should be -1.
2686 */
2687Oid
2689{
2690 /*
2691 * We loop to find the bottom base type in a stack of domains.
2692 */
2693 for (;;)
2694 {
2695 HeapTuple tup;
2696 Form_pg_type typTup;
2697
2698 tup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2699 if (!HeapTupleIsValid(tup))
2700 elog(ERROR, "cache lookup failed for type %u", typid);
2701 typTup = (Form_pg_type) GETSTRUCT(tup);
2702 if (typTup->typtype != TYPTYPE_DOMAIN)
2703 {
2704 /* Not a domain, so done */
2705 ReleaseSysCache(tup);
2706 break;
2707 }
2708
2709 Assert(*typmod == -1);
2710 typid = typTup->typbasetype;
2711 *typmod = typTup->typtypmod;
2712
2713 ReleaseSysCache(tup);
2714 }
2715
2716 return typid;
2717}
2718
2719/*
2720 * get_typavgwidth
2721 *
2722 * Given a type OID and a typmod value (pass -1 if typmod is unknown),
2723 * estimate the average width of values of the type. This is used by
2724 * the planner, which doesn't require absolutely correct results;
2725 * it's OK (and expected) to guess if we don't know for sure.
2726 */
2727int32
2729{
2730 int typlen = get_typlen(typid);
2731 int32 maxwidth;
2732
2733 /*
2734 * Easy if it's a fixed-width type
2735 */
2736 if (typlen > 0)
2737 return typlen;
2738
2739 /*
2740 * type_maximum_size knows the encoding of typmod for some datatypes;
2741 * don't duplicate that knowledge here.
2742 */
2743 maxwidth = type_maximum_size(typid, typmod);
2744 if (maxwidth > 0)
2745 {
2746 /*
2747 * For BPCHAR, the max width is also the only width. Otherwise we
2748 * need to guess about the typical data width given the max. A sliding
2749 * scale for percentage of max width seems reasonable.
2750 */
2751 if (typid == BPCHAROID)
2752 return maxwidth;
2753 if (maxwidth <= 32)
2754 return maxwidth; /* assume full width */
2755 if (maxwidth < 1000)
2756 return 32 + (maxwidth - 32) / 2; /* assume 50% */
2757
2758 /*
2759 * Beyond 1000, assume we're looking at something like
2760 * "varchar(10000)" where the limit isn't actually reached often, and
2761 * use a fixed estimate.
2762 */
2763 return 32 + (1000 - 32) / 2;
2764 }
2765
2766 /*
2767 * Oops, we have no idea ... wild guess time.
2768 */
2769 return 32;
2770}
2771
2772/*
2773 * get_typtype
2774 *
2775 * Given the type OID, find if it is a basic type, a complex type, etc.
2776 * It returns the null char if the cache lookup fails...
2777 */
2778char
2780{
2781 HeapTuple tp;
2782
2783 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2784 if (HeapTupleIsValid(tp))
2785 {
2786 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2787 char result;
2788
2789 result = typtup->typtype;
2790 ReleaseSysCache(tp);
2791 return result;
2792 }
2793 else
2794 return '\0';
2795}
2796
2797/*
2798 * type_is_rowtype
2799 *
2800 * Convenience function to determine whether a type OID represents
2801 * a "rowtype" type --- either RECORD or a named composite type
2802 * (including a domain over a named composite type).
2803 */
2804bool
2806{
2807 if (typid == RECORDOID)
2808 return true; /* easy case */
2809 switch (get_typtype(typid))
2810 {
2811 case TYPTYPE_COMPOSITE:
2812 return true;
2813 case TYPTYPE_DOMAIN:
2814 if (get_typtype(getBaseType(typid)) == TYPTYPE_COMPOSITE)
2815 return true;
2816 break;
2817 default:
2818 break;
2819 }
2820 return false;
2821}
2822
2823/*
2824 * type_is_enum
2825 * Returns true if the given type is an enum type.
2826 */
2827bool
2829{
2830 return (get_typtype(typid) == TYPTYPE_ENUM);
2831}
2832
2833/*
2834 * type_is_range
2835 * Returns true if the given type is a range type.
2836 */
2837bool
2839{
2840 return (get_typtype(typid) == TYPTYPE_RANGE);
2841}
2842
2843/*
2844 * type_is_multirange
2845 * Returns true if the given type is a multirange type.
2846 */
2847bool
2849{
2850 return (get_typtype(typid) == TYPTYPE_MULTIRANGE);
2851}
2852
2853/*
2854 * get_type_category_preferred
2855 *
2856 * Given the type OID, fetch its category and preferred-type status.
2857 * Throws error on failure.
2858 */
2859void
2860get_type_category_preferred(Oid typid, char *typcategory, bool *typispreferred)
2861{
2862 HeapTuple tp;
2863 Form_pg_type typtup;
2864
2865 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2866 if (!HeapTupleIsValid(tp))
2867 elog(ERROR, "cache lookup failed for type %u", typid);
2868 typtup = (Form_pg_type) GETSTRUCT(tp);
2869 *typcategory = typtup->typcategory;
2870 *typispreferred = typtup->typispreferred;
2871 ReleaseSysCache(tp);
2872}
2873
2874/*
2875 * get_typ_typrelid
2876 *
2877 * Given the type OID, get the typrelid (InvalidOid if not a complex
2878 * type).
2879 */
2880Oid
2882{
2883 HeapTuple tp;
2884
2885 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2886 if (HeapTupleIsValid(tp))
2887 {
2888 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2889 Oid result;
2890
2891 result = typtup->typrelid;
2892 ReleaseSysCache(tp);
2893 return result;
2894 }
2895 else
2896 return InvalidOid;
2897}
2898
2899/*
2900 * get_element_type
2901 *
2902 * Given the type OID, get the typelem (InvalidOid if not an array type).
2903 *
2904 * NB: this only succeeds for "true" arrays having array_subscript_handler
2905 * as typsubscript. For other types, InvalidOid is returned independently
2906 * of whether they have typelem or typsubscript set.
2907 */
2908Oid
2910{
2911 HeapTuple tp;
2912
2913 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2914 if (HeapTupleIsValid(tp))
2915 {
2916 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
2917 Oid result;
2918
2919 if (IsTrueArrayType(typtup))
2920 result = typtup->typelem;
2921 else
2922 result = InvalidOid;
2923 ReleaseSysCache(tp);
2924 return result;
2925 }
2926 else
2927 return InvalidOid;
2928}
2929
2930/*
2931 * get_array_type
2932 *
2933 * Given the type OID, get the corresponding "true" array type.
2934 * Returns InvalidOid if no array type can be found.
2935 */
2936Oid
2938{
2939 HeapTuple tp;
2940 Oid result = InvalidOid;
2941
2942 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2943 if (HeapTupleIsValid(tp))
2944 {
2945 result = ((Form_pg_type) GETSTRUCT(tp))->typarray;
2946 ReleaseSysCache(tp);
2947 }
2948 return result;
2949}
2950
2951/*
2952 * get_promoted_array_type
2953 *
2954 * The "promoted" type is what you'd get from an ARRAY(SELECT ...)
2955 * construct, that is, either the corresponding "true" array type
2956 * if the input is a scalar type that has such an array type,
2957 * or the same type if the input is already a "true" array type.
2958 * Returns InvalidOid if neither rule is satisfied.
2959 */
2960Oid
2962{
2963 Oid array_type = get_array_type(typid);
2964
2965 if (OidIsValid(array_type))
2966 return array_type;
2967 if (OidIsValid(get_element_type(typid)))
2968 return typid;
2969 return InvalidOid;
2970}
2971
2972/*
2973 * get_base_element_type
2974 * Given the type OID, get the typelem, looking "through" any domain
2975 * to its underlying array type.
2976 *
2977 * This is equivalent to get_element_type(getBaseType(typid)), but avoids
2978 * an extra cache lookup. Note that it fails to provide any information
2979 * about the typmod of the array.
2980 */
2981Oid
2983{
2984 /*
2985 * We loop to find the bottom base type in a stack of domains.
2986 */
2987 for (;;)
2988 {
2989 HeapTuple tup;
2990 Form_pg_type typTup;
2991
2992 tup = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
2993 if (!HeapTupleIsValid(tup))
2994 break;
2995 typTup = (Form_pg_type) GETSTRUCT(tup);
2996 if (typTup->typtype != TYPTYPE_DOMAIN)
2997 {
2998 /* Not a domain, so stop descending */
2999 Oid result;
3000
3001 /* This test must match get_element_type */
3002 if (IsTrueArrayType(typTup))
3003 result = typTup->typelem;
3004 else
3005 result = InvalidOid;
3006 ReleaseSysCache(tup);
3007 return result;
3008 }
3009
3010 typid = typTup->typbasetype;
3011 ReleaseSysCache(tup);
3012 }
3013
3014 /* Like get_element_type, silently return InvalidOid for bogus input */
3015 return InvalidOid;
3016}
3017
3018/*
3019 * getTypeInputInfo
3020 *
3021 * Get info needed for converting values of a type to internal form
3022 */
3023void
3024getTypeInputInfo(Oid type, Oid *typInput, Oid *typIOParam)
3025{
3026 HeapTuple typeTuple;
3027 Form_pg_type pt;
3028
3029 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(type));
3030 if (!HeapTupleIsValid(typeTuple))
3031 elog(ERROR, "cache lookup failed for type %u", type);
3032 pt = (Form_pg_type) GETSTRUCT(typeTuple);
3033
3034 if (!pt->typisdefined)
3035 ereport(ERROR,
3036 (errcode(ERRCODE_UNDEFINED_OBJECT),
3037 errmsg("type %s is only a shell",
3038 format_type_be(type))));
3039 if (!OidIsValid(pt->typinput))
3040 ereport(ERROR,
3041 (errcode(ERRCODE_UNDEFINED_FUNCTION),
3042 errmsg("no input function available for type %s",
3043 format_type_be(type))));
3044
3045 *typInput = pt->typinput;
3046 *typIOParam = getTypeIOParam(typeTuple);
3047
3048 ReleaseSysCache(typeTuple);
3049}
3050
3051/*
3052 * getTypeOutputInfo
3053 *
3054 * Get info needed for printing values of a type
3055 */
3056void
3057getTypeOutputInfo(Oid type, Oid *typOutput, bool *typIsVarlena)
3058{
3059 HeapTuple typeTuple;
3060 Form_pg_type pt;
3061
3062 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(type));
3063 if (!HeapTupleIsValid(typeTuple))
3064 elog(ERROR, "cache lookup failed for type %u", type);
3065 pt = (Form_pg_type) GETSTRUCT(typeTuple);
3066
3067 if (!pt->typisdefined)
3068 ereport(ERROR,
3069 (errcode(ERRCODE_UNDEFINED_OBJECT),
3070 errmsg("type %s is only a shell",
3071 format_type_be(type))));
3072 if (!OidIsValid(pt->typoutput))
3073 ereport(ERROR,
3074 (errcode(ERRCODE_UNDEFINED_FUNCTION),
3075 errmsg("no output function available for type %s",
3076 format_type_be(type))));
3077
3078 *typOutput = pt->typoutput;
3079 *typIsVarlena = (!pt->typbyval) && (pt->typlen == -1);
3080
3081 ReleaseSysCache(typeTuple);
3082}
3083
3084/*
3085 * getTypeBinaryInputInfo
3086 *
3087 * Get info needed for binary input of values of a type
3088 */
3089void
3090getTypeBinaryInputInfo(Oid type, Oid *typReceive, Oid *typIOParam)
3091{
3092 HeapTuple typeTuple;
3093 Form_pg_type pt;
3094
3095 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(type));
3096 if (!HeapTupleIsValid(typeTuple))
3097 elog(ERROR, "cache lookup failed for type %u", type);
3098 pt = (Form_pg_type) GETSTRUCT(typeTuple);
3099
3100 if (!pt->typisdefined)
3101 ereport(ERROR,
3102 (errcode(ERRCODE_UNDEFINED_OBJECT),
3103 errmsg("type %s is only a shell",
3104 format_type_be(type))));
3105 if (!OidIsValid(pt->typreceive))
3106 ereport(ERROR,
3107 (errcode(ERRCODE_UNDEFINED_FUNCTION),
3108 errmsg("no binary input function available for type %s",
3109 format_type_be(type))));
3110
3111 *typReceive = pt->typreceive;
3112 *typIOParam = getTypeIOParam(typeTuple);
3113
3114 ReleaseSysCache(typeTuple);
3115}
3116
3117/*
3118 * getTypeBinaryOutputInfo
3119 *
3120 * Get info needed for binary output of values of a type
3121 */
3122void
3123getTypeBinaryOutputInfo(Oid type, Oid *typSend, bool *typIsVarlena)
3124{
3125 HeapTuple typeTuple;
3126 Form_pg_type pt;
3127
3128 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(type));
3129 if (!HeapTupleIsValid(typeTuple))
3130 elog(ERROR, "cache lookup failed for type %u", type);
3131 pt = (Form_pg_type) GETSTRUCT(typeTuple);
3132
3133 if (!pt->typisdefined)
3134 ereport(ERROR,
3135 (errcode(ERRCODE_UNDEFINED_OBJECT),
3136 errmsg("type %s is only a shell",
3137 format_type_be(type))));
3138 if (!OidIsValid(pt->typsend))
3139 ereport(ERROR,
3140 (errcode(ERRCODE_UNDEFINED_FUNCTION),
3141 errmsg("no binary output function available for type %s",
3142 format_type_be(type))));
3143
3144 *typSend = pt->typsend;
3145 *typIsVarlena = (!pt->typbyval) && (pt->typlen == -1);
3146
3147 ReleaseSysCache(typeTuple);
3148}
3149
3150/*
3151 * get_typmodin
3152 *
3153 * Given the type OID, return the type's typmodin procedure, if any.
3154 */
3155Oid
3157{
3158 HeapTuple tp;
3159
3160 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
3161 if (HeapTupleIsValid(tp))
3162 {
3163 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
3164 Oid result;
3165
3166 result = typtup->typmodin;
3167 ReleaseSysCache(tp);
3168 return result;
3169 }
3170 else
3171 return InvalidOid;
3172}
3173
3174#ifdef NOT_USED
3175/*
3176 * get_typmodout
3177 *
3178 * Given the type OID, return the type's typmodout procedure, if any.
3179 */
3180Oid
3181get_typmodout(Oid typid)
3182{
3183 HeapTuple tp;
3184
3185 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
3186 if (HeapTupleIsValid(tp))
3187 {
3188 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
3189 Oid result;
3190
3191 result = typtup->typmodout;
3192 ReleaseSysCache(tp);
3193 return result;
3194 }
3195 else
3196 return InvalidOid;
3197}
3198#endif /* NOT_USED */
3199
3200/*
3201 * get_typcollation
3202 *
3203 * Given the type OID, return the type's typcollation attribute.
3204 */
3205Oid
3207{
3208 HeapTuple tp;
3209
3210 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
3211 if (HeapTupleIsValid(tp))
3212 {
3213 Form_pg_type typtup = (Form_pg_type) GETSTRUCT(tp);
3214 Oid result;
3215
3216 result = typtup->typcollation;
3217 ReleaseSysCache(tp);
3218 return result;
3219 }
3220 else
3221 return InvalidOid;
3222}
3223
3224
3225/*
3226 * type_is_collatable
3227 *
3228 * Return whether the type cares about collations
3229 */
3230bool
3232{
3233 return OidIsValid(get_typcollation(typid));
3234}
3235
3236
3237/*
3238 * get_typsubscript
3239 *
3240 * Given the type OID, return the type's subscripting handler's OID,
3241 * if it has one.
3242 *
3243 * If typelemp isn't NULL, we also store the type's typelem value there.
3244 * This saves some callers an extra catalog lookup.
3245 */
3247get_typsubscript(Oid typid, Oid *typelemp)
3248{
3249 HeapTuple tp;
3250
3251 tp = SearchSysCache1(TYPEOID, ObjectIdGetDatum(typid));
3252 if (HeapTupleIsValid(tp))
3253 {
3254 Form_pg_type typform = (Form_pg_type) GETSTRUCT(tp);
3255 RegProcedure handler = typform->typsubscript;
3256
3257 if (typelemp)
3258 *typelemp = typform->typelem;
3259 ReleaseSysCache(tp);
3260 return handler;
3261 }
3262 else
3263 {
3264 if (typelemp)
3265 *typelemp = InvalidOid;
3266 return InvalidOid;
3267 }
3268}
3269
3270/*
3271 * getSubscriptingRoutines
3272 *
3273 * Given the type OID, fetch the type's subscripting methods struct.
3274 * Return NULL if type is not subscriptable.
3275 *
3276 * If typelemp isn't NULL, we also store the type's typelem value there.
3277 * This saves some callers an extra catalog lookup.
3278 */
3279const struct SubscriptRoutines *
3281{
3282 RegProcedure typsubscript = get_typsubscript(typid, typelemp);
3283
3284 if (!OidIsValid(typsubscript))
3285 return NULL;
3286
3287 return (const struct SubscriptRoutines *)
3288 DatumGetPointer(OidFunctionCall0(typsubscript));
3289}
3290
3291
3292/* ---------- STATISTICS CACHE ---------- */
3293
3294/*
3295 * get_attavgwidth
3296 *
3297 * Given the table and attribute number of a column, get the average
3298 * width of entries in the column. Return zero if no data available.
3299 *
3300 * Currently this is only consulted for individual tables, not for inheritance
3301 * trees, so we don't need an "inh" parameter.
3302 *
3303 * Calling a hook at this point looks somewhat strange, but is required
3304 * because the optimizer calls this function without any other way for
3305 * plug-ins to control the result.
3306 */
3307int32
3309{
3310 HeapTuple tp;
3311 int32 stawidth;
3312
3314 {
3315 stawidth = (*get_attavgwidth_hook) (relid, attnum);
3316 if (stawidth > 0)
3317 return stawidth;
3318 }
3319 tp = SearchSysCache3(STATRELATTINH,
3320 ObjectIdGetDatum(relid),
3322 BoolGetDatum(false));
3323 if (HeapTupleIsValid(tp))
3324 {
3325 stawidth = ((Form_pg_statistic) GETSTRUCT(tp))->stawidth;
3326 ReleaseSysCache(tp);
3327 if (stawidth > 0)
3328 return stawidth;
3329 }
3330 return 0;
3331}
3332
3333/*
3334 * get_attstatsslot
3335 *
3336 * Extract the contents of a "slot" of a pg_statistic tuple.
3337 * Returns true if requested slot type was found, else false.
3338 *
3339 * Unlike other routines in this file, this takes a pointer to an
3340 * already-looked-up tuple in the pg_statistic cache. We do this since
3341 * most callers will want to extract more than one value from the cache
3342 * entry, and we don't want to repeat the cache lookup unnecessarily.
3343 * Also, this API allows this routine to be used with statistics tuples
3344 * that have been provided by a stats hook and didn't really come from
3345 * pg_statistic.
3346 *
3347 * sslot: pointer to output area (typically, a local variable in the caller).
3348 * statstuple: pg_statistic tuple to be examined.
3349 * reqkind: STAKIND code for desired statistics slot kind.
3350 * reqop: STAOP value wanted, or InvalidOid if don't care.
3351 * flags: bitmask of ATTSTATSSLOT_VALUES and/or ATTSTATSSLOT_NUMBERS.
3352 *
3353 * If a matching slot is found, true is returned, and *sslot is filled thus:
3354 * staop: receives the actual STAOP value.
3355 * stacoll: receives the actual STACOLL value.
3356 * valuetype: receives actual datatype of the elements of stavalues.
3357 * values: receives pointer to an array of the slot's stavalues.
3358 * nvalues: receives number of stavalues.
3359 * numbers: receives pointer to an array of the slot's stanumbers (as float4).
3360 * nnumbers: receives number of stanumbers.
3361 *
3362 * valuetype/values/nvalues are InvalidOid/NULL/0 if ATTSTATSSLOT_VALUES
3363 * wasn't specified. Likewise, numbers/nnumbers are NULL/0 if
3364 * ATTSTATSSLOT_NUMBERS wasn't specified.
3365 *
3366 * If no matching slot is found, false is returned, and *sslot is zeroed.
3367 *
3368 * Note that the current API doesn't allow for searching for a slot with
3369 * a particular collation. If we ever actually support recording more than
3370 * one collation, we'll have to extend the API, but for now simple is good.
3371 *
3372 * The data referred to by the fields of sslot is locally palloc'd and
3373 * is independent of the original pg_statistic tuple. When the caller
3374 * is done with it, call free_attstatsslot to release the palloc'd data.
3375 *
3376 * If it's desirable to call free_attstatsslot when get_attstatsslot might
3377 * not have been called, memset'ing sslot to zeroes will allow that.
3378 *
3379 * Passing flags=0 can be useful to quickly check if the requested slot type
3380 * exists. In this case no arrays are extracted, so free_attstatsslot need
3381 * not be called.
3382 */
3383bool
3385 int reqkind, Oid reqop, int flags)
3386{
3387 Form_pg_statistic stats = (Form_pg_statistic) GETSTRUCT(statstuple);
3388 int i;
3389 Datum val;
3390 ArrayType *statarray;
3391 Oid arrayelemtype;
3392 int narrayelem;
3393 HeapTuple typeTuple;
3394 Form_pg_type typeForm;
3395
3396 /* initialize *sslot properly */
3397 memset(sslot, 0, sizeof(AttStatsSlot));
3398
3399 for (i = 0; i < STATISTIC_NUM_SLOTS; i++)
3400 {
3401 if ((&stats->stakind1)[i] == reqkind &&
3402 (reqop == InvalidOid || (&stats->staop1)[i] == reqop))
3403 break;
3404 }
3405 if (i >= STATISTIC_NUM_SLOTS)
3406 return false; /* not there */
3407
3408 sslot->staop = (&stats->staop1)[i];
3409 sslot->stacoll = (&stats->stacoll1)[i];
3410
3411 if (flags & ATTSTATSSLOT_VALUES)
3412 {
3413 val = SysCacheGetAttrNotNull(STATRELATTINH, statstuple,
3414 Anum_pg_statistic_stavalues1 + i);
3415
3416 /*
3417 * Detoast the array if needed, and in any case make a copy that's
3418 * under control of this AttStatsSlot.
3419 */
3420 statarray = DatumGetArrayTypePCopy(val);
3421
3422 /*
3423 * Extract the actual array element type, and pass it back in case the
3424 * caller needs it.
3425 */
3426 sslot->valuetype = arrayelemtype = ARR_ELEMTYPE(statarray);
3427
3428 /* Need info about element type */
3429 typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(arrayelemtype));
3430 if (!HeapTupleIsValid(typeTuple))
3431 elog(ERROR, "cache lookup failed for type %u", arrayelemtype);
3432 typeForm = (Form_pg_type) GETSTRUCT(typeTuple);
3433
3434 /* Deconstruct array into Datum elements; NULLs not expected */
3435 deconstruct_array(statarray,
3436 arrayelemtype,
3437 typeForm->typlen,
3438 typeForm->typbyval,
3439 typeForm->typalign,
3440 &sslot->values, NULL, &sslot->nvalues);
3441
3442 /*
3443 * If the element type is pass-by-reference, we now have a bunch of
3444 * Datums that are pointers into the statarray, so we need to keep
3445 * that until free_attstatsslot. Otherwise, all the useful info is in
3446 * sslot->values[], so we can free the array object immediately.
3447 */
3448 if (!typeForm->typbyval)
3449 sslot->values_arr = statarray;
3450 else
3451 pfree(statarray);
3452
3453 ReleaseSysCache(typeTuple);
3454 }
3455
3456 if (flags & ATTSTATSSLOT_NUMBERS)
3457 {
3458 val = SysCacheGetAttrNotNull(STATRELATTINH, statstuple,
3459 Anum_pg_statistic_stanumbers1 + i);
3460
3461 /*
3462 * Detoast the array if needed, and in any case make a copy that's
3463 * under control of this AttStatsSlot.
3464 */
3465 statarray = DatumGetArrayTypePCopy(val);
3466
3467 /*
3468 * We expect the array to be a 1-D float4 array; verify that. We don't
3469 * need to use deconstruct_array() since the array data is just going
3470 * to look like a C array of float4 values.
3471 */
3472 narrayelem = ARR_DIMS(statarray)[0];
3473 if (ARR_NDIM(statarray) != 1 || narrayelem <= 0 ||
3474 ARR_HASNULL(statarray) ||
3475 ARR_ELEMTYPE(statarray) != FLOAT4OID)
3476 elog(ERROR, "stanumbers is not a 1-D float4 array");
3477
3478 /* Give caller a pointer directly into the statarray */
3479 sslot->numbers = (float4 *) ARR_DATA_PTR(statarray);
3480 sslot->nnumbers = narrayelem;
3481
3482 /* We'll free the statarray in free_attstatsslot */
3483 sslot->numbers_arr = statarray;
3484 }
3485
3486 return true;
3487}
3488
3489/*
3490 * free_attstatsslot
3491 * Free data allocated by get_attstatsslot
3492 */
3493void
3495{
3496 /* The values[] array was separately palloc'd by deconstruct_array */
3497 if (sslot->values)
3498 pfree(sslot->values);
3499 /* The numbers[] array points into numbers_arr, do not pfree it */
3500 /* Free the detoasted array objects, if any */
3501 if (sslot->values_arr)
3502 pfree(sslot->values_arr);
3503 if (sslot->numbers_arr)
3504 pfree(sslot->numbers_arr);
3505}
3506
3507/* ---------- PG_NAMESPACE CACHE ---------- */
3508
3509/*
3510 * get_namespace_name
3511 * Returns the name of a given namespace
3512 *
3513 * Returns a palloc'd copy of the string, or NULL if no such namespace.
3514 */
3515char *
3517{
3518 HeapTuple tp;
3519
3520 tp = SearchSysCache1(NAMESPACEOID, ObjectIdGetDatum(nspid));
3521 if (HeapTupleIsValid(tp))
3522 {
3524 char *result;
3525
3526 result = pstrdup(NameStr(nsptup->nspname));
3527 ReleaseSysCache(tp);
3528 return result;
3529 }
3530 else
3531 return NULL;
3532}
3533
3534/*
3535 * get_namespace_name_or_temp
3536 * As above, but if it is this backend's temporary namespace, return
3537 * "pg_temp" instead.
3538 */
3539char *
3541{
3543 return pstrdup("pg_temp");
3544 else
3545 return get_namespace_name(nspid);
3546}
3547
3548/* ---------- PG_RANGE CACHES ---------- */
3549
3550/*
3551 * get_range_subtype
3552 * Returns the subtype of a given range type
3553 *
3554 * Returns InvalidOid if the type is not a range type.
3555 */
3556Oid
3558{
3559 HeapTuple tp;
3560
3561 tp = SearchSysCache1(RANGETYPE, ObjectIdGetDatum(rangeOid));
3562 if (HeapTupleIsValid(tp))
3563 {
3564 Form_pg_range rngtup = (Form_pg_range) GETSTRUCT(tp);
3565 Oid result;
3566
3567 result = rngtup->rngsubtype;
3568 ReleaseSysCache(tp);
3569 return result;
3570 }
3571 else
3572 return InvalidOid;
3573}
3574
3575/*
3576 * get_range_collation
3577 * Returns the collation of a given range type
3578 *
3579 * Returns InvalidOid if the type is not a range type,
3580 * or if its subtype is not collatable.
3581 */
3582Oid
3584{
3585 HeapTuple tp;
3586
3587 tp = SearchSysCache1(RANGETYPE, ObjectIdGetDatum(rangeOid));
3588 if (HeapTupleIsValid(tp))
3589 {
3590 Form_pg_range rngtup = (Form_pg_range) GETSTRUCT(tp);
3591 Oid result;
3592
3593 result = rngtup->rngcollation;
3594 ReleaseSysCache(tp);
3595 return result;
3596 }
3597 else
3598 return InvalidOid;
3599}
3600
3601/*
3602 * get_range_multirange
3603 * Returns the multirange type of a given range type
3604 *
3605 * Returns InvalidOid if the type is not a range type.
3606 */
3607Oid
3609{
3610 HeapTuple tp;
3611
3612 tp = SearchSysCache1(RANGETYPE, ObjectIdGetDatum(rangeOid));
3613 if (HeapTupleIsValid(tp))
3614 {
3615 Form_pg_range rngtup = (Form_pg_range) GETSTRUCT(tp);
3616 Oid result;
3617
3618 result = rngtup->rngmultitypid;
3619 ReleaseSysCache(tp);
3620 return result;
3621 }
3622 else
3623 return InvalidOid;
3624}
3625
3626/*
3627 * get_multirange_range
3628 * Returns the range type of a given multirange
3629 *
3630 * Returns InvalidOid if the type is not a multirange.
3631 */
3632Oid
3634{
3635 HeapTuple tp;
3636
3637 tp = SearchSysCache1(RANGEMULTIRANGE, ObjectIdGetDatum(multirangeOid));
3638 if (HeapTupleIsValid(tp))
3639 {
3640 Form_pg_range rngtup = (Form_pg_range) GETSTRUCT(tp);
3641 Oid result;
3642
3643 result = rngtup->rngtypid;
3644 ReleaseSysCache(tp);
3645 return result;
3646 }
3647 else
3648 return InvalidOid;
3649}
3650
3651/* ---------- PG_INDEX CACHE ---------- */
3652
3653/*
3654 * get_index_column_opclass
3655 *
3656 * Given the index OID and column number,
3657 * return opclass of the index column
3658 * or InvalidOid if the index was not found
3659 * or column is non-key one.
3660 */
3661Oid
3662get_index_column_opclass(Oid index_oid, int attno)
3663{
3664 HeapTuple tuple;
3665 Form_pg_index rd_index;
3666 Datum datum;
3667 oidvector *indclass;
3668 Oid opclass;
3669
3670 /* First we need to know the column's opclass. */
3671
3672 tuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(index_oid));
3673 if (!HeapTupleIsValid(tuple))
3674 return InvalidOid;
3675
3676 rd_index = (Form_pg_index) GETSTRUCT(tuple);
3677
3678 /* caller is supposed to guarantee this */
3679 Assert(attno > 0 && attno <= rd_index->indnatts);
3680
3681 /* Non-key attributes don't have an opclass */
3682 if (attno > rd_index->indnkeyatts)
3683 {
3684 ReleaseSysCache(tuple);
3685 return InvalidOid;
3686 }
3687
3688 datum = SysCacheGetAttrNotNull(INDEXRELID, tuple, Anum_pg_index_indclass);
3689 indclass = ((oidvector *) DatumGetPointer(datum));
3690
3691 Assert(attno <= indclass->dim1);
3692 opclass = indclass->values[attno - 1];
3693
3694 ReleaseSysCache(tuple);
3695
3696 return opclass;
3697}
3698
3699/*
3700 * get_index_isreplident
3701 *
3702 * Given the index OID, return pg_index.indisreplident.
3703 */
3704bool
3706{
3707 HeapTuple tuple;
3708 Form_pg_index rd_index;
3709 bool result;
3710
3711 tuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(index_oid));
3712 if (!HeapTupleIsValid(tuple))
3713 return false;
3714
3715 rd_index = (Form_pg_index) GETSTRUCT(tuple);
3716 result = rd_index->indisreplident;
3717 ReleaseSysCache(tuple);
3718
3719 return result;
3720}
3721
3722/*
3723 * get_index_isvalid
3724 *
3725 * Given the index OID, return pg_index.indisvalid.
3726 */
3727bool
3729{
3730 bool isvalid;
3731 HeapTuple tuple;
3732 Form_pg_index rd_index;
3733
3734 tuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(index_oid));
3735 if (!HeapTupleIsValid(tuple))
3736 elog(ERROR, "cache lookup failed for index %u", index_oid);
3737
3738 rd_index = (Form_pg_index) GETSTRUCT(tuple);
3739 isvalid = rd_index->indisvalid;
3740 ReleaseSysCache(tuple);
3741
3742 return isvalid;
3743}
3744
3745/*
3746 * get_index_isclustered
3747 *
3748 * Given the index OID, return pg_index.indisclustered.
3749 */
3750bool
3752{
3753 bool isclustered;
3754 HeapTuple tuple;
3755 Form_pg_index rd_index;
3756
3757 tuple = SearchSysCache1(INDEXRELID, ObjectIdGetDatum(index_oid));
3758 if (!HeapTupleIsValid(tuple))
3759 elog(ERROR, "cache lookup failed for index %u", index_oid);
3760
3761 rd_index = (Form_pg_index) GETSTRUCT(tuple);
3762 isclustered = rd_index->indisclustered;
3763 ReleaseSysCache(tuple);
3764
3765 return isclustered;
3766}
3767
3768/*
3769 * get_publication_oid - given a publication name, look up the OID
3770 *
3771 * If missing_ok is false, throw an error if name not found. If true, just
3772 * return InvalidOid.
3773 */
3774Oid
3775get_publication_oid(const char *pubname, bool missing_ok)
3776{
3777 Oid oid;
3778
3779 oid = GetSysCacheOid1(PUBLICATIONNAME, Anum_pg_publication_oid,
3780 CStringGetDatum(pubname));
3781 if (!OidIsValid(oid) && !missing_ok)
3782 ereport(ERROR,
3783 (errcode(ERRCODE_UNDEFINED_OBJECT),
3784 errmsg("publication \"%s\" does not exist", pubname)));
3785 return oid;
3786}
3787
3788/*
3789 * get_publication_name - given a publication Oid, look up the name
3790 *
3791 * If missing_ok is false, throw an error if name not found. If true, just
3792 * return NULL.
3793 */
3794char *
3795get_publication_name(Oid pubid, bool missing_ok)
3796{
3797 HeapTuple tup;
3798 char *pubname;
3799 Form_pg_publication pubform;
3800
3801 tup = SearchSysCache1(PUBLICATIONOID, ObjectIdGetDatum(pubid));
3802
3803 if (!HeapTupleIsValid(tup))
3804 {
3805 if (!missing_ok)
3806 elog(ERROR, "cache lookup failed for publication %u", pubid);
3807 return NULL;
3808 }
3809
3810 pubform = (Form_pg_publication) GETSTRUCT(tup);
3811 pubname = pstrdup(NameStr(pubform->pubname));
3812
3813 ReleaseSysCache(tup);
3814
3815 return pubname;
3816}
3817
3818/*
3819 * get_subscription_oid - given a subscription name, look up the OID
3820 *
3821 * If missing_ok is false, throw an error if name not found. If true, just
3822 * return InvalidOid.
3823 */
3824Oid
3825get_subscription_oid(const char *subname, bool missing_ok)
3826{
3827 Oid oid;
3828
3829 oid = GetSysCacheOid2(SUBSCRIPTIONNAME, Anum_pg_subscription_oid,
3831 if (!OidIsValid(oid) && !missing_ok)
3832 ereport(ERROR,
3833 (errcode(ERRCODE_UNDEFINED_OBJECT),
3834 errmsg("subscription \"%s\" does not exist", subname)));
3835 return oid;
3836}
3837
3838/*
3839 * get_subscription_name - given a subscription OID, look up the name
3840 *
3841 * If missing_ok is false, throw an error if name not found. If true, just
3842 * return NULL.
3843 */
3844char *
3845get_subscription_name(Oid subid, bool missing_ok)
3846{
3847 HeapTuple tup;
3848 char *subname;
3849 Form_pg_subscription subform;
3850
3851 tup = SearchSysCache1(SUBSCRIPTIONOID, ObjectIdGetDatum(subid));
3852
3853 if (!HeapTupleIsValid(tup))
3854 {
3855 if (!missing_ok)
3856 elog(ERROR, "cache lookup failed for subscription %u", subid);
3857 return NULL;
3858 }
3859
3860 subform = (Form_pg_subscription) GETSTRUCT(tup);
3861 subname = pstrdup(NameStr(subform->subname));
3862
3863 ReleaseSysCache(tup);
3864
3865 return subname;
3866}
const IndexAmRoutine * GetIndexAmRoutineByAmId(Oid amoid, bool noerror)
Definition: amapi.c:69
StrategyNumber IndexAmTranslateCompareType(CompareType cmptype, Oid amoid, Oid opfamily, bool missing_ok)
Definition: amapi.c:161
CompareType IndexAmTranslateStrategy(StrategyNumber strategy, Oid amoid, Oid opfamily, bool missing_ok)
Definition: amapi.c:131
#define DatumGetArrayTypePCopy(X)
Definition: array.h:262
#define ARR_NDIM(a)
Definition: array.h:290
#define ARR_DATA_PTR(a)
Definition: array.h:322
#define ARR_ELEMTYPE(a)
Definition: array.h:292
#define ARR_DIMS(a)
Definition: array.h:294
#define ARR_HASNULL(a)
Definition: array.h:291
void deconstruct_array(const ArrayType *array, Oid elmtype, int elmlen, bool elmbyval, char elmalign, Datum **elemsp, bool **nullsp, int *nelemsp)
Definition: arrayfuncs.c:3632
int16 AttrNumber
Definition: attnum.h:21
#define InvalidAttrNumber
Definition: attnum.h:23
void boot_get_type_io_data(Oid typid, int16 *typlen, bool *typbyval, char *typalign, char *typdelim, Oid *typioparam, Oid *typinput, Oid *typoutput)
Definition: bootstrap.c:839
#define TextDatumGetCString(d)
Definition: builtins.h:98
#define NameStr(name)
Definition: c.h:765
int16_t int16
Definition: c.h:547
regproc RegProcedure
Definition: c.h:669
int32_t int32
Definition: c.h:548
float float4
Definition: c.h:648
#define OidIsValid(objectId)
Definition: c.h:788
CompareType
Definition: cmptype.h:32
@ COMPARE_INVALID
Definition: cmptype.h:33
@ COMPARE_GT
Definition: cmptype.h:38
@ COMPARE_EQ
Definition: cmptype.h:36
@ COMPARE_NE
Definition: cmptype.h:39
@ COMPARE_LT
Definition: cmptype.h:34
int nspid
Oid collid
Datum datumCopy(Datum value, bool typByVal, int typLen)
Definition: datum.c:132
int errcode(int sqlerrcode)
Definition: elog.c:863
int errmsg(const char *fmt,...)
Definition: elog.c:1080
#define ERROR
Definition: elog.h:39
#define elog(elevel,...)
Definition: elog.h:226
#define ereport(elevel,...)
Definition: elog.h:150
#define palloc_object(type)
Definition: fe_memutils.h:74
Datum OidInputFunctionCall(Oid functionId, char *str, Oid typioparam, int32 typmod)
Definition: fmgr.c:1754
#define OidFunctionCall0(functionId)
Definition: fmgr.h:718
int32 type_maximum_size(Oid type_oid, int32 typemod)
Definition: format_type.c:412
char * format_type_be(Oid type_oid)
Definition: format_type.c:343
Oid MyDatabaseId
Definition: globals.c:94
#define HASHSTANDARD_PROC
Definition: hash.h:355
Assert(PointerIsAligned(start, uint64))
#define HeapTupleIsValid(tuple)
Definition: htup.h:78
static void * GETSTRUCT(const HeapTupleData *tuple)
Definition: htup_details.h:728
long val
Definition: informix.c:689
int i
Definition: isn.c:77
List * lappend(List *list, void *datum)
Definition: list.c:339
List * lappend_oid(List *list, Oid datum)
Definition: list.c:375
bool list_member_oid(const List *list, Oid datum)
Definition: list.c:722
Oid get_range_subtype(Oid rangeOid)
Definition: lsyscache.c:3557
char * get_rel_name(Oid relid)
Definition: lsyscache.c:2078
void get_op_opfamily_properties(Oid opno, Oid opfamily, bool ordering_op, int *strategy, Oid *lefttype, Oid *righttype)
Definition: lsyscache.c:138
Oid get_func_variadictype(Oid funcid)
Definition: lsyscache.c:1873
Oid get_opclass_method(Oid opclass)
Definition: lsyscache.c:1362
bool get_compatible_hash_operators(Oid opno, Oid *lhs_opno, Oid *rhs_opno)
Definition: lsyscache.c:475
bool get_rel_relispartition(Oid relid)
Definition: lsyscache.c:2177
Oid get_op_opfamily_sortfamily(Oid opno, Oid opfamily)
Definition: lsyscache.c:110
char get_rel_persistence(Oid relid)
Definition: lsyscache.c:2228
char get_func_prokind(Oid funcid)
Definition: lsyscache.c:1968
bool get_index_isvalid(Oid index_oid)
Definition: lsyscache.c:3728
Oid get_cast_oid(Oid sourcetypeid, Oid targettypeid, bool missing_ok)
Definition: lsyscache.c:1083
void getTypeBinaryOutputInfo(Oid type, Oid *typSend, bool *typIsVarlena)
Definition: lsyscache.c:3123
AttrNumber get_attnum(Oid relid, const char *attname)
Definition: lsyscache.c:934
RegProcedure get_oprrest(Oid opno)
Definition: lsyscache.c:1707
void free_attstatsslot(AttStatsSlot *sslot)
Definition: lsyscache.c:3494
bool comparison_ops_are_compatible(Oid opno1, Oid opno2)
Definition: lsyscache.c:823
Oid get_constraint_index(Oid conoid)
Definition: lsyscache.c:1189
bool get_func_retset(Oid funcid)
Definition: lsyscache.c:1892
bool get_ordering_op_properties(Oid opno, Oid *opfamily, Oid *opcintype, CompareType *cmptype)
Definition: lsyscache.c:259
Oid get_element_type(Oid typid)
Definition: lsyscache.c:2909
Oid get_opclass_input_type(Oid opclass)
Definition: lsyscache.c:1314
bool type_is_rowtype(Oid typid)
Definition: lsyscache.c:2805
bool type_is_range(Oid typid)
Definition: lsyscache.c:2838
char func_parallel(Oid funcid)
Definition: lsyscache.c:1949
Oid get_opclass_family(Oid opclass)
Definition: lsyscache.c:1292
char get_attgenerated(Oid relid, AttrNumber attnum)
Definition: lsyscache.c:964
bool type_is_enum(Oid typid)
Definition: lsyscache.c:2828
Oid get_multirange_range(Oid multirangeOid)
Definition: lsyscache.c:3633
Oid get_typmodin(Oid typid)
Definition: lsyscache.c:3156
Oid get_opfamily_member_for_cmptype(Oid opfamily, Oid lefttype, Oid righttype, CompareType cmptype)
Definition: lsyscache.c:197
char get_typstorage(Oid typid)
Definition: lsyscache.c:2569
bool get_opclass_opfamily_and_input_type(Oid opclass, Oid *opfamily, Oid *opcintype)
Definition: lsyscache.c:1337
RegProcedure get_func_support(Oid funcid)
Definition: lsyscache.c:2008
char * get_database_name(Oid dbid)
Definition: lsyscache.c:1242
void getTypeOutputInfo(Oid type, Oid *typOutput, bool *typIsVarlena)
Definition: lsyscache.c:3057
bool get_typisdefined(Oid typid)
Definition: lsyscache.c:2323
char * get_opname(Oid opno)
Definition: lsyscache.c:1460
Datum get_attoptions(Oid relid, int16 attnum)
Definition: lsyscache.c:1046
void get_typlenbyvalalign(Oid typid, int16 *typlen, bool *typbyval, char *typalign)
Definition: lsyscache.c:2421
int32 get_attavgwidth(Oid relid, AttrNumber attnum)
Definition: lsyscache.c:3308
bool get_index_isreplident(Oid index_oid)
Definition: lsyscache.c:3705
Oid get_opfamily_proc(Oid opfamily, Oid lefttype, Oid righttype, int16 procnum)
Definition: lsyscache.c:872
RegProcedure get_oprjoin(Oid opno)
Definition: lsyscache.c:1731
Oid get_equality_op_for_ordering_op(Oid opno, bool *reverse)
Definition: lsyscache.c:324
bool op_strict(Oid opno)
Definition: lsyscache.c:1627
bool op_hashjoinable(Oid opno, Oid inputtype)
Definition: lsyscache.c:1587
char get_rel_relkind(Oid relid)
Definition: lsyscache.c:2153
void get_typlenbyval(Oid typid, int16 *typlen, bool *typbyval)
Definition: lsyscache.c:2401
Oid get_func_signature(Oid funcid, Oid **argtypes, int *nargs)
Definition: lsyscache.c:1846
Oid get_publication_oid(const char *pubname, bool missing_ok)
Definition: lsyscache.c:3775
Oid get_rel_namespace(Oid relid)
Definition: lsyscache.c:2102
static bool get_opmethod_canorder(Oid amoid)
Definition: lsyscache.c:221
RegProcedure get_opcode(Oid opno)
Definition: lsyscache.c:1435
Oid get_typcollation(Oid typid)
Definition: lsyscache.c:3206
Oid get_op_rettype(Oid opno)
Definition: lsyscache.c:1483
int get_op_opfamily_strategy(Oid opno, Oid opfamily)
Definition: lsyscache.c:85
char * get_collation_name(Oid colloid)
Definition: lsyscache.c:1111
Oid get_rel_type_id(Oid relid)
Definition: lsyscache.c:2129
char * get_language_name(Oid langoid, bool missing_ok)
Definition: lsyscache.c:1263
char * get_namespace_name_or_temp(Oid nspid)
Definition: lsyscache.c:3540
void getTypeInputInfo(Oid type, Oid *typInput, Oid *typIOParam)
Definition: lsyscache.c:3024
char func_volatile(Oid funcid)
Definition: lsyscache.c:1930
bool equality_ops_are_compatible(Oid opno1, Oid opno2)
Definition: lsyscache.c:771
get_attavgwidth_hook_type get_attavgwidth_hook
Definition: lsyscache.c:55
bool get_index_isclustered(Oid index_oid)
Definition: lsyscache.c:3751
Oid get_opfamily_member(Oid opfamily, Oid lefttype, Oid righttype, int16 strategy)
Definition: lsyscache.c:168
Oid get_ordering_op_for_equality_op(Oid opno, bool use_lhs_type)
Definition: lsyscache.c:362
Oid get_transform_tosql(Oid typid, Oid langid, List *trftypes)
Definition: lsyscache.c:2292
bool func_strict(Oid funcid)
Definition: lsyscache.c:1911
Oid get_index_column_opclass(Oid index_oid, int attno)
Definition: lsyscache.c:3662
char * get_constraint_name(Oid conoid)
Definition: lsyscache.c:1157
char * get_attname(Oid relid, AttrNumber attnum, bool missing_ok)
Definition: lsyscache.c:903
bool get_func_leakproof(Oid funcid)
Definition: lsyscache.c:1987
const struct SubscriptRoutines * getSubscriptingRoutines(Oid typid, Oid *typelemp)
Definition: lsyscache.c:3280
Node * get_typdefault(Oid typid)
Definition: lsyscache.c:2598
bool get_collation_isdeterministic(Oid colloid)
Definition: lsyscache.c:1130
List * get_op_index_interpretation(Oid opno)
Definition: lsyscache.c:666
Oid get_subscription_oid(const char *subname, bool missing_ok)
Definition: lsyscache.c:3825
char * get_subscription_name(Oid subid, bool missing_ok)
Definition: lsyscache.c:3845
Oid get_range_collation(Oid rangeOid)
Definition: lsyscache.c:3583
char * get_opfamily_name(Oid opfid, bool missing_ok)
Definition: lsyscache.c:1403
char * get_func_name(Oid funcid)
Definition: lsyscache.c:1758
Oid get_range_multirange(Oid rangeOid)
Definition: lsyscache.c:3608
Oid get_rel_relam(Oid relid)
Definition: lsyscache.c:2250
char op_volatile(Oid opno)
Definition: lsyscache.c:1643
Oid get_func_namespace(Oid funcid)
Definition: lsyscache.c:1782
bool type_is_collatable(Oid typid)
Definition: lsyscache.c:3231
Oid get_rel_tablespace(Oid relid)
Definition: lsyscache.c:2204
int get_func_nargs(Oid funcid)
Definition: lsyscache.c:1824
void get_type_io_data(Oid typid, IOFuncSelector which_func, int16 *typlen, bool *typbyval, char *typalign, char *typdelim, Oid *typioparam, Oid *func)
Definition: lsyscache.c:2475
int16 get_typlen(Oid typid)
Definition: lsyscache.c:2347
Oid get_typ_typrelid(Oid typid)
Definition: lsyscache.c:2881
char get_typtype(Oid typid)
Definition: lsyscache.c:2779
Oid get_base_element_type(Oid typid)
Definition: lsyscache.c:2982
Oid getTypeIOParam(HeapTuple typeTuple)
Definition: lsyscache.c:2453
Oid get_opfamily_method(Oid opfid)
Definition: lsyscache.c:1386
Oid getBaseTypeAndTypmod(Oid typid, int32 *typmod)
Definition: lsyscache.c:2688
Oid get_transform_fromsql(Oid typid, Oid langid, List *trftypes)
Definition: lsyscache.c:2270
char * get_publication_name(Oid pubid, bool missing_ok)
Definition: lsyscache.c:3795
Oid getBaseType(Oid typid)
Definition: lsyscache.c:2671
bool get_op_hash_functions(Oid opno, RegProcedure *lhs_procno, RegProcedure *rhs_procno)
Definition: lsyscache.c:575
bool get_typbyval(Oid typid)
Definition: lsyscache.c:2372
bool op_mergejoinable(Oid opno, Oid inputtype)
Definition: lsyscache.c:1536
List * get_mergejoin_opfamilies(Oid opno)
Definition: lsyscache.c:428
char * get_namespace_name(Oid nspid)
Definition: lsyscache.c:3516
Oid get_array_type(Oid typid)
Definition: lsyscache.c:2937
Oid get_func_rettype(Oid funcid)
Definition: lsyscache.c:1805
Oid get_promoted_array_type(Oid typid)
Definition: lsyscache.c:2961
Oid get_atttype(Oid relid, AttrNumber attnum)
Definition: lsyscache.c:989
char get_constraint_type(Oid conoid)
Definition: lsyscache.c:1219
int32 get_typavgwidth(Oid typid, int32 typmod)
Definition: lsyscache.c:2728
bool op_in_opfamily(Oid opno, Oid opfamily)
Definition: lsyscache.c:68
RegProcedure get_typsubscript(Oid typid, Oid *typelemp)
Definition: lsyscache.c:3247
void get_type_category_preferred(Oid typid, char *typcategory, bool *typispreferred)
Definition: lsyscache.c:2860
bool get_attstatsslot(AttStatsSlot *sslot, HeapTuple statstuple, int reqkind, Oid reqop, int flags)
Definition: lsyscache.c:3384
Oid get_relname_relid(const char *relname, Oid relnamespace)
Definition: lsyscache.c:2035
Oid get_negator(Oid opno)
Definition: lsyscache.c:1683
Oid get_commutator(Oid opno)
Definition: lsyscache.c:1659
void op_input_types(Oid opno, Oid *lefttype, Oid *righttype)
Definition: lsyscache.c:1508
bool type_is_multirange(Oid typid)
Definition: lsyscache.c:2848
void getTypeBinaryInputInfo(Oid type, Oid *typReceive, Oid *typIOParam)
Definition: lsyscache.c:3090
void get_atttypetypmodcoll(Oid relid, AttrNumber attnum, Oid *typid, int32 *typmod, Oid *collid)
Definition: lsyscache.c:1019
#define ATTSTATSSLOT_NUMBERS
Definition: lsyscache.h:44
#define ATTSTATSSLOT_VALUES
Definition: lsyscache.h:43
IOFuncSelector
Definition: lsyscache.h:35
@ IOFunc_output
Definition: lsyscache.h:37
@ IOFunc_input
Definition: lsyscache.h:36
@ IOFunc_send
Definition: lsyscache.h:39
@ IOFunc_receive
Definition: lsyscache.h:38
int32(* get_attavgwidth_hook_type)(Oid relid, AttrNumber attnum)
Definition: lsyscache.h:66
Const * makeConst(Oid consttype, int32 consttypmod, Oid constcollid, int constlen, Datum constvalue, bool constisnull, bool constbyval)
Definition: makefuncs.c:350
char * pstrdup(const char *in)
Definition: mcxt.c:1781
void pfree(void *pointer)
Definition: mcxt.c:1616
void * palloc(Size size)
Definition: mcxt.c:1387
#define IsBootstrapProcessingMode()
Definition: miscadmin.h:477
bool isTempNamespace(Oid namespaceId)
Definition: namespace.c:3719
FormData_pg_amop * Form_pg_amop
Definition: pg_amop.h:88
FormData_pg_amproc * Form_pg_amproc
Definition: pg_amproc.h:68
NameData attname
Definition: pg_attribute.h:41
int16 attnum
Definition: pg_attribute.h:74
FormData_pg_attribute * Form_pg_attribute
Definition: pg_attribute.h:202
NameData relname
Definition: pg_class.h:38
FormData_pg_class * Form_pg_class
Definition: pg_class.h:156
FormData_pg_collation * Form_pg_collation
Definition: pg_collation.h:58
FormData_pg_constraint * Form_pg_constraint
FormData_pg_database * Form_pg_database
Definition: pg_database.h:96
NameData datname
Definition: pg_database.h:35
FormData_pg_index * Form_pg_index
Definition: pg_index.h:70
FormData_pg_language * Form_pg_language
Definition: pg_language.h:65
#define NIL
Definition: pg_list.h:68
FormData_pg_namespace * Form_pg_namespace
Definition: pg_namespace.h:52
FormData_pg_opclass * Form_pg_opclass
Definition: pg_opclass.h:83
FormData_pg_operator * Form_pg_operator
Definition: pg_operator.h:83
FormData_pg_opfamily * Form_pg_opfamily
Definition: pg_opfamily.h:51
FormData_pg_proc * Form_pg_proc
Definition: pg_proc.h:136
int16 pronargs
Definition: pg_proc.h:81
FormData_pg_publication * Form_pg_publication
FormData_pg_range * Form_pg_range
Definition: pg_range.h:58
#define STATISTIC_NUM_SLOTS
Definition: pg_statistic.h:127
FormData_pg_statistic * Form_pg_statistic
Definition: pg_statistic.h:135
NameData subname
FormData_pg_subscription * Form_pg_subscription
FormData_pg_transform * Form_pg_transform
Definition: pg_transform.h:43
char typalign
Definition: pg_type.h:176
FormData_pg_type * Form_pg_type
Definition: pg_type.h:261
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:332
static Datum Int16GetDatum(int16 X)
Definition: postgres.h:182
static Datum BoolGetDatum(bool X)
Definition: postgres.h:112
static Datum ObjectIdGetDatum(Oid X)
Definition: postgres.h:262
uint64_t Datum
Definition: postgres.h:70
static Pointer DatumGetPointer(Datum X)
Definition: postgres.h:322
static Datum CStringGetDatum(const char *X)
Definition: postgres.h:360
static Datum CharGetDatum(char X)
Definition: postgres.h:132
#define InvalidOid
Definition: postgres_ext.h:37
unsigned int Oid
Definition: postgres_ext.h:32
void * stringToNode(const char *str)
Definition: read.c:90
uint16 StrategyNumber
Definition: stratnum.h:22
#define HTEqualStrategyNumber
Definition: stratnum.h:41
Oid valuetype
Definition: lsyscache.h:53
Datum * values
Definition: lsyscache.h:54
void * numbers_arr
Definition: lsyscache.h:62
float4 * numbers
Definition: lsyscache.h:57
int nnumbers
Definition: lsyscache.h:58
void * values_arr
Definition: lsyscache.h:61
bool amconsistentordering
Definition: amapi.h:254
bool amcanorder
Definition: amapi.h:246
bool amconsistentequality
Definition: amapi.h:252
Definition: pg_list.h:54
Definition: nodes.h:135
CompareType cmptype
Definition: lsyscache.h:28
CatCTup * members[FLEXIBLE_ARRAY_MEMBER]
Definition: catcache.h:182
int n_members
Definition: catcache.h:180
HeapTupleData tuple
Definition: catcache.h:124
Definition: c.h:745
Oid values[FLEXIBLE_ARRAY_MEMBER]
Definition: c.h:752
void ReleaseSysCache(HeapTuple tuple)
Definition: syscache.c:264
HeapTuple SearchSysCache1(int cacheId, Datum key1)
Definition: syscache.c:220
HeapTuple SearchSysCache3(int cacheId, Datum key1, Datum key2, Datum key3)
Definition: syscache.c:240
Datum SysCacheGetAttr(int cacheId, HeapTuple tup, AttrNumber attributeNumber, bool *isNull)
Definition: syscache.c:595
HeapTuple SearchSysCache2(int cacheId, Datum key1, Datum key2)
Definition: syscache.c:230
HeapTuple SearchSysCacheAttName(Oid relid, const char *attname)
Definition: syscache.c:475
HeapTuple SearchSysCache4(int cacheId, Datum key1, Datum key2, Datum key3, Datum key4)
Definition: syscache.c:250
Datum SysCacheGetAttrNotNull(int cacheId, HeapTuple tup, AttrNumber attributeNumber)
Definition: syscache.c:625
#define ReleaseSysCacheList(x)
Definition: syscache.h:134
#define SearchSysCacheList1(cacheId, key1)
Definition: syscache.h:127
#define SearchSysCacheExists3(cacheId, key1, key2, key3)
Definition: syscache.h:104
#define GetSysCacheOid1(cacheId, oidcol, key1)
Definition: syscache.h:109
#define GetSysCacheOid2(cacheId, oidcol, key1, key2)
Definition: syscache.h:111
TypeCacheEntry * lookup_type_cache(Oid type_id, int flags)
Definition: typcache.c:386
#define TYPECACHE_CMP_PROC
Definition: typcache.h:141
#define TYPECACHE_HASH_PROC
Definition: typcache.h:142
const char * type