printtup.c 7.3 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
11
 *	  $Header: /cvsroot/pgsql/src/backend/access/common/printtup.c,v 1.39 1999/01/24 22:50:58 tgl Exp $
12 13 14 15
 *
 *-------------------------------------------------------------------------
 */

16
#include <string.h>
M
Marc G. Fournier 已提交
17
#include <postgres.h>
18

19 20 21
#include <fmgr.h>
#include <access/heapam.h>
#include <access/printtup.h>
M
Marc G. Fournier 已提交
22 23 24
#include <catalog/pg_type.h>
#include <libpq/libpq.h>
#include <utils/syscache.h>
M
Marc G. Fournier 已提交
25

26
#ifdef MULTIBYTE
M
 
Marc G. Fournier 已提交
27
#include <mb/pg_wchar.h>
28 29
#endif

30
/* ----------------------------------------------------------------
31
 *		printtup / debugtup support
32 33 34 35
 * ----------------------------------------------------------------
 */

/* ----------------
36 37 38 39 40
 *		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.
41 42
 * ----------------
 */
43 44
int
getTypeOutAndElem(Oid type, Oid* typOutput, Oid* typElem)
45
{
46
	HeapTuple	typeTuple;
47 48 49 50 51 52

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

	if (HeapTupleIsValid(typeTuple))
53 54 55 56 57 58
	{
		Form_pg_type pt = (Form_pg_type) GETSTRUCT(typeTuple);
		*typOutput = (Oid) pt->typoutput;
		*typElem = (Oid) pt->typelem;
		return OidIsValid(*typOutput);
	}
59

60
	elog(ERROR, "getTypeOutAndElem: Cache lookup of type %d failed", type);
61

62 63 64
	*typOutput = InvalidOid;
	*typElem = InvalidOid;
	return 0;
65 66 67
}

/* ----------------
68
 *		printtup
69 70 71 72 73
 * ----------------
 */
void
printtup(HeapTuple tuple, TupleDesc typeinfo)
{
74 75
	int			i,
				j,
76 77
				k,
				outputlen;
78 79
	char	   *outputstr;
	Datum		attr;
80
	bool		isnull;
81 82
	Oid			typoutput,
				typelem;
83
#ifdef MULTIBYTE
84 85
	unsigned char *p;
#endif
86 87

	/* ----------------
88
	 *	tell the frontend to expect new tuple data (in ASCII style)
89 90 91 92 93
	 * ----------------
	 */
	pq_putnchar("D", 1);

	/* ----------------
94
	 *	send a bitmap of which attributes are not null
95 96 97 98
	 * ----------------
	 */
	j = 0;
	k = 1 << 7;
99
	for (i = 0; i < tuple->t_data->t_natts; ++i)
100
	{
101 102
		if (! heap_attisnull(tuple, i + 1))
			j |= k;				/* set bit if not null */
103
		k >>= 1;
104
		if (k == 0)				/* end of byte? */
105 106 107 108 109
		{
			pq_putint(j, 1);
			j = 0;
			k = 1 << 7;
		}
110
	}
111
	if (k != (1 << 7))			/* flush last partial byte */
112 113 114 115 116 117
		pq_putint(j, 1);

	/* ----------------
	 *	send the attributes of this tuple
	 * ----------------
	 */
118
	for (i = 0; i < tuple->t_data->t_natts; ++i)
119
	{
120
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
M
 
Marc G. Fournier 已提交
121 122
		if (isnull)
			continue;
123 124
		if (getTypeOutAndElem((Oid) typeinfo->attrs[i]->atttypid,
							  &typoutput, &typelem))
125
		{
126
			outputstr = fmgr(typoutput, attr, typelem,
127
							 typeinfo->attrs[i]->atttypmod);
128
#ifdef MULTIBYTE
129
			p = pg_server_to_client(outputstr, strlen(outputstr));
130 131 132
			outputlen = strlen(p);
			pq_putint(outputlen + VARHDRSZ, VARHDRSZ);
			pq_putnchar(p, outputlen);
133
#else
134 135 136
			outputlen = strlen(outputstr);
			pq_putint(outputlen + VARHDRSZ, VARHDRSZ);
			pq_putnchar(outputstr, outputlen);
137
#endif
138 139
			pfree(outputstr);
		}
M
 
Marc G. Fournier 已提交
140 141 142
		else
		{
			outputstr = "<unprintable>";
143 144 145
			outputlen = strlen(outputstr);
			pq_putint(outputlen + VARHDRSZ, VARHDRSZ);
			pq_putnchar(outputstr, outputlen);
146
		}
147 148 149 150
	}
}

/* ----------------
151
 *		printatt
152 153 154 155
 * ----------------
 */
static void
printatt(unsigned attributeId,
156
		 Form_pg_attribute attributeP,
157
		 char *value)
158
{
B
Bruce Momjian 已提交
159
	printf("\t%2d: %s%s%s%s\t(typeid = %u, len = %d, typmod = %d, byval = %c)\n",
160 161 162 163 164 165 166
		   attributeId,
		   attributeP->attname.data,
		   value != NULL ? " = \"" : "",
		   value != NULL ? value : "",
		   value != NULL ? "\"" : "",
		   (unsigned int) (attributeP->atttypid),
		   attributeP->attlen,
B
Bruce Momjian 已提交
167
		   attributeP->atttypmod,
168
		   attributeP->attbyval ? 't' : 'f');
169 170 171
}

