LCOV - code coverage report
Current view: top level - src/backend/catalog - pg_type.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 263 281 93.6 %
Date: 2025-01-18 03:14:54 Functions: 7 7 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * pg_type.c
       4             :  *    routines to support manipulation of the pg_type relation
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  * Portions Copyright (c) 1994, Regents of the University of California
       8             :  *
       9             :  *
      10             :  * IDENTIFICATION
      11             :  *    src/backend/catalog/pg_type.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include "access/htup_details.h"
      18             : #include "access/table.h"
      19             : #include "access/xact.h"
      20             : #include "catalog/binary_upgrade.h"
      21             : #include "catalog/catalog.h"
      22             : #include "catalog/dependency.h"
      23             : #include "catalog/indexing.h"
      24             : #include "catalog/objectaccess.h"
      25             : #include "catalog/pg_collation.h"
      26             : #include "catalog/pg_namespace.h"
      27             : #include "catalog/pg_proc.h"
      28             : #include "catalog/pg_type.h"
      29             : #include "commands/defrem.h"
      30             : #include "commands/typecmds.h"
      31             : #include "mb/pg_wchar.h"
      32             : #include "miscadmin.h"
      33             : #include "utils/acl.h"
      34             : #include "utils/builtins.h"
      35             : #include "utils/fmgroids.h"
      36             : #include "utils/lsyscache.h"
      37             : #include "utils/rel.h"
      38             : #include "utils/syscache.h"
      39             : 
      40             : /* Potentially set by pg_upgrade_support functions */
      41             : Oid         binary_upgrade_next_pg_type_oid = InvalidOid;
      42             : 
      43             : /* ----------------------------------------------------------------
      44             :  *      TypeShellMake
      45             :  *
      46             :  *      This procedure inserts a "shell" tuple into the pg_type relation.
      47             :  *      The type tuple inserted has valid but dummy values, and its
      48             :  *      "typisdefined" field is false indicating it's not really defined.
      49             :  *
      50             :  *      This is used so that a tuple exists in the catalogs.  The I/O
      51             :  *      functions for the type will link to this tuple.  When the full
      52             :  *      CREATE TYPE command is issued, the bogus values will be replaced
      53             :  *      with correct ones, and "typisdefined" will be set to true.
      54             :  * ----------------------------------------------------------------
      55             :  */
      56             : ObjectAddress
      57         216 : TypeShellMake(const char *typeName, Oid typeNamespace, Oid ownerId)
      58             : {
      59             :     Relation    pg_type_desc;
      60             :     TupleDesc   tupDesc;
      61             :     int         i;
      62             :     HeapTuple   tup;
      63             :     Datum       values[Natts_pg_type];
      64             :     bool        nulls[Natts_pg_type];
      65             :     Oid         typoid;
      66             :     NameData    name;
      67             :     ObjectAddress address;
      68             : 
      69             :     Assert(PointerIsValid(typeName));
      70             : 
      71             :     /*
      72             :      * open pg_type
      73             :      */
      74         216 :     pg_type_desc = table_open(TypeRelationId, RowExclusiveLock);
      75         216 :     tupDesc = pg_type_desc->rd_att;
      76             : 
      77             :     /*
      78             :      * initialize our *nulls and *values arrays
      79             :      */
      80        7128 :     for (i = 0; i < Natts_pg_type; ++i)
      81             :     {
      82        6912 :         nulls[i] = false;
      83        6912 :         values[i] = (Datum) NULL;   /* redundant, but safe */
      84             :     }
      85             : 
      86             :     /*
      87             :      * initialize *values with the type name and dummy values
      88             :      *
      89             :      * The representational details are the same as int4 ... it doesn't really
      90             :      * matter what they are so long as they are consistent.  Also note that we
      91             :      * give it typtype = TYPTYPE_PSEUDO as extra insurance that it won't be
      92             :      * mistaken for a usable type.
      93             :      */
      94         216 :     namestrcpy(&name, typeName);
      95         216 :     values[Anum_pg_type_typname - 1] = NameGetDatum(&name);
      96         216 :     values[Anum_pg_type_typnamespace - 1] = ObjectIdGetDatum(typeNamespace);
      97         216 :     values[Anum_pg_type_typowner - 1] = ObjectIdGetDatum(ownerId);
      98         216 :     values[Anum_pg_type_typlen - 1] = Int16GetDatum(sizeof(int32));
      99         216 :     values[Anum_pg_type_typbyval - 1] = BoolGetDatum(true);
     100         216 :     values[Anum_pg_type_typtype - 1] = CharGetDatum(TYPTYPE_PSEUDO);
     101         216 :     values[Anum_pg_type_typcategory - 1] = CharGetDatum(TYPCATEGORY_PSEUDOTYPE);
     102         216 :     values[Anum_pg_type_typispreferred - 1] = BoolGetDatum(false);
     103         216 :     values[Anum_pg_type_typisdefined - 1] = BoolGetDatum(false);
     104         216 :     values[Anum_pg_type_typdelim - 1] = CharGetDatum(DEFAULT_TYPDELIM);
     105         216 :     values[Anum_pg_type_typrelid - 1] = ObjectIdGetDatum(InvalidOid);
     106         216 :     values[Anum_pg_type_typsubscript - 1] = ObjectIdGetDatum(InvalidOid);
     107         216 :     values[Anum_pg_type_typelem - 1] = ObjectIdGetDatum(InvalidOid);
     108         216 :     values[Anum_pg_type_typarray - 1] = ObjectIdGetDatum(InvalidOid);
     109         216 :     values[Anum_pg_type_typinput - 1] = ObjectIdGetDatum(F_SHELL_IN);
     110         216 :     values[Anum_pg_type_typoutput - 1] = ObjectIdGetDatum(F_SHELL_OUT);
     111         216 :     values[Anum_pg_type_typreceive - 1] = ObjectIdGetDatum(InvalidOid);
     112         216 :     values[Anum_pg_type_typsend - 1] = ObjectIdGetDatum(InvalidOid);
     113         216 :     values[Anum_pg_type_typmodin - 1] = ObjectIdGetDatum(InvalidOid);
     114         216 :     values[Anum_pg_type_typmodout - 1] = ObjectIdGetDatum(InvalidOid);
     115         216 :     values[Anum_pg_type_typanalyze - 1] = ObjectIdGetDatum(InvalidOid);
     116         216 :     values[Anum_pg_type_typalign - 1] = CharGetDatum(TYPALIGN_INT);
     117         216 :     values[Anum_pg_type_typstorage - 1] = CharGetDatum(TYPSTORAGE_PLAIN);
     118         216 :     values[Anum_pg_type_typnotnull - 1] = BoolGetDatum(false);
     119         216 :     values[Anum_pg_type_typbasetype - 1] = ObjectIdGetDatum(InvalidOid);
     120         216 :     values[Anum_pg_type_typtypmod - 1] = Int32GetDatum(-1);
     121         216 :     values[Anum_pg_type_typndims - 1] = Int32GetDatum(0);
     122         216 :     values[Anum_pg_type_typcollation - 1] = ObjectIdGetDatum(InvalidOid);
     123         216 :     nulls[Anum_pg_type_typdefaultbin - 1] = true;
     124         216 :     nulls[Anum_pg_type_typdefault - 1] = true;
     125         216 :     nulls[Anum_pg_type_typacl - 1] = true;
     126             : 
     127             :     /* Use binary-upgrade override for pg_type.oid? */
     128         216 :     if (IsBinaryUpgrade)
     129             :     {
     130          16 :         if (!OidIsValid(binary_upgrade_next_pg_type_oid))
     131           0 :             ereport(ERROR,
     132             :                     (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     133             :                      errmsg("pg_type OID value not set when in binary upgrade mode")));
     134             : 
     135          16 :         typoid = binary_upgrade_next_pg_type_oid;
     136          16 :         binary_upgrade_next_pg_type_oid = InvalidOid;
     137             :     }
     138             :     else
     139             :     {
     140         200 :         typoid = GetNewOidWithIndex(pg_type_desc, TypeOidIndexId,
     141             :                                     Anum_pg_type_oid);
     142             :     }
     143             : 
     144         216 :     values[Anum_pg_type_oid - 1] = ObjectIdGetDatum(typoid);
     145             : 
     146             :     /*
     147             :      * create a new type tuple
     148             :      */
     149         216 :     tup = heap_form_tuple(tupDesc, values, nulls);
     150             : 
     151             :     /*
     152             :      * insert the tuple in the relation and get the tuple's oid.
     153             :      */
     154         216 :     CatalogTupleInsert(pg_type_desc, tup);
     155             : 
     156             :     /*
     157             :      * Create dependencies.  We can/must skip this in bootstrap mode.
     158             :      */
     159         216 :     if (!IsBootstrapProcessingMode())
     160         216 :         GenerateTypeDependencies(tup,
     161             :                                  pg_type_desc,
     162             :                                  NULL,
     163             :                                  NULL,
     164             :                                  0,
     165             :                                  false,
     166             :                                  false,
     167             :                                  true,  /* make extension dependency */
     168             :                                  false);
     169             : 
     170             :     /* Post creation hook for new shell type */
     171         216 :     InvokeObjectPostCreateHook(TypeRelationId, typoid, 0);
     172             : 
     173         216 :     ObjectAddressSet(address, TypeRelationId, typoid);
     174             : 
     175             :     /*
     176             :      * clean up and return the type-oid
     177             :      */
     178         216 :     heap_freetuple(tup);
     179         216 :     table_close(pg_type_desc, RowExclusiveLock);
     180             : 
     181         216 :     return address;
     182             : }
     183             : 
     184             : /* ----------------------------------------------------------------
     185             :  *      TypeCreate
     186             :  *
     187             :  *      This does all the necessary work needed to define a new type.
     188             :  *
     189             :  *      Returns the ObjectAddress assigned to the new type.
     190             :  *      If newTypeOid is zero (the normal case), a new OID is created;
     191             :  *      otherwise we use exactly that OID.
     192             :  * ----------------------------------------------------------------
     193             :  */
     194             : ObjectAddress
     195      129468 : TypeCreate(Oid newTypeOid,
     196             :            const char *typeName,
     197             :            Oid typeNamespace,
     198             :            Oid relationOid,     /* only for relation rowtypes */
     199             :            char relationKind,   /* ditto */
     200             :            Oid ownerId,
     201             :            int16 internalSize,
     202             :            char typeType,
     203             :            char typeCategory,
     204             :            bool typePreferred,
     205             :            char typDelim,
     206             :            Oid inputProcedure,
     207             :            Oid outputProcedure,
     208             :            Oid receiveProcedure,
     209             :            Oid sendProcedure,
     210             :            Oid typmodinProcedure,
     211             :            Oid typmodoutProcedure,
     212             :            Oid analyzeProcedure,
     213             :            Oid subscriptProcedure,
     214             :            Oid elementType,
     215             :            bool isImplicitArray,
     216             :            Oid arrayType,
     217             :            Oid baseType,
     218             :            const char *defaultTypeValue,    /* human-readable rep */
     219             :            char *defaultTypeBin,    /* cooked rep */
     220             :            bool passedByValue,
     221             :            char alignment,
     222             :            char storage,
     223             :            int32 typeMod,
     224             :            int32 typNDims,      /* Array dimensions for baseType */
     225             :            bool typeNotNull,
     226             :            Oid typeCollation)
     227             : {
     228             :     Relation    pg_type_desc;
     229             :     Oid         typeObjectId;
     230             :     bool        isDependentType;
     231      129468 :     bool        rebuildDeps = false;
     232             :     Acl        *typacl;
     233             :     HeapTuple   tup;
     234             :     bool        nulls[Natts_pg_type];
     235             :     bool        replaces[Natts_pg_type];
     236             :     Datum       values[Natts_pg_type];
     237             :     NameData    name;
     238             :     int         i;
     239             :     ObjectAddress address;
     240             : 
     241             :     /*
     242             :      * We assume that the caller validated the arguments individually, but did
     243             :      * not check for bad combinations.
     244             :      *
     245             :      * Validate size specifications: either positive (fixed-length) or -1
     246             :      * (varlena) or -2 (cstring).
     247             :      */
     248      129468 :     if (!(internalSize > 0 ||
     249             :           internalSize == -1 ||
     250             :           internalSize == -2))
     251           0 :         ereport(ERROR,
     252             :                 (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     253             :                  errmsg("invalid type internal size %d",
     254             :                         internalSize)));
     255             : 
     256      129468 :     if (passedByValue)
     257             :     {
     258             :         /*
     259             :          * Pass-by-value types must have a fixed length that is one of the
     260             :          * values supported by fetch_att() and store_att_byval(); and the
     261             :          * alignment had better agree, too.  All this code must match
     262             :          * access/tupmacs.h!
     263             :          */
     264        1014 :         if (internalSize == (int16) sizeof(char))
     265             :         {
     266          12 :             if (alignment != TYPALIGN_CHAR)
     267           0 :                 ereport(ERROR,
     268             :                         (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     269             :                          errmsg("alignment \"%c\" is invalid for passed-by-value type of size %d",
     270             :                                 alignment, internalSize)));
     271             :         }
     272        1002 :         else if (internalSize == (int16) sizeof(int16))
     273             :         {
     274           4 :             if (alignment != TYPALIGN_SHORT)
     275           0 :                 ereport(ERROR,
     276             :                         (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     277             :                          errmsg("alignment \"%c\" is invalid for passed-by-value type of size %d",
     278             :                                 alignment, internalSize)));
     279             :         }
     280         998 :         else if (internalSize == (int16) sizeof(int32))
     281             :         {
     282         864 :             if (alignment != TYPALIGN_INT)
     283           0 :                 ereport(ERROR,
     284             :                         (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     285             :                          errmsg("alignment \"%c\" is invalid for passed-by-value type of size %d",
     286             :                                 alignment, internalSize)));
     287             :         }
     288             : #if SIZEOF_DATUM == 8
     289         134 :         else if (internalSize == (int16) sizeof(Datum))
     290             :         {
     291         134 :             if (alignment != TYPALIGN_DOUBLE)
     292           0 :                 ereport(ERROR,
     293             :                         (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     294             :                          errmsg("alignment \"%c\" is invalid for passed-by-value type of size %d",
     295             :                                 alignment, internalSize)));
     296             :         }
     297             : #endif
     298             :         else
     299           0 :             ereport(ERROR,
     300             :                     (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     301             :                      errmsg("internal size %d is invalid for passed-by-value type",
     302             :                             internalSize)));
     303             :     }
     304             :     else
     305             :     {
     306             :         /* varlena types must have int align or better */
     307      128454 :         if (internalSize == -1 &&
     308      125582 :             !(alignment == TYPALIGN_INT || alignment == TYPALIGN_DOUBLE))
     309           0 :             ereport(ERROR,
     310             :                     (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     311             :                      errmsg("alignment \"%c\" is invalid for variable-length type",
     312             :                             alignment)));
     313             :         /* cstring must have char alignment */
     314      128454 :         if (internalSize == -2 && !(alignment == TYPALIGN_CHAR))
     315           0 :             ereport(ERROR,
     316             :                     (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     317             :                      errmsg("alignment \"%c\" is invalid for variable-length type",
     318             :                             alignment)));
     319             :     }
     320             : 
     321             :     /* Only varlena types can be toasted */
     322      129468 :     if (storage != TYPSTORAGE_PLAIN && internalSize != -1)
     323           0 :         ereport(ERROR,
     324             :                 (errcode(ERRCODE_INVALID_OBJECT_DEFINITION),
     325             :                  errmsg("fixed-size types must have storage PLAIN")));
     326             : 
     327             :     /*
     328             :      * This is a dependent type if it's an implicitly-created array type or
     329             :      * multirange type, or if it's a relation rowtype that's not a composite
     330             :      * type.  For such types we'll leave the ACL empty, and we'll skip
     331             :      * creating some dependency records because there will be a dependency
     332             :      * already through the depended-on type or relation.  (Caution: this is
     333             :      * closely intertwined with some behavior in GenerateTypeDependencies.)
     334             :      */
     335       64744 :     isDependentType = isImplicitArray ||
     336      256798 :         typeType == TYPTYPE_MULTIRANGE ||
     337       62586 :         (OidIsValid(relationOid) && relationKind != RELKIND_COMPOSITE_TYPE);
     338             : 
     339             :     /*
     340             :      * initialize arrays needed for heap_form_tuple or heap_modify_tuple
     341             :      */
     342     4272444 :     for (i = 0; i < Natts_pg_type; ++i)
     343             :     {
     344     4142976 :         nulls[i] = false;
     345     4142976 :         replaces[i] = true;
     346     4142976 :         values[i] = (Datum) 0;
     347             :     }
     348             : 
     349             :     /*
     350             :      * insert data values
     351             :      */
     352      129468 :     namestrcpy(&name, typeName);
     353      129468 :     values[Anum_pg_type_typname - 1] = NameGetDatum(&name);
     354      129468 :     values[Anum_pg_type_typnamespace - 1] = ObjectIdGetDatum(typeNamespace);
     355      129468 :     values[Anum_pg_type_typowner - 1] = ObjectIdGetDatum(ownerId);
     356      129468 :     values[Anum_pg_type_typlen - 1] = Int16GetDatum(internalSize);
     357      129468 :     values[Anum_pg_type_typbyval - 1] = BoolGetDatum(passedByValue);
     358      129468 :     values[Anum_pg_type_typtype - 1] = CharGetDatum(typeType);
     359      129468 :     values[Anum_pg_type_typcategory - 1] = CharGetDatum(typeCategory);
     360      129468 :     values[Anum_pg_type_typispreferred - 1] = BoolGetDatum(typePreferred);
     361      129468 :     values[Anum_pg_type_typisdefined - 1] = BoolGetDatum(true);
     362      129468 :     values[Anum_pg_type_typdelim - 1] = CharGetDatum(typDelim);
     363      129468 :     values[Anum_pg_type_typrelid - 1] = ObjectIdGetDatum(relationOid);
     364      129468 :     values[Anum_pg_type_typsubscript - 1] = ObjectIdGetDatum(subscriptProcedure);
     365      129468 :     values[Anum_pg_type_typelem - 1] = ObjectIdGetDatum(elementType);
     366      129468 :     values[Anum_pg_type_typarray - 1] = ObjectIdGetDatum(arrayType);
     367      129468 :     values[Anum_pg_type_typinput - 1] = ObjectIdGetDatum(inputProcedure);
     368      129468 :     values[Anum_pg_type_typoutput - 1] = ObjectIdGetDatum(outputProcedure);
     369      129468 :     values[Anum_pg_type_typreceive - 1] = ObjectIdGetDatum(receiveProcedure);
     370      129468 :     values[Anum_pg_type_typsend - 1] = ObjectIdGetDatum(sendProcedure);
     371      129468 :     values[Anum_pg_type_typmodin - 1] = ObjectIdGetDatum(typmodinProcedure);
     372      129468 :     values[Anum_pg_type_typmodout - 1] = ObjectIdGetDatum(typmodoutProcedure);
     373      129468 :     values[Anum_pg_type_typanalyze - 1] = ObjectIdGetDatum(analyzeProcedure);
     374      129468 :     values[Anum_pg_type_typalign - 1] = CharGetDatum(alignment);
     375      129468 :     values[Anum_pg_type_typstorage - 1] = CharGetDatum(storage);
     376      129468 :     values[Anum_pg_type_typnotnull - 1] = BoolGetDatum(typeNotNull);
     377      129468 :     values[Anum_pg_type_typbasetype - 1] = ObjectIdGetDatum(baseType);
     378      129468 :     values[Anum_pg_type_typtypmod - 1] = Int32GetDatum(typeMod);
     379      129468 :     values[Anum_pg_type_typndims - 1] = Int32GetDatum(typNDims);
     380      129468 :     values[Anum_pg_type_typcollation - 1] = ObjectIdGetDatum(typeCollation);
     381             : 
     382             :     /*
     383             :      * initialize the default binary value for this type.  Check for nulls of
     384             :      * course.
     385             :      */
     386      129468 :     if (defaultTypeBin)
     387         128 :         values[Anum_pg_type_typdefaultbin - 1] = CStringGetTextDatum(defaultTypeBin);
     388             :     else
     389      129340 :         nulls[Anum_pg_type_typdefaultbin - 1] = true;
     390             : 
     391             :     /*
     392             :      * initialize the default value for this type.
     393             :      */
     394      129468 :     if (defaultTypeValue)
     395         146 :         values[Anum_pg_type_typdefault - 1] = CStringGetTextDatum(defaultTypeValue);
     396             :     else
     397      129322 :         nulls[Anum_pg_type_typdefault - 1] = true;
     398             : 
     399             :     /*
     400             :      * Initialize the type's ACL, too.  But dependent types don't get one.
     401             :      */
     402      129468 :     if (isDependentType)
     403      123980 :         typacl = NULL;
     404             :     else
     405        5488 :         typacl = get_user_default_acl(OBJECT_TYPE, ownerId,
     406             :                                       typeNamespace);
     407      129468 :     if (typacl != NULL)
     408           6 :         values[Anum_pg_type_typacl - 1] = PointerGetDatum(typacl);
     409             :     else
     410      129462 :         nulls[Anum_pg_type_typacl - 1] = true;
     411             : 
     412             :     /*
     413             :      * open pg_type and prepare to insert or update a row.
     414             :      *
     415             :      * NOTE: updating will not work correctly in bootstrap mode; but we don't
     416             :      * expect to be overwriting any shell types in bootstrap mode.
     417             :      */
     418      129468 :     pg_type_desc = table_open(TypeRelationId, RowExclusiveLock);
     419             : 
     420      129468 :     tup = SearchSysCacheCopy2(TYPENAMENSP,
     421             :                               CStringGetDatum(typeName),
     422             :                               ObjectIdGetDatum(typeNamespace));
     423      129468 :     if (HeapTupleIsValid(tup))
     424             :     {
     425         192 :         Form_pg_type typform = (Form_pg_type) GETSTRUCT(tup);
     426             : 
     427             :         /*
     428             :          * check that the type is not already defined.  It may exist as a
     429             :          * shell type, however.
     430             :          */
     431         192 :         if (typform->typisdefined)
     432           0 :             ereport(ERROR,
     433             :                     (errcode(ERRCODE_DUPLICATE_OBJECT),
     434             :                      errmsg("type \"%s\" already exists", typeName)));
     435             : 
     436             :         /*
     437             :          * shell type must have been created by same owner
     438             :          */
     439         192 :         if (typform->typowner != ownerId)
     440           0 :             aclcheck_error(ACLCHECK_NOT_OWNER, OBJECT_TYPE, typeName);
     441             : 
     442             :         /* trouble if caller wanted to force the OID */
     443         192 :         if (OidIsValid(newTypeOid))
     444           0 :             elog(ERROR, "cannot assign new OID to existing shell type");
     445             : 
     446         192 :         replaces[Anum_pg_type_oid - 1] = false;
     447             : 
     448             :         /*
     449             :          * Okay to update existing shell type tuple
     450             :          */
     451         192 :         tup = heap_modify_tuple(tup,
     452             :                                 RelationGetDescr(pg_type_desc),
     453             :                                 values,
     454             :                                 nulls,
     455             :                                 replaces);
     456             : 
     457         192 :         CatalogTupleUpdate(pg_type_desc, &tup->t_self, tup);
     458             : 
     459         192 :         typeObjectId = typform->oid;
     460             : 
     461         192 :         rebuildDeps = true;     /* get rid of shell type's dependencies */
     462             :     }
     463             :     else
     464             :     {
     465             :         /* Force the OID if requested by caller */
     466      129276 :         if (OidIsValid(newTypeOid))
     467       65318 :             typeObjectId = newTypeOid;
     468             :         /* Use binary-upgrade override for pg_type.oid, if supplied. */
     469       63958 :         else if (IsBinaryUpgrade)
     470             :         {
     471        1512 :             if (!OidIsValid(binary_upgrade_next_pg_type_oid))
     472           0 :                 ereport(ERROR,
     473             :                         (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     474             :                          errmsg("pg_type OID value not set when in binary upgrade mode")));
     475             : 
     476        1512 :             typeObjectId = binary_upgrade_next_pg_type_oid;
     477        1512 :             binary_upgrade_next_pg_type_oid = InvalidOid;
     478             :         }
     479             :         else
     480             :         {
     481       62446 :             typeObjectId = GetNewOidWithIndex(pg_type_desc, TypeOidIndexId,
     482             :                                               Anum_pg_type_oid);
     483             :         }
     484             : 
     485      129276 :         values[Anum_pg_type_oid - 1] = ObjectIdGetDatum(typeObjectId);
     486             : 
     487      129276 :         tup = heap_form_tuple(RelationGetDescr(pg_type_desc),
     488             :                               values, nulls);
     489             : 
     490      129276 :         CatalogTupleInsert(pg_type_desc, tup);
     491             :     }
     492             : 
     493             :     /*
     494             :      * Create dependencies.  We can/must skip this in bootstrap mode.
     495             :      */
     496      129468 :     if (!IsBootstrapProcessingMode())
     497      118796 :         GenerateTypeDependencies(tup,
     498             :                                  pg_type_desc,
     499             :                                  (defaultTypeBin ?
     500         128 :                                   stringToNode(defaultTypeBin) :
     501             :                                   NULL),
     502             :                                  typacl,
     503             :                                  relationKind,
     504             :                                  isImplicitArray,
     505             :                                  isDependentType,
     506             :                                  true,  /* make extension dependency */
     507             :                                  rebuildDeps);
     508             : 
     509             :     /* Post creation hook for new type */
     510      129466 :     InvokeObjectPostCreateHook(TypeRelationId, typeObjectId, 0);
     511             : 
     512      129466 :     ObjectAddressSet(address, TypeRelationId, typeObjectId);
     513             : 
     514             :     /*
     515             :      * finish up
     516             :      */
     517      129466 :     table_close(pg_type_desc, RowExclusiveLock);
     518             : 
     519      129466 :     return address;
     520             : }
     521             : 
     522             : /*
     523             :  * GenerateTypeDependencies: build the dependencies needed for a type
     524             :  *
     525             :  * Most of what this function needs to know about the type is passed as the
     526             :  * new pg_type row, typeTuple.  We make callers pass the pg_type Relation
     527             :  * as well, so that we have easy access to a tuple descriptor for the row.
     528             :  *
     529             :  * While this is able to extract the defaultExpr and typacl from the tuple,
     530             :  * doing so is relatively expensive, and callers may have those values at
     531             :  * hand already.  Pass those if handy, otherwise pass NULL.  (typacl is really
     532             :  * "Acl *", but we declare it "void *" to avoid including acl.h in pg_type.h.)
     533             :  *
     534             :  * relationKind and isImplicitArray are likewise somewhat expensive to deduce
     535             :  * from the tuple, so we make callers pass those (they're not optional).
     536             :  *
     537             :  * isDependentType is true if this is an implicit array, multirange, or
     538             :  * relation rowtype; that means it doesn't need its own dependencies on owner
     539             :  * etc.
     540             :  *
     541             :  * We make an extension-membership dependency if we're in an extension
     542             :  * script and makeExtensionDep is true.
     543             :  * makeExtensionDep should be true when creating a new type or replacing a
     544             :  * shell type, but not for ALTER TYPE on an existing type.  Passing false
     545             :  * causes the type's extension membership to be left alone.
     546             :  *
     547             :  * rebuild should be true if this is a pre-existing type.  We will remove
     548             :  * existing dependencies and rebuild them from scratch.  This is needed for
     549             :  * ALTER TYPE, and also when replacing a shell type.  We don't remove any
     550             :  * existing extension dependency, though; hence, if makeExtensionDep is also
     551             :  * true and we're in an extension script, an error will occur unless the
     552             :  * type already belongs to the current extension.  That's the behavior we
     553             :  * want when replacing a shell type, which is the only case where both flags
     554             :  * are true.
     555             :  */
     556             : void
     557      118964 : GenerateTypeDependencies(HeapTuple typeTuple,
     558             :                          Relation typeCatalog,
     559             :                          Node *defaultExpr,
     560             :                          void *typacl,
     561             :                          char relationKind, /* only for relation rowtypes */
     562             :                          bool isImplicitArray,
     563             :                          bool isDependentType,
     564             :                          bool makeExtensionDep,
     565             :                          bool rebuild)
     566             : {
     567      118964 :     Form_pg_type typeForm = (Form_pg_type) GETSTRUCT(typeTuple);
     568      118964 :     Oid         typeObjectId = typeForm->oid;
     569             :     Datum       datum;
     570             :     bool        isNull;
     571             :     ObjectAddress myself,
     572             :                 referenced;
     573             :     ObjectAddresses *addrs_normal;
     574             : 
     575             :     /* Extract defaultExpr if caller didn't pass it */
     576      118964 :     if (defaultExpr == NULL)
     577             :     {
     578      118828 :         datum = heap_getattr(typeTuple, Anum_pg_type_typdefaultbin,
     579             :                              RelationGetDescr(typeCatalog), &isNull);
     580      118828 :         if (!isNull)
     581           0 :             defaultExpr = stringToNode(TextDatumGetCString(datum));
     582             :     }
     583             :     /* Extract typacl if caller didn't pass it */
     584      118964 :     if (typacl == NULL)
     585             :     {
     586      118958 :         datum = heap_getattr(typeTuple, Anum_pg_type_typacl,
     587             :                              RelationGetDescr(typeCatalog), &isNull);
     588      118958 :         if (!isNull)
     589           0 :             typacl = DatumGetAclPCopy(datum);
     590             :     }
     591             : 
     592             :     /* If rebuild, first flush old dependencies, except extension deps */
     593      118964 :     if (rebuild)
     594             :     {
     595         272 :         deleteDependencyRecordsFor(TypeRelationId, typeObjectId, true);
     596         272 :         deleteSharedDependencyRecordsFor(TypeRelationId, typeObjectId, 0);
     597             :     }
     598             : 
     599      118964 :     ObjectAddressSet(myself, TypeRelationId, typeObjectId);
     600             : 
     601             :     /*
     602             :      * Make dependencies on namespace, owner, ACL.
     603             :      *
     604             :      * Skip these for a dependent type, since it will have such dependencies
     605             :      * indirectly through its depended-on type or relation.  An exception is
     606             :      * that multiranges need their own namespace dependency, since we don't
     607             :      * force them to be in the same schema as their range type.
     608             :      */
     609             : 
     610             :     /* collects normal dependencies for bulk recording */
     611      118964 :     addrs_normal = new_object_addresses();
     612             : 
     613      118964 :     if (!isDependentType || typeForm->typtype == TYPTYPE_MULTIRANGE)
     614             :     {
     615        5922 :         ObjectAddressSet(referenced, NamespaceRelationId,
     616             :                          typeForm->typnamespace);
     617        5922 :         add_exact_object_address(&referenced, addrs_normal);
     618             :     }
     619             : 
     620      118964 :     if (!isDependentType)
     621             :     {
     622        5778 :         recordDependencyOnOwner(TypeRelationId, typeObjectId,
     623             :                                 typeForm->typowner);
     624             : 
     625        5778 :         recordDependencyOnNewAcl(TypeRelationId, typeObjectId, 0,
     626             :                                  typeForm->typowner, typacl);
     627             :     }
     628             : 
     629             :     /*
     630             :      * Make extension dependency if requested.
     631             :      *
     632             :      * We used to skip this for dependent types, but it seems better to record
     633             :      * their extension membership explicitly; otherwise code such as
     634             :      * postgres_fdw's shippability test will be fooled.
     635             :      */
     636      118964 :     if (makeExtensionDep)
     637      118884 :         recordDependencyOnCurrentExtension(&myself, rebuild);
     638             : 
     639             :     /* Normal dependencies on the I/O and support functions */
     640      118962 :     if (OidIsValid(typeForm->typinput))
     641             :     {
     642      118962 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typinput);
     643      118962 :         add_exact_object_address(&referenced, addrs_normal);
     644             :     }
     645             : 
     646      118962 :     if (OidIsValid(typeForm->typoutput))
     647             :     {
     648      118962 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typoutput);
     649      118962 :         add_exact_object_address(&referenced, addrs_normal);
     650             :     }
     651             : 
     652      118962 :     if (OidIsValid(typeForm->typreceive))
     653             :     {
     654      118568 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typreceive);
     655      118568 :         add_exact_object_address(&referenced, addrs_normal);
     656             :     }
     657             : 
     658      118962 :     if (OidIsValid(typeForm->typsend))
     659             :     {
     660      118556 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typsend);
     661      118556 :         add_exact_object_address(&referenced, addrs_normal);
     662             :     }
     663             : 
     664      118962 :     if (OidIsValid(typeForm->typmodin))
     665             :     {
     666          28 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typmodin);
     667          28 :         add_exact_object_address(&referenced, addrs_normal);
     668             :     }
     669             : 
     670      118962 :     if (OidIsValid(typeForm->typmodout))
     671             :     {
     672          28 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typmodout);
     673          28 :         add_exact_object_address(&referenced, addrs_normal);
     674             :     }
     675             : 
     676      118962 :     if (OidIsValid(typeForm->typanalyze))
     677             :     {
     678       59788 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typanalyze);
     679       59788 :         add_exact_object_address(&referenced, addrs_normal);
     680             :     }
     681             : 
     682      118962 :     if (OidIsValid(typeForm->typsubscript))
     683             :     {
     684       59358 :         ObjectAddressSet(referenced, ProcedureRelationId, typeForm->typsubscript);
     685       59358 :         add_exact_object_address(&referenced, addrs_normal);
     686             :     }
     687             : 
     688             :     /* Normal dependency from a domain to its base type. */
     689      118962 :     if (OidIsValid(typeForm->typbasetype))
     690             :     {
     691        1258 :         ObjectAddressSet(referenced, TypeRelationId, typeForm->typbasetype);
     692        1258 :         add_exact_object_address(&referenced, addrs_normal);
     693             :     }
     694             : 
     695             :     /*
     696             :      * Normal dependency from a domain to its collation.  We know the default
     697             :      * collation is pinned, so don't bother recording it.
     698             :      */
     699      118962 :     if (OidIsValid(typeForm->typcollation) &&
     700         956 :         typeForm->typcollation != DEFAULT_COLLATION_OID)
     701             :     {
     702         556 :         ObjectAddressSet(referenced, CollationRelationId, typeForm->typcollation);
     703         556 :         add_exact_object_address(&referenced, addrs_normal);
     704             :     }
     705             : 
     706      118962 :     record_object_address_dependencies(&myself, addrs_normal, DEPENDENCY_NORMAL);
     707      118962 :     free_object_addresses(addrs_normal);
     708             : 
     709             :     /* Normal dependency on the default expression. */
     710      118962 :     if (defaultExpr)
     711         136 :         recordDependencyOnExpr(&myself, defaultExpr, NIL, DEPENDENCY_NORMAL);
     712             : 
     713             :     /*
     714             :      * If the type is a rowtype for a relation, mark it as internally
     715             :      * dependent on the relation, *unless* it is a stand-alone composite type
     716             :      * relation. For the latter case, we have to reverse the dependency.
     717             :      *
     718             :      * In the former case, this allows the type to be auto-dropped when the
     719             :      * relation is, and not otherwise. And in the latter, of course we get the
     720             :      * opposite effect.
     721             :      */
     722      118962 :     if (OidIsValid(typeForm->typrelid))
     723             :     {
     724       57186 :         ObjectAddressSet(referenced, RelationRelationId, typeForm->typrelid);
     725             : 
     726       57186 :         if (relationKind != RELKIND_COMPOSITE_TYPE)
     727       53712 :             recordDependencyOn(&myself, &referenced, DEPENDENCY_INTERNAL);
     728             :         else
     729        3474 :             recordDependencyOn(&referenced, &myself, DEPENDENCY_INTERNAL);
     730             :     }
     731             : 
     732             :     /*
     733             :      * If the type is an implicitly-created array type, mark it as internally
     734             :      * dependent on the element type.  Otherwise, if it has an element type,
     735             :      * the dependency is a normal one.
     736             :      */
     737      118962 :     if (OidIsValid(typeForm->typelem))
     738             :     {
     739       59338 :         ObjectAddressSet(referenced, TypeRelationId, typeForm->typelem);
     740       59338 :         recordDependencyOn(&myself, &referenced,
     741             :                            isImplicitArray ? DEPENDENCY_INTERNAL : DEPENDENCY_NORMAL);
     742             :     }
     743             : 
     744             :     /*
     745             :      * Note: you might expect that we should record an internal dependency of
     746             :      * a multirange on its range type here, by analogy with the cases above.
     747             :      * But instead, that is done by RangeCreate(), which also handles
     748             :      * recording of other range-type-specific dependencies.  That's pretty
     749             :      * bogus.  It's okay for now, because there are no cases where we need to
     750             :      * regenerate the dependencies of a range or multirange type.  But someday
     751             :      * we might need to move that logic here to allow such regeneration.
     752             :      */
     753      118962 : }
     754             : 
     755             : /*
     756             :  * RenameTypeInternal
     757             :  *      This renames a type, as well as any associated array type.
     758             :  *
     759             :  * Caller must have already checked privileges.
     760             :  *
     761             :  * Currently this is used for renaming table rowtypes and for
     762             :  * ALTER TYPE RENAME TO command.
     763             :  */
     764             : void
     765         328 : RenameTypeInternal(Oid typeOid, const char *newTypeName, Oid typeNamespace)
     766             : {
     767             :     Relation    pg_type_desc;
     768             :     HeapTuple   tuple;
     769             :     Form_pg_type typ;
     770             :     Oid         arrayOid;
     771             :     Oid         oldTypeOid;
     772             : 
     773         328 :     pg_type_desc = table_open(TypeRelationId, RowExclusiveLock);
     774             : 
     775         328 :     tuple = SearchSysCacheCopy1(TYPEOID, ObjectIdGetDatum(typeOid));
     776         328 :     if (!HeapTupleIsValid(tuple))
     777           0 :         elog(ERROR, "cache lookup failed for type %u", typeOid);
     778         328 :     typ = (Form_pg_type) GETSTRUCT(tuple);
     779             : 
     780             :     /* We are not supposed to be changing schemas here */
     781             :     Assert(typeNamespace == typ->typnamespace);
     782             : 
     783         328 :     arrayOid = typ->typarray;
     784             : 
     785             :     /* Check for a conflicting type name. */
     786         328 :     oldTypeOid = GetSysCacheOid2(TYPENAMENSP, Anum_pg_type_oid,
     787             :                                  CStringGetDatum(newTypeName),
     788             :                                  ObjectIdGetDatum(typeNamespace));
     789             : 
     790             :     /*
     791             :      * If there is one, see if it's an autogenerated array type, and if so
     792             :      * rename it out of the way.  (But we must skip that for a shell type
     793             :      * because moveArrayTypeName will do the wrong thing in that case.)
     794             :      * Otherwise, we can at least give a more friendly error than unique-index
     795             :      * violation.
     796             :      */
     797         328 :     if (OidIsValid(oldTypeOid))
     798             :     {
     799          24 :         if (get_typisdefined(oldTypeOid) &&
     800          12 :             moveArrayTypeName(oldTypeOid, newTypeName, typeNamespace))
     801             :              /* successfully dodged the problem */ ;
     802             :         else
     803           0 :             ereport(ERROR,
     804             :                     (errcode(ERRCODE_DUPLICATE_OBJECT),
     805             :                      errmsg("type \"%s\" already exists", newTypeName)));
     806             :     }
     807             : 
     808             :     /* OK, do the rename --- tuple is a copy, so OK to scribble on it */
     809         328 :     namestrcpy(&(typ->typname), newTypeName);
     810             : 
     811         328 :     CatalogTupleUpdate(pg_type_desc, &tuple->t_self, tuple);
     812             : 
     813         328 :     InvokeObjectPostAlterHook(TypeRelationId, typeOid, 0);
     814             : 
     815         328 :     heap_freetuple(tuple);
     816         328 :     table_close(pg_type_desc, RowExclusiveLock);
     817             : 
     818             :     /*
     819             :      * If the type has an array type, recurse to handle that.  But we don't
     820             :      * need to do anything more if we already renamed that array type above
     821             :      * (which would happen when, eg, renaming "foo" to "_foo").
     822             :      */
     823         328 :     if (OidIsValid(arrayOid) && arrayOid != oldTypeOid)
     824             :     {
     825         148 :         char       *arrname = makeArrayTypeName(newTypeName, typeNamespace);
     826             : 
     827         148 :         RenameTypeInternal(arrayOid, arrname, typeNamespace);
     828         148 :         pfree(arrname);
     829             :     }
     830         328 : }
     831             : 
     832             : 
     833             : /*
     834             :  * makeArrayTypeName
     835             :  *    - given a base type name, make an array type name for it
     836             :  *
     837             :  * the caller is responsible for pfreeing the result
     838             :  */
     839             : char *
     840       64898 : makeArrayTypeName(const char *typeName, Oid typeNamespace)
     841             : {
     842             :     char       *arr_name;
     843       64898 :     int         pass = 0;
     844             :     char        suffix[NAMEDATALEN];
     845             : 
     846             :     /*
     847             :      * Per ancient Postgres tradition, array type names are made by prepending
     848             :      * an underscore to the base type name.  Much client code knows that
     849             :      * convention, so don't muck with it.  However, the tradition is less
     850             :      * clear about what to do in the corner cases where the resulting name is
     851             :      * too long or conflicts with an existing name.  Our current rules are (1)
     852             :      * truncate the base name on the right as needed, and (2) if there is a
     853             :      * conflict, append another underscore and some digits chosen to make it
     854             :      * unique.  This is similar to what ChooseRelationName() does.
     855             :      *
     856             :      * The actual name generation can be farmed out to makeObjectName() by
     857             :      * giving it an empty first name component.
     858             :      */
     859             : 
     860             :     /* First, try with no numeric suffix */
     861       64898 :     arr_name = makeObjectName("", typeName, NULL);
     862             : 
     863             :     for (;;)
     864             :     {
     865       64918 :         if (!SearchSysCacheExists2(TYPENAMENSP,
     866             :                                    CStringGetDatum(arr_name),
     867             :                                    ObjectIdGetDatum(typeNamespace)))
     868       64898 :             break;
     869             : 
     870             :         /* That attempt conflicted.  Prepare a new name with some digits. */
     871          20 :         pfree(arr_name);
     872          20 :         snprintf(suffix, sizeof(suffix), "%d", ++pass);
     873          20 :         arr_name = makeObjectName("", typeName, suffix);
     874             :     }
     875             : 
     876       64898 :     return arr_name;
     877             : }
     878             : 
     879             : 
     880             : /*
     881             :  * moveArrayTypeName
     882             :  *    - try to reassign an array type name that the user wants to use.
     883             :  *
     884             :  * The given type name has been discovered to already exist (with the given
     885             :  * OID).  If it is an autogenerated array type, change the array type's name
     886             :  * to not conflict.  This allows the user to create type "foo" followed by
     887             :  * type "_foo" without problems.  (Of course, there are race conditions if
     888             :  * two backends try to create similarly-named types concurrently, but the
     889             :  * worst that can happen is an unnecessary failure --- anything we do here
     890             :  * will be rolled back if the type creation fails due to conflicting names.)
     891             :  *
     892             :  * Note that this must be called *before* calling makeArrayTypeName to
     893             :  * determine the new type's own array type name; else the latter will
     894             :  * certainly pick the same name.
     895             :  *
     896             :  * Returns true if successfully moved the type, false if not.
     897             :  *
     898             :  * We also return true if the given type is a shell type.  In this case
     899             :  * the type has not been renamed out of the way, but nonetheless it can
     900             :  * be expected that TypeCreate will succeed.  This behavior is convenient
     901             :  * for most callers --- those that need to distinguish the shell-type case
     902             :  * must do their own typisdefined test.
     903             :  */
     904             : bool
     905          40 : moveArrayTypeName(Oid typeOid, const char *typeName, Oid typeNamespace)
     906             : {
     907             :     Oid         elemOid;
     908             :     char       *newname;
     909             : 
     910             :     /* We need do nothing if it's a shell type. */
     911          40 :     if (!get_typisdefined(typeOid))
     912           2 :         return true;
     913             : 
     914             :     /* Can't change it if it's not an autogenerated array type. */
     915          38 :     elemOid = get_element_type(typeOid);
     916          64 :     if (!OidIsValid(elemOid) ||
     917          26 :         get_array_type(elemOid) != typeOid)
     918          12 :         return false;
     919             : 
     920             :     /*
     921             :      * OK, use makeArrayTypeName to pick an unused modification of the name.
     922             :      * Note that since makeArrayTypeName is an iterative process, this will
     923             :      * produce a name that it might have produced the first time, had the
     924             :      * conflicting type we are about to create already existed.
     925             :      */
     926          26 :     newname = makeArrayTypeName(typeName, typeNamespace);
     927             : 
     928             :     /* Apply the rename */
     929          26 :     RenameTypeInternal(typeOid, newname, typeNamespace);
     930             : 
     931             :     /*
     932             :      * We must bump the command counter so that any subsequent use of
     933             :      * makeArrayTypeName sees what we just did and doesn't pick the same name.
     934             :      */
     935          26 :     CommandCounterIncrement();
     936             : 
     937          26 :     pfree(newname);
     938             : 
     939          26 :     return true;
     940             : }
     941             : 
     942             : 
     943             : /*
     944             :  * makeMultirangeTypeName
     945             :  *    - given a range type name, make a multirange type name for it
     946             :  *
     947             :  * caller is responsible for pfreeing the result
     948             :  */
     949             : char *
     950         130 : makeMultirangeTypeName(const char *rangeTypeName, Oid typeNamespace)
     951             : {
     952             :     char       *buf;
     953             :     char       *rangestr;
     954             : 
     955             :     /*
     956             :      * If the range type name contains "range" then change that to
     957             :      * "multirange". Otherwise add "_multirange" to the end.
     958             :      */
     959         130 :     rangestr = strstr(rangeTypeName, "range");
     960         130 :     if (rangestr)
     961             :     {
     962         112 :         char       *prefix = pnstrdup(rangeTypeName, rangestr - rangeTypeName);
     963             : 
     964         112 :         buf = psprintf("%s%s%s", prefix, "multi", rangestr);
     965             :     }
     966             :     else
     967          18 :         buf = psprintf("%s_multirange", pnstrdup(rangeTypeName, NAMEDATALEN - 12));
     968             : 
     969             :     /* clip it at NAMEDATALEN-1 bytes */
     970         130 :     buf[pg_mbcliplen(buf, strlen(buf), NAMEDATALEN - 1)] = '\0';
     971             : 
     972         130 :     if (SearchSysCacheExists2(TYPENAMENSP,
     973             :                               CStringGetDatum(buf),
     974             :                               ObjectIdGetDatum(typeNamespace)))
     975          12 :         ereport(ERROR,
     976             :                 (errcode(ERRCODE_DUPLICATE_OBJECT),
     977             :                  errmsg("type \"%s\" already exists", buf),
     978             :                  errdetail("Failed while creating a multirange type for type \"%s\".", rangeTypeName),
     979             :                  errhint("You can manually specify a multirange type name using the \"multirange_type_name\" attribute.")));
     980             : 
     981         118 :     return pstrdup(buf);
     982             : }

Generated by: LCOV version 1.14