LCOV - code coverage report
Current view: top level - src/backend/utils/adt - pseudotypes.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 19 76 25.0 %
Date: 2025-01-18 04:15:08 Functions: 9 62 14.5 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * pseudotypes.c
       4             :  *    Functions for the system pseudo-types.
       5             :  *
       6             :  * A pseudo-type isn't really a type and never has any operations, but
       7             :  * we do need to supply input and output functions to satisfy the links
       8             :  * in the pseudo-type's entry in pg_type.  In most cases the functions
       9             :  * just throw an error if invoked.  (XXX the error messages here cover
      10             :  * the most common case, but might be confusing in some contexts.  Can
      11             :  * we do better?)
      12             :  *
      13             :  *
      14             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
      15             :  * Portions Copyright (c) 1994, Regents of the University of California
      16             :  *
      17             :  *
      18             :  * IDENTIFICATION
      19             :  *    src/backend/utils/adt/pseudotypes.c
      20             :  *
      21             :  *-------------------------------------------------------------------------
      22             :  */
      23             : #include "postgres.h"
      24             : 
      25             : #include "libpq/pqformat.h"
      26             : #include "utils/fmgrprotos.h"
      27             : 
      28             : 
      29             : /*
      30             :  * These macros generate input and output functions for a pseudo-type that
      31             :  * will reject all input and output attempts.  (But for some types, only
      32             :  * the input function need be dummy.)
      33             :  */
      34             : #define PSEUDOTYPE_DUMMY_INPUT_FUNC(typname) \
      35             : Datum \
      36             : typname##_in(PG_FUNCTION_ARGS) \
      37             : { \
      38             :     ereport(ERROR, \
      39             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED), \
      40             :              errmsg("cannot accept a value of type %s", #typname))); \
      41             : \
      42             :     PG_RETURN_VOID();           /* keep compiler quiet */ \
      43             : } \
      44             : \
      45             : extern int no_such_variable
      46             : 
      47             : #define PSEUDOTYPE_DUMMY_IO_FUNCS(typname) \
      48             : PSEUDOTYPE_DUMMY_INPUT_FUNC(typname); \
      49             : \
      50             : Datum \
      51             : typname##_out(PG_FUNCTION_ARGS) \
      52             : { \
      53             :     ereport(ERROR, \
      54             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED), \
      55             :              errmsg("cannot display a value of type %s", #typname))); \
      56             : \
      57             :     PG_RETURN_VOID();           /* keep compiler quiet */ \
      58             : } \
      59             : \
      60             : extern int no_such_variable
      61             : 
      62             : /*
      63             :  * Likewise for binary send/receive functions.  We don't bother with these
      64             :  * at all for many pseudotypes, but some have them.  (By convention, if
      65             :  * a type has a send function it should have a receive function, even if
      66             :  * that's only dummy.)
      67             :  */
      68             : #define PSEUDOTYPE_DUMMY_RECEIVE_FUNC(typname) \
      69             : Datum \
      70             : typname##_recv(PG_FUNCTION_ARGS) \
      71             : { \
      72             :     ereport(ERROR, \
      73             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED), \
      74             :              errmsg("cannot accept a value of type %s", #typname))); \
      75             : \
      76             :     PG_RETURN_VOID();           /* keep compiler quiet */ \
      77             : } \
      78             : \
      79             : extern int no_such_variable
      80             : 
      81             : #define PSEUDOTYPE_DUMMY_BINARY_IO_FUNCS(typname) \
      82             : PSEUDOTYPE_DUMMY_RECEIVE_FUNC(typname); \
      83             : \
      84             : Datum \
      85             : typname##_send(PG_FUNCTION_ARGS) \
      86             : { \
      87             :     ereport(ERROR, \
      88             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED), \
      89             :              errmsg("cannot display a value of type %s", #typname))); \
      90             : \
      91             :     PG_RETURN_VOID();           /* keep compiler quiet */ \
      92             : } \
      93             : \
      94             : extern int no_such_variable
      95             : 
      96             : 
      97             : /*
      98             :  * cstring
      99             :  *
     100             :  * cstring is marked as a pseudo-type because we don't want people using it
     101             :  * in tables.  But it's really a perfectly functional type, so provide
     102             :  * a full set of working I/O functions for it.  Among other things, this
     103             :  * allows manual invocation of datatype I/O functions, along the lines of
     104             :  * "SELECT foo_in('blah')" or "SELECT foo_out(some-foo-value)".
     105             :  */
     106             : Datum
     107          18 : cstring_in(PG_FUNCTION_ARGS)
     108             : {
     109          18 :     char       *str = PG_GETARG_CSTRING(0);
     110             : 
     111          18 :     PG_RETURN_CSTRING(pstrdup(str));
     112             : }
     113             : 
     114             : Datum
     115         246 : cstring_out(PG_FUNCTION_ARGS)
     116             : {
     117         246 :     char       *str = PG_GETARG_CSTRING(0);
     118             : 
     119         246 :     PG_RETURN_CSTRING(pstrdup(str));
     120             : }
     121             : 
     122             : Datum
     123           0 : cstring_recv(PG_FUNCTION_ARGS)
     124             : {
     125           0 :     StringInfo  buf = (StringInfo) PG_GETARG_POINTER(0);
     126             :     char       *str;
     127             :     int         nbytes;
     128             : 
     129           0 :     str = pq_getmsgtext(buf, buf->len - buf->cursor, &nbytes);
     130           0 :     PG_RETURN_CSTRING(str);
     131             : }
     132             : 
     133             : Datum
     134           0 : cstring_send(PG_FUNCTION_ARGS)
     135             : {
     136           0 :     char       *str = PG_GETARG_CSTRING(0);
     137             :     StringInfoData buf;
     138             : 
     139           0 :     pq_begintypsend(&buf);
     140           0 :     pq_sendtext(&buf, str, strlen(str));
     141           0 :     PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
     142             : }
     143             : 
     144             : /*
     145             :  * anyarray
     146             :  *
     147             :  * We need to allow output of anyarray so that, e.g., pg_statistic columns
     148             :  * can be printed.  Input has to be disallowed, however.
     149             :  *
     150             :  * XXX anyarray_recv could actually be made to work, since the incoming
     151             :  * array data would contain the element type OID.  It seems unlikely that
     152             :  * it'd be sufficiently type-safe, though.
     153             :  */
     154           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anyarray);
     155           0 : PSEUDOTYPE_DUMMY_RECEIVE_FUNC(anyarray);
     156             : 
     157             : Datum
     158        6084 : anyarray_out(PG_FUNCTION_ARGS)
     159             : {
     160        6084 :     return array_out(fcinfo);
     161             : }
     162             : 
     163             : Datum
     164           0 : anyarray_send(PG_FUNCTION_ARGS)
     165             : {
     166           0 :     return array_send(fcinfo);
     167             : }
     168             : 
     169             : /*
     170             :  * anycompatiblearray
     171             :  *
     172             :  * We may as well allow output, since we do for anyarray.
     173             :  */
     174           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anycompatiblearray);
     175           0 : PSEUDOTYPE_DUMMY_RECEIVE_FUNC(anycompatiblearray);
     176             : 
     177             : Datum
     178           0 : anycompatiblearray_out(PG_FUNCTION_ARGS)
     179             : {
     180           0 :     return array_out(fcinfo);
     181             : }
     182             : 
     183             : Datum
     184           0 : anycompatiblearray_send(PG_FUNCTION_ARGS)
     185             : {
     186           0 :     return array_send(fcinfo);
     187             : }
     188             : 
     189             : /*
     190             :  * anyenum
     191             :  *
     192             :  * We may as well allow output, since enum_out will in fact work.
     193             :  */
     194           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anyenum);
     195             : 
     196             : Datum
     197           0 : anyenum_out(PG_FUNCTION_ARGS)
     198             : {
     199           0 :     return enum_out(fcinfo);
     200             : }
     201             : 
     202             : /*
     203             :  * anyrange
     204             :  *
     205             :  * We may as well allow output, since range_out will in fact work.
     206             :  */
     207           6 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anyrange);
     208             : 
     209             : Datum
     210         320 : anyrange_out(PG_FUNCTION_ARGS)
     211             : {
     212         320 :     return range_out(fcinfo);
     213             : }
     214             : 
     215             : /*
     216             :  * anycompatiblerange
     217             :  *
     218             :  * We may as well allow output, since range_out will in fact work.
     219             :  */
     220           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anycompatiblerange);
     221             : 
     222             : Datum
     223           0 : anycompatiblerange_out(PG_FUNCTION_ARGS)
     224             : {
     225           0 :     return range_out(fcinfo);
     226             : }
     227             : 
     228             : /*
     229             :  * anymultirange
     230             :  *
     231             :  * We may as well allow output, since multirange_out will in fact work.
     232             :  */
     233           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anymultirange);
     234             : 
     235             : Datum
     236          84 : anymultirange_out(PG_FUNCTION_ARGS)
     237             : {
     238          84 :     return multirange_out(fcinfo);
     239             : }
     240             : 
     241             : /*
     242             :  * anycompatiblemultirange
     243             :  *
     244             :  * We may as well allow output, since multirange_out will in fact work.
     245             :  */
     246           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(anycompatiblemultirange);
     247             : 
     248             : Datum
     249           0 : anycompatiblemultirange_out(PG_FUNCTION_ARGS)
     250             : {
     251           0 :     return multirange_out(fcinfo);
     252             : }
     253             : 
     254             : /*
     255             :  * void
     256             :  *
     257             :  * We support void_in so that PL functions can return VOID without any
     258             :  * special hack in the PL handler.  Whatever value the PL thinks it's
     259             :  * returning will just be ignored.  Conversely, void_out and void_send
     260             :  * are needed so that "SELECT function_returning_void(...)" works.
     261             :  */
     262             : Datum
     263          28 : void_in(PG_FUNCTION_ARGS)
     264             : {
     265          28 :     PG_RETURN_VOID();           /* you were expecting something different? */
     266             : }
     267             : 
     268             : Datum
     269       20938 : void_out(PG_FUNCTION_ARGS)
     270             : {
     271       20938 :     PG_RETURN_CSTRING(pstrdup(""));
     272             : }
     273             : 
     274             : Datum
     275           0 : void_recv(PG_FUNCTION_ARGS)
     276             : {
     277             :     /*
     278             :      * Note that since we consume no bytes, an attempt to send anything but an
     279             :      * empty string will result in an "invalid message format" error.
     280             :      */
     281           0 :     PG_RETURN_VOID();
     282             : }
     283             : 
     284             : Datum
     285           0 : void_send(PG_FUNCTION_ARGS)
     286             : {
     287             :     StringInfoData buf;
     288             : 
     289             :     /* send an empty string */
     290           0 :     pq_begintypsend(&buf);
     291           0 :     PG_RETURN_BYTEA_P(pq_endtypsend(&buf));
     292             : }
     293             : 
     294             : /*
     295             :  * shell
     296             :  *
     297             :  * shell_in and shell_out are entered in pg_type for "shell" types
     298             :  * (those not yet filled in).  They should be unreachable, but we
     299             :  * set them up just in case some code path tries to do I/O without
     300             :  * having checked pg_type.typisdefined anywhere along the way.
     301             :  */
     302             : Datum
     303           0 : shell_in(PG_FUNCTION_ARGS)
     304             : {
     305           0 :     ereport(ERROR,
     306             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     307             :              errmsg("cannot accept a value of a shell type")));
     308             : 
     309             :     PG_RETURN_VOID();           /* keep compiler quiet */
     310             : }
     311             : 
     312             : Datum
     313           0 : shell_out(PG_FUNCTION_ARGS)
     314             : {
     315           0 :     ereport(ERROR,
     316             :             (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
     317             :              errmsg("cannot display a value of a shell type")));
     318             : 
     319             :     PG_RETURN_VOID();           /* keep compiler quiet */
     320             : }
     321             : 
     322             : 
     323             : /*
     324             :  * pg_node_tree
     325             :  *
     326             :  * pg_node_tree isn't really a pseudotype --- it's real enough to be a table
     327             :  * column --- but it presently has no operations of its own, and disallows
     328             :  * input too, so its I/O functions seem to fit here as much as anywhere.
     329             :  *
     330             :  * We must disallow input of pg_node_tree values because the SQL functions
     331             :  * that operate on the type are not secure against malformed input.
     332             :  * We do want to allow output, though.
     333             :  */
     334           0 : PSEUDOTYPE_DUMMY_INPUT_FUNC(pg_node_tree);
     335           0 : PSEUDOTYPE_DUMMY_RECEIVE_FUNC(pg_node_tree);
     336             : 
     337             : Datum
     338       12728 : pg_node_tree_out(PG_FUNCTION_ARGS)
     339             : {
     340       12728 :     return textout(fcinfo);
     341             : }
     342             : 
     343             : Datum
     344           0 : pg_node_tree_send(PG_FUNCTION_ARGS)
     345             : {
     346           0 :     return textsend(fcinfo);
     347             : }
     348             : 
     349             : /*
     350             :  * pg_ddl_command
     351             :  *
     352             :  * Like pg_node_tree, pg_ddl_command isn't really a pseudotype; it's here
     353             :  * for the same reasons as that one.
     354             :  *
     355             :  * We don't have any good way to output this type directly, so punt
     356             :  * for output as well as input.
     357             :  */
     358           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(pg_ddl_command);
     359           0 : PSEUDOTYPE_DUMMY_BINARY_IO_FUNCS(pg_ddl_command);
     360             : 
     361             : 
     362             : /*
     363             :  * Dummy I/O functions for various other pseudotypes.
     364             :  */
     365           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(any);
     366           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(trigger);
     367           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(event_trigger);
     368           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(language_handler);
     369           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(fdw_handler);
     370           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(table_am_handler);
     371           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(index_am_handler);
     372           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(tsm_handler);
     373           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(internal);
     374           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(anyelement);
     375           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(anynonarray);
     376           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(anycompatible);
     377           0 : PSEUDOTYPE_DUMMY_IO_FUNCS(anycompatiblenonarray);

Generated by: LCOV version 1.14