/* ----------------
172
 *		showatts
173 174 175 176 177
 * ----------------
 */
void
showatts(char *name, TupleDesc tupleDesc)
{
178 179
	int			i;
	int			natts = tupleDesc->natts;
180
	Form_pg_attribute *attinfo = tupleDesc->attrs;
181

182 183 184 185
	puts(name);
	for (i = 0; i < natts; ++i)
		printatt((unsigned) i + 1, attinfo[i], (char *) NULL);
	printf("\t----\n");
186 187 188
}

/* ----------------
189
 *		debugtup
190 191 192 193 194
 * ----------------
 */
void
debugtup(HeapTuple tuple, TupleDesc typeinfo)
{
195 196
	int			i;
	Datum		attr;
197
	char	   *value;
198
	bool		isnull;
199 200
	Oid			typoutput,
				typelem;
201

202
	for (i = 0; i < tuple->t_data->t_natts; ++i)
203
	{
204
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
205 206 207 208
		if (isnull)
			continue;
		if (getTypeOutAndElem((Oid) typeinfo->attrs[i]->atttypid,
							  &typoutput, &typelem))
209
		{
210
			value = fmgr(typoutput, attr, typelem,
211
						 typeinfo->attrs[i]->atttypmod);
212 213 214
			printatt((unsigned) i + 1, typeinfo->attrs[i], value);
			pfree(value);
		}
215
	}
216
	printf("\t----\n");
217 218 219
}

/* ----------------
220 221 222
 *		printtup_internal
 *		We use a different data prefix, e.g. 'B' instead of 'D' to
 *		indicate a tuple in internal (binary) form.
223
 *
224
 *		This is same as printtup, except we don't use the typout func.
225 226 227 228 229
 * ----------------
 */
void
printtup_internal(HeapTuple tuple, TupleDesc typeinfo)
{
230 231 232
	int			i,
				j,
				k;
233
	Datum		attr;
234
	bool		isnull;
235 236

	/* ----------------
237
	 *	tell the frontend to expect new tuple data (in binary style)
238 239 240 241 242
	 * ----------------
	 */
	pq_putnchar("B", 1);

	/* ----------------
243
	 *	send a bitmap of which attributes are not null
244 245 246 247
	 * ----------------
	 */
	j = 0;
	k = 1 << 7;
248
	for (i = 0; i < tuple->t_data->t_natts; ++i)
249
	{
250 251
		if (! heap_attisnull(tuple, i + 1))
			j |= k;				/* set bit if not null */
252
		k >>= 1;
253
		if (k == 0)				/* end of byte? */
254 255 256 257 258
		{
			pq_putint(j, 1);
			j = 0;
			k = 1 << 7;
		}
259
	}
260
	if (k != (1 << 7))			/* flush last partial byte */
261 262 263 264 265 266
		pq_putint(j, 1);

	/* ----------------
	 *	send the attributes of this tuple
	 * ----------------
	 */
267
#ifdef IPORTAL_DEBUG
268
	fprintf(stderr, "sending tuple with %d atts\n", tuple->t_data->t_natts);
269
#endif
270
	for (i = 0; i < tuple->t_data->t_natts; ++i)
271
	{
272
		int32		len = typeinfo->attrs[i]->attlen;
273

274
		attr = heap_getattr(tuple, i + 1, typeinfo, &isnull);
275
		if (!isnull)
276
		{
277 278 279 280 281 282
			/* # of bytes, and opaque data */
			if (len == -1)
			{
				/* variable length, assume a varlena structure */
				len = VARSIZE(attr) - VARHDRSZ;

B
Bruce Momjian 已提交
283
				pq_putint(len, VARHDRSZ);
284
				pq_putnchar(VARDATA(attr), len);
285

286 287
#ifdef IPORTAL_DEBUG
				{
288
					char	   *d = VARDATA(attr);
289 290 291 292

					fprintf(stderr, "length %d data %x%x%x%x\n",
							len, *d, *(d + 1), *(d + 2), *(d + 3));
				}
293
#endif
294 295 296 297 298 299
			}
			else
			{
				/* fixed size */
				if (typeinfo->attrs[i]->attbyval)
				{
300 301 302
					int8		i8;
					int16		i16;
					int32		i32;
303 304 305 306

					pq_putint(len, sizeof(int32));
					switch (len)
					{
307 308 309 310 311 312 313 314 315 316 317 318
						case sizeof(int8):
							i8 = DatumGetChar(attr);
							pq_putnchar((char *) &i8, len);
							break;
						case sizeof(int16):
							i16 = DatumGetInt16(attr);
							pq_putnchar((char *) &i16, len);
							break;
						case sizeof(int32):
							i32 = DatumGetInt32(attr);
							pq_putnchar((char *) &i32, len);
							break;
319
					}
320
#ifdef IPORTAL_DEBUG
321
					fprintf(stderr, "byval length %d data %d\n", len, attr);
322
#endif
323 324 325 326
				}
				else
				{
					pq_putint(len, sizeof(int32));
327
					pq_putnchar(DatumGetPointer(attr), len);
328
#ifdef IPORTAL_DEBUG
329
					fprintf(stderr, "byref length %d data %x\n", len,
330
							DatumGetPointer(attr));
331
#endif
332 333
				}
			}
334 335 336
		}
	}
}