55
55
/*
56
56
* struct varatt_external is a traditional "TOAST pointer", that is, the
57
57
* information needed to fetch a Datum stored out-of-line in a TOAST table.
58
- * The data is compressed if and only if the size stored in va_extinfo <
59
- * va_rawsize - VARHDRSZ.
58
+ * The data is compressed if and only if the external size stored in
59
+ * va_extinfo is less than va_rawsize - VARHDRSZ.
60
60
*
61
61
* This struct must not contain any padding, because we sometimes compare
62
62
* these pointers using memcmp.
@@ -75,6 +75,13 @@ typedef struct varatt_external
75
75
Oid va_toastrelid ; /* RelID of TOAST table containing it */
76
76
} varatt_external ;
77
77
78
+ /*
79
+ * These macros define the "saved size" portion of va_extinfo. Its remaining
80
+ * two high-order bits identify the compression method.
81
+ */
82
+ #define VARLENA_EXTSIZE_BITS 30
83
+ #define VARLENA_EXTSIZE_MASK ((1U << VARLENA_EXTSIZE_BITS) - 1)
84
+
78
85
/*
79
86
* struct varatt_indirect is a "TOAST pointer" representing an out-of-line
80
87
* Datum that's stored in memory, not in an external toast relation.
@@ -149,7 +156,7 @@ typedef union
149
156
{
150
157
uint32 va_header ;
151
158
uint32 va_tcinfo ; /* Original data size (excludes header) and
152
- * compression method */
159
+ * compression method; see va_extinfo */
153
160
char va_data [FLEXIBLE_ARRAY_MEMBER ]; /* Compressed data */
154
161
} va_compressed ;
155
162
} varattrib_4b ;
@@ -235,6 +242,7 @@ typedef struct
235
242
#define SET_VARTAG_1B_E (PTR ,tag ) \
236
243
(((varattrib_1b_e *) (PTR))->va_header = 0x80, \
237
244
((varattrib_1b_e *) (PTR))->va_tag = (tag))
245
+
238
246
#else /* !WORDS_BIGENDIAN */
239
247
240
248
#define VARATT_IS_4B (PTR ) \
@@ -267,36 +275,28 @@ typedef struct
267
275
#define SET_VARTAG_1B_E (PTR ,tag ) \
268
276
(((varattrib_1b_e *) (PTR))->va_header = 0x01, \
269
277
((varattrib_1b_e *) (PTR))->va_tag = (tag))
270
- #endif /* WORDS_BIGENDIAN */
271
278
272
- #define VARHDRSZ_SHORT offsetof(varattrib_1b, va_data)
273
- #define VARATT_SHORT_MAX 0x7F
274
- #define VARATT_CAN_MAKE_SHORT (PTR ) \
275
- (VARATT_IS_4B_U(PTR) && \
276
- (VARSIZE(PTR) - VARHDRSZ + VARHDRSZ_SHORT) <= VARATT_SHORT_MAX)
277
- #define VARATT_CONVERTED_SHORT_SIZE (PTR ) \
278
- (VARSIZE(PTR) - VARHDRSZ + VARHDRSZ_SHORT)
279
-
280
- #define VARHDRSZ_EXTERNAL offsetof(varattrib_1b_e, va_data)
281
- #define VARHDRSZ_COMPRESS offsetof(varattrib_4b, va_compressed.va_data)
279
+ #endif /* WORDS_BIGENDIAN */
282
280
283
281
#define VARDATA_4B (PTR ) (((varattrib_4b *) (PTR))->va_4byte.va_data)
284
282
#define VARDATA_4B_C (PTR ) (((varattrib_4b *) (PTR))->va_compressed.va_data)
285
283
#define VARDATA_1B (PTR ) (((varattrib_1b *) (PTR))->va_data)
286
284
#define VARDATA_1B_E (PTR ) (((varattrib_1b_e *) (PTR))->va_data)
287
285
288
- #define VARLENA_RAWSIZE_BITS 30
289
- #define VARLENA_RAWSIZE_MASK ((1U << VARLENA_RAWSIZE_BITS) - 1)
290
-
291
286
/*
292
- * va_tcinfo in va_compress contains raw size of datum and compression method .
287
+ * Externally visible TOAST macros begin here .
293
288
*/
294
- #define VARRAWSIZE_4B_C (PTR ) \
295
- (((varattrib_4b *) (PTR))->va_compressed.va_tcinfo & VARLENA_RAWSIZE_MASK)
296
- #define VARCOMPRESS_4B_C (PTR ) \
297
- (((varattrib_4b *) (PTR))->va_compressed.va_tcinfo >> VARLENA_RAWSIZE_BITS)
298
289
299
- /* Externally visible macros */
290
+ #define VARHDRSZ_EXTERNAL offsetof(varattrib_1b_e, va_data)
291
+ #define VARHDRSZ_COMPRESSED offsetof(varattrib_4b, va_compressed.va_data)
292
+ #define VARHDRSZ_SHORT offsetof(varattrib_1b, va_data)
293
+
294
+ #define VARATT_SHORT_MAX 0x7F
295
+ #define VARATT_CAN_MAKE_SHORT (PTR ) \
296
+ (VARATT_IS_4B_U(PTR) && \
297
+ (VARSIZE(PTR) - VARHDRSZ + VARHDRSZ_SHORT) <= VARATT_SHORT_MAX)
298
+ #define VARATT_CONVERTED_SHORT_SIZE (PTR ) \
299
+ (VARSIZE(PTR) - VARHDRSZ + VARHDRSZ_SHORT)
300
300
301
301
/*
302
302
* In consumers oblivious to data alignment, call PG_DETOAST_DATUM_PACKED(),
@@ -336,35 +336,6 @@ typedef struct
336
336
(VARATT_IS_EXTERNAL(PTR) && VARTAG_IS_EXPANDED(VARTAG_EXTERNAL(PTR)))
337
337
#define VARATT_IS_EXTERNAL_NON_EXPANDED (PTR ) \
338
338
(VARATT_IS_EXTERNAL(PTR) && !VARTAG_IS_EXPANDED(VARTAG_EXTERNAL(PTR)))
339
-
340
- /*
341
- * va_extinfo in varatt_external contains actual length of the external data
342
- * and compression method if external data is compressed.
343
- */
344
- #define VARATT_EXTERNAL_GET_EXTSIZE (toast_pointer ) \
345
- ((toast_pointer).va_extinfo & VARLENA_RAWSIZE_MASK)
346
-
347
- #define VARATT_EXTERNAL_SET_SIZE_AND_COMPRESSION (toast_pointer , len , cm ) \
348
- do { \
349
- Assert((cm) == TOAST_PGLZ_COMPRESSION_ID || \
350
- (cm) == TOAST_LZ4_COMPRESSION_ID); \
351
- ((toast_pointer).va_extinfo = (len) | (cm) << VARLENA_RAWSIZE_BITS); \
352
- } while (0)
353
-
354
- #define VARATT_EXTERNAL_GET_COMPRESSION (PTR ) \
355
- ((toast_pointer).va_extinfo >> VARLENA_RAWSIZE_BITS)
356
-
357
- /*
358
- * Testing whether an externally-stored value is compressed now requires
359
- * comparing size stored in va_extinfo (the actual length of the external data)
360
- * to rawsize (the original uncompressed datum's size). The latter includes
361
- * VARHDRSZ overhead, the former doesn't. We never use compression unless it
362
- * actually saves space, so we expect either equality or less-than.
363
- */
364
- #define VARATT_EXTERNAL_IS_COMPRESSED (toast_pointer ) \
365
- (VARATT_EXTERNAL_GET_EXTSIZE(toast_pointer) < \
366
- (toast_pointer).va_rawsize - VARHDRSZ)
367
-
368
339
#define VARATT_IS_SHORT (PTR ) VARATT_IS_1B(PTR)
369
340
#define VARATT_IS_EXTENDED (PTR ) (!VARATT_IS_4B_U(PTR))
370
341
@@ -390,6 +361,37 @@ typedef struct
390
361
#define VARDATA_ANY (PTR ) \
391
362
(VARATT_IS_1B(PTR) ? VARDATA_1B(PTR) : VARDATA_4B(PTR))
392
363
364
+ /* Decompressed size and compression method of an external compressed Datum */
365
+ #define VARDATA_COMPRESSED_GET_EXTSIZE (PTR ) \
366
+ (((varattrib_4b *) (PTR))->va_compressed.va_tcinfo & VARLENA_EXTSIZE_MASK)
367
+ #define VARDATA_COMPRESSED_GET_COMPRESS_METHOD (PTR ) \
368
+ (((varattrib_4b *) (PTR))->va_compressed.va_tcinfo >> VARLENA_EXTSIZE_BITS)
369
+
370
+ /* Same, when working directly with a struct varatt_external */
371
+ #define VARATT_EXTERNAL_GET_EXTSIZE (toast_pointer ) \
372
+ ((toast_pointer).va_extinfo & VARLENA_EXTSIZE_MASK)
373
+ #define VARATT_EXTERNAL_GET_COMPRESS_METHOD (toast_pointer ) \
374
+ ((toast_pointer).va_extinfo >> VARLENA_EXTSIZE_BITS)
375
+
376
+ #define VARATT_EXTERNAL_SET_SIZE_AND_COMPRESS_METHOD (toast_pointer , len , cm ) \
377
+ do { \
378
+ Assert((cm) == TOAST_PGLZ_COMPRESSION_ID || \
379
+ (cm) == TOAST_LZ4_COMPRESSION_ID); \
380
+ ((toast_pointer).va_extinfo = \
381
+ (len) | ((uint32) (cm) << VARLENA_EXTSIZE_BITS)); \
382
+ } while (0)
383
+
384
+ /*
385
+ * Testing whether an externally-stored value is compressed now requires
386
+ * comparing size stored in va_extinfo (the actual length of the external data)
387
+ * to rawsize (the original uncompressed datum's size). The latter includes
388
+ * VARHDRSZ overhead, the former doesn't. We never use compression unless it
389
+ * actually saves space, so we expect either equality or less-than.
390
+ */
391
+ #define VARATT_EXTERNAL_IS_COMPRESSED (toast_pointer ) \
392
+ (VARATT_EXTERNAL_GET_EXTSIZE(toast_pointer) < \
393
+ (toast_pointer).va_rawsize - VARHDRSZ)
394
+
393
395
394
396
/* ----------------------------------------------------------------
395
397
* Section 2: Datum type + support macros
0 commit comments