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

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * pg_conversion.c
       4             :  *    routines to support manipulation of the pg_conversion 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_conversion.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_conversion.h"
      24             : #include "catalog/pg_namespace.h"
      25             : #include "catalog/pg_proc.h"
      26             : #include "mb/pg_wchar.h"
      27             : #include "utils/builtins.h"
      28             : #include "utils/catcache.h"
      29             : #include "utils/rel.h"
      30             : #include "utils/syscache.h"
      31             : 
      32             : /*
      33             :  * ConversionCreate
      34             :  *
      35             :  * Add a new tuple to pg_conversion.
      36             :  */
      37             : ObjectAddress
      38          64 : ConversionCreate(const char *conname, Oid connamespace,
      39             :                  Oid conowner,
      40             :                  int32 conforencoding, int32 contoencoding,
      41             :                  Oid conproc, bool def)
      42             : {
      43             :     int         i;
      44             :     Relation    rel;
      45             :     TupleDesc   tupDesc;
      46             :     HeapTuple   tup;
      47             :     Oid         oid;
      48             :     bool        nulls[Natts_pg_conversion];
      49             :     Datum       values[Natts_pg_conversion];
      50             :     NameData    cname;
      51             :     ObjectAddress myself,
      52             :                 referenced;
      53             : 
      54             :     /* sanity checks */
      55          64 :     if (!conname)
      56           0 :         elog(ERROR, "no conversion name supplied");
      57             : 
      58             :     /* make sure there is no existing conversion of same name */
      59          64 :     if (SearchSysCacheExists2(CONNAMENSP,
      60             :                               PointerGetDatum(conname),
      61             :                               ObjectIdGetDatum(connamespace)))
      62           6 :         ereport(ERROR,
      63             :                 (errcode(ERRCODE_DUPLICATE_OBJECT),
      64             :                  errmsg("conversion \"%s\" already exists", conname)));
      65             : 
      66          58 :     if (def)
      67             :     {
      68             :         /*
      69             :          * make sure there is no existing default <for encoding><to encoding>
      70             :          * pair in this name space
      71             :          */
      72          14 :         if (FindDefaultConversion(connamespace,
      73             :                                   conforencoding,
      74             :                                   contoencoding))
      75           6 :             ereport(ERROR,
      76             :                     (errcode(ERRCODE_DUPLICATE_OBJECT),
      77             :                      errmsg("default conversion for %s to %s already exists",
      78             :                             pg_encoding_to_char(conforencoding),
      79             :                             pg_encoding_to_char(contoencoding))));
      80             :     }
      81             : 
      82             :     /* open pg_conversion */
      83          52 :     rel = table_open(ConversionRelationId, RowExclusiveLock);
      84          52 :     tupDesc = rel->rd_att;
      85             : 
      86             :     /* initialize nulls and values */
      87         468 :     for (i = 0; i < Natts_pg_conversion; i++)
      88             :     {
      89         416 :         nulls[i] = false;
      90         416 :         values[i] = (Datum) NULL;
      91             :     }
      92             : 
      93             :     /* form a tuple */
      94          52 :     namestrcpy(&cname, conname);
      95          52 :     oid = GetNewOidWithIndex(rel, ConversionOidIndexId,
      96             :                              Anum_pg_conversion_oid);
      97          52 :     values[Anum_pg_conversion_oid - 1] = ObjectIdGetDatum(oid);
      98          52 :     values[Anum_pg_conversion_conname - 1] = NameGetDatum(&cname);
      99          52 :     values[Anum_pg_conversion_connamespace - 1] = ObjectIdGetDatum(connamespace);
     100          52 :     values[Anum_pg_conversion_conowner - 1] = ObjectIdGetDatum(conowner);
     101          52 :     values[Anum_pg_conversion_conforencoding - 1] = Int32GetDatum(conforencoding);
     102          52 :     values[Anum_pg_conversion_contoencoding - 1] = Int32GetDatum(contoencoding);
     103          52 :     values[Anum_pg_conversion_conproc - 1] = ObjectIdGetDatum(conproc);
     104          52 :     values[Anum_pg_conversion_condefault - 1] = BoolGetDatum(def);
     105             : 
     106          52 :     tup = heap_form_tuple(tupDesc, values, nulls);
     107             : 
     108             :     /* insert a new tuple */
     109          52 :     CatalogTupleInsert(rel, tup);
     110             : 
     111          52 :     myself.classId = ConversionRelationId;
     112          52 :     myself.objectId = oid;
     113          52 :     myself.objectSubId = 0;
     114             : 
     115             :     /* create dependency on conversion procedure */
     116          52 :     referenced.classId = ProcedureRelationId;
     117          52 :     referenced.objectId = conproc;
     118          52 :     referenced.objectSubId = 0;
     119          52 :     recordDependencyOn(&myself, &referenced, DEPENDENCY_NORMAL);
     120             : 
     121             :     /* create dependency on namespace */
     122          52 :     referenced.classId = NamespaceRelationId;
     123          52 :     referenced.objectId = connamespace;
     124          52 :     referenced.objectSubId = 0;
     125          52 :     recordDependencyOn(&myself, &referenced, DEPENDENCY_NORMAL);
     126             : 
     127             :     /* create dependency on owner */
     128          52 :     recordDependencyOnOwner(ConversionRelationId, oid, conowner);
     129             : 
     130             :     /* dependency on extension */
     131          52 :     recordDependencyOnCurrentExtension(&myself, false);
     132             : 
     133             :     /* Post creation hook for new conversion */
     134          52 :     InvokeObjectPostCreateHook(ConversionRelationId, oid, 0);
     135             : 
     136          52 :     heap_freetuple(tup);
     137          52 :     table_close(rel, RowExclusiveLock);
     138             : 
     139          52 :     return myself;
     140             : }
     141             : 
     142             : /*
     143             :  * FindDefaultConversion
     144             :  *
     145             :  * Find "default" conversion proc by for_encoding and to_encoding in the
     146             :  * given namespace.
     147             :  *
     148             :  * If found, returns the procedure's oid, otherwise InvalidOid.  Note that
     149             :  * you get the procedure's OID not the conversion's OID!
     150             :  */
     151             : Oid
     152        6750 : FindDefaultConversion(Oid name_space, int32 for_encoding, int32 to_encoding)
     153             : {
     154             :     CatCList   *catlist;
     155             :     HeapTuple   tuple;
     156             :     Form_pg_conversion body;
     157        6750 :     Oid         proc = InvalidOid;
     158             :     int         i;
     159             : 
     160        6750 :     catlist = SearchSysCacheList3(CONDEFAULT,
     161             :                                   ObjectIdGetDatum(name_space),
     162             :                                   Int32GetDatum(for_encoding),
     163             :                                   Int32GetDatum(to_encoding));
     164             : 
     165        6762 :     for (i = 0; i < catlist->n_members; i++)
     166             :     {
     167        6754 :         tuple = &catlist->members[i]->tuple;
     168        6754 :         body = (Form_pg_conversion) GETSTRUCT(tuple);
     169        6754 :         if (body->condefault)
     170             :         {
     171        6742 :             proc = body->conproc;
     172        6742 :             break;
     173             :         }
     174             :     }
     175        6750 :     ReleaseSysCacheList(catlist);
     176        6750 :     return proc;
     177             : }

Generated by: LCOV version 1.14