LCOV - code coverage report
Current view: top level - src/backend/access/index - amvalidate.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 87 95 91.6 %
Date: 2025-01-18 03:14:54 Functions: 6 6 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * amvalidate.c
       4             :  *    Support routines for index access methods' amvalidate and
       5             :  *    amadjustmembers functions.
       6             :  *
       7             :  * Copyright (c) 2016-2025, PostgreSQL Global Development Group
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/access/index/amvalidate.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include "access/amvalidate.h"
      18             : #include "access/htup_details.h"
      19             : #include "catalog/pg_am.h"
      20             : #include "catalog/pg_amop.h"
      21             : #include "catalog/pg_amproc.h"
      22             : #include "catalog/pg_opclass.h"
      23             : #include "catalog/pg_operator.h"
      24             : #include "catalog/pg_proc.h"
      25             : #include "catalog/pg_type.h"
      26             : #include "parser/parse_coerce.h"
      27             : #include "utils/syscache.h"
      28             : 
      29             : 
      30             : /*
      31             :  * identify_opfamily_groups() returns a List of OpFamilyOpFuncGroup structs,
      32             :  * one for each combination of lefttype/righttype present in the family's
      33             :  * operator and support function lists.  If amopstrategy K is present for
      34             :  * this datatype combination, we set bit 1 << K in operatorset, and similarly
      35             :  * for the support functions.  With uint64 fields we can handle operator and
      36             :  * function numbers up to 63, which is plenty for the foreseeable future.
      37             :  *
      38             :  * The given CatCLists are expected to represent a single opfamily fetched
      39             :  * from the AMOPSTRATEGY and AMPROCNUM caches, so that they will be in
      40             :  * order by those caches' second and third cache keys, namely the datatypes.
      41             :  */
      42             : List *
      43        1264 : identify_opfamily_groups(CatCList *oprlist, CatCList *proclist)
      44             : {
      45        1264 :     List       *result = NIL;
      46             :     OpFamilyOpFuncGroup *thisgroup;
      47             :     Form_pg_amop oprform;
      48             :     Form_pg_amproc procform;
      49             :     int         io,
      50             :                 ip;
      51             : 
      52             :     /* We need the lists to be ordered; should be true in normal operation */
      53        1264 :     if (!oprlist->ordered || !proclist->ordered)
      54           0 :         elog(ERROR, "cannot validate operator family without ordered data");
      55             : 
      56             :     /*
      57             :      * Advance through the lists concurrently.  Thanks to the ordering, we
      58             :      * should see all operators and functions of a given datatype pair
      59             :      * consecutively.
      60             :      */
      61        1264 :     thisgroup = NULL;
      62        1264 :     io = ip = 0;
      63        1264 :     if (io < oprlist->n_members)
      64             :     {
      65        1264 :         oprform = (Form_pg_amop) GETSTRUCT(&oprlist->members[io]->tuple);
      66        1264 :         io++;
      67             :     }
      68             :     else
      69           0 :         oprform = NULL;
      70        1264 :     if (ip < proclist->n_members)
      71             :     {
      72        1264 :         procform = (Form_pg_amproc) GETSTRUCT(&proclist->members[ip]->tuple);
      73        1264 :         ip++;
      74             :     }
      75             :     else
      76           0 :         procform = NULL;
      77             : 
      78       26358 :     while (oprform || procform)
      79             :     {
      80       25094 :         if (oprform && thisgroup &&
      81       19002 :             oprform->amoplefttype == thisgroup->lefttype &&
      82       17440 :             oprform->amoprighttype == thisgroup->righttype)
      83             :         {
      84             :             /* Operator belongs to current group; include it and advance */
      85             : 
      86             :             /* Ignore strategy numbers outside supported range */
      87       13416 :             if (oprform->amopstrategy > 0 && oprform->amopstrategy < 64)
      88       13410 :                 thisgroup->operatorset |= ((uint64) 1) << oprform->amopstrategy;
      89             : 
      90       13416 :             if (io < oprlist->n_members)
      91             :             {
      92       12152 :                 oprform = (Form_pg_amop) GETSTRUCT(&oprlist->members[io]->tuple);
      93       12152 :                 io++;
      94             :             }
      95             :             else
      96        1264 :                 oprform = NULL;
      97       13416 :             continue;
      98             :         }
      99             : 
     100       11678 :         if (procform && thisgroup &&
     101       10314 :             procform->amproclefttype == thisgroup->lefttype &&
     102        9182 :             procform->amprocrighttype == thisgroup->righttype)
     103             :         {
     104             :             /* Procedure belongs to current group; include it and advance */
     105             : 
     106             :             /* Ignore function numbers outside supported range */
     107        7946 :             if (procform->amprocnum > 0 && procform->amprocnum < 64)
     108        7946 :                 thisgroup->functionset |= ((uint64) 1) << procform->amprocnum;
     109             : 
     110        7946 :             if (ip < proclist->n_members)
     111             :             {
     112        6682 :                 procform = (Form_pg_amproc) GETSTRUCT(&proclist->members[ip]->tuple);
     113        6682 :                 ip++;
     114             :             }
     115             :             else
     116        1264 :                 procform = NULL;
     117        7946 :             continue;
     118             :         }
     119             : 
     120             :         /* Time for a new group */
     121        3732 :         thisgroup = (OpFamilyOpFuncGroup *) palloc(sizeof(OpFamilyOpFuncGroup));
     122        3732 :         if (oprform &&
     123        3606 :             (!procform ||
     124        3606 :              (oprform->amoplefttype < procform->amproclefttype ||
     125        3118 :               (oprform->amoplefttype == procform->amproclefttype &&
     126        3078 :                oprform->amoprighttype < procform->amprocrighttype))))
     127             :         {
     128        1162 :             thisgroup->lefttype = oprform->amoplefttype;
     129        1162 :             thisgroup->righttype = oprform->amoprighttype;
     130             :         }
     131             :         else
     132             :         {
     133        2570 :             thisgroup->lefttype = procform->amproclefttype;
     134        2570 :             thisgroup->righttype = procform->amprocrighttype;
     135             :         }
     136        3732 :         thisgroup->operatorset = thisgroup->functionset = 0;
     137        3732 :         result = lappend(result, thisgroup);
     138             :     }
     139             : 
     140        1264 :     return result;
     141             : }
     142             : 
     143             : /*
     144             :  * Validate the signature (argument and result types) of an opclass support
     145             :  * function.  Return true if OK, false if not.
     146             :  *
     147             :  * The "..." represents maxargs argument-type OIDs.  If "exact" is true, they
     148             :  * must match the function arg types exactly, else only binary-coercibly.
     149             :  * In any case the function result type must match restype exactly.
     150             :  */
     151             : bool
     152        7472 : check_amproc_signature(Oid funcid, Oid restype, bool exact,
     153             :                        int minargs, int maxargs,...)
     154             : {
     155        7472 :     bool        result = true;
     156             :     HeapTuple   tp;
     157             :     Form_pg_proc procform;
     158             :     va_list     ap;
     159             :     int         i;
     160             : 
     161        7472 :     tp = SearchSysCache1(PROCOID, ObjectIdGetDatum(funcid));
     162        7472 :     if (!HeapTupleIsValid(tp))
     163           0 :         elog(ERROR, "cache lookup failed for function %u", funcid);
     164        7472 :     procform = (Form_pg_proc) GETSTRUCT(tp);
     165             : 
     166        7472 :     if (procform->prorettype != restype || procform->proretset ||
     167        7472 :         procform->pronargs < minargs || procform->pronargs > maxargs)
     168           0 :         result = false;
     169             : 
     170        7472 :     va_start(ap, maxargs);
     171       26168 :     for (i = 0; i < maxargs; i++)
     172             :     {
     173       18696 :         Oid         argtype = va_arg(ap, Oid);
     174             : 
     175       18696 :         if (i >= procform->pronargs)
     176         560 :             continue;
     177       21504 :         if (exact ? (argtype != procform->proargtypes.values[i]) :
     178        3368 :             !IsBinaryCoercible(argtype, procform->proargtypes.values[i]))
     179           0 :             result = false;
     180             :     }
     181        7472 :     va_end(ap);
     182             : 
     183        7472 :     ReleaseSysCache(tp);
     184        7472 :     return result;
     185             : }
     186             : 
     187             : /*
     188             :  * Validate the signature of an opclass options support function, that should
     189             :  * be 'void(internal)'.
     190             :  */
     191             : bool
     192         444 : check_amoptsproc_signature(Oid funcid)
     193             : {
     194         444 :     return check_amproc_signature(funcid, VOIDOID, true, 1, 1, INTERNALOID);
     195             : }
     196             : 
     197             : /*
     198             :  * Validate the signature (argument and result types) of an opclass operator.
     199             :  * Return true if OK, false if not.
     200             :  *
     201             :  * Currently, we can hard-wire this as accepting only binary operators.  Also,
     202             :  * we can insist on exact type matches, since the given lefttype/righttype
     203             :  * come from pg_amop and should always match the operator exactly.
     204             :  */
     205             : bool
     206       13416 : check_amop_signature(Oid opno, Oid restype, Oid lefttype, Oid righttype)
     207             : {
     208       13416 :     bool        result = true;
     209             :     HeapTuple   tp;
     210             :     Form_pg_operator opform;
     211             : 
     212       13416 :     tp = SearchSysCache1(OPEROID, ObjectIdGetDatum(opno));
     213       13416 :     if (!HeapTupleIsValid(tp))  /* shouldn't happen */
     214           0 :         elog(ERROR, "cache lookup failed for operator %u", opno);
     215       13416 :     opform = (Form_pg_operator) GETSTRUCT(tp);
     216             : 
     217       13416 :     if (opform->oprresult != restype || opform->oprkind != 'b' ||
     218       13416 :         opform->oprleft != lefttype || opform->oprright != righttype)
     219           0 :         result = false;
     220             : 
     221       13416 :     ReleaseSysCache(tp);
     222       13416 :     return result;
     223             : }
     224             : 
     225             : /*
     226             :  * Get the OID of the opclass belonging to an opfamily and accepting
     227             :  * the specified type as input type.  Returns InvalidOid if no such opclass.
     228             :  *
     229             :  * If there is more than one such opclass, you get a random one of them.
     230             :  * Since that shouldn't happen, we don't waste cycles checking.
     231             :  *
     232             :  * We could look up the AM's OID from the opfamily, but all existing callers
     233             :  * know that or can get it without an extra lookup, so we make them pass it.
     234             :  */
     235             : Oid
     236         138 : opclass_for_family_datatype(Oid amoid, Oid opfamilyoid, Oid datatypeoid)
     237             : {
     238         138 :     Oid         result = InvalidOid;
     239             :     CatCList   *opclist;
     240             :     int         i;
     241             : 
     242             :     /*
     243             :      * We search through all the AM's opclasses to see if one matches.  This
     244             :      * is a bit inefficient but there is no better index available.  It also
     245             :      * saves making an explicit check that the opfamily belongs to the AM.
     246             :      */
     247         138 :     opclist = SearchSysCacheList1(CLAAMNAMENSP, ObjectIdGetDatum(amoid));
     248             : 
     249        3280 :     for (i = 0; i < opclist->n_members; i++)
     250             :     {
     251        3234 :         HeapTuple   classtup = &opclist->members[i]->tuple;
     252        3234 :         Form_pg_opclass classform = (Form_pg_opclass) GETSTRUCT(classtup);
     253             : 
     254        3234 :         if (classform->opcfamily == opfamilyoid &&
     255         244 :             classform->opcintype == datatypeoid)
     256             :         {
     257          92 :             result = classform->oid;
     258          92 :             break;
     259             :         }
     260             :     }
     261             : 
     262         138 :     ReleaseCatCacheList(opclist);
     263             : 
     264         138 :     return result;
     265             : }
     266             : 
     267             : /*
     268             :  * Is the datatype a legitimate input type for the btree opfamily?
     269             :  */
     270             : bool
     271          92 : opfamily_can_sort_type(Oid opfamilyoid, Oid datatypeoid)
     272             : {
     273          92 :     return OidIsValid(opclass_for_family_datatype(BTREE_AM_OID,
     274             :                                                   opfamilyoid,
     275             :                                                   datatypeoid));
     276             : }

Generated by: LCOV version 1.14