LCOV - code coverage report
Current view: top level - src/backend/access/common - printtup.c (source / functions) Coverage Total Hit
Test: PostgreSQL 20devel Lines: 98.0 % 148 145
Test Date: 2026-07-03 19:57:34 Functions: 100.0 % 11 11
Legend: Lines:     hit not hit
Branches: + taken - not taken # not executed
Branches: 85.7 % 56 48

             Branch data     Line data    Source code
       1                 :             : /*-------------------------------------------------------------------------
       2                 :             :  *
       3                 :             :  * printtup.c
       4                 :             :  *    Routines to print out tuples to the destination (both frontend
       5                 :             :  *    clients and standalone backends are supported here).
       6                 :             :  *
       7                 :             :  *
       8                 :             :  * Portions Copyright (c) 1996-2026, PostgreSQL Global Development Group
       9                 :             :  * Portions Copyright (c) 1994, Regents of the University of California
      10                 :             :  *
      11                 :             :  * IDENTIFICATION
      12                 :             :  *    src/backend/access/common/printtup.c
      13                 :             :  *
      14                 :             :  *-------------------------------------------------------------------------
      15                 :             :  */
      16                 :             : #include "postgres.h"
      17                 :             : 
      18                 :             : #include "access/printtup.h"
      19                 :             : #include "libpq/pqformat.h"
      20                 :             : #include "libpq/protocol.h"
      21                 :             : #include "tcop/pquery.h"
      22                 :             : #include "utils/lsyscache.h"
      23                 :             : #include "utils/memdebug.h"
      24                 :             : #include "utils/memutils.h"
      25                 :             : #include "varatt.h"
      26                 :             : 
      27                 :             : 
      28                 :             : static void printtup_startup(DestReceiver *self, int operation,
      29                 :             :                              TupleDesc typeinfo);
      30                 :             : static bool printtup(TupleTableSlot *slot, DestReceiver *self);
      31                 :             : static void printtup_shutdown(DestReceiver *self);
      32                 :             : static void printtup_destroy(DestReceiver *self);
      33                 :             : 
      34                 :             : /* ----------------------------------------------------------------
      35                 :             :  *      printtup / debugtup support
      36                 :             :  * ----------------------------------------------------------------
      37                 :             :  */
      38                 :             : 
      39                 :             : /* ----------------
      40                 :             :  *      Private state for a printtup destination object
      41                 :             :  *
      42                 :             :  * NOTE: finfo is the lookup info for either typoutput or typsend, whichever
      43                 :             :  * we are using for this column.
      44                 :             :  * ----------------
      45                 :             :  */
      46                 :             : typedef struct
      47                 :             : {                               /* Per-attribute information */
      48                 :             :     Oid         typoutput;      /* Oid for the type's text output fn */
      49                 :             :     Oid         typsend;        /* Oid for the type's binary output fn */
      50                 :             :     bool        typisvarlena;   /* is it varlena (ie possibly toastable)? */
      51                 :             :     int16       format;         /* format code for this column */
      52                 :             :     FmgrInfo    finfo;          /* Precomputed call info for output fn */
      53                 :             : } PrinttupAttrInfo;
      54                 :             : 
      55                 :             : typedef struct
      56                 :             : {
      57                 :             :     DestReceiver pub;           /* publicly-known function pointers */
      58                 :             :     Portal      portal;         /* the Portal we are printing from */
      59                 :             :     bool        sendDescrip;    /* send RowDescription at startup? */
      60                 :             :     TupleDesc   attrinfo;       /* The attr info we are set up for */
      61                 :             :     int         nattrs;
      62                 :             :     PrinttupAttrInfo *myinfo;   /* Cached info about each attr */
      63                 :             :     StringInfoData buf;         /* output buffer (*not* in tmpcontext) */
      64                 :             :     MemoryContext tmpcontext;   /* Memory context for per-row workspace */
      65                 :             : } DR_printtup;
      66                 :             : 
      67                 :             : /* ----------------
      68                 :             :  *      Initialize: create a DestReceiver for printtup
      69                 :             :  * ----------------
      70                 :             :  */
      71                 :             : DestReceiver *
      72                 :      408918 : printtup_create_DR(CommandDest dest)
      73                 :             : {
      74                 :      408918 :     DR_printtup *self = palloc0_object(DR_printtup);
      75                 :             : 
      76                 :      408918 :     self->pub.receiveSlot = printtup;    /* might get changed later */
      77                 :      408918 :     self->pub.rStartup = printtup_startup;
      78                 :      408918 :     self->pub.rShutdown = printtup_shutdown;
      79                 :      408918 :     self->pub.rDestroy = printtup_destroy;
      80                 :      408918 :     self->pub.mydest = dest;
      81                 :             : 
      82                 :             :     /*
      83                 :             :      * Send T message automatically if DestRemote, but not if
      84                 :             :      * DestRemoteExecute
      85                 :             :      */
      86                 :      408918 :     self->sendDescrip = (dest == DestRemote);
      87                 :             : 
      88                 :      408918 :     self->attrinfo = NULL;
      89                 :      408918 :     self->nattrs = 0;
      90                 :      408918 :     self->myinfo = NULL;
      91                 :      408918 :     self->buf.data = NULL;
      92                 :      408918 :     self->tmpcontext = NULL;
      93                 :             : 
      94                 :      408918 :     return (DestReceiver *) self;
      95                 :             : }
      96                 :             : 
      97                 :             : /*
      98                 :             :  * Set parameters for a DestRemote (or DestRemoteExecute) receiver
      99                 :             :  */
     100                 :             : void
     101                 :      408918 : SetRemoteDestReceiverParams(DestReceiver *self, Portal portal)
     102                 :             : {
     103                 :      408918 :     DR_printtup *myState = (DR_printtup *) self;
     104                 :             : 
     105                 :             :     Assert(myState->pub.mydest == DestRemote ||
     106                 :             :            myState->pub.mydest == DestRemoteExecute);
     107                 :             : 
     108                 :      408918 :     myState->portal = portal;
     109                 :      408918 : }
     110                 :             : 
     111                 :             : static void
     112                 :      196260 : printtup_startup(DestReceiver *self, int operation, TupleDesc typeinfo)
     113                 :             : {
     114                 :      196260 :     DR_printtup *myState = (DR_printtup *) self;
     115                 :      196260 :     Portal      portal = myState->portal;
     116                 :             : 
     117                 :             :     /*
     118                 :             :      * Create I/O buffer to be used for all messages.  This cannot be inside
     119                 :             :      * tmpcontext, since we want to re-use it across rows.
     120                 :             :      */
     121                 :      196260 :     initStringInfo(&myState->buf);
     122                 :             : 
     123                 :             :     /*
     124                 :             :      * Create a temporary memory context that we can reset once per row to
     125                 :             :      * recover palloc'd memory.  This avoids any problems with leaks inside
     126                 :             :      * datatype output routines, and should be faster than retail pfree's
     127                 :             :      * anyway.
     128                 :             :      */
     129                 :      196260 :     myState->tmpcontext = AllocSetContextCreate(CurrentMemoryContext,
     130                 :             :                                                 "printtup",
     131                 :             :                                                 ALLOCSET_DEFAULT_SIZES);
     132                 :             : 
     133                 :             :     /*
     134                 :             :      * If we are supposed to emit row descriptions, then send the tuple
     135                 :             :      * descriptor of the tuples.
     136                 :             :      */
     137         [ +  + ]:      196260 :     if (myState->sendDescrip)
     138                 :      191074 :         SendRowDescriptionMessage(&myState->buf,
     139                 :             :                                   typeinfo,
     140                 :             :                                   FetchPortalTargetList(portal),
     141                 :             :                                   portal->formats);
     142                 :             : 
     143                 :             :     /* ----------------
     144                 :             :      * We could set up the derived attr info at this time, but we postpone it
     145                 :             :      * until the first call of printtup, for 2 reasons:
     146                 :             :      * 1. We don't waste time (compared to the old way) if there are no
     147                 :             :      *    tuples at all to output.
     148                 :             :      * 2. Checking in printtup allows us to handle the case that the tuples
     149                 :             :      *    change type midway through (although this probably can't happen in
     150                 :             :      *    the current executor).
     151                 :             :      * ----------------
     152                 :             :      */
     153                 :      196260 : }
     154                 :             : 
     155                 :             : /*
     156                 :             :  * SendRowDescriptionMessage --- send a RowDescription message to the frontend
     157                 :             :  *
     158                 :             :  * Notes: the TupleDesc has typically been manufactured by ExecTypeFromTL()
     159                 :             :  * or some similar function; it does not contain a full set of fields.
     160                 :             :  * The targetlist will be NIL when executing a utility function that does
     161                 :             :  * not have a plan.  If the targetlist isn't NIL then it is a Query node's
     162                 :             :  * targetlist; it is up to us to ignore resjunk columns in it.  The formats[]
     163                 :             :  * array pointer might be NULL (if we are doing Describe on a prepared stmt);
     164                 :             :  * send zeroes for the format codes in that case.
     165                 :             :  */
     166                 :             : void
     167                 :      196334 : SendRowDescriptionMessage(StringInfo buf, TupleDesc typeinfo,
     168                 :             :                           List *targetlist, int16 *formats)
     169                 :             : {
     170                 :      196334 :     int         natts = typeinfo->natts;
     171                 :             :     int         i;
     172                 :      196334 :     ListCell   *tlist_item = list_head(targetlist);
     173                 :             : 
     174                 :             :     /* tuple descriptor message type */
     175                 :      196334 :     pq_beginmessage_reuse(buf, PqMsg_RowDescription);
     176                 :             :     /* # of attrs in tuples */
     177                 :      196334 :     pq_sendint16(buf, natts);
     178                 :             : 
     179                 :             :     /*
     180                 :             :      * Preallocate memory for the entire message to be sent. That allows to
     181                 :             :      * use the significantly faster inline pqformat.h functions and to avoid
     182                 :             :      * reallocations.
     183                 :             :      *
     184                 :             :      * Have to overestimate the size of the column-names, to account for
     185                 :             :      * character set overhead.
     186                 :             :      */
     187                 :      196334 :     enlargeStringInfo(buf, (NAMEDATALEN * MAX_CONVERSION_GROWTH /* attname */
     188                 :             :                             + sizeof(Oid)   /* resorigtbl */
     189                 :             :                             + sizeof(AttrNumber)    /* resorigcol */
     190                 :             :                             + sizeof(Oid)   /* atttypid */
     191                 :             :                             + sizeof(int16) /* attlen */
     192                 :             :                             + sizeof(int32) /* attypmod */
     193                 :             :                             + sizeof(int16) /* format */
     194                 :             :                             ) * natts);
     195                 :             : 
     196         [ +  + ]:      798582 :     for (i = 0; i < natts; ++i)
     197                 :             :     {
     198                 :      602248 :         Form_pg_attribute att = TupleDescAttr(typeinfo, i);
     199                 :      602248 :         Oid         atttypid = att->atttypid;
     200                 :      602248 :         int32       atttypmod = att->atttypmod;
     201                 :             :         Oid         resorigtbl;
     202                 :             :         AttrNumber  resorigcol;
     203                 :             :         int16       format;
     204                 :             : 
     205                 :             :         /*
     206                 :             :          * If column is a domain, send the base type and typmod instead.
     207                 :             :          * Lookup before sending any ints, for efficiency.
     208                 :             :          */
     209                 :      602248 :         atttypid = getBaseTypeAndTypmod(atttypid, &atttypmod);
     210                 :             : 
     211                 :             :         /* Do we have a non-resjunk tlist item? */
     212         [ +  + ]:      602248 :         while (tlist_item &&
     213         [ -  + ]:      590332 :                ((TargetEntry *) lfirst(tlist_item))->resjunk)
     214                 :           0 :             tlist_item = lnext(targetlist, tlist_item);
     215         [ +  + ]:      602248 :         if (tlist_item)
     216                 :             :         {
     217                 :      590332 :             TargetEntry *tle = (TargetEntry *) lfirst(tlist_item);
     218                 :             : 
     219                 :      590332 :             resorigtbl = tle->resorigtbl;
     220                 :      590332 :             resorigcol = tle->resorigcol;
     221                 :      590332 :             tlist_item = lnext(targetlist, tlist_item);
     222                 :             :         }
     223                 :             :         else
     224                 :             :         {
     225                 :             :             /* No info available, so send zeroes */
     226                 :       11916 :             resorigtbl = 0;
     227                 :       11916 :             resorigcol = 0;
     228                 :             :         }
     229                 :             : 
     230         [ +  + ]:      602248 :         if (formats)
     231                 :      601993 :             format = formats[i];
     232                 :             :         else
     233                 :         255 :             format = 0;
     234                 :             : 
     235                 :      602248 :         pq_writestring(buf, NameStr(att->attname));
     236                 :      602248 :         pq_writeint32(buf, resorigtbl);
     237                 :      602248 :         pq_writeint16(buf, resorigcol);
     238                 :      602248 :         pq_writeint32(buf, atttypid);
     239                 :      602248 :         pq_writeint16(buf, att->attlen);
     240                 :      602248 :         pq_writeint32(buf, atttypmod);
     241                 :      602248 :         pq_writeint16(buf, format);
     242                 :             :     }
     243                 :             : 
     244                 :      196334 :     pq_endmessage_reuse(buf);
     245                 :      196334 : }
     246                 :             : 
     247                 :             : /*
     248                 :             :  * Get the lookup info that printtup() needs
     249                 :             :  */
     250                 :             : static void
     251                 :      160241 : printtup_prepare_info(DR_printtup *myState, TupleDesc typeinfo, int numAttrs)
     252                 :             : {
     253                 :      160241 :     int16      *formats = myState->portal->formats;
     254                 :             :     int         i;
     255                 :             : 
     256                 :             :     /* get rid of any old data */
     257         [ +  + ]:      160241 :     if (myState->myinfo)
     258                 :         867 :         pfree(myState->myinfo);
     259                 :      160241 :     myState->myinfo = NULL;
     260                 :             : 
     261                 :      160241 :     myState->attrinfo = typeinfo;
     262                 :      160241 :     myState->nattrs = numAttrs;
     263         [ +  + ]:      160241 :     if (numAttrs <= 0)
     264                 :         195 :         return;
     265                 :             : 
     266                 :      160046 :     myState->myinfo = (PrinttupAttrInfo *)
     267                 :      160046 :         palloc0(numAttrs * sizeof(PrinttupAttrInfo));
     268                 :             : 
     269         [ +  + ]:      627547 :     for (i = 0; i < numAttrs; i++)
     270                 :             :     {
     271                 :      467501 :         PrinttupAttrInfo *thisState = myState->myinfo + i;
     272         [ +  + ]:      467501 :         int16       format = (formats ? formats[i] : 0);
     273                 :      467501 :         Form_pg_attribute attr = TupleDescAttr(typeinfo, i);
     274                 :             : 
     275                 :      467501 :         thisState->format = format;
     276         [ +  + ]:      467501 :         if (format == 0)
     277                 :             :         {
     278                 :      467458 :             getTypeOutputInfo(attr->atttypid,
     279                 :             :                               &thisState->typoutput,
     280                 :             :                               &thisState->typisvarlena);
     281                 :      467458 :             fmgr_info(thisState->typoutput, &thisState->finfo);
     282                 :             :         }
     283         [ +  - ]:          43 :         else if (format == 1)
     284                 :             :         {
     285                 :          43 :             getTypeBinaryOutputInfo(attr->atttypid,
     286                 :             :                                     &thisState->typsend,
     287                 :             :                                     &thisState->typisvarlena);
     288                 :          43 :             fmgr_info(thisState->typsend, &thisState->finfo);
     289                 :             :         }
     290                 :             :         else
     291         [ #  # ]:           0 :             ereport(ERROR,
     292                 :             :                     (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     293                 :             :                      errmsg("unsupported format code: %d", format)));
     294                 :             :     }
     295                 :             : }
     296                 :             : 
     297                 :             : /* ----------------
     298                 :             :  *      printtup --- send a tuple to the client
     299                 :             :  *
     300                 :             :  * Note: if you change this function, see also serializeAnalyzeReceive
     301                 :             :  * in explain.c, which is meant to replicate the computations done here.
     302                 :             :  * ----------------
     303                 :             :  */
     304                 :             : static bool
     305                 :     3733019 : printtup(TupleTableSlot *slot, DestReceiver *self)
     306                 :             : {
     307                 :     3733019 :     TupleDesc   typeinfo = slot->tts_tupleDescriptor;
     308                 :     3733019 :     DR_printtup *myState = (DR_printtup *) self;
     309                 :             :     MemoryContext oldcontext;
     310                 :     3733019 :     StringInfo  buf = &myState->buf;
     311                 :     3733019 :     int         natts = typeinfo->natts;
     312                 :             :     int         i;
     313                 :             : 
     314                 :             :     /* Set or update my derived attribute info, if needed */
     315   [ +  +  -  + ]:     3733019 :     if (myState->attrinfo != typeinfo || myState->nattrs != natts)
     316                 :      160241 :         printtup_prepare_info(myState, typeinfo, natts);
     317                 :             : 
     318                 :             :     /* Make sure the tuple is fully deconstructed */
     319                 :     3733019 :     slot_getallattrs(slot);
     320                 :             : 
     321                 :             :     /* Switch into per-row context so we can recover memory below */
     322                 :     3733019 :     oldcontext = MemoryContextSwitchTo(myState->tmpcontext);
     323                 :             : 
     324                 :             :     /*
     325                 :             :      * Prepare a DataRow message (note buffer is in per-query context)
     326                 :             :      */
     327                 :     3733019 :     pq_beginmessage_reuse(buf, PqMsg_DataRow);
     328                 :             : 
     329                 :     3733019 :     pq_sendint16(buf, natts);
     330                 :             : 
     331                 :             :     /*
     332                 :             :      * send the attributes of this tuple
     333                 :             :      */
     334         [ +  + ]:    22482933 :     for (i = 0; i < natts; ++i)
     335                 :             :     {
     336                 :    18749922 :         PrinttupAttrInfo *thisState = myState->myinfo + i;
     337                 :    18749922 :         Datum       attr = slot->tts_values[i];
     338                 :             : 
     339         [ +  + ]:    18749922 :         if (slot->tts_isnull[i])
     340                 :             :         {
     341                 :     1116195 :             pq_sendint32(buf, -1);
     342                 :     1116195 :             continue;
     343                 :             :         }
     344                 :             : 
     345                 :             :         /*
     346                 :             :          * Here we catch undefined bytes in datums that are returned to the
     347                 :             :          * client without hitting disk; see comments at the related check in
     348                 :             :          * PageAddItem().  This test is most useful for uncompressed,
     349                 :             :          * non-external datums, but we're quite likely to see such here when
     350                 :             :          * testing new C functions.
     351                 :             :          */
     352                 :    17633727 :         if (thisState->typisvarlena)
     353                 :             :             VALGRIND_CHECK_MEM_IS_DEFINED(DatumGetPointer(attr),
     354                 :             :                                           VARSIZE_ANY(DatumGetPointer(attr)));
     355                 :             : 
     356         [ +  + ]:    17633727 :         if (thisState->format == 0)
     357                 :             :         {
     358                 :             :             /* Text output */
     359                 :             :             char       *outputstr;
     360                 :             : 
     361                 :    17626607 :             outputstr = OutputFunctionCall(&thisState->finfo, attr);
     362                 :    17626599 :             pq_sendcountedtext(buf, outputstr, strlen(outputstr));
     363                 :             :         }
     364                 :             :         else
     365                 :             :         {
     366                 :             :             /* Binary output */
     367                 :             :             bytea      *outputbytes;
     368                 :             : 
     369                 :        7120 :             outputbytes = SendFunctionCall(&thisState->finfo, attr);
     370                 :        7120 :             pq_sendint32(buf, VARSIZE(outputbytes) - VARHDRSZ);
     371                 :        7120 :             pq_sendbytes(buf, VARDATA(outputbytes),
     372                 :        7120 :                          VARSIZE(outputbytes) - VARHDRSZ);
     373                 :             :         }
     374                 :             :     }
     375                 :             : 
     376                 :     3733011 :     pq_endmessage_reuse(buf);
     377                 :             : 
     378                 :             :     /* Return to caller's context, and flush row's temporary memory */
     379                 :     3733011 :     MemoryContextSwitchTo(oldcontext);
     380                 :     3733011 :     MemoryContextReset(myState->tmpcontext);
     381                 :             : 
     382                 :     3733011 :     return true;
     383                 :             : }
     384                 :             : 
     385                 :             : /* ----------------
     386                 :             :  *      printtup_shutdown
     387                 :             :  * ----------------
     388                 :             :  */
     389                 :             : static void
     390                 :      191255 : printtup_shutdown(DestReceiver *self)
     391                 :             : {
     392                 :      191255 :     DR_printtup *myState = (DR_printtup *) self;
     393                 :             : 
     394         [ +  + ]:      191255 :     if (myState->myinfo)
     395                 :      159047 :         pfree(myState->myinfo);
     396                 :      191255 :     myState->myinfo = NULL;
     397                 :             : 
     398                 :      191255 :     myState->attrinfo = NULL;
     399                 :             : 
     400         [ +  - ]:      191255 :     if (myState->buf.data)
     401                 :      191255 :         pfree(myState->buf.data);
     402                 :      191255 :     myState->buf.data = NULL;
     403                 :             : 
     404         [ +  - ]:      191255 :     if (myState->tmpcontext)
     405                 :      191255 :         MemoryContextDelete(myState->tmpcontext);
     406                 :      191255 :     myState->tmpcontext = NULL;
     407                 :      191255 : }
     408                 :             : 
     409                 :             : /* ----------------
     410                 :             :  *      printtup_destroy
     411                 :             :  * ----------------
     412                 :             :  */
     413                 :             : static void
     414                 :      388586 : printtup_destroy(DestReceiver *self)
     415                 :             : {
     416                 :      388586 :     pfree(self);
     417                 :      388586 : }
     418                 :             : 
     419                 :             : /* ----------------
     420                 :             :  *      printatt
     421                 :             :  * ----------------
     422                 :             :  */
     423                 :             : static void
     424                 :         242 : printatt(unsigned attributeId,
     425                 :             :          Form_pg_attribute attributeP,
     426                 :             :          char *value)
     427                 :             : {
     428   [ +  +  +  +  :         242 :     printf("\t%2d: %s%s%s%s\t(typeid = %u, len = %d, typmod = %d, byval = %c)\n",
             +  +  +  + ]
     429                 :             :            attributeId,
     430                 :             :            NameStr(attributeP->attname),
     431                 :             :            value != NULL ? " = \"" : "",
     432                 :             :            value != NULL ? value : "",
     433                 :             :            value != NULL ? "\"" : "",
     434                 :             :            attributeP->atttypid,
     435                 :             :            attributeP->attlen,
     436                 :             :            attributeP->atttypmod,
     437                 :             :            attributeP->attbyval ? 't' : 'f');
     438                 :         242 : }
     439                 :             : 
     440                 :             : /* ----------------
     441                 :             :  *      debugStartup - prepare to print tuples for an interactive backend
     442                 :             :  * ----------------
     443                 :             :  */
     444                 :             : void
     445                 :         120 : debugStartup(DestReceiver *self, int operation, TupleDesc typeinfo)
     446                 :             : {
     447                 :         120 :     int         natts = typeinfo->natts;
     448                 :             :     int         i;
     449                 :             : 
     450                 :             :     /*
     451                 :             :      * show the return type of the tuples
     452                 :             :      */
     453         [ +  + ]:         240 :     for (i = 0; i < natts; ++i)
     454                 :         120 :         printatt((unsigned) i + 1, TupleDescAttr(typeinfo, i), NULL);
     455                 :         120 :     printf("\t----\n");
     456                 :         120 : }
     457                 :             : 
     458                 :             : /* ----------------
     459                 :             :  *      debugtup - print one tuple for an interactive backend
     460                 :             :  * ----------------
     461                 :             :  */
     462                 :             : bool
     463                 :         122 : debugtup(TupleTableSlot *slot, DestReceiver *self)
     464                 :             : {
     465                 :         122 :     TupleDesc   typeinfo = slot->tts_tupleDescriptor;
     466                 :         122 :     int         natts = typeinfo->natts;
     467                 :             :     int         i;
     468                 :             :     Datum       attr;
     469                 :             :     char       *value;
     470                 :             :     bool        isnull;
     471                 :             :     Oid         typoutput;
     472                 :             :     bool        typisvarlena;
     473                 :             : 
     474         [ +  + ]:         244 :     for (i = 0; i < natts; ++i)
     475                 :             :     {
     476                 :         122 :         attr = slot_getattr(slot, i + 1, &isnull);
     477         [ -  + ]:         122 :         if (isnull)
     478                 :           0 :             continue;
     479                 :         122 :         getTypeOutputInfo(TupleDescAttr(typeinfo, i)->atttypid,
     480                 :             :                           &typoutput, &typisvarlena);
     481                 :             : 
     482                 :         122 :         value = OidOutputFunctionCall(typoutput, attr);
     483                 :             : 
     484                 :         122 :         printatt((unsigned) i + 1, TupleDescAttr(typeinfo, i), value);
     485                 :             :     }
     486                 :         122 :     printf("\t----\n");
     487                 :             : 
     488                 :         122 :     return true;
     489                 :             : }
        

Generated by: LCOV version 2.0-1