printtup.c 10.1 KB
Newer Older
1 2
/*-------------------------------------------------------------------------
 *
3
 * printtup.c
4 5
 *	  Routines to print out tuples to the destination (binary or non-binary
 *	  portals, frontend/interactive backend, etc.).
6 7 8 9 10
 *
 * Copyright (c) 1994, Regents of the University of California
 *
 *
 * IDENTIFICATION
B
Bruce Momjian 已提交
11
 *	  $Header: /cvsroot/pgsql/src/backend/access/common/printtup.c,v 1.46 1999/05/25 16:06:39 momjian Exp $
12 13 14 15
 *
 *-------------------------------------------------------------------------
 */

16
#include <string.h>
17

18 19 20 21 22 23 24 25 26
#include "postgres.h"

#include "fmgr.h"
#include "access/heapam.h"
#include "access/printtup.h"
#include "catalog/pg_type.h"
#include "libpq/libpq.h"
#include "libpq/pqformat.h"
#include "utils/syscache.h"
M
Marc G. Fournier 已提交
27

B
Bruce Momjian 已提交
28 29 30
static void printtup_setup(DestReceiver * self, TupleDesc typeinfo);
static void printtup(HeapTuple tuple, TupleDesc typeinfo, DestReceiver * self);
static void printtup_cleanup(DestReceiver * self);
31

32
/* ----------------------------------------------------------------
33
 *		printtup / debugtup support
34 35 36 37
 * ----------------------------------------------------------------
 */

/* ----------------
38 39 40 41 42
 *		getTypeOutAndElem -- get both typoutput and typelem for a type
 *
 * We used to fetch these with two separate function calls,
 * typtoout() and gettypelem(), which each called SearchSysCacheTuple.
 * This way takes half the time.
43 44
 * ----------------
 */
45
int
B
Bruce Momjian 已提交
46
getTypeOutAndElem(Oid type, Oid *typOutput, Oid *typElem)
47
{
48
	HeapTuple	typeTuple;
49 50 51 52 53 54

	typeTuple = SearchSysCacheTuple(TYPOID,
									ObjectIdGetDatum(type),
									0, 0, 0);

	if (HeapTupleIsValid(typeTuple))
55 56
	{
		Form_pg_type pt = (Form_pg_type) GETSTRUCT(typeTuple);
B
Bruce Momjian 已提交
57

58 59 60 61
		*typOutput = (Oid) pt->typoutput;
		*typElem = (Oid) pt->typelem;
		return OidIsValid(*typOutput);
	}
62

63
	elog(ERROR, "getTypeOutAndElem: Cache lookup of type %u failed", type);
64

65 66 67
	*typOutput = InvalidOid;
	*typElem = InvalidOid;
	return 0;
68 69
}

70 71 72 73
/* ----------------
 *		Private state for a printtup destination object
 * ----------------
 */
B
Bruce Momjian 已提交
74 75
typedef struct
{								/* Per-attribute information */
76 77
	Oid			typoutput;		/* Oid for the attribute's type output fn */
	Oid			typelem;		/* typelem value to pass to the output fn */
78
	FmgrInfo	finfo;			/* Precomputed call info for typoutput */
B
Bruce Momjian 已提交
79
}			PrinttupAttrInfo;
80

B
Bruce Momjian 已提交
81 82 83 84 85 86 87
typedef struct
{
	DestReceiver pub;			/* publicly-known function pointers */
	TupleDesc	attrinfo;		/* The attr info we are set up for */
	int			nattrs;
	PrinttupAttrInfo *myinfo;	/* Cached info about each attr */
}			DR_printtup;
88 89 90 91 92

/* ----------------
 *		Initialize: create a DestReceiver for printtup
 * ----------------
 */
