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