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

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * pg_cast.c
       4             :  *    routines to support manipulation of the pg_cast 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_cast.c
      12             :  *
      13             :  *-------------------------------------------------------------------------
      14             :  */
      15             : #include "postgres.h"
      16             : 
      17             : #include "access/htup_details.h"
      18             : #include "access/table.h"
      19             : #include "catalog/catalog.h"
      20             : #include "catalog/dependency.h"
      21             : #include "catalog/indexing.h"
      22             : #include "catalog/objectaccess.h"
      23             : #include "catalog/pg_cast.h"
      24             : #include "catalog/pg_proc.h"
      25             : #include "catalog/pg_type.h"
      26             : #include "utils/builtins.h"
      27             : #include "utils/rel.h"
      28             : #include "utils/syscache.h"
      29             : 
      30             : /*
      31             :  * ----------------------------------------------------------------
      32             :  *      CastCreate
      33             :  *
      34             :  * Forms and inserts catalog tuples for a new cast being created.
      35             :  * Caller must have already checked privileges, and done consistency
      36             :  * checks on the given datatypes and cast function (if applicable).
      37             :  *
      38             :  * Since we allow binary coercibility of the datatypes to the cast
      39             :  * function's input and result, there could be one or two WITHOUT FUNCTION
      40             :  * casts that this one depends on.  We don't record that explicitly
      41             :  * in pg_cast, but we still need to make dependencies on those casts.
      42             :  *
      43             :  * 'behavior' indicates the types of the dependencies that the new
      44             :  * cast will have on its input and output types, the cast function,
      45             :  * and the other casts if any.
      46             :  * ----------------------------------------------------------------
      47             :  */
      48             : ObjectAddress
      49         408 : CastCreate(Oid sourcetypeid, Oid targettypeid,
      50             :            Oid funcid, Oid incastid, Oid outcastid,
      51             :            char castcontext, char castmethod, DependencyType behavior)
      52             : {
      53             :     Relation    relation;
      54             :     HeapTuple   tuple;
      55             :     Oid         castid;
      56             :     Datum       values[Natts_pg_cast];
      57         408 :     bool        nulls[Natts_pg_cast] = {0};
      58             :     ObjectAddress myself,
      59             :                 referenced;
      60             :     ObjectAddresses *addrs;
      61             : 
      62         408 :     relation = table_open(CastRelationId, RowExclusiveLock);
      63             : 
      64             :     /*
      65             :      * Check for duplicate.  This is just to give a friendly error message,
      66             :      * the unique index would catch it anyway (so no need to sweat about race
      67             :      * conditions).
      68             :      */
      69         408 :     tuple = SearchSysCache2(CASTSOURCETARGET,
      70             :                             ObjectIdGetDatum(sourcetypeid),
      71             :                             ObjectIdGetDatum(targettypeid));
      72         408 :     if (HeapTupleIsValid(tuple))
      73           0 :         ereport(ERROR,
      74             :                 (errcode(ERRCODE_DUPLICATE_OBJECT),
      75             :                  errmsg("cast from type %s to type %s already exists",
      76             :                         format_type_be(sourcetypeid),
      77             :                         format_type_be(targettypeid))));
      78             : 
      79             :     /* ready to go */
      80         408 :     castid = GetNewOidWithIndex(relation, CastOidIndexId, Anum_pg_cast_oid);
      81         408 :     values[Anum_pg_cast_oid - 1] = ObjectIdGetDatum(castid);
      82         408 :     values[Anum_pg_cast_castsource - 1] = ObjectIdGetDatum(sourcetypeid);
      83         408 :     values[Anum_pg_cast_casttarget - 1] = ObjectIdGetDatum(targettypeid);
      84         408 :     values[Anum_pg_cast_castfunc - 1] = ObjectIdGetDatum(funcid);
      85         408 :     values[Anum_pg_cast_castcontext - 1] = CharGetDatum(castcontext);
      86         408 :     values[Anum_pg_cast_castmethod - 1] = CharGetDatum(castmethod);
      87             : 
      88         408 :     tuple = heap_form_tuple(RelationGetDescr(relation), values, nulls);
      89             : 
      90         408 :     CatalogTupleInsert(relation, tuple);
      91             : 
      92         408 :     addrs = new_object_addresses();
      93             : 
      94             :     /* make dependency entries */
      95         408 :     ObjectAddressSet(myself, CastRelationId, castid);
      96             : 
      97             :     /* dependency on source type */
      98         408 :     ObjectAddressSet(referenced, TypeRelationId, sourcetypeid);
      99         408 :     add_exact_object_address(&referenced, addrs);
     100             : 
     101             :     /* dependency on target type */
     102         408 :     ObjectAddressSet(referenced, TypeRelationId, targettypeid);
     103         408 :     add_exact_object_address(&referenced, addrs);
     104             : 
     105             :     /* dependency on function */
     106         408 :     if (OidIsValid(funcid))
     107             :     {
     108         240 :         ObjectAddressSet(referenced, ProcedureRelationId, funcid);
     109         240 :         add_exact_object_address(&referenced, addrs);
     110             :     }
     111             : 
     112             :     /* dependencies on casts required for function */
     113         408 :     if (OidIsValid(incastid))
     114             :     {
     115           0 :         ObjectAddressSet(referenced, CastRelationId, incastid);
     116           0 :         add_exact_object_address(&referenced, addrs);
     117             :     }
     118         408 :     if (OidIsValid(outcastid))
     119             :     {
     120           8 :         ObjectAddressSet(referenced, CastRelationId, outcastid);
     121           8 :         add_exact_object_address(&referenced, addrs);
     122             :     }
     123             : 
     124         408 :     record_object_address_dependencies(&myself, addrs, behavior);
     125         408 :     free_object_addresses(addrs);
     126             : 
     127             :     /* dependency on extension */
     128         408 :     recordDependencyOnCurrentExtension(&myself, false);
     129             : 
     130             :     /* Post creation hook for new cast */
     131         408 :     InvokeObjectPostCreateHook(CastRelationId, castid, 0);
     132             : 
     133         408 :     heap_freetuple(tuple);
     134             : 
     135         408 :     table_close(relation, RowExclusiveLock);
     136             : 
     137         408 :     return myself;
     138             : }

Generated by: LCOV version 1.14