B
Bruce Momjian 已提交
93
DestReceiver *
94 95
printtup_create_DR()
{
B
Bruce Momjian 已提交
96
	DR_printtup *self = (DR_printtup *) palloc(sizeof(DR_printtup));
97 98 99 100 101 102 103 104 105

	self->pub.receiveTuple = printtup;
	self->pub.setup = printtup_setup;
	self->pub.cleanup = printtup_cleanup;

	self->attrinfo = NULL;
	self->nattrs = 0;
	self->myinfo = NULL;

B
Bruce Momjian 已提交
106
	return (DestReceiver *) self;
107 108 109
}

static void
B
Bruce Momjian 已提交
110
printtup_setup(DestReceiver * self, TupleDesc typeinfo)
111 112 113 114 115
{
	/* ----------------
	 * We could set up the derived attr info at this time, but we postpone it
	 * until the first call of printtup, for 3 reasons:
	 * 1. We don't waste time (compared to the old way) if there are no
B
Bruce Momjian 已提交
116
	 *	  tuples at all to output.
117
	 * 2. Checking in printtup allows us to handle the case that the tuples
B
Bruce Momjian 已提交
118 119
	 *	  change type midway through (although this probably can't happen in
	 *	  the current executor).
120 121 122 123 124 125
	 * 3. Right now, ExecutorRun passes a NULL for typeinfo anyway :-(
	 * ----------------
	 */
}

static void
B
Bruce Momjian 已提交
126
printtup_prepare_info(DR_printtup * myState, TupleDesc typeinfo, int numAttrs)
127
{
B
Bruce Momjian 已提交
128
	int			i;
129 130

	if (myState->myinfo)
B
Bruce Momjian 已提交
131
		pfree(myState->myinfo); /* get rid of any old data */
132 133 134 135 136
	myState->myinfo = NULL;
	myState->attrinfo = typeinfo;
	myState->nattrs = numAttrs;
	if (numAttrs <= 0)
		return;
B
Bruce Momjian 已提交
137
	myState->myinfo = (PrinttupAttrInfo *)
138 139 140
		palloc(numAttrs * sizeof(PrinttupAttrInfo));
	for (i = 0; i < numAttrs; i++)
	{
B
Bruce Momjian 已提交
141 142
		PrinttupAttrInfo *thisState = myState->myinfo + i;

143 144 145
		if (getTypeOutAndElem((Oid) typeinfo->attrs[i]->atttypid,
							  &thisState->typoutput, &thisState->typelem))
			fmgr_info(thisState->typoutput, &thisState->finfo);
146 147 148
	}
}

149
/* ----------------
150
 *		printtup
151 152
 * ----------------
 */
153
static void
B
Bruce Momjian 已提交
154
printtup(HeapTuple tuple, TupleDesc typeinfo, DestReceiver * self)
155
{
B
Bruce Momjian 已提交
156
	DR_printtup *myState = (DR_printtup *) self;
157
	StringInfoData buf;
158 159
	int			i,
				j,
160
				k;
161 162
	char	   *outputstr;
	Datum		attr;
163
	bool		isnull;
164

165 166 167 168 169
	/* Set or update my derived attribute info, if needed */
	if (myState->attrinfo != typeinfo ||
		myState->nattrs != tuple->t_data->t_natts)
		printtup_prepare_info(myState, typeinfo, tuple->t_data->t_natts);

170
	/* ----------------
171
	 *	tell the frontend to expect new tuple data (in ASCII style)
172 173
	 * ----------------
	 */
174 175
	pq_beginmessage(&buf);
	pq_sendbyte(&buf, 'D');
176 177

	/* ----------------
178
	 *	send a bitmap of which attributes are not null
179 180 181 182
	 * ----------------
	 */
	j = 0;
	k = 1 << 7;
183
	for (i = 0; i < tuple->t_data->t_natts; ++i)
184
	{
B
Bruce Momjian 已提交
185
		if (!heap_attisnull(tuple, i + 1))
186
			j |= k;				/* set bit if not null */
187
		k >>= 1;
188
		if (k == 0)				/* end of byte? */
189
		{
190
			pq_sendint(&buf, j, 1);
191 192 193
			j = 0;
			k = 1 << 7;
		}
194
	}
195
	if (k != (1 << 7))			/* flush last partial byte */
196
		pq_sendint(&buf, j, 1);
197 198 199 200 201

	/* ----------------
	 *	send the attributes of this tuple
	 * ----------------
	 */
202
	for (i = 0; i < tuple->t_data->t_natts; ++i)
203
	{
B
Bruce Momjian 已提交
204 205
		PrinttupAttrInfo *thisState = myState->myinfo + i;

206
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
M
 
Marc G. Fournier 已提交
207 208
		if (isnull)
			continue;
209
		if (OidIsValid(thisState->typoutput))
210
		{
211 212
			outputstr = (char *) (*fmgr_faddr(&thisState->finfo))
				(attr, thisState->typelem, typeinfo->attrs[i]->atttypmod);
213
			pq_sendcountedtext(&buf, outputstr, strlen(outputstr));
214 215
			pfree(outputstr);
		}
M
 
Marc G. Fournier 已提交
216 217 218
		else
		{
			outputstr = "<unprintable>";
219
			pq_sendcountedtext(&buf, outputstr, strlen(outputstr));
220
		}
221
	}
222 223

	pq_endmessage(&buf);
224 225
}

