Commit fb00d066 authored by marko's avatar marko

branches/zip: In page_zip_compress(), do not touch page_zip unless the

operation succeeds.

page_zip_dir_set(): Remove, as the slots will be written to a temporary
buffer.

page_zip_dir_encode(): Replace page_zip with buf.  Make some assertions
more accurate.  Write the dense page directory to buf.

page_zip_compress(): Allocate a temporary buffer for the compressed page
except the page header.  Make some assertions more accurate.

page_zip_compress(), page_zip_decompress(): Remove 2 bytes of empty space
adjacent to the area reserved for storing uncompressed columns.
parent 20acdb79
...@@ -184,17 +184,6 @@ page_zip_dir_get( ...@@ -184,17 +184,6 @@ page_zip_dir_get(
ulint slot) /* in: slot ulint slot) /* in: slot
(0=first user record) */ (0=first user record) */
__attribute__((pure)); __attribute__((pure));
/*****************************************************************
Write a given slot in the dense page directory. */
UNIV_INLINE
void
page_zip_dir_set(
/*=============*/
page_zip_des_t* page_zip, /* in: compressed page */
ulint slot, /* in: slot (0=first user record) */
ulint offs); /* in: offset, possibly ORed with
PAGE_ZIP_DIR_SLOT_DEL or
PAGE_ZIP_DIR_SLOT_OWNED */
/************************************************************************** /**************************************************************************
Ensure that enough space is available in the modification log. Ensure that enough space is available in the modification log.
......
...@@ -255,23 +255,6 @@ page_zip_dir_get( ...@@ -255,23 +255,6 @@ page_zip_dir_get(
return(mach_read_from_2(page_zip->data + page_zip->size return(mach_read_from_2(page_zip->data + page_zip->size
- PAGE_ZIP_DIR_SLOT_SIZE * (slot + 1))); - PAGE_ZIP_DIR_SLOT_SIZE * (slot + 1)));
} }
/*****************************************************************
Write a given slot in the dense page directory. */
UNIV_INLINE
void
page_zip_dir_set(
/*=============*/
page_zip_des_t* page_zip, /* in: compressed page */
ulint slot, /* in: slot (0=first user record) */
ulint offs) /* in: offset, possibly ORed with
PAGE_ZIP_DIR_SLOT_DEL or
PAGE_ZIP_DIR_SLOT_OWNED */
{
ut_ad(page_zip_simple_validate(page_zip));
mach_write_to_2(page_zip->data + page_zip->size
- PAGE_ZIP_DIR_SLOT_SIZE * (slot + 1),
offs);
}
/************************************************************************** /**************************************************************************
Ensure that enough space is available in the modification log. Ensure that enough space is available in the modification log.
......
...@@ -190,8 +190,9 @@ void ...@@ -190,8 +190,9 @@ void
page_zip_dir_encode( page_zip_dir_encode(
/*================*/ /*================*/
const page_t* page, /* in: compact page */ const page_t* page, /* in: compact page */
page_zip_des_t* page_zip,/* out: dense directory on compressed page */ byte* buf, /* in: pointer to dense page directory[-1];
const rec_t** recs) /* in: pointer to array of 0, or NULL; out: dense directory on compressed page */
const rec_t** recs) /* in: pointer to an array of 0, or NULL;
out: dense page directory sorted by ascending out: dense page directory sorted by ascending
address (and heap_no) */ address (and heap_no) */
{ {
...@@ -232,11 +233,13 @@ page_zip_dir_encode( ...@@ -232,11 +233,13 @@ page_zip_dir_encode(
} }
rec = (page_t*) page + offs; rec = (page_t*) page + offs;
heap_no = rec_get_heap_no_new(rec); heap_no = rec_get_heap_no_new(rec);
ut_a(heap_no > 0); ut_a(heap_no >= 2); /* not infimum or supremum */
ut_a(heap_no < n_heap); ut_a(heap_no < n_heap);
ut_a(!(offs & ~PAGE_ZIP_DIR_SLOT_MASK)); ut_a(offs < UNIV_PAGE_SIZE - PAGE_DIR);
ut_a(offs); ut_a(offs >= PAGE_ZIP_START);
#if PAGE_ZIP_DIR_SLOT_MASK & UNIV_PAGE_SIZE
# error "PAGE_ZIP_DIR_SLOT_MASK & UNIV_PAGE_SIZE"
#endif
if (UNIV_UNLIKELY(rec_get_n_owned_new(rec))) { if (UNIV_UNLIKELY(rec_get_n_owned_new(rec))) {
offs |= PAGE_ZIP_DIR_SLOT_OWNED; offs |= PAGE_ZIP_DIR_SLOT_OWNED;
} }
...@@ -251,7 +254,7 @@ page_zip_dir_encode( ...@@ -251,7 +254,7 @@ page_zip_dir_encode(
REC_INFO_MIN_REC_FLAG set. */ REC_INFO_MIN_REC_FLAG set. */
min_mark = 0; min_mark = 0;
page_zip_dir_set(page_zip, i++, offs); mach_write_to_2(buf - PAGE_ZIP_DIR_SLOT_SIZE * ++i, offs);
if (UNIV_LIKELY_NULL(recs)) { if (UNIV_LIKELY_NULL(recs)) {
/* Ensure that each heap_no occurs at most once. */ /* Ensure that each heap_no occurs at most once. */
...@@ -271,13 +274,13 @@ page_zip_dir_encode( ...@@ -271,13 +274,13 @@ page_zip_dir_encode(
rec = (page_t*) page + offs; rec = (page_t*) page + offs;
heap_no = rec_get_heap_no_new(rec); heap_no = rec_get_heap_no_new(rec);
ut_a(heap_no >= 2); /* only user records can be deleted */ ut_a(heap_no >= 2); /* not infimum or supremum */
ut_a(heap_no < n_heap); ut_a(heap_no < n_heap);
ut_a(!rec[-REC_N_NEW_EXTRA_BYTES]); /* info_bits and n_owned */ ut_a(!rec[-REC_N_NEW_EXTRA_BYTES]); /* info_bits and n_owned */
ut_a(rec_get_status(rec) == status); ut_a(rec_get_status(rec) == status);
page_zip_dir_set(page_zip, i++, offs); mach_write_to_2(buf - PAGE_ZIP_DIR_SLOT_SIZE * ++i, offs);
if (UNIV_LIKELY_NULL(recs)) { if (UNIV_LIKELY_NULL(recs)) {
/* Ensure that each heap_no occurs at most once. */ /* Ensure that each heap_no occurs at most once. */
...@@ -313,6 +316,7 @@ page_zip_compress( ...@@ -313,6 +316,7 @@ page_zip_compress(
ulint n_fields;/* number of index fields needed */ ulint n_fields;/* number of index fields needed */
byte* fields; /* index field information */ byte* fields; /* index field information */
byte* buf; /* compressed payload of the page */ byte* buf; /* compressed payload of the page */
byte* buf_end;/* end of buf */
ulint n_dense; ulint n_dense;
const rec_t** recs; /* dense page directory, sorted by address */ const rec_t** recs; /* dense page directory, sorted by address */
mem_heap_t* heap; mem_heap_t* heap;
...@@ -363,8 +367,8 @@ page_zip_compress( ...@@ -363,8 +367,8 @@ page_zip_compress(
fields = mem_heap_alloc(heap, (n_fields + 1) * 2); fields = mem_heap_alloc(heap, (n_fields + 1) * 2);
buf = mem_heap_alloc(heap, page_zip->size buf = mem_heap_alloc(heap, page_zip->size - PAGE_DATA);
- PAGE_DATA - PAGE_ZIP_DIR_SLOT_SIZE * n_dense); buf_end = buf + page_zip->size - PAGE_DATA;
/* Compress the data payload. */ /* Compress the data payload. */
c_stream.zalloc = (alloc_func) 0; c_stream.zalloc = (alloc_func) 0;
...@@ -377,8 +381,7 @@ page_zip_compress( ...@@ -377,8 +381,7 @@ page_zip_compress(
c_stream.next_out = buf; c_stream.next_out = buf;
/* Subtract the space reserved for uncompressed data. */ /* Subtract the space reserved for uncompressed data. */
/* Page header, end marker of modification log */ /* Page header, end marker of modification log */
c_stream.avail_out = page_zip->size c_stream.avail_out = buf_end - buf - PAGE_ZIP_DIR_SLOT_SIZE;
- (PAGE_DATA + PAGE_ZIP_DIR_SLOT_SIZE);
/* Dense page directory and uncompressed columns, if any */ /* Dense page directory and uncompressed columns, if any */
if (page_is_leaf(page)) { if (page_is_leaf(page)) {
trx_id_col = dict_index_get_sys_col_pos(index, DATA_TRX_ID); trx_id_col = dict_index_get_sys_col_pos(index, DATA_TRX_ID);
...@@ -412,15 +415,11 @@ page_zip_compress( ...@@ -412,15 +415,11 @@ page_zip_compress(
ut_ad(!c_stream.avail_in); ut_ad(!c_stream.avail_in);
/* TODO: do not write to page_zip->data until deflateEnd() */ page_zip_dir_encode(page, buf_end, recs);
page_zip_dir_encode(page, page_zip, recs);
c_stream.next_in = (byte*) page + PAGE_ZIP_START; c_stream.next_in = (byte*) page + PAGE_ZIP_START;
/* TODO: do not write to page_zip->data until deflateEnd() */ storage = buf_end - n_dense * PAGE_ZIP_DIR_SLOT_SIZE;
storage = page_zip->data + page_zip->size
- (n_dense + 1)
* PAGE_ZIP_DIR_SLOT_SIZE;
if (page_is_leaf(page)) { if (page_is_leaf(page)) {
/* BTR_EXTERN_FIELD_REF storage */ /* BTR_EXTERN_FIELD_REF storage */
...@@ -527,9 +526,9 @@ page_zip_compress( ...@@ -527,9 +526,9 @@ page_zip_compress(
-= BTR_EXTERN_FIELD_REF_SIZE; -= BTR_EXTERN_FIELD_REF_SIZE;
externs -= BTR_EXTERN_FIELD_REF_SIZE; externs -= BTR_EXTERN_FIELD_REF_SIZE;
ut_ad(externs - page_zip->data ut_ad(externs == c_stream.next_out
> c_stream.next_out + c_stream.avail_out
+ c_stream.avail_out - buf); + PAGE_ZIP_DIR_SLOT_SIZE/* mod log */);
/* Copy the BLOB pointer */ /* Copy the BLOB pointer */
memcpy(externs, c_stream.next_in, memcpy(externs, c_stream.next_in,
...@@ -619,16 +618,18 @@ zlib_error: ...@@ -619,16 +618,18 @@ zlib_error:
ut_a(err == Z_OK); ut_a(err == Z_OK);
ut_ad(buf + c_stream.total_out == c_stream.next_out); ut_ad(buf + c_stream.total_out == c_stream.next_out);
ut_ad((ulint) (storage - c_stream.next_out) >= c_stream.avail_out);
/* Zero out the area reserved for the modification log */
memset(c_stream.next_out, 0, c_stream.avail_out
+ PAGE_ZIP_DIR_SLOT_SIZE);
page_zip->m_end = page_zip->m_start = PAGE_DATA + c_stream.total_out; page_zip->m_end = page_zip->m_start = PAGE_DATA + c_stream.total_out;
page_zip->n_blobs = n_blobs; page_zip->n_blobs = n_blobs;
/* Copy the page header */ /* Copy the page header */
memcpy(page_zip->data, page, PAGE_DATA); memcpy(page_zip->data, page, PAGE_DATA);
/* Copy the compressed data */ /* Copy the rest of the compressed page */
memcpy(page_zip->data + PAGE_DATA, buf, c_stream.total_out); memcpy(page_zip->data + PAGE_DATA, buf, page_zip->size - PAGE_DATA);
/* Zero out the area reserved for the modification log */
memset(page_zip->data + page_zip->m_start, 0,
c_stream.avail_out + PAGE_ZIP_DIR_SLOT_SIZE);
mem_heap_free(heap); mem_heap_free(heap);
#if defined UNIV_DEBUG || defined UNIV_ZIP_DEBUG #if defined UNIV_DEBUG || defined UNIV_ZIP_DEBUG
ut_a(page_zip_validate(page_zip, page)); ut_a(page_zip_validate(page_zip, page));
...@@ -1536,7 +1537,7 @@ err_exit: ...@@ -1536,7 +1537,7 @@ err_exit:
/* Copy the uncompressed fields. */ /* Copy the uncompressed fields. */
storage = page_zip->data + page_zip->size storage = page_zip->data + page_zip->size
- (n_dense + 1) * PAGE_ZIP_DIR_SLOT_SIZE; - n_dense * PAGE_ZIP_DIR_SLOT_SIZE;
externs = storage - n_dense * (DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN); externs = storage - n_dense * (DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN);
page_zip->n_blobs = 0; page_zip->n_blobs = 0;
recsc = recs; recsc = recs;
...@@ -2053,7 +2054,9 @@ page_zip_dir_rewrite( ...@@ -2053,7 +2054,9 @@ page_zip_dir_rewrite(
page_zip_des_t* page_zip,/* out: dense directory on compressed page */ page_zip_des_t* page_zip,/* out: dense directory on compressed page */
const page_t* page) /* in: uncompressed page */ const page_t* page) /* in: uncompressed page */
{ {
page_zip_dir_encode(page, page_zip, NULL); ut_ad(page_zip_simple_validate(page_zip));
page_zip_dir_encode(page, page_zip->data + page_zip->size, NULL);
} }
/************************************************************************** /**************************************************************************
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment