mirror of
https://github.com/postgres/postgres.git
synced 2025-10-24 01:29:19 +03:00
Fix tuptoaster bugs induced by making bytea toastable. Durn thing was
trying to toast tuples inserted into toast tables! Fix is two-pronged: first, ensure all columns of a toast table are marked attstorage='p', and second, alter the target chunk size so that it's less than the threshold for trying to toast a tuple. (Code tried to do that but the expression was wrong.) A few cosmetic cleanups in tuptoaster too. NOTE: initdb forced due to change in toaster chunk-size.
This commit is contained in:
@@ -8,7 +8,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/access/heap/heapam.c,v 1.83 2000/08/03 19:18:54 tgl Exp $
|
* $Header: /cvsroot/pgsql/src/backend/access/heap/heapam.c,v 1.84 2000/08/04 04:16:06 tgl Exp $
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* INTERFACE ROUTINES
|
* INTERFACE ROUTINES
|
||||||
@@ -1362,7 +1362,7 @@ heap_insert(Relation relation, HeapTuple tup)
|
|||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
if (HeapTupleHasExtended(tup) ||
|
if (HeapTupleHasExtended(tup) ||
|
||||||
(MAXALIGN(tup->t_len) > (MaxTupleSize / 4)))
|
(MAXALIGN(tup->t_len) > TOAST_TUPLE_THRESHOLD))
|
||||||
heap_tuple_toast_attrs(relation, tup, NULL);
|
heap_tuple_toast_attrs(relation, tup, NULL);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -1621,13 +1621,15 @@ l2:
|
|||||||
#ifdef TUPLE_TOASTER_ACTIVE
|
#ifdef TUPLE_TOASTER_ACTIVE
|
||||||
/* ----------
|
/* ----------
|
||||||
* If this relation is enabled for toasting, let the toaster
|
* If this relation is enabled for toasting, let the toaster
|
||||||
* delete not any longer needed entries and create new ones to
|
* delete any no-longer-needed entries and create new ones to
|
||||||
* make the new tuple fit again.
|
* make the new tuple fit again. Also, if there are already-
|
||||||
|
* toasted values from some other relation, the toaster must
|
||||||
|
* fix them.
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
if (HeapTupleHasExtended(&oldtup) ||
|
if (HeapTupleHasExtended(&oldtup) ||
|
||||||
HeapTupleHasExtended(newtup) ||
|
HeapTupleHasExtended(newtup) ||
|
||||||
(MAXALIGN(newtup->t_len) > (MaxTupleSize / 4)))
|
(MAXALIGN(newtup->t_len) > TOAST_TUPLE_THRESHOLD))
|
||||||
heap_tuple_toast_attrs(relation, newtup, &oldtup);
|
heap_tuple_toast_attrs(relation, newtup, &oldtup);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/access/heap/tuptoaster.c,v 1.11 2000/08/03 16:33:40 tgl Exp $
|
* $Header: /cvsroot/pgsql/src/backend/access/heap/tuptoaster.c,v 1.12 2000/08/04 04:16:07 tgl Exp $
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* INTERFACE ROUTINES
|
* INTERFACE ROUTINES
|
||||||
@@ -22,11 +22,10 @@
|
|||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <stdio.h>
|
#include "postgres.h"
|
||||||
#include <stdlib.h>
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
#include "postgres.h"
|
|
||||||
|
|
||||||
#include "access/heapam.h"
|
#include "access/heapam.h"
|
||||||
#include "access/genam.h"
|
#include "access/genam.h"
|
||||||
@@ -39,6 +38,7 @@
|
|||||||
|
|
||||||
|
|
||||||
#ifdef TUPLE_TOASTER_ACTIVE
|
#ifdef TUPLE_TOASTER_ACTIVE
|
||||||
|
|
||||||
#undef TOAST_DEBUG
|
#undef TOAST_DEBUG
|
||||||
|
|
||||||
static void toast_delete(Relation rel, HeapTuple oldtup);
|
static void toast_delete(Relation rel, HeapTuple oldtup);
|
||||||
@@ -47,7 +47,6 @@ static void toast_insert_or_update(Relation rel, HeapTuple newtup,
|
|||||||
HeapTuple oldtup);
|
HeapTuple oldtup);
|
||||||
static Datum toast_compress_datum(Datum value);
|
static Datum toast_compress_datum(Datum value);
|
||||||
static Datum toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value);
|
static Datum toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value);
|
||||||
|
|
||||||
static varattrib *toast_fetch_datum(varattrib *attr);
|
static varattrib *toast_fetch_datum(varattrib *attr);
|
||||||
|
|
||||||
|
|
||||||
@@ -209,7 +208,7 @@ toast_delete(Relation rel, HeapTuple oldtup)
|
|||||||
/* ----------
|
/* ----------
|
||||||
* toast_insert_or_update -
|
* toast_insert_or_update -
|
||||||
*
|
*
|
||||||
* Delete no more used toast-entries and create new ones to
|
* Delete no-longer-used toast-entries and create new ones to
|
||||||
* make the new tuple fit on INSERT or UPDATE
|
* make the new tuple fit on INSERT or UPDATE
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
@@ -375,7 +374,7 @@ toast_insert_or_update(Relation rel, HeapTuple newtup, HeapTuple oldtup)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Compress and/or save external until data fits
|
* Compress and/or save external until data fits into target length
|
||||||
*
|
*
|
||||||
* 1: Inline compress attributes with attstorage 'x'
|
* 1: Inline compress attributes with attstorage 'x'
|
||||||
* 2: Store attributes with attstorage 'x' or 'e' external
|
* 2: Store attributes with attstorage 'x' or 'e' external
|
||||||
@@ -386,7 +385,7 @@ toast_insert_or_update(Relation rel, HeapTuple newtup, HeapTuple oldtup)
|
|||||||
maxDataLen = offsetof(HeapTupleHeaderData, t_bits);
|
maxDataLen = offsetof(HeapTupleHeaderData, t_bits);
|
||||||
if (has_nulls)
|
if (has_nulls)
|
||||||
maxDataLen += BITMAPLEN(numAttrs);
|
maxDataLen += BITMAPLEN(numAttrs);
|
||||||
maxDataLen = (MaxTupleSize / 4) - MAXALIGN(maxDataLen);
|
maxDataLen = TOAST_TUPLE_TARGET - MAXALIGN(maxDataLen);
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Look for attributes with attstorage 'x' to compress
|
* Look for attributes with attstorage 'x' to compress
|
||||||
@@ -560,7 +559,7 @@ toast_insert_or_update(Relation rel, HeapTuple newtup, HeapTuple oldtup)
|
|||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Search for the biggest yet inlined attribute with
|
* Search for the biggest yet inlined attribute with
|
||||||
* attstorage = 'x' or 'e'
|
* attstorage = 'm'
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
for (i = 0; i < numAttrs; i++)
|
for (i = 0; i < numAttrs; i++)
|
||||||
@@ -684,15 +683,13 @@ toast_insert_or_update(Relation rel, HeapTuple newtup, HeapTuple oldtup)
|
|||||||
if (toast_delold[i])
|
if (toast_delold[i])
|
||||||
toast_delete_datum(rel,
|
toast_delete_datum(rel,
|
||||||
heap_getattr(oldtup, i + 1, tupleDesc, &old_isnull));
|
heap_getattr(oldtup, i + 1, tupleDesc, &old_isnull));
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* toast_compress_datum -
|
* toast_compress_datum -
|
||||||
*
|
*
|
||||||
* Create a compressed version of a datum
|
* Create a compressed version of a varlena datum
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
static Datum
|
static Datum
|
||||||
@@ -726,9 +723,9 @@ toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value)
|
|||||||
InsertIndexResult idxres;
|
InsertIndexResult idxres;
|
||||||
TupleDesc toasttupDesc;
|
TupleDesc toasttupDesc;
|
||||||
Datum t_values[3];
|
Datum t_values[3];
|
||||||
char t_nulls[4];
|
char t_nulls[3];
|
||||||
varattrib *result;
|
varattrib *result;
|
||||||
char chunk_data[MaxTupleSize];
|
char chunk_data[VARHDRSZ + TOAST_MAX_CHUNK_SIZE];
|
||||||
int32 chunk_size;
|
int32 chunk_size;
|
||||||
int32 chunk_seq = 0;
|
int32 chunk_seq = 0;
|
||||||
char *data_p;
|
char *data_p;
|
||||||
@@ -769,7 +766,6 @@ toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value)
|
|||||||
t_nulls[0] = ' ';
|
t_nulls[0] = ' ';
|
||||||
t_nulls[1] = ' ';
|
t_nulls[1] = ' ';
|
||||||
t_nulls[2] = ' ';
|
t_nulls[2] = ' ';
|
||||||
t_nulls[3] = '\0';
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Get the data to process
|
* Get the data to process
|
||||||
@@ -783,16 +779,8 @@ toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value)
|
|||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
toastrel = heap_open(rel->rd_rel->reltoastrelid, RowExclusiveLock);
|
toastrel = heap_open(rel->rd_rel->reltoastrelid, RowExclusiveLock);
|
||||||
if (toastrel == NULL)
|
|
||||||
elog(ERROR, "Failed to open secondary relation of %s",
|
|
||||||
DatumGetCString(DirectFunctionCall1(nameout,
|
|
||||||
NameGetDatum(&(rel->rd_rel->relname)))));
|
|
||||||
toasttupDesc = toastrel->rd_att;
|
toasttupDesc = toastrel->rd_att;
|
||||||
toastidx = index_open(rel->rd_rel->reltoastidxid);
|
toastidx = index_open(rel->rd_rel->reltoastidxid);
|
||||||
if (toastidx == NULL)
|
|
||||||
elog(ERROR, "Failed to open index for secondary relation of %s",
|
|
||||||
DatumGetCString(DirectFunctionCall1(nameout,
|
|
||||||
NameGetDatum(&(rel->rd_rel->relname)))));
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Split up the item into chunks
|
* Split up the item into chunks
|
||||||
@@ -804,14 +792,13 @@ toast_save_datum(Relation rel, Oid mainoid, int16 attno, Datum value)
|
|||||||
* Calculate the size of this chunk
|
* Calculate the size of this chunk
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
chunk_size = (TOAST_MAX_CHUNK_SIZE < data_todo) ?
|
chunk_size = Min(TOAST_MAX_CHUNK_SIZE, data_todo);
|
||||||
TOAST_MAX_CHUNK_SIZE : data_todo;
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Build a tuple
|
* Build a tuple
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
t_values[1] = (Datum)(chunk_seq++);
|
t_values[1] = Int32GetDatum(chunk_seq++);
|
||||||
VARATT_SIZEP(chunk_data) = chunk_size + VARHDRSZ;
|
VARATT_SIZEP(chunk_data) = chunk_size + VARHDRSZ;
|
||||||
memcpy(VARATT_DATA(chunk_data), data_p, chunk_size);
|
memcpy(VARATT_DATA(chunk_data), data_p, chunk_size);
|
||||||
toasttup = heap_formtuple(toasttupDesc, t_values, t_nulls);
|
toasttup = heap_formtuple(toasttupDesc, t_values, t_nulls);
|
||||||
@@ -882,11 +869,7 @@ toast_delete_datum(Relation rel, Datum value)
|
|||||||
*/
|
*/
|
||||||
toastrel = heap_open(attr->va_content.va_external.va_toastrelid,
|
toastrel = heap_open(attr->va_content.va_external.va_toastrelid,
|
||||||
RowExclusiveLock);
|
RowExclusiveLock);
|
||||||
if (toastrel == NULL)
|
|
||||||
elog(ERROR, "Failed to open secondary relation at TOAST fetch");
|
|
||||||
toastidx = index_open(attr->va_content.va_external.va_toastidxid);
|
toastidx = index_open(attr->va_content.va_external.va_toastidxid);
|
||||||
if (toastidx == NULL)
|
|
||||||
elog(ERROR, "Failed to open index of secondary relation at TOAST fetch");
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Setup a scan key to fetch from the index by va_valueid
|
* Setup a scan key to fetch from the index by va_valueid
|
||||||
@@ -928,8 +911,6 @@ toast_delete_datum(Relation rel, Datum value)
|
|||||||
index_endscan(toastscan);
|
index_endscan(toastscan);
|
||||||
index_close(toastidx);
|
index_close(toastidx);
|
||||||
heap_close(toastrel, RowExclusiveLock);
|
heap_close(toastrel, RowExclusiveLock);
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -957,14 +938,15 @@ toast_fetch_datum(varattrib *attr)
|
|||||||
int32 ressize;
|
int32 ressize;
|
||||||
int32 residx;
|
int32 residx;
|
||||||
int numchunks;
|
int numchunks;
|
||||||
Datum chunk;
|
Pointer chunk;
|
||||||
bool isnull;
|
bool isnull;
|
||||||
|
int32 chunksize;
|
||||||
|
|
||||||
char *chunks_found;
|
char *chunks_found;
|
||||||
char *chunks_expected;
|
char *chunks_expected;
|
||||||
|
|
||||||
ressize = attr->va_content.va_external.va_extsize;
|
ressize = attr->va_content.va_external.va_extsize;
|
||||||
numchunks = (ressize / TOAST_MAX_CHUNK_SIZE) + 1;
|
numchunks = ((ressize - 1) / TOAST_MAX_CHUNK_SIZE) + 1;
|
||||||
|
|
||||||
chunks_found = palloc(numchunks);
|
chunks_found = palloc(numchunks);
|
||||||
chunks_expected = palloc(numchunks);
|
chunks_expected = palloc(numchunks);
|
||||||
@@ -982,12 +964,8 @@ toast_fetch_datum(varattrib *attr)
|
|||||||
*/
|
*/
|
||||||
toastrel = heap_open(attr->va_content.va_external.va_toastrelid,
|
toastrel = heap_open(attr->va_content.va_external.va_toastrelid,
|
||||||
AccessShareLock);
|
AccessShareLock);
|
||||||
if (toastrel == NULL)
|
|
||||||
elog(ERROR, "Failed to open secondary relation at TOAST fetch");
|
|
||||||
toasttupDesc = toastrel->rd_att;
|
toasttupDesc = toastrel->rd_att;
|
||||||
toastidx = index_open(attr->va_content.va_external.va_toastidxid);
|
toastidx = index_open(attr->va_content.va_external.va_toastidxid);
|
||||||
if (toastidx == NULL)
|
|
||||||
elog(ERROR, "Failed to open index of secondary relation at TOAST fetch");
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Setup a scan key to fetch from the index by va_valueid
|
* Setup a scan key to fetch from the index by va_valueid
|
||||||
@@ -1001,6 +979,8 @@ toast_fetch_datum(varattrib *attr)
|
|||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Read the chunks by index
|
* Read the chunks by index
|
||||||
|
*
|
||||||
|
* Note we will not necessarily see the chunks in sequence-number order.
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
toastscan = index_beginscan(toastidx, false, 1, &toastkey);
|
toastscan = index_beginscan(toastidx, false, 1, &toastkey);
|
||||||
@@ -1018,30 +998,46 @@ toast_fetch_datum(varattrib *attr)
|
|||||||
* Have a chunk, extract the sequence number and the data
|
* Have a chunk, extract the sequence number and the data
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
residx = (int32)heap_getattr(ttup, 2, toasttupDesc, &isnull);
|
residx = DatumGetInt32(heap_getattr(ttup, 2, toasttupDesc, &isnull));
|
||||||
chunk = heap_getattr(ttup, 3, toasttupDesc, &isnull);
|
Assert(!isnull);
|
||||||
|
chunk = DatumGetPointer(heap_getattr(ttup, 3, toasttupDesc, &isnull));
|
||||||
|
Assert(!isnull);
|
||||||
|
chunksize = VARATT_SIZE(chunk) - VARHDRSZ;
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Some checks on the data we've found
|
* Some checks on the data we've found
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
if (residx * TOAST_MAX_CHUNK_SIZE + VARATT_SIZE(chunk) - VARHDRSZ
|
if (residx < 0 || residx >= numchunks)
|
||||||
> ressize)
|
elog(ERROR, "unexpected chunk number %d for toast value %d",
|
||||||
elog(ERROR, "chunk data exceeds original data size for "
|
residx,
|
||||||
"toast value %d",
|
|
||||||
attr->va_content.va_external.va_valueid);
|
attr->va_content.va_external.va_valueid);
|
||||||
|
if (residx < numchunks-1)
|
||||||
|
{
|
||||||
|
if (chunksize != TOAST_MAX_CHUNK_SIZE)
|
||||||
|
elog(ERROR, "unexpected chunk size %d in chunk %d for toast value %d",
|
||||||
|
chunksize, residx,
|
||||||
|
attr->va_content.va_external.va_valueid);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if ((residx * TOAST_MAX_CHUNK_SIZE + chunksize) != ressize)
|
||||||
|
elog(ERROR, "unexpected chunk size %d in chunk %d for toast value %d",
|
||||||
|
chunksize, residx,
|
||||||
|
attr->va_content.va_external.va_valueid);
|
||||||
|
}
|
||||||
if (chunks_found[residx]++ > 0)
|
if (chunks_found[residx]++ > 0)
|
||||||
elog(ERROR, "chunk %d for toast value %d appears multiple times",
|
elog(ERROR, "chunk %d for toast value %d appears multiple times",
|
||||||
residx,
|
residx,
|
||||||
attr->va_content.va_external.va_valueid);
|
attr->va_content.va_external.va_valueid);
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* Copy the data into our result
|
* Copy the data into proper place in our result
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
memcpy(((char *)VARATT_DATA(result)) + residx * TOAST_MAX_CHUNK_SIZE,
|
memcpy(((char *)VARATT_DATA(result)) + residx * TOAST_MAX_CHUNK_SIZE,
|
||||||
VARATT_DATA(chunk),
|
VARATT_DATA(chunk),
|
||||||
VARATT_SIZE(chunk) - VARHDRSZ);
|
chunksize);
|
||||||
|
|
||||||
ReleaseBuffer(buffer);
|
ReleaseBuffer(buffer);
|
||||||
}
|
}
|
||||||
|
@@ -8,7 +8,7 @@
|
|||||||
*
|
*
|
||||||
*
|
*
|
||||||
* IDENTIFICATION
|
* IDENTIFICATION
|
||||||
* $Header: /cvsroot/pgsql/src/backend/commands/Attic/command.c,v 1.93 2000/08/03 19:19:18 tgl Exp $
|
* $Header: /cvsroot/pgsql/src/backend/commands/Attic/command.c,v 1.94 2000/08/04 04:16:06 tgl Exp $
|
||||||
*
|
*
|
||||||
* NOTES
|
* NOTES
|
||||||
* The PerformAddAttribute() code, like most of the relation
|
* The PerformAddAttribute() code, like most of the relation
|
||||||
@@ -1503,6 +1503,14 @@ AlterTableCreateToastTable(const char *relationName, bool silent)
|
|||||||
"chunk_data",
|
"chunk_data",
|
||||||
BYTEAOID,
|
BYTEAOID,
|
||||||
-1, 0, false);
|
-1, 0, false);
|
||||||
|
/*
|
||||||
|
* Ensure that the toast table doesn't itself get toasted,
|
||||||
|
* or we'll be toast :-(. This is essential for chunk_data because
|
||||||
|
* type bytea is toastable; hit the other two just to be sure.
|
||||||
|
*/
|
||||||
|
tupdesc->attrs[0]->attstorage = 'p';
|
||||||
|
tupdesc->attrs[1]->attstorage = 'p';
|
||||||
|
tupdesc->attrs[2]->attstorage = 'p';
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Note: the toast relation is considered a "normal" relation even if
|
* Note: the toast relation is considered a "normal" relation even if
|
||||||
|
@@ -6,7 +6,7 @@
|
|||||||
*
|
*
|
||||||
* Copyright (c) 2000, PostgreSQL Development Team
|
* Copyright (c) 2000, PostgreSQL Development Team
|
||||||
*
|
*
|
||||||
* $Id: tuptoaster.h,v 1.7 2000/07/22 11:18:47 wieck Exp $
|
* $Id: tuptoaster.h,v 1.8 2000/08/04 04:16:10 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@@ -20,25 +20,57 @@
|
|||||||
#include "access/tupmacs.h"
|
#include "access/tupmacs.h"
|
||||||
#include "utils/rel.h"
|
#include "utils/rel.h"
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This enables de-toasting of index entries. Needed until VACUUM is
|
||||||
|
* smart enough to rebuild indexes from scratch.
|
||||||
|
*/
|
||||||
#define TOAST_INDEX_HACK
|
#define TOAST_INDEX_HACK
|
||||||
|
|
||||||
|
|
||||||
#define TOAST_MAX_CHUNK_SIZE ((MaxTupleSize - \
|
/*
|
||||||
|
* These symbols control toaster activation. If a tuple is larger than
|
||||||
|
* TOAST_TUPLE_THRESHOLD, we will try to toast it down to no more than
|
||||||
|
* TOAST_TUPLE_TARGET bytes. Both numbers include all tuple header and
|
||||||
|
* alignment-padding overhead.
|
||||||
|
*
|
||||||
|
* The numbers need not be the same, though they currently are.
|
||||||
|
*/
|
||||||
|
#define TOAST_TUPLE_THRESHOLD (MaxTupleSize / 4)
|
||||||
|
|
||||||
|
#define TOAST_TUPLE_TARGET (MaxTupleSize / 4)
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* When we store an oversize datum externally, we divide it into chunks
|
||||||
|
* containing at most TOAST_MAX_CHUNK_SIZE data bytes. This number *must*
|
||||||
|
* be small enough that the completed toast-table tuple (including the
|
||||||
|
* ID and sequence fields and all overhead) is no more than MaxTupleSize
|
||||||
|
* bytes. It *should* be small enough to make toast-table tuples no more
|
||||||
|
* than TOAST_TUPLE_THRESHOLD bytes, else heapam.c will uselessly invoke
|
||||||
|
* the toaster on toast-table tuples.
|
||||||
|
*
|
||||||
|
* NB: you cannot change this value without forcing initdb, at least not
|
||||||
|
* if your DB contains any multi-chunk toasted values.
|
||||||
|
*/
|
||||||
|
#define TOAST_MAX_CHUNK_SIZE (TOAST_TUPLE_THRESHOLD - \
|
||||||
MAXALIGN( \
|
MAXALIGN( \
|
||||||
MAXALIGN(offsetof(HeapTupleHeaderData, t_bits)) + \
|
MAXALIGN(offsetof(HeapTupleHeaderData, t_bits)) + \
|
||||||
MAXALIGN(sizeof(Oid)) + \
|
sizeof(Oid) + \
|
||||||
MAXALIGN(sizeof(int32)) + \
|
sizeof(int32) + \
|
||||||
MAXALIGN(VARHDRSZ))) / 4)
|
VARHDRSZ))
|
||||||
|
|
||||||
|
|
||||||
/* ----------
|
/* ----------
|
||||||
* heap_tuple_toast_attrs() -
|
* heap_tuple_toast_attrs() -
|
||||||
*
|
*
|
||||||
* Called by heap_insert(), heap_update() and heap_delete().
|
* Called by heap_insert(), heap_update() and heap_delete().
|
||||||
* Outdates not any longer needed toast entries referenced
|
* Outdates any no-longer-needed toast entries referenced
|
||||||
* by oldtup and creates new ones until newtup is smaller
|
* by oldtup and creates new ones until newtup is no more than
|
||||||
* that ~2K (or running out of toastable values).
|
* TOAST_TUPLE_TARGET (or we run out of toastable values).
|
||||||
* Possibly modifies newtup by replacing the t_data part!
|
* Possibly modifies newtup by replacing the t_data part!
|
||||||
|
*
|
||||||
|
* oldtup is NULL if insert, newtup is NULL if delete.
|
||||||
* ----------
|
* ----------
|
||||||
*/
|
*/
|
||||||
extern void heap_tuple_toast_attrs(Relation rel,
|
extern void heap_tuple_toast_attrs(Relation rel,
|
||||||
|
@@ -37,7 +37,7 @@
|
|||||||
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
|
* Portions Copyright (c) 1996-2000, PostgreSQL, Inc
|
||||||
* Portions Copyright (c) 1994, Regents of the University of California
|
* Portions Copyright (c) 1994, Regents of the University of California
|
||||||
*
|
*
|
||||||
* $Id: catversion.h,v 1.38 2000/07/30 22:13:59 tgl Exp $
|
* $Id: catversion.h,v 1.39 2000/08/04 04:16:17 tgl Exp $
|
||||||
*
|
*
|
||||||
*-------------------------------------------------------------------------
|
*-------------------------------------------------------------------------
|
||||||
*/
|
*/
|
||||||
@@ -53,6 +53,6 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/* yyyymmddN */
|
/* yyyymmddN */
|
||||||
#define CATALOG_VERSION_NO 200007301
|
#define CATALOG_VERSION_NO 200008031
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Reference in New Issue
Block a user