226 227 228 229 230
/* ----------------
 *		printtup_cleanup
 * ----------------
 */
static void
B
Bruce Momjian 已提交
231
printtup_cleanup(DestReceiver * self)
232
{
B
Bruce Momjian 已提交
233 234
	DR_printtup *myState = (DR_printtup *) self;

235 236 237 238 239
	if (myState->myinfo)
		pfree(myState->myinfo);
	pfree(myState);
}

240
/* ----------------
241
 *		printatt
242 243 244 245
 * ----------------
 */
static void
printatt(unsigned attributeId,
246
		 Form_pg_attribute attributeP,
247
		 char *value)
248
{
B
Bruce Momjian 已提交
249
	printf("\t%2d: %s%s%s%s\t(typeid = %u, len = %d, typmod = %d, byval = %c)\n",
250 251 252 253 254 255 256
		   attributeId,
		   attributeP->attname.data,
		   value != NULL ? " = \"" : "",
		   value != NULL ? value : "",
		   value != NULL ? "\"" : "",
		   (unsigned int) (attributeP->atttypid),
		   attributeP->attlen,
B
Bruce Momjian 已提交
257
		   attributeP->atttypmod,
258
		   attributeP->attbyval ? 't' : 'f');
259 260 261
}

/* ----------------
262
 *		showatts
263 264 265 266 267
 * ----------------
 */
void
showatts(char *name, TupleDesc tupleDesc)
{
268 269
	int			i;
	int			natts = tupleDesc->natts;
270
	Form_pg_attribute *attinfo = tupleDesc->attrs;
271

272 273 274 275
	puts(name);
	for (i = 0; i < natts; ++i)
		printatt((unsigned) i + 1, attinfo[i], (char *) NULL);
	printf("\t----\n");
276 277 278
}

/* ----------------
279
 *		debugtup
280 281 282
 * ----------------
 */
void
B
Bruce Momjian 已提交
283
debugtup(HeapTuple tuple, TupleDesc typeinfo, DestReceiver * self)
284
{
285 286
	int			i;
	Datum		attr;
287
	char	   *value;
288
	bool		isnull;
289 290
	Oid			typoutput,
				typelem;
291

292
	for (i = 0; i < tuple->t_data->t_natts; ++i)
293
	{
294
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
295 296 297 298
		if (isnull)
			continue;
		if (getTypeOutAndElem((Oid) typeinfo->attrs[i]->atttypid,
							  &typoutput, &typelem))
299
		{
300
			value = fmgr(typoutput, attr, typelem,
301
						 typeinfo->attrs[i]->atttypmod);
302 303 304
			printatt((unsigned) i + 1, typeinfo->attrs[i], value);
			pfree(value);
		}
305
	}
306
	printf("\t----\n");
307 308 309
}

/* ----------------
310 311 312
 *		printtup_internal
 *		We use a different data prefix, e.g. 'B' instead of 'D' to
 *		indicate a tuple in internal (binary) form.
313
 *
314 315
 *		This is same as printtup, except we don't use the typout func,
 *		and therefore have no need for persistent state.
316 317 318
 * ----------------
 */
void
B
Bruce Momjian 已提交
319
printtup_internal(HeapTuple tuple, TupleDesc typeinfo, DestReceiver * self)
320
{
321
	StringInfoData buf;
322 323 324
	int			i,
				j,
				k;
325
	Datum		attr;
326
	bool		isnull;
327 328

	/* ----------------
329
	 *	tell the frontend to expect new tuple data (in binary style)
330 331
	 * ----------------
	 */
332 333
	pq_beginmessage(&buf);
	pq_sendbyte(&buf, 'B');
334 335

	/* ----------------
336
	 *	send a bitmap of which attributes are not null
337 338 339 340
	 * ----------------
	 */
	j = 0;
	k = 1 << 7;
341
	for (i = 0; i < tuple->t_data->t_natts; ++i)
342
	{
B
Bruce Momjian 已提交
343
		if (!heap_attisnull(tuple, i + 1))
344
			j |= k;				/* set bit if not null */
345
		k >>= 1;
346
		if (k == 0)				/* end of byte? */
347
		{
348
			pq_sendint(&buf, j, 1);
349 350 351
			j = 0;
			k = 1 << 7;
		}
352
	}
353
	if (k != (1 << 7))			/* flush last partial byte */
354
		pq_sendint(&buf, j, 1);
355 356 357 358 359

	/* ----------------
	 *	send the attributes of this tuple
	 * ----------------
	 */
360
#ifdef IPORTAL_DEBUG
361
	fprintf(stderr, "sending tuple with %d atts\n", tuple->t_data->t_natts);
362
#endif
363
	for (i = 0; i < tuple->t_data->t_natts; ++i)
364
	{
365
		int32		len = typeinfo->attrs[i]->attlen;
366

367
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
368
		if (!isnull)
369
		{
370 371 372 373 374 375
			/* # of bytes, and opaque data */
			if (len == -1)
			{
				/* variable length, assume a varlena structure */
				len = VARSIZE(attr) - VARHDRSZ;

376 377
				pq_sendint(&buf, len, VARHDRSZ);
				pq_sendbytes(&buf, VARDATA(attr), len);
378

379 380
#ifdef IPORTAL_DEBUG
				{
381
					char	   *d = VARDATA(attr);
382 383 384 385

					fprintf(stderr, "length %d data %x%x%x%x\n",
							len, *d, *(d + 1), *(d + 2), *(d + 3));
				}
386
#endif
387 388 389 390 391 392
			}
			else
			{
				/* fixed size */
				if (typeinfo->attrs[i]->attbyval)
				{
393 394 395
					int8		i8;
					int16		i16;
					int32		i32;
396

397
					pq_sendint(&buf, len, sizeof(int32));
398 399
					switch (len)
					{
400 401
						case sizeof(int8):
							i8 = DatumGetChar(attr);
402
							pq_sendbytes(&buf, (char *) &i8, len);
403 404 405
							break;
						case sizeof(int16):
							i16 = DatumGetInt16(attr);
406
							pq_sendbytes(&buf, (char *) &i16, len);
407 408 409
							break;
						case sizeof(int32):
							i32 = DatumGetInt32(attr);
410
							pq_sendbytes(&buf, (char *) &i32, len);
411
							break;
412
					}
413
#ifdef IPORTAL_DEBUG
414
					fprintf(stderr, "byval length %d data %d\n", len, attr);
415
#endif
416 417 418
				}
				else
				{
419 420
					pq_sendint(&buf, len, sizeof(int32));
					pq_sendbytes(&buf, DatumGetPointer(attr), len);
421
#ifdef IPORTAL_DEBUG
422
					fprintf(stderr, "byref length %d data %x\n", len,
423
							DatumGetPointer(attr));
424
#endif
425 426
				}
			}
427 428
		}
	}
429 430

	pq_endmessage(&buf);
431
}