1
/*****************************************************************************
3
Copyright (c) 1994, 2009, Innobase Oy. All Rights Reserved.
5
This program is free software; you can redistribute it and/or modify it under
6
the terms of the GNU General Public License as published by the Free Software
7
Foundation; version 2 of the License.
9
This program is distributed in the hope that it will be useful, but WITHOUT
10
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
11
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
13
You should have received a copy of the GNU General Public License along with
14
this program; if not, write to the Free Software Foundation, Inc., 59 Temple
15
Place, Suite 330, Boston, MA 02111-1307 USA
17
*****************************************************************************/
19
/**************************************************//**
20
@file page/page0page.c
23
Created 2/2/1994 Heikki Tuuri
24
*******************************************************/
26
#include "page0page.h"
28
#include "page0page.ic"
35
#ifndef UNIV_HOTBACKUP
37
# include "lock0lock.h"
40
#endif /* !UNIV_HOTBACKUP */
45
The index page consists of a page header which contains the page's
46
id and other information. On top of it are the the index records
47
in a heap linked into a one way linear list according to alphabetic order.
49
Just below page end is an array of pointers which we call page directory,
50
to about every sixth record in the list. The pointers are placed in
51
the directory in the alphabetical order of the records pointed to,
52
enabling us to make binary search using the array. Each slot n:o I
53
in the directory points to a record, where a 4-bit field contains a count
54
of those records which are in the linear list between pointer I and
55
the pointer I - 1 in the directory, including the record
56
pointed to by pointer I and not including the record pointed to by I - 1.
57
We say that the record pointed to by slot I, or that slot I, owns
58
these records. The count is always kept in the range 4 to 8, with
59
the exception that it is 1 for the first slot, and 1--8 for the second slot.
61
An essentially binary search can be performed in the list of index
62
records, like we could do if we had pointer to every record in the
63
page directory. The data structure is, however, more efficient when
64
we are doing inserts, because most inserts are just pushed on a heap.
65
Only every 8th insert requires block move in the directory pointer
66
table, which itself is quite small. A record is deleted from the page
67
by just taking it off the linear list and updating the number of owned
68
records-field of the record which owns it, and updating the page directory,
69
if necessary. A special case is the one when the record owns itself.
70
Because the overhead of inserts is so small, we may also increase the
71
page size from the projected default of 8 kB to 64 kB without too
72
much loss of efficiency in inserts. Bigger page becomes actual
73
when the disk transfer rate compared to seek and latency time rises.
74
On the present system, the page size is set so that the page transfer
75
time (3 ms) is 20 % of the disk random access time (15 ms).
77
When the page is split, merged, or becomes full but contains deleted
78
records, we have to reorganize the page.
80
Assuming a page size of 8 kB, a typical index page of a secondary
81
index contains 300 index entries, and the size of the page directory
82
is 50 x 4 bytes = 200 bytes. */
84
/***************************************************************//**
85
Looks for the directory slot which owns the given record.
86
@return the directory slot number */
89
page_dir_find_owner_slot(
90
/*=====================*/
91
const rec_t* rec) /*!< in: the physical record */
94
register uint16 rec_offs_bytes;
95
register const page_dir_slot_t* slot;
96
register const page_dir_slot_t* first_slot;
97
register const rec_t* r = rec;
99
ut_ad(page_rec_check(rec));
101
page = page_align(rec);
102
first_slot = page_dir_get_nth_slot(page, 0);
103
slot = page_dir_get_nth_slot(page, page_dir_get_n_slots(page) - 1);
105
if (page_is_comp(page)) {
106
while (rec_get_n_owned_new(r) == 0) {
107
r = rec_get_next_ptr_const(r, TRUE);
108
ut_ad(r >= page + PAGE_NEW_SUPREMUM);
109
ut_ad(r < page + (UNIV_PAGE_SIZE - PAGE_DIR));
112
while (rec_get_n_owned_old(r) == 0) {
113
r = rec_get_next_ptr_const(r, FALSE);
114
ut_ad(r >= page + PAGE_OLD_SUPREMUM);
115
ut_ad(r < page + (UNIV_PAGE_SIZE - PAGE_DIR));
119
rec_offs_bytes = mach_encode_2(r - page);
121
while (UNIV_LIKELY(*(uint16*) slot != rec_offs_bytes)) {
123
if (UNIV_UNLIKELY(slot == first_slot)) {
125
"InnoDB: Probable data corruption on"
127
"InnoDB: Original record ",
128
(ulong) page_get_page_no(page));
130
if (page_is_comp(page)) {
131
fputs("(compact record)", stderr);
133
rec_print_old(stderr, rec);
137
"InnoDB: on that page.\n"
138
"InnoDB: Cannot find the dir slot for record ",
140
if (page_is_comp(page)) {
141
fputs("(compact record)", stderr);
143
rec_print_old(stderr, page
144
+ mach_decode_2(rec_offs_bytes));
147
"InnoDB: on that page!\n", stderr);
149
buf_page_print(page, 0);
154
slot += PAGE_DIR_SLOT_SIZE;
157
return(((ulint) (first_slot - slot)) / PAGE_DIR_SLOT_SIZE);
160
/**************************************************************//**
161
Used to check the consistency of a directory slot.
162
@return TRUE if succeed */
167
page_dir_slot_t* slot) /*!< in: slot */
175
page = page_align(slot);
177
n_slots = page_dir_get_n_slots(page);
179
ut_a(slot <= page_dir_get_nth_slot(page, 0));
180
ut_a(slot >= page_dir_get_nth_slot(page, n_slots - 1));
182
ut_a(page_rec_check(page_dir_slot_get_rec(slot)));
184
if (page_is_comp(page)) {
185
n_owned = rec_get_n_owned_new(page_dir_slot_get_rec(slot));
187
n_owned = rec_get_n_owned_old(page_dir_slot_get_rec(slot));
190
if (slot == page_dir_get_nth_slot(page, 0)) {
192
} else if (slot == page_dir_get_nth_slot(page, n_slots - 1)) {
194
ut_a(n_owned <= PAGE_DIR_SLOT_MAX_N_OWNED);
196
ut_a(n_owned >= PAGE_DIR_SLOT_MIN_N_OWNED);
197
ut_a(n_owned <= PAGE_DIR_SLOT_MAX_N_OWNED);
203
/*************************************************************//**
204
Sets the max trx id field value. */
209
buf_block_t* block, /*!< in/out: page */
210
page_zip_des_t* page_zip,/*!< in/out: compressed page, or NULL */
211
trx_id_t trx_id, /*!< in: transaction id */
212
mtr_t* mtr) /*!< in/out: mini-transaction, or NULL */
214
page_t* page = buf_block_get_frame(block);
215
#ifndef UNIV_HOTBACKUP
216
const ibool is_hashed = block->is_hashed;
219
rw_lock_x_lock(&btr_search_latch);
222
ut_ad(!mtr || mtr_memo_contains(mtr, block, MTR_MEMO_PAGE_X_FIX));
223
#endif /* !UNIV_HOTBACKUP */
225
/* It is not necessary to write this change to the redo log, as
226
during a database recovery we assume that the max trx id of every
227
page is the maximum trx id assigned before the crash. */
229
if (UNIV_LIKELY_NULL(page_zip)) {
230
mach_write_to_8(page + (PAGE_HEADER + PAGE_MAX_TRX_ID), trx_id);
231
page_zip_write_header(page_zip,
232
page + (PAGE_HEADER + PAGE_MAX_TRX_ID),
234
#ifndef UNIV_HOTBACKUP
236
mlog_write_dulint(page + (PAGE_HEADER + PAGE_MAX_TRX_ID),
238
#endif /* !UNIV_HOTBACKUP */
240
mach_write_to_8(page + (PAGE_HEADER + PAGE_MAX_TRX_ID), trx_id);
243
#ifndef UNIV_HOTBACKUP
245
rw_lock_x_unlock(&btr_search_latch);
247
#endif /* !UNIV_HOTBACKUP */
250
/************************************************************//**
251
Allocates a block of memory from the heap of an index page.
252
@return pointer to start of allocated buffer, or NULL if allocation fails */
257
page_t* page, /*!< in/out: index page */
258
page_zip_des_t* page_zip,/*!< in/out: compressed page with enough
259
space available for inserting the record,
261
ulint need, /*!< in: total number of bytes needed */
262
ulint* heap_no)/*!< out: this contains the heap number
263
of the allocated record
264
if allocation succeeds */
269
ut_ad(page && heap_no);
271
avl_space = page_get_max_insert_size(page, 1);
273
if (avl_space >= need) {
274
block = page_header_get_ptr(page, PAGE_HEAP_TOP);
276
page_header_set_ptr(page, page_zip, PAGE_HEAP_TOP,
278
*heap_no = page_dir_get_n_heap(page);
280
page_dir_set_n_heap(page, page_zip, 1 + *heap_no);
288
#ifndef UNIV_HOTBACKUP
289
/**********************************************************//**
290
Writes a log record of page creation. */
293
page_create_write_log(
294
/*==================*/
295
buf_frame_t* frame, /*!< in: a buffer frame where the page is
297
mtr_t* mtr, /*!< in: mini-transaction handle */
298
ibool comp) /*!< in: TRUE=compact page format */
300
mlog_write_initial_log_record(frame, comp
301
? MLOG_COMP_PAGE_CREATE
302
: MLOG_PAGE_CREATE, mtr);
304
#else /* !UNIV_HOTBACKUP */
305
# define page_create_write_log(frame,mtr,comp) ((void) 0)
306
#endif /* !UNIV_HOTBACKUP */
308
/***********************************************************//**
309
Parses a redo log record of creating a page.
310
@return end of log record or NULL */
315
byte* ptr, /*!< in: buffer */
316
byte* end_ptr __attribute__((unused)), /*!< in: buffer end */
317
ulint comp, /*!< in: nonzero=compact page format */
318
buf_block_t* block, /*!< in: block or NULL */
319
mtr_t* mtr) /*!< in: mtr or NULL */
321
ut_ad(ptr && end_ptr);
323
/* The record is empty, except for the record initial part */
326
page_create(block, mtr, comp);
332
/**********************************************************//**
333
The index page creation function.
334
@return pointer to the page */
339
buf_block_t* block, /*!< in: a buffer block where the
341
ulint comp) /*!< in: nonzero=compact page format */
343
page_dir_slot_t* slot;
355
#if PAGE_BTR_IBUF_FREE_LIST + FLST_BASE_NODE_SIZE > PAGE_DATA
356
# error "PAGE_BTR_IBUF_FREE_LIST + FLST_BASE_NODE_SIZE > PAGE_DATA"
358
#if PAGE_BTR_IBUF_FREE_LIST_NODE + FLST_NODE_SIZE > PAGE_DATA
359
# error "PAGE_BTR_IBUF_FREE_LIST_NODE + FLST_NODE_SIZE > PAGE_DATA"
362
/* The infimum and supremum records use a dummy index. */
363
if (UNIV_LIKELY(comp)) {
364
index = dict_ind_compact;
366
index = dict_ind_redundant;
369
/* 1. INCREMENT MODIFY CLOCK */
370
buf_block_modify_clock_inc(block);
372
page = buf_block_get_frame(block);
374
fil_page_set_type(page, FIL_PAGE_INDEX);
376
heap = mem_heap_create(200);
378
/* 3. CREATE THE INFIMUM AND SUPREMUM RECORDS */
380
/* Create first a data tuple for infimum record */
381
tuple = dtuple_create(heap, 1);
382
dtuple_set_info_bits(tuple, REC_STATUS_INFIMUM);
383
field = dtuple_get_nth_field(tuple, 0);
385
dfield_set_data(field, "infimum", 8);
386
dtype_set(dfield_get_type(field),
387
DATA_VARCHAR, DATA_ENGLISH | DATA_NOT_NULL, 8);
388
/* Set the corresponding physical record to its place in the page
391
heap_top = page + PAGE_DATA;
393
infimum_rec = rec_convert_dtuple_to_rec(heap_top, index, tuple, 0);
395
if (UNIV_LIKELY(comp)) {
396
ut_a(infimum_rec == page + PAGE_NEW_INFIMUM);
398
rec_set_n_owned_new(infimum_rec, NULL, 1);
399
rec_set_heap_no_new(infimum_rec, 0);
401
ut_a(infimum_rec == page + PAGE_OLD_INFIMUM);
403
rec_set_n_owned_old(infimum_rec, 1);
404
rec_set_heap_no_old(infimum_rec, 0);
407
offsets = rec_get_offsets(infimum_rec, index, NULL,
408
ULINT_UNDEFINED, &heap);
410
heap_top = rec_get_end(infimum_rec, offsets);
412
/* Create then a tuple for supremum */
414
tuple = dtuple_create(heap, 1);
415
dtuple_set_info_bits(tuple, REC_STATUS_SUPREMUM);
416
field = dtuple_get_nth_field(tuple, 0);
418
dfield_set_data(field, "supremum", comp ? 8 : 9);
419
dtype_set(dfield_get_type(field),
420
DATA_VARCHAR, DATA_ENGLISH | DATA_NOT_NULL, comp ? 8 : 9);
422
supremum_rec = rec_convert_dtuple_to_rec(heap_top, index, tuple, 0);
424
if (UNIV_LIKELY(comp)) {
425
ut_a(supremum_rec == page + PAGE_NEW_SUPREMUM);
427
rec_set_n_owned_new(supremum_rec, NULL, 1);
428
rec_set_heap_no_new(supremum_rec, 1);
430
ut_a(supremum_rec == page + PAGE_OLD_SUPREMUM);
432
rec_set_n_owned_old(supremum_rec, 1);
433
rec_set_heap_no_old(supremum_rec, 1);
436
offsets = rec_get_offsets(supremum_rec, index, offsets,
437
ULINT_UNDEFINED, &heap);
438
heap_top = rec_get_end(supremum_rec, offsets);
440
ut_ad(heap_top == page
441
+ (comp ? PAGE_NEW_SUPREMUM_END : PAGE_OLD_SUPREMUM_END));
445
/* 4. INITIALIZE THE PAGE */
447
page_header_set_field(page, NULL, PAGE_N_DIR_SLOTS, 2);
448
page_header_set_ptr(page, NULL, PAGE_HEAP_TOP, heap_top);
449
page_header_set_field(page, NULL, PAGE_N_HEAP, comp
450
? 0x8000 | PAGE_HEAP_NO_USER_LOW
451
: PAGE_HEAP_NO_USER_LOW);
452
page_header_set_ptr(page, NULL, PAGE_FREE, NULL);
453
page_header_set_field(page, NULL, PAGE_GARBAGE, 0);
454
page_header_set_ptr(page, NULL, PAGE_LAST_INSERT, NULL);
455
page_header_set_field(page, NULL, PAGE_DIRECTION, PAGE_NO_DIRECTION);
456
page_header_set_field(page, NULL, PAGE_N_DIRECTION, 0);
457
page_header_set_field(page, NULL, PAGE_N_RECS, 0);
458
page_set_max_trx_id(block, NULL, ut_dulint_zero, NULL);
459
memset(heap_top, 0, UNIV_PAGE_SIZE - PAGE_EMPTY_DIR_START
460
- page_offset(heap_top));
462
/* 5. SET POINTERS IN RECORDS AND DIR SLOTS */
464
/* Set the slots to point to infimum and supremum. */
466
slot = page_dir_get_nth_slot(page, 0);
467
page_dir_slot_set_rec(slot, infimum_rec);
469
slot = page_dir_get_nth_slot(page, 1);
470
page_dir_slot_set_rec(slot, supremum_rec);
472
/* Set the next pointers in infimum and supremum */
474
if (UNIV_LIKELY(comp)) {
475
rec_set_next_offs_new(infimum_rec, PAGE_NEW_SUPREMUM);
476
rec_set_next_offs_new(supremum_rec, 0);
478
rec_set_next_offs_old(infimum_rec, PAGE_OLD_SUPREMUM);
479
rec_set_next_offs_old(supremum_rec, 0);
485
/**********************************************************//**
486
Create an uncompressed B-tree index page.
487
@return pointer to the page */
492
buf_block_t* block, /*!< in: a buffer block where the
494
mtr_t* mtr, /*!< in: mini-transaction handle */
495
ulint comp) /*!< in: nonzero=compact page format */
497
page_create_write_log(buf_block_get_frame(block), mtr, comp);
498
return(page_create_low(block, comp));
501
/**********************************************************//**
502
Create a compressed B-tree index page.
503
@return pointer to the page */
508
buf_block_t* block, /*!< in/out: a buffer frame where the
510
dict_index_t* index, /*!< in: the index of the page */
511
ulint level, /*!< in: the B-tree level of the page */
512
mtr_t* mtr) /*!< in: mini-transaction handle */
515
page_zip_des_t* page_zip = buf_block_get_page_zip(block);
520
ut_ad(dict_table_is_comp(index->table));
522
page = page_create_low(block, TRUE);
523
mach_write_to_2(page + PAGE_HEADER + PAGE_LEVEL, level);
525
if (UNIV_UNLIKELY(!page_zip_compress(page_zip, page, index, mtr))) {
526
/* The compression of a newly created page
527
should always succeed. */
534
/*************************************************************//**
535
Differs from page_copy_rec_list_end, because this function does not
536
touch the lock table and max trx id on page or compress the page. */
539
page_copy_rec_list_end_no_locks(
540
/*============================*/
541
buf_block_t* new_block, /*!< in: index page to copy to */
542
buf_block_t* block, /*!< in: index page of rec */
543
rec_t* rec, /*!< in: record on page */
544
dict_index_t* index, /*!< in: record descriptor */
545
mtr_t* mtr) /*!< in: mtr */
547
page_t* new_page = buf_block_get_frame(new_block);
550
mem_heap_t* heap = NULL;
551
ulint offsets_[REC_OFFS_NORMAL_SIZE];
552
ulint* offsets = offsets_;
553
rec_offs_init(offsets_);
555
page_cur_position(rec, block, &cur1);
557
if (page_cur_is_before_first(&cur1)) {
559
page_cur_move_to_next(&cur1);
562
ut_a((ibool)!!page_is_comp(new_page)
563
== dict_table_is_comp(index->table));
564
ut_a(page_is_comp(new_page) == page_rec_is_comp(rec));
565
ut_a(mach_read_from_2(new_page + UNIV_PAGE_SIZE - 10) == (ulint)
566
(page_is_comp(new_page) ? PAGE_NEW_INFIMUM : PAGE_OLD_INFIMUM));
568
cur2 = page_get_infimum_rec(buf_block_get_frame(new_block));
570
/* Copy records from the original page to the new page */
572
while (!page_cur_is_after_last(&cur1)) {
573
rec_t* cur1_rec = page_cur_get_rec(&cur1);
575
offsets = rec_get_offsets(cur1_rec, index, offsets,
576
ULINT_UNDEFINED, &heap);
577
ins_rec = page_cur_insert_rec_low(cur2, index,
578
cur1_rec, offsets, mtr);
579
if (UNIV_UNLIKELY(!ins_rec)) {
580
/* Track an assertion failure reported on the mailing
581
list on June 18th, 2003 */
583
buf_page_print(new_page, 0);
584
buf_page_print(page_align(rec), 0);
585
ut_print_timestamp(stderr);
588
"InnoDB: rec offset %lu, cur1 offset %lu,"
589
" cur2 offset %lu\n",
590
(ulong) page_offset(rec),
591
(ulong) page_offset(page_cur_get_rec(&cur1)),
592
(ulong) page_offset(cur2));
596
page_cur_move_to_next(&cur1);
600
if (UNIV_LIKELY_NULL(heap)) {
605
#ifndef UNIV_HOTBACKUP
606
/*************************************************************//**
607
Copies records from page to new_page, from a given record onward,
608
including that record. Infimum and supremum records are not copied.
609
The records are copied to the start of the record list on new_page.
610
@return pointer to the original successor of the infimum record on
611
new_page, or NULL on zip overflow (new_block will be decompressed) */
614
page_copy_rec_list_end(
615
/*===================*/
616
buf_block_t* new_block, /*!< in/out: index page to copy to */
617
buf_block_t* block, /*!< in: index page containing rec */
618
rec_t* rec, /*!< in: record on page */
619
dict_index_t* index, /*!< in: record descriptor */
620
mtr_t* mtr) /*!< in: mtr */
622
page_t* new_page = buf_block_get_frame(new_block);
623
page_zip_des_t* new_page_zip = buf_block_get_page_zip(new_block);
624
page_t* page = page_align(rec);
625
rec_t* ret = page_rec_get_next(
626
page_get_infimum_rec(new_page));
627
ulint log_mode = 0; /* remove warning */
629
#ifdef UNIV_ZIP_DEBUG
631
page_zip_des_t* page_zip = buf_block_get_page_zip(block);
634
/* Strict page_zip_validate() may fail here.
635
Furthermore, btr_compress() may set FIL_PAGE_PREV to
636
FIL_NULL on new_page while leaving it intact on
637
new_page_zip. So, we cannot validate new_page_zip. */
638
ut_a(page_zip_validate_low(page_zip, page, TRUE));
640
#endif /* UNIV_ZIP_DEBUG */
641
ut_ad(buf_block_get_frame(block) == page);
642
ut_ad(page_is_leaf(page) == page_is_leaf(new_page));
643
ut_ad(page_is_comp(page) == page_is_comp(new_page));
644
/* Here, "ret" may be pointing to a user record or the
645
predefined supremum record. */
647
if (UNIV_LIKELY_NULL(new_page_zip)) {
648
log_mode = mtr_set_log_mode(mtr, MTR_LOG_NONE);
651
if (page_dir_get_n_heap(new_page) == PAGE_HEAP_NO_USER_LOW) {
652
page_copy_rec_list_end_to_created_page(new_page, rec,
655
page_copy_rec_list_end_no_locks(new_block, block, rec,
659
if (UNIV_LIKELY_NULL(new_page_zip)) {
660
mtr_set_log_mode(mtr, log_mode);
663
(!page_zip_compress(new_page_zip, new_page, index, mtr))) {
664
/* Before trying to reorganize the page,
665
store the number of preceding records on the page. */
667
= page_rec_get_n_recs_before(ret);
668
/* Before copying, "ret" was the successor of
669
the predefined infimum record. It must still
670
have at least one predecessor (the predefined
671
infimum record, or a freshly copied record
672
that is smaller than "ret"). */
676
(!page_zip_reorganize(new_block, index, mtr))) {
679
(!page_zip_decompress(new_page_zip,
683
ut_ad(page_validate(new_page, index));
686
/* The page was reorganized:
688
ret = new_page + PAGE_NEW_INFIMUM;
691
ret = rec_get_next_ptr(ret, TRUE);
697
/* Update the lock table, MAX_TRX_ID, and possible hash index */
699
lock_move_rec_list_end(new_block, block, rec);
701
if (dict_index_is_sec_or_ibuf(index) && page_is_leaf(page)) {
702
page_update_max_trx_id(new_block, new_page_zip,
703
page_get_max_trx_id(page), mtr);
706
btr_search_move_or_delete_hash_entries(new_block, block, index);
711
/*************************************************************//**
712
Copies records from page to new_page, up to the given record,
713
NOT including that record. Infimum and supremum records are not copied.
714
The records are copied to the end of the record list on new_page.
715
@return pointer to the original predecessor of the supremum record on
716
new_page, or NULL on zip overflow (new_block will be decompressed) */
719
page_copy_rec_list_start(
720
/*=====================*/
721
buf_block_t* new_block, /*!< in/out: index page to copy to */
722
buf_block_t* block, /*!< in: index page containing rec */
723
rec_t* rec, /*!< in: record on page */
724
dict_index_t* index, /*!< in: record descriptor */
725
mtr_t* mtr) /*!< in: mtr */
727
page_t* new_page = buf_block_get_frame(new_block);
728
page_zip_des_t* new_page_zip = buf_block_get_page_zip(new_block);
731
ulint log_mode = 0 /* remove warning */;
732
mem_heap_t* heap = NULL;
734
= page_rec_get_prev(page_get_supremum_rec(new_page));
735
ulint offsets_[REC_OFFS_NORMAL_SIZE];
736
ulint* offsets = offsets_;
737
rec_offs_init(offsets_);
739
/* Here, "ret" may be pointing to a user record or the
740
predefined infimum record. */
742
if (page_rec_is_infimum(rec)) {
747
if (UNIV_LIKELY_NULL(new_page_zip)) {
748
log_mode = mtr_set_log_mode(mtr, MTR_LOG_NONE);
751
page_cur_set_before_first(block, &cur1);
752
page_cur_move_to_next(&cur1);
756
/* Copy records from the original page to the new page */
758
while (page_cur_get_rec(&cur1) != rec) {
759
rec_t* cur1_rec = page_cur_get_rec(&cur1);
760
offsets = rec_get_offsets(cur1_rec, index, offsets,
761
ULINT_UNDEFINED, &heap);
762
cur2 = page_cur_insert_rec_low(cur2, index,
763
cur1_rec, offsets, mtr);
766
page_cur_move_to_next(&cur1);
769
if (UNIV_LIKELY_NULL(heap)) {
773
if (UNIV_LIKELY_NULL(new_page_zip)) {
774
mtr_set_log_mode(mtr, log_mode);
777
(!page_zip_compress(new_page_zip, new_page, index, mtr))) {
778
/* Before trying to reorganize the page,
779
store the number of preceding records on the page. */
781
= page_rec_get_n_recs_before(ret);
782
/* Before copying, "ret" was the predecessor
783
of the predefined supremum record. If it was
784
the predefined infimum record, then it would
785
still be the infimum. Thus, the assertion
786
ut_a(ret_pos > 0) would fail here. */
789
(!page_zip_reorganize(new_block, index, mtr))) {
792
(!page_zip_decompress(new_page_zip,
796
ut_ad(page_validate(new_page, index));
799
/* The page was reorganized:
801
ret = new_page + PAGE_NEW_INFIMUM;
804
ret = rec_get_next_ptr(ret, TRUE);
810
/* Update MAX_TRX_ID, the lock table, and possible hash index */
812
if (dict_index_is_sec_or_ibuf(index)
813
&& page_is_leaf(page_align(rec))) {
814
page_update_max_trx_id(new_block, new_page_zip,
815
page_get_max_trx_id(page_align(rec)),
819
lock_move_rec_list_start(new_block, block, rec, ret);
821
btr_search_move_or_delete_hash_entries(new_block, block, index);
826
/**********************************************************//**
827
Writes a log record of a record list end or start deletion. */
830
page_delete_rec_list_write_log(
831
/*===========================*/
832
rec_t* rec, /*!< in: record on page */
833
dict_index_t* index, /*!< in: record descriptor */
834
byte type, /*!< in: operation type:
835
MLOG_LIST_END_DELETE, ... */
836
mtr_t* mtr) /*!< in: mtr */
839
ut_ad(type == MLOG_LIST_END_DELETE
840
|| type == MLOG_LIST_START_DELETE
841
|| type == MLOG_COMP_LIST_END_DELETE
842
|| type == MLOG_COMP_LIST_START_DELETE);
844
log_ptr = mlog_open_and_write_index(mtr, rec, index, type, 2);
846
/* Write the parameter as a 2-byte ulint */
847
mach_write_to_2(log_ptr, page_offset(rec));
848
mlog_close(mtr, log_ptr + 2);
851
#else /* !UNIV_HOTBACKUP */
852
# define page_delete_rec_list_write_log(rec,index,type,mtr) ((void) 0)
853
#endif /* !UNIV_HOTBACKUP */
855
/**********************************************************//**
856
Parses a log record of a record list end or start deletion.
857
@return end of log record or NULL */
860
page_parse_delete_rec_list(
861
/*=======================*/
862
byte type, /*!< in: MLOG_LIST_END_DELETE,
863
MLOG_LIST_START_DELETE,
864
MLOG_COMP_LIST_END_DELETE or
865
MLOG_COMP_LIST_START_DELETE */
866
byte* ptr, /*!< in: buffer */
867
byte* end_ptr,/*!< in: buffer end */
868
buf_block_t* block, /*!< in/out: buffer block or NULL */
869
dict_index_t* index, /*!< in: record descriptor */
870
mtr_t* mtr) /*!< in: mtr or NULL */
875
ut_ad(type == MLOG_LIST_END_DELETE
876
|| type == MLOG_LIST_START_DELETE
877
|| type == MLOG_COMP_LIST_END_DELETE
878
|| type == MLOG_COMP_LIST_START_DELETE);
880
/* Read the record offset as a 2-byte ulint */
882
if (end_ptr < ptr + 2) {
887
offset = mach_read_from_2(ptr);
895
page = buf_block_get_frame(block);
897
ut_ad(!!page_is_comp(page) == dict_table_is_comp(index->table));
899
if (type == MLOG_LIST_END_DELETE
900
|| type == MLOG_COMP_LIST_END_DELETE) {
901
page_delete_rec_list_end(page + offset, block, index,
902
ULINT_UNDEFINED, ULINT_UNDEFINED,
905
page_delete_rec_list_start(page + offset, block, index, mtr);
911
/*************************************************************//**
912
Deletes records from a page from a given record onward, including that record.
913
The infimum and supremum records are not deleted. */
916
page_delete_rec_list_end(
917
/*=====================*/
918
rec_t* rec, /*!< in: pointer to record on page */
919
buf_block_t* block, /*!< in: buffer block of the page */
920
dict_index_t* index, /*!< in: record descriptor */
921
ulint n_recs, /*!< in: number of records to delete,
922
or ULINT_UNDEFINED if not known */
923
ulint size, /*!< in: the sum of the sizes of the
924
records in the end of the chain to
925
delete, or ULINT_UNDEFINED if not known */
926
mtr_t* mtr) /*!< in: mtr */
928
page_dir_slot_t*slot;
933
page_zip_des_t* page_zip = buf_block_get_page_zip(block);
934
page_t* page = page_align(rec);
935
mem_heap_t* heap = NULL;
936
ulint offsets_[REC_OFFS_NORMAL_SIZE];
937
ulint* offsets = offsets_;
938
rec_offs_init(offsets_);
940
ut_ad(size == ULINT_UNDEFINED || size < UNIV_PAGE_SIZE);
941
ut_ad(!page_zip || page_rec_is_comp(rec));
942
#ifdef UNIV_ZIP_DEBUG
943
ut_a(!page_zip || page_zip_validate(page_zip, page));
944
#endif /* UNIV_ZIP_DEBUG */
946
if (page_rec_is_infimum(rec)) {
947
rec = page_rec_get_next(rec);
950
if (page_rec_is_supremum(rec)) {
955
/* Reset the last insert info in the page header and increment
956
the modify clock for the frame */
958
page_header_set_ptr(page, page_zip, PAGE_LAST_INSERT, NULL);
960
/* The page gets invalid for optimistic searches: increment the
961
frame modify clock */
963
buf_block_modify_clock_inc(block);
965
page_delete_rec_list_write_log(rec, index, page_is_comp(page)
966
? MLOG_COMP_LIST_END_DELETE
967
: MLOG_LIST_END_DELETE, mtr);
969
if (UNIV_LIKELY_NULL(page_zip)) {
972
ut_a(page_is_comp(page));
973
/* Individual deletes are not logged */
975
log_mode = mtr_set_log_mode(mtr, MTR_LOG_NONE);
979
page_cur_position(rec, block, &cur);
981
offsets = rec_get_offsets(rec, index, offsets,
982
ULINT_UNDEFINED, &heap);
983
rec = rec_get_next_ptr(rec, TRUE);
984
#ifdef UNIV_ZIP_DEBUG
985
ut_a(page_zip_validate(page_zip, page));
986
#endif /* UNIV_ZIP_DEBUG */
987
page_cur_delete_rec(&cur, index, offsets, mtr);
988
} while (page_offset(rec) != PAGE_NEW_SUPREMUM);
990
if (UNIV_LIKELY_NULL(heap)) {
994
/* Restore log mode */
996
mtr_set_log_mode(mtr, log_mode);
1000
prev_rec = page_rec_get_prev(rec);
1002
last_rec = page_rec_get_prev(page_get_supremum_rec(page));
1004
if ((size == ULINT_UNDEFINED) || (n_recs == ULINT_UNDEFINED)) {
1006
/* Calculate the sum of sizes and the number of records */
1012
offsets = rec_get_offsets(rec2, index, offsets,
1013
ULINT_UNDEFINED, &heap);
1014
s = rec_offs_size(offsets);
1015
ut_ad(rec2 - page + s - rec_offs_extra_size(offsets)
1017
ut_ad(size + s < UNIV_PAGE_SIZE);
1021
rec2 = page_rec_get_next(rec2);
1022
} while (!page_rec_is_supremum(rec2));
1024
if (UNIV_LIKELY_NULL(heap)) {
1025
mem_heap_free(heap);
1029
ut_ad(size < UNIV_PAGE_SIZE);
1031
/* Update the page directory; there is no need to balance the number
1032
of the records owned by the supremum record, as it is allowed to be
1033
less than PAGE_DIR_SLOT_MIN_N_OWNED */
1035
if (page_is_comp(page)) {
1039
while (rec_get_n_owned_new(rec2) == 0) {
1042
rec2 = rec_get_next_ptr(rec2, TRUE);
1045
ut_ad(rec_get_n_owned_new(rec2) > count);
1047
n_owned = rec_get_n_owned_new(rec2) - count;
1048
slot_index = page_dir_find_owner_slot(rec2);
1049
slot = page_dir_get_nth_slot(page, slot_index);
1054
while (rec_get_n_owned_old(rec2) == 0) {
1057
rec2 = rec_get_next_ptr(rec2, FALSE);
1060
ut_ad(rec_get_n_owned_old(rec2) > count);
1062
n_owned = rec_get_n_owned_old(rec2) - count;
1063
slot_index = page_dir_find_owner_slot(rec2);
1064
slot = page_dir_get_nth_slot(page, slot_index);
1067
page_dir_slot_set_rec(slot, page_get_supremum_rec(page));
1068
page_dir_slot_set_n_owned(slot, NULL, n_owned);
1070
page_dir_set_n_slots(page, NULL, slot_index + 1);
1072
/* Remove the record chain segment from the record chain */
1073
page_rec_set_next(prev_rec, page_get_supremum_rec(page));
1075
/* Catenate the deleted chain segment to the page free list */
1077
page_rec_set_next(last_rec, page_header_get_ptr(page, PAGE_FREE));
1078
page_header_set_ptr(page, NULL, PAGE_FREE, rec);
1080
page_header_set_field(page, NULL, PAGE_GARBAGE, size
1081
+ page_header_get_field(page, PAGE_GARBAGE));
1083
page_header_set_field(page, NULL, PAGE_N_RECS,
1084
(ulint)(page_get_n_recs(page) - n_recs));
1087
/*************************************************************//**
1088
Deletes records from page, up to the given record, NOT including
1089
that record. Infimum and supremum records are not deleted. */
1092
page_delete_rec_list_start(
1093
/*=======================*/
1094
rec_t* rec, /*!< in: record on page */
1095
buf_block_t* block, /*!< in: buffer block of the page */
1096
dict_index_t* index, /*!< in: record descriptor */
1097
mtr_t* mtr) /*!< in: mtr */
1101
ulint offsets_[REC_OFFS_NORMAL_SIZE];
1102
ulint* offsets = offsets_;
1103
mem_heap_t* heap = NULL;
1106
rec_offs_init(offsets_);
1108
ut_ad((ibool) !!page_rec_is_comp(rec)
1109
== dict_table_is_comp(index->table));
1110
#ifdef UNIV_ZIP_DEBUG
1112
page_zip_des_t* page_zip= buf_block_get_page_zip(block);
1113
page_t* page = buf_block_get_frame(block);
1115
/* page_zip_validate() would detect a min_rec_mark mismatch
1116
in btr_page_split_and_insert()
1117
between btr_attach_half_pages() and insert_page = ...
1118
when btr_page_get_split_rec_to_left() holds
1119
(direction == FSP_DOWN). */
1120
ut_a(!page_zip || page_zip_validate_low(page_zip, page, TRUE));
1122
#endif /* UNIV_ZIP_DEBUG */
1124
if (page_rec_is_infimum(rec)) {
1129
if (page_rec_is_comp(rec)) {
1130
type = MLOG_COMP_LIST_START_DELETE;
1132
type = MLOG_LIST_START_DELETE;
1135
page_delete_rec_list_write_log(rec, index, type, mtr);
1137
page_cur_set_before_first(block, &cur1);
1138
page_cur_move_to_next(&cur1);
1140
/* Individual deletes are not logged */
1142
log_mode = mtr_set_log_mode(mtr, MTR_LOG_NONE);
1144
while (page_cur_get_rec(&cur1) != rec) {
1145
offsets = rec_get_offsets(page_cur_get_rec(&cur1), index,
1146
offsets, ULINT_UNDEFINED, &heap);
1147
page_cur_delete_rec(&cur1, index, offsets, mtr);
1150
if (UNIV_LIKELY_NULL(heap)) {
1151
mem_heap_free(heap);
1154
/* Restore log mode */
1156
mtr_set_log_mode(mtr, log_mode);
1159
#ifndef UNIV_HOTBACKUP
1160
/*************************************************************//**
1161
Moves record list end to another page. Moved records include
1163
@return TRUE on success; FALSE on compression failure (new_block will
1167
page_move_rec_list_end(
1168
/*===================*/
1169
buf_block_t* new_block, /*!< in/out: index page where to move */
1170
buf_block_t* block, /*!< in: index page from where to move */
1171
rec_t* split_rec, /*!< in: first record to move */
1172
dict_index_t* index, /*!< in: record descriptor */
1173
mtr_t* mtr) /*!< in: mtr */
1175
page_t* new_page = buf_block_get_frame(new_block);
1176
ulint old_data_size;
1177
ulint new_data_size;
1181
old_data_size = page_get_data_size(new_page);
1182
old_n_recs = page_get_n_recs(new_page);
1183
#ifdef UNIV_ZIP_DEBUG
1185
page_zip_des_t* new_page_zip
1186
= buf_block_get_page_zip(new_block);
1187
page_zip_des_t* page_zip
1188
= buf_block_get_page_zip(block);
1189
ut_a(!new_page_zip == !page_zip);
1191
|| page_zip_validate(new_page_zip, new_page));
1193
|| page_zip_validate(page_zip, page_align(split_rec)));
1195
#endif /* UNIV_ZIP_DEBUG */
1197
if (UNIV_UNLIKELY(!page_copy_rec_list_end(new_block, block,
1198
split_rec, index, mtr))) {
1202
new_data_size = page_get_data_size(new_page);
1203
new_n_recs = page_get_n_recs(new_page);
1205
ut_ad(new_data_size >= old_data_size);
1207
page_delete_rec_list_end(split_rec, block, index,
1208
new_n_recs - old_n_recs,
1209
new_data_size - old_data_size, mtr);
1214
/*************************************************************//**
1215
Moves record list start to another page. Moved records do not include
1217
@return TRUE on success; FALSE on compression failure */
1220
page_move_rec_list_start(
1221
/*=====================*/
1222
buf_block_t* new_block, /*!< in/out: index page where to move */
1223
buf_block_t* block, /*!< in/out: page containing split_rec */
1224
rec_t* split_rec, /*!< in: first record not to move */
1225
dict_index_t* index, /*!< in: record descriptor */
1226
mtr_t* mtr) /*!< in: mtr */
1228
if (UNIV_UNLIKELY(!page_copy_rec_list_start(new_block, block,
1229
split_rec, index, mtr))) {
1233
page_delete_rec_list_start(split_rec, block, index, mtr);
1238
/***********************************************************************//**
1239
This is a low-level operation which is used in a database index creation
1240
to update the page number of a created B-tree to a data dictionary record. */
1243
page_rec_write_index_page_no(
1244
/*=========================*/
1245
rec_t* rec, /*!< in: record to update */
1246
ulint i, /*!< in: index of the field to update */
1247
ulint page_no,/*!< in: value to write */
1248
mtr_t* mtr) /*!< in: mtr */
1253
data = rec_get_nth_field_old(rec, i, &len);
1257
mlog_write_ulint(data, page_no, MLOG_4BYTES, mtr);
1259
#endif /* !UNIV_HOTBACKUP */
1261
/**************************************************************//**
1262
Used to delete n slots from the directory. This function updates
1263
also n_owned fields in the records, so that the first slot after
1264
the deleted ones inherits the records of the deleted slots. */
1267
page_dir_delete_slot(
1268
/*=================*/
1269
page_t* page, /*!< in/out: the index page */
1270
page_zip_des_t* page_zip,/*!< in/out: compressed page, or NULL */
1271
ulint slot_no)/*!< in: slot to be deleted */
1273
page_dir_slot_t* slot;
1278
ut_ad(!page_zip || page_is_comp(page));
1280
ut_ad(slot_no + 1 < page_dir_get_n_slots(page));
1282
n_slots = page_dir_get_n_slots(page);
1284
/* 1. Reset the n_owned fields of the slots to be
1286
slot = page_dir_get_nth_slot(page, slot_no);
1287
n_owned = page_dir_slot_get_n_owned(slot);
1288
page_dir_slot_set_n_owned(slot, page_zip, 0);
1290
/* 2. Update the n_owned value of the first non-deleted slot */
1292
slot = page_dir_get_nth_slot(page, slot_no + 1);
1293
page_dir_slot_set_n_owned(slot, page_zip,
1294
n_owned + page_dir_slot_get_n_owned(slot));
1296
/* 3. Destroy the slot by copying slots */
1297
for (i = slot_no + 1; i < n_slots; i++) {
1298
rec_t* rec = (rec_t*)
1299
page_dir_slot_get_rec(page_dir_get_nth_slot(page, i));
1300
page_dir_slot_set_rec(page_dir_get_nth_slot(page, i - 1), rec);
1303
/* 4. Zero out the last slot, which will be removed */
1304
mach_write_to_2(page_dir_get_nth_slot(page, n_slots - 1), 0);
1306
/* 5. Update the page header */
1307
page_header_set_field(page, page_zip, PAGE_N_DIR_SLOTS, n_slots - 1);
1310
/**************************************************************//**
1311
Used to add n slots to the directory. Does not set the record pointers
1312
in the added slots or update n_owned values: this is the responsibility
1318
page_t* page, /*!< in/out: the index page */
1319
page_zip_des_t* page_zip,/*!< in/out: comprssed page, or NULL */
1320
ulint start) /*!< in: the slot above which the new slots
1323
page_dir_slot_t* slot;
1326
n_slots = page_dir_get_n_slots(page);
1328
ut_ad(start < n_slots - 1);
1330
/* Update the page header */
1331
page_dir_set_n_slots(page, page_zip, n_slots + 1);
1334
slot = page_dir_get_nth_slot(page, n_slots);
1335
memmove(slot, slot + PAGE_DIR_SLOT_SIZE,
1336
(n_slots - 1 - start) * PAGE_DIR_SLOT_SIZE);
1339
/****************************************************************//**
1340
Splits a directory slot which owns too many records. */
1343
page_dir_split_slot(
1344
/*================*/
1345
page_t* page, /*!< in/out: index page */
1346
page_zip_des_t* page_zip,/*!< in/out: compressed page whose
1347
uncompressed part will be written, or NULL */
1348
ulint slot_no)/*!< in: the directory slot */
1351
page_dir_slot_t* new_slot;
1352
page_dir_slot_t* prev_slot;
1353
page_dir_slot_t* slot;
1358
ut_ad(!page_zip || page_is_comp(page));
1361
slot = page_dir_get_nth_slot(page, slot_no);
1363
n_owned = page_dir_slot_get_n_owned(slot);
1364
ut_ad(n_owned == PAGE_DIR_SLOT_MAX_N_OWNED + 1);
1366
/* 1. We loop to find a record approximately in the middle of the
1367
records owned by the slot. */
1369
prev_slot = page_dir_get_nth_slot(page, slot_no - 1);
1370
rec = (rec_t*) page_dir_slot_get_rec(prev_slot);
1372
for (i = 0; i < n_owned / 2; i++) {
1373
rec = page_rec_get_next(rec);
1376
ut_ad(n_owned / 2 >= PAGE_DIR_SLOT_MIN_N_OWNED);
1378
/* 2. We add one directory slot immediately below the slot to be
1381
page_dir_add_slot(page, page_zip, slot_no - 1);
1383
/* The added slot is now number slot_no, and the old slot is
1384
now number slot_no + 1 */
1386
new_slot = page_dir_get_nth_slot(page, slot_no);
1387
slot = page_dir_get_nth_slot(page, slot_no + 1);
1389
/* 3. We store the appropriate values to the new slot. */
1391
page_dir_slot_set_rec(new_slot, rec);
1392
page_dir_slot_set_n_owned(new_slot, page_zip, n_owned / 2);
1394
/* 4. Finally, we update the number of records field of the
1397
page_dir_slot_set_n_owned(slot, page_zip, n_owned - (n_owned / 2));
1400
/*************************************************************//**
1401
Tries to balance the given directory slot with too few records with the upper
1402
neighbor, so that there are at least the minimum number of records owned by
1403
the slot; this may result in the merging of two slots. */
1406
page_dir_balance_slot(
1407
/*==================*/
1408
page_t* page, /*!< in/out: index page */
1409
page_zip_des_t* page_zip,/*!< in/out: compressed page, or NULL */
1410
ulint slot_no)/*!< in: the directory slot */
1412
page_dir_slot_t* slot;
1413
page_dir_slot_t* up_slot;
1420
ut_ad(!page_zip || page_is_comp(page));
1423
slot = page_dir_get_nth_slot(page, slot_no);
1425
/* The last directory slot cannot be balanced with the upper
1426
neighbor, as there is none. */
1428
if (UNIV_UNLIKELY(slot_no == page_dir_get_n_slots(page) - 1)) {
1433
up_slot = page_dir_get_nth_slot(page, slot_no + 1);
1435
n_owned = page_dir_slot_get_n_owned(slot);
1436
up_n_owned = page_dir_slot_get_n_owned(up_slot);
1438
ut_ad(n_owned == PAGE_DIR_SLOT_MIN_N_OWNED - 1);
1440
/* If the upper slot has the minimum value of n_owned, we will merge
1441
the two slots, therefore we assert: */
1442
ut_ad(2 * PAGE_DIR_SLOT_MIN_N_OWNED - 1 <= PAGE_DIR_SLOT_MAX_N_OWNED);
1444
if (up_n_owned > PAGE_DIR_SLOT_MIN_N_OWNED) {
1446
/* In this case we can just transfer one record owned
1447
by the upper slot to the property of the lower slot */
1448
old_rec = (rec_t*) page_dir_slot_get_rec(slot);
1450
if (page_is_comp(page)) {
1451
new_rec = rec_get_next_ptr(old_rec, TRUE);
1453
rec_set_n_owned_new(old_rec, page_zip, 0);
1454
rec_set_n_owned_new(new_rec, page_zip, n_owned + 1);
1456
new_rec = rec_get_next_ptr(old_rec, FALSE);
1458
rec_set_n_owned_old(old_rec, 0);
1459
rec_set_n_owned_old(new_rec, n_owned + 1);
1462
page_dir_slot_set_rec(slot, new_rec);
1464
page_dir_slot_set_n_owned(up_slot, page_zip, up_n_owned -1);
1466
/* In this case we may merge the two slots */
1467
page_dir_delete_slot(page, page_zip, slot_no);
1471
#ifndef UNIV_HOTBACKUP
1472
/************************************************************//**
1473
Returns the middle record of the record list. If there are an even number
1474
of records in the list, returns the first record of the upper half-list.
1475
@return middle record */
1478
page_get_middle_rec(
1479
/*================*/
1480
page_t* page) /*!< in: page */
1482
page_dir_slot_t* slot;
1489
/* This many records we must leave behind */
1490
middle = (page_get_n_recs(page) + PAGE_HEAP_NO_USER_LOW) / 2;
1496
slot = page_dir_get_nth_slot(page, i);
1497
n_owned = page_dir_slot_get_n_owned(slot);
1499
if (count + n_owned > middle) {
1507
slot = page_dir_get_nth_slot(page, i - 1);
1508
rec = (rec_t*) page_dir_slot_get_rec(slot);
1509
rec = page_rec_get_next(rec);
1511
/* There are now count records behind rec */
1513
for (i = 0; i < middle - count; i++) {
1514
rec = page_rec_get_next(rec);
1519
#endif /* !UNIV_HOTBACKUP */
1521
/***************************************************************//**
1522
Returns the number of records before the given record in chain.
1523
The number includes infimum and supremum records.
1524
@return number of records */
1527
page_rec_get_n_recs_before(
1528
/*=======================*/
1529
const rec_t* rec) /*!< in: the physical record */
1531
const page_dir_slot_t* slot;
1532
const rec_t* slot_rec;
1537
ut_ad(page_rec_check(rec));
1539
page = page_align(rec);
1540
if (page_is_comp(page)) {
1541
while (rec_get_n_owned_new(rec) == 0) {
1543
rec = rec_get_next_ptr_const(rec, TRUE);
1547
for (i = 0; ; i++) {
1548
slot = page_dir_get_nth_slot(page, i);
1549
slot_rec = page_dir_slot_get_rec(slot);
1551
n += rec_get_n_owned_new(slot_rec);
1553
if (rec == slot_rec) {
1559
while (rec_get_n_owned_old(rec) == 0) {
1561
rec = rec_get_next_ptr_const(rec, FALSE);
1565
for (i = 0; ; i++) {
1566
slot = page_dir_get_nth_slot(page, i);
1567
slot_rec = page_dir_slot_get_rec(slot);
1569
n += rec_get_n_owned_old(slot_rec);
1571
if (rec == slot_rec) {
1585
#ifndef UNIV_HOTBACKUP
1586
/************************************************************//**
1587
Prints record contents including the data relevant only in
1588
the index page context. */
1593
const rec_t* rec, /*!< in: physical record */
1594
const ulint* offsets)/*!< in: record descriptor */
1596
ut_a(!page_rec_is_comp(rec) == !rec_offs_comp(offsets));
1597
rec_print_new(stderr, rec, offsets);
1598
if (page_rec_is_comp(rec)) {
1600
" n_owned: %lu; heap_no: %lu; next rec: %lu\n",
1601
(ulong) rec_get_n_owned_new(rec),
1602
(ulong) rec_get_heap_no_new(rec),
1603
(ulong) rec_get_next_offs(rec, TRUE));
1606
" n_owned: %lu; heap_no: %lu; next rec: %lu\n",
1607
(ulong) rec_get_n_owned_old(rec),
1608
(ulong) rec_get_heap_no_old(rec),
1609
(ulong) rec_get_next_offs(rec, TRUE));
1612
page_rec_check(rec);
1613
rec_validate(rec, offsets);
1616
/***************************************************************//**
1617
This is used to print the contents of the directory for
1618
debugging purposes. */
1623
page_t* page, /*!< in: index page */
1624
ulint pr_n) /*!< in: print n first and n last entries */
1628
page_dir_slot_t* slot;
1630
n = page_dir_get_n_slots(page);
1632
fprintf(stderr, "--------------------------------\n"
1635
"Directory stack top at offs: %lu; number of slots: %lu\n",
1636
page, (ulong) page_offset(page_dir_get_nth_slot(page, n - 1)),
1638
for (i = 0; i < n; i++) {
1639
slot = page_dir_get_nth_slot(page, i);
1640
if ((i == pr_n) && (i < n - pr_n)) {
1641
fputs(" ... \n", stderr);
1643
if ((i < pr_n) || (i >= n - pr_n)) {
1645
"Contents of slot: %lu: n_owned: %lu,"
1648
(ulong) page_dir_slot_get_n_owned(slot),
1650
page_offset(page_dir_slot_get_rec(slot)));
1653
fprintf(stderr, "Total of %lu records\n"
1654
"--------------------------------\n",
1655
(ulong) (PAGE_HEAP_NO_USER_LOW + page_get_n_recs(page)));
1658
/***************************************************************//**
1659
This is used to print the contents of the page record list for
1660
debugging purposes. */
1665
buf_block_t* block, /*!< in: index page */
1666
dict_index_t* index, /*!< in: dictionary index of the page */
1667
ulint pr_n) /*!< in: print n first and n last entries */
1669
page_t* page = block->frame;
1673
mem_heap_t* heap = NULL;
1674
ulint offsets_[REC_OFFS_NORMAL_SIZE];
1675
ulint* offsets = offsets_;
1676
rec_offs_init(offsets_);
1678
ut_a((ibool)!!page_is_comp(page) == dict_table_is_comp(index->table));
1681
"--------------------------------\n"
1682
"PAGE RECORD LIST\n"
1683
"Page address %p\n", page);
1685
n_recs = page_get_n_recs(page);
1687
page_cur_set_before_first(block, &cur);
1690
offsets = rec_get_offsets(cur.rec, index, offsets,
1691
ULINT_UNDEFINED, &heap);
1692
page_rec_print(cur.rec, offsets);
1694
if (count == pr_n) {
1697
if (page_cur_is_after_last(&cur)) {
1700
page_cur_move_to_next(&cur);
1704
if (n_recs > 2 * pr_n) {
1705
fputs(" ... \n", stderr);
1708
while (!page_cur_is_after_last(&cur)) {
1709
page_cur_move_to_next(&cur);
1711
if (count + pr_n >= n_recs) {
1712
offsets = rec_get_offsets(cur.rec, index, offsets,
1713
ULINT_UNDEFINED, &heap);
1714
page_rec_print(cur.rec, offsets);
1720
"Total of %lu records \n"
1721
"--------------------------------\n",
1722
(ulong) (count + 1));
1724
if (UNIV_LIKELY_NULL(heap)) {
1725
mem_heap_free(heap);
1729
/***************************************************************//**
1730
Prints the info in a page header. */
1738
"--------------------------------\n"
1739
"PAGE HEADER INFO\n"
1740
"Page address %p, n records %lu (%s)\n"
1741
"n dir slots %lu, heap top %lu\n"
1742
"Page n heap %lu, free %lu, garbage %lu\n"
1743
"Page last insert %lu, direction %lu, n direction %lu\n",
1744
page, (ulong) page_header_get_field(page, PAGE_N_RECS),
1745
page_is_comp(page) ? "compact format" : "original format",
1746
(ulong) page_header_get_field(page, PAGE_N_DIR_SLOTS),
1747
(ulong) page_header_get_field(page, PAGE_HEAP_TOP),
1748
(ulong) page_dir_get_n_heap(page),
1749
(ulong) page_header_get_field(page, PAGE_FREE),
1750
(ulong) page_header_get_field(page, PAGE_GARBAGE),
1751
(ulong) page_header_get_field(page, PAGE_LAST_INSERT),
1752
(ulong) page_header_get_field(page, PAGE_DIRECTION),
1753
(ulong) page_header_get_field(page, PAGE_N_DIRECTION));
1756
/***************************************************************//**
1757
This is used to print the contents of the page for
1758
debugging purposes. */
1763
buf_block_t* block, /*!< in: index page */
1764
dict_index_t* index, /*!< in: dictionary index of the page */
1765
ulint dn, /*!< in: print dn first and last entries
1767
ulint rn) /*!< in: print rn first and last records
1770
page_t* page = block->frame;
1772
page_header_print(page);
1773
page_dir_print(page, dn);
1774
page_print_list(block, index, rn);
1776
#endif /* !UNIV_HOTBACKUP */
1778
/***************************************************************//**
1779
The following is used to validate a record on a page. This function
1780
differs from rec_validate as it can also check the n_owned field and
1782
@return TRUE if ok */
1787
rec_t* rec, /*!< in: physical record */
1788
const ulint* offsets)/*!< in: array returned by rec_get_offsets() */
1794
page = page_align(rec);
1795
ut_a(!page_is_comp(page) == !rec_offs_comp(offsets));
1797
page_rec_check(rec);
1798
rec_validate(rec, offsets);
1800
if (page_rec_is_comp(rec)) {
1801
n_owned = rec_get_n_owned_new(rec);
1802
heap_no = rec_get_heap_no_new(rec);
1804
n_owned = rec_get_n_owned_old(rec);
1805
heap_no = rec_get_heap_no_old(rec);
1808
if (UNIV_UNLIKELY(!(n_owned <= PAGE_DIR_SLOT_MAX_N_OWNED))) {
1810
"InnoDB: Dir slot of rec %lu, n owned too big %lu\n",
1811
(ulong) page_offset(rec), (ulong) n_owned);
1815
if (UNIV_UNLIKELY(!(heap_no < page_dir_get_n_heap(page)))) {
1817
"InnoDB: Heap no of rec %lu too big %lu %lu\n",
1818
(ulong) page_offset(rec), (ulong) heap_no,
1819
(ulong) page_dir_get_n_heap(page));
1826
#ifndef UNIV_HOTBACKUP
1827
/***************************************************************//**
1828
Checks that the first directory slot points to the infimum record and
1829
the last to the supremum. This function is intended to track if the
1830
bug fixed in 4.0.14 has caused corruption to users' databases. */
1835
const page_t* page) /*!< in: index page */
1839
ulint supremum_offs;
1841
n_slots = page_dir_get_n_slots(page);
1842
infimum_offs = mach_read_from_2(page_dir_get_nth_slot(page, 0));
1843
supremum_offs = mach_read_from_2(page_dir_get_nth_slot(page,
1846
if (UNIV_UNLIKELY(!page_rec_is_infimum_low(infimum_offs))) {
1849
"InnoDB: Page directory corruption:"
1850
" infimum not pointed to\n");
1851
buf_page_print(page, 0);
1854
if (UNIV_UNLIKELY(!page_rec_is_supremum_low(supremum_offs))) {
1857
"InnoDB: Page directory corruption:"
1858
" supremum not pointed to\n");
1859
buf_page_print(page, 0);
1862
#endif /* !UNIV_HOTBACKUP */
1864
/***************************************************************//**
1865
This function checks the consistency of an index page when we do not
1866
know the index. This is also resilient so that this should never crash
1867
even if the page is total garbage.
1868
@return TRUE if ok */
1871
page_simple_validate_old(
1872
/*=====================*/
1873
page_t* page) /*!< in: old-style index page */
1875
page_dir_slot_t* slot;
1884
ut_a(!page_is_comp(page));
1886
/* Check first that the record heap and the directory do not
1889
n_slots = page_dir_get_n_slots(page);
1891
if (UNIV_UNLIKELY(n_slots > UNIV_PAGE_SIZE / 4)) {
1893
"InnoDB: Nonsensical number %lu of page dir slots\n",
1899
rec_heap_top = page_header_get_ptr(page, PAGE_HEAP_TOP);
1901
if (UNIV_UNLIKELY(rec_heap_top
1902
> page_dir_get_nth_slot(page, n_slots - 1))) {
1905
"InnoDB: Record heap and dir overlap on a page,"
1906
" heap top %lu, dir %lu\n",
1907
(ulong) page_header_get_field(page, PAGE_HEAP_TOP),
1909
page_offset(page_dir_get_nth_slot(page, n_slots - 1)));
1914
/* Validate the record list in a loop checking also that it is
1915
consistent with the page record directory. */
1920
slot = page_dir_get_nth_slot(page, slot_no);
1922
rec = page_get_infimum_rec(page);
1925
if (UNIV_UNLIKELY(rec > rec_heap_top)) {
1927
"InnoDB: Record %lu is above"
1928
" rec heap top %lu\n",
1929
(ulong)(rec - page),
1930
(ulong)(rec_heap_top - page));
1935
if (UNIV_UNLIKELY(rec_get_n_owned_old(rec))) {
1936
/* This is a record pointed to by a dir slot */
1937
if (UNIV_UNLIKELY(rec_get_n_owned_old(rec)
1941
"InnoDB: Wrong owned count %lu, %lu,"
1943
(ulong) rec_get_n_owned_old(rec),
1945
(ulong)(rec - page));
1951
(page_dir_slot_get_rec(slot) != rec)) {
1953
"InnoDB: Dir slot does not point"
1954
" to right rec %lu\n",
1955
(ulong)(rec - page));
1962
if (!page_rec_is_supremum(rec)) {
1964
slot = page_dir_get_nth_slot(page, slot_no);
1968
if (page_rec_is_supremum(rec)) {
1974
(rec_get_next_offs(rec, FALSE) < FIL_PAGE_DATA
1975
|| rec_get_next_offs(rec, FALSE) >= UNIV_PAGE_SIZE)) {
1977
"InnoDB: Next record offset"
1978
" nonsensical %lu for rec %lu\n",
1979
(ulong) rec_get_next_offs(rec, FALSE),
1980
(ulong) (rec - page));
1987
if (UNIV_UNLIKELY(count > UNIV_PAGE_SIZE)) {
1989
"InnoDB: Page record list appears"
1990
" to be circular %lu\n",
1995
rec = page_rec_get_next(rec);
1999
if (UNIV_UNLIKELY(rec_get_n_owned_old(rec) == 0)) {
2000
fprintf(stderr, "InnoDB: n owned is zero in a supremum rec\n");
2005
if (UNIV_UNLIKELY(slot_no != n_slots - 1)) {
2006
fprintf(stderr, "InnoDB: n slots wrong %lu, %lu\n",
2007
(ulong) slot_no, (ulong) (n_slots - 1));
2011
if (UNIV_UNLIKELY(page_header_get_field(page, PAGE_N_RECS)
2012
+ PAGE_HEAP_NO_USER_LOW
2014
fprintf(stderr, "InnoDB: n recs wrong %lu %lu\n",
2015
(ulong) page_header_get_field(page, PAGE_N_RECS)
2016
+ PAGE_HEAP_NO_USER_LOW,
2017
(ulong) (count + 1));
2022
/* Check then the free list */
2023
rec = page_header_get_ptr(page, PAGE_FREE);
2025
while (rec != NULL) {
2026
if (UNIV_UNLIKELY(rec < page + FIL_PAGE_DATA
2027
|| rec >= page + UNIV_PAGE_SIZE)) {
2029
"InnoDB: Free list record has"
2030
" a nonsensical offset %lu\n",
2031
(ulong) (rec - page));
2036
if (UNIV_UNLIKELY(rec > rec_heap_top)) {
2038
"InnoDB: Free list record %lu"
2039
" is above rec heap top %lu\n",
2040
(ulong) (rec - page),
2041
(ulong) (rec_heap_top - page));
2048
if (UNIV_UNLIKELY(count > UNIV_PAGE_SIZE)) {
2050
"InnoDB: Page free list appears"
2051
" to be circular %lu\n",
2056
rec = page_rec_get_next(rec);
2059
if (UNIV_UNLIKELY(page_dir_get_n_heap(page) != count + 1)) {
2061
fprintf(stderr, "InnoDB: N heap is wrong %lu, %lu\n",
2062
(ulong) page_dir_get_n_heap(page),
2063
(ulong) (count + 1));
2074
/***************************************************************//**
2075
This function checks the consistency of an index page when we do not
2076
know the index. This is also resilient so that this should never crash
2077
even if the page is total garbage.
2078
@return TRUE if ok */
2081
page_simple_validate_new(
2082
/*=====================*/
2083
page_t* page) /*!< in: new-style index page */
2085
page_dir_slot_t* slot;
2094
ut_a(page_is_comp(page));
2096
/* Check first that the record heap and the directory do not
2099
n_slots = page_dir_get_n_slots(page);
2101
if (UNIV_UNLIKELY(n_slots > UNIV_PAGE_SIZE / 4)) {
2103
"InnoDB: Nonsensical number %lu"
2104
" of page dir slots\n", (ulong) n_slots);
2109
rec_heap_top = page_header_get_ptr(page, PAGE_HEAP_TOP);
2111
if (UNIV_UNLIKELY(rec_heap_top
2112
> page_dir_get_nth_slot(page, n_slots - 1))) {
2115
"InnoDB: Record heap and dir overlap on a page,"
2116
" heap top %lu, dir %lu\n",
2117
(ulong) page_header_get_field(page, PAGE_HEAP_TOP),
2119
page_offset(page_dir_get_nth_slot(page, n_slots - 1)));
2124
/* Validate the record list in a loop checking also that it is
2125
consistent with the page record directory. */
2130
slot = page_dir_get_nth_slot(page, slot_no);
2132
rec = page_get_infimum_rec(page);
2135
if (UNIV_UNLIKELY(rec > rec_heap_top)) {
2137
"InnoDB: Record %lu is above rec"
2139
(ulong) page_offset(rec),
2140
(ulong) page_offset(rec_heap_top));
2145
if (UNIV_UNLIKELY(rec_get_n_owned_new(rec))) {
2146
/* This is a record pointed to by a dir slot */
2147
if (UNIV_UNLIKELY(rec_get_n_owned_new(rec)
2151
"InnoDB: Wrong owned count %lu, %lu,"
2153
(ulong) rec_get_n_owned_new(rec),
2155
(ulong) page_offset(rec));
2161
(page_dir_slot_get_rec(slot) != rec)) {
2163
"InnoDB: Dir slot does not point"
2164
" to right rec %lu\n",
2165
(ulong) page_offset(rec));
2172
if (!page_rec_is_supremum(rec)) {
2174
slot = page_dir_get_nth_slot(page, slot_no);
2178
if (page_rec_is_supremum(rec)) {
2184
(rec_get_next_offs(rec, TRUE) < FIL_PAGE_DATA
2185
|| rec_get_next_offs(rec, TRUE) >= UNIV_PAGE_SIZE)) {
2187
"InnoDB: Next record offset nonsensical %lu"
2189
(ulong) rec_get_next_offs(rec, TRUE),
2190
(ulong) page_offset(rec));
2197
if (UNIV_UNLIKELY(count > UNIV_PAGE_SIZE)) {
2199
"InnoDB: Page record list appears"
2200
" to be circular %lu\n",
2205
rec = page_rec_get_next(rec);
2209
if (UNIV_UNLIKELY(rec_get_n_owned_new(rec) == 0)) {
2210
fprintf(stderr, "InnoDB: n owned is zero"
2211
" in a supremum rec\n");
2216
if (UNIV_UNLIKELY(slot_no != n_slots - 1)) {
2217
fprintf(stderr, "InnoDB: n slots wrong %lu, %lu\n",
2218
(ulong) slot_no, (ulong) (n_slots - 1));
2222
if (UNIV_UNLIKELY(page_header_get_field(page, PAGE_N_RECS)
2223
+ PAGE_HEAP_NO_USER_LOW
2225
fprintf(stderr, "InnoDB: n recs wrong %lu %lu\n",
2226
(ulong) page_header_get_field(page, PAGE_N_RECS)
2227
+ PAGE_HEAP_NO_USER_LOW,
2228
(ulong) (count + 1));
2233
/* Check then the free list */
2234
rec = page_header_get_ptr(page, PAGE_FREE);
2236
while (rec != NULL) {
2237
if (UNIV_UNLIKELY(rec < page + FIL_PAGE_DATA
2238
|| rec >= page + UNIV_PAGE_SIZE)) {
2240
"InnoDB: Free list record has"
2241
" a nonsensical offset %lu\n",
2242
(ulong) page_offset(rec));
2247
if (UNIV_UNLIKELY(rec > rec_heap_top)) {
2249
"InnoDB: Free list record %lu"
2250
" is above rec heap top %lu\n",
2251
(ulong) page_offset(rec),
2252
(ulong) page_offset(rec_heap_top));
2259
if (UNIV_UNLIKELY(count > UNIV_PAGE_SIZE)) {
2261
"InnoDB: Page free list appears"
2262
" to be circular %lu\n",
2267
rec = page_rec_get_next(rec);
2270
if (UNIV_UNLIKELY(page_dir_get_n_heap(page) != count + 1)) {
2272
fprintf(stderr, "InnoDB: N heap is wrong %lu, %lu\n",
2273
(ulong) page_dir_get_n_heap(page),
2274
(ulong) (count + 1));
2285
/***************************************************************//**
2286
This function checks the consistency of an index page.
2287
@return TRUE if ok */
2292
page_t* page, /*!< in: index page */
2293
dict_index_t* index) /*!< in: data dictionary index containing
2294
the page record type definition */
2296
page_dir_slot_t*slot;
2301
ulint rec_own_count;
2305
rec_t* old_rec = NULL;
2310
ulint* offsets = NULL;
2311
ulint* old_offsets = NULL;
2313
if (UNIV_UNLIKELY((ibool) !!page_is_comp(page)
2314
!= dict_table_is_comp(index->table))) {
2315
fputs("InnoDB: 'compact format' flag mismatch\n", stderr);
2318
if (page_is_comp(page)) {
2319
if (UNIV_UNLIKELY(!page_simple_validate_new(page))) {
2323
if (UNIV_UNLIKELY(!page_simple_validate_old(page))) {
2328
heap = mem_heap_create(UNIV_PAGE_SIZE + 200);
2330
/* The following buffer is used to check that the
2331
records in the page record heap do not overlap */
2333
buf = mem_heap_zalloc(heap, UNIV_PAGE_SIZE);
2335
/* Check first that the record heap and the directory do not
2338
n_slots = page_dir_get_n_slots(page);
2340
if (UNIV_UNLIKELY(!(page_header_get_ptr(page, PAGE_HEAP_TOP)
2341
<= page_dir_get_nth_slot(page, n_slots - 1)))) {
2344
"InnoDB: Record heap and dir overlap"
2345
" on space %lu page %lu index %s, %p, %p\n",
2346
(ulong) page_get_space_id(page),
2347
(ulong) page_get_page_no(page), index->name,
2348
page_header_get_ptr(page, PAGE_HEAP_TOP),
2349
page_dir_get_nth_slot(page, n_slots - 1));
2354
/* Validate the record list in a loop checking also that
2355
it is consistent with the directory. */
2360
slot = page_dir_get_nth_slot(page, slot_no);
2362
rec = page_get_infimum_rec(page);
2365
offsets = rec_get_offsets(rec, index, offsets,
2366
ULINT_UNDEFINED, &heap);
2368
if (page_is_comp(page) && page_rec_is_user_rec(rec)
2369
&& UNIV_UNLIKELY(rec_get_node_ptr_flag(rec)
2370
== page_is_leaf(page))) {
2371
fputs("InnoDB: node_ptr flag mismatch\n", stderr);
2375
if (UNIV_UNLIKELY(!page_rec_validate(rec, offsets))) {
2379
#ifndef UNIV_HOTBACKUP
2380
/* Check that the records are in the ascending order */
2381
if (UNIV_LIKELY(count >= PAGE_HEAP_NO_USER_LOW)
2382
&& !page_rec_is_supremum(rec)) {
2384
(1 != cmp_rec_rec(rec, old_rec,
2385
offsets, old_offsets, index))) {
2387
"InnoDB: Records in wrong order"
2388
" on space %lu page %lu index %s\n",
2389
(ulong) page_get_space_id(page),
2390
(ulong) page_get_page_no(page),
2392
fputs("\nInnoDB: previous record ", stderr);
2393
rec_print_new(stderr, old_rec, old_offsets);
2394
fputs("\nInnoDB: record ", stderr);
2395
rec_print_new(stderr, rec, offsets);
2401
#endif /* !UNIV_HOTBACKUP */
2403
if (page_rec_is_user_rec(rec)) {
2405
data_size += rec_offs_size(offsets);
2408
offs = page_offset(rec_get_start(rec, offsets));
2410
for (i = rec_offs_size(offsets); i--; ) {
2411
if (UNIV_UNLIKELY(buf[offs + i])) {
2412
/* No other record may overlap this */
2414
fputs("InnoDB: Record overlaps another\n",
2422
if (page_is_comp(page)) {
2423
rec_own_count = rec_get_n_owned_new(rec);
2425
rec_own_count = rec_get_n_owned_old(rec);
2428
if (UNIV_UNLIKELY(rec_own_count)) {
2429
/* This is a record pointed to by a dir slot */
2430
if (UNIV_UNLIKELY(rec_own_count != own_count)) {
2432
"InnoDB: Wrong owned count %lu, %lu\n",
2433
(ulong) rec_own_count,
2438
if (page_dir_slot_get_rec(slot) != rec) {
2439
fputs("InnoDB: Dir slot does not"
2440
" point to right rec\n",
2445
page_dir_slot_check(slot);
2448
if (!page_rec_is_supremum(rec)) {
2450
slot = page_dir_get_nth_slot(page, slot_no);
2454
if (page_rec_is_supremum(rec)) {
2461
rec = page_rec_get_next(rec);
2463
/* set old_offsets to offsets; recycle offsets */
2465
ulint* offs = old_offsets;
2466
old_offsets = offsets;
2471
if (page_is_comp(page)) {
2472
if (UNIV_UNLIKELY(rec_get_n_owned_new(rec) == 0)) {
2476
} else if (UNIV_UNLIKELY(rec_get_n_owned_old(rec) == 0)) {
2478
fputs("InnoDB: n owned is zero\n", stderr);
2482
if (UNIV_UNLIKELY(slot_no != n_slots - 1)) {
2483
fprintf(stderr, "InnoDB: n slots wrong %lu %lu\n",
2484
(ulong) slot_no, (ulong) (n_slots - 1));
2488
if (UNIV_UNLIKELY(page_header_get_field(page, PAGE_N_RECS)
2489
+ PAGE_HEAP_NO_USER_LOW
2491
fprintf(stderr, "InnoDB: n recs wrong %lu %lu\n",
2492
(ulong) page_header_get_field(page, PAGE_N_RECS)
2493
+ PAGE_HEAP_NO_USER_LOW,
2494
(ulong) (count + 1));
2498
if (UNIV_UNLIKELY(data_size != page_get_data_size(page))) {
2500
"InnoDB: Summed data size %lu, returned by func %lu\n",
2501
(ulong) data_size, (ulong) page_get_data_size(page));
2505
/* Check then the free list */
2506
rec = page_header_get_ptr(page, PAGE_FREE);
2508
while (rec != NULL) {
2509
offsets = rec_get_offsets(rec, index, offsets,
2510
ULINT_UNDEFINED, &heap);
2511
if (UNIV_UNLIKELY(!page_rec_validate(rec, offsets))) {
2517
offs = page_offset(rec_get_start(rec, offsets));
2519
for (i = rec_offs_size(offsets); i--; ) {
2521
if (UNIV_UNLIKELY(buf[offs + i])) {
2522
fputs("InnoDB: Record overlaps another"
2523
" in free list\n", stderr);
2530
rec = page_rec_get_next(rec);
2533
if (UNIV_UNLIKELY(page_dir_get_n_heap(page) != count + 1)) {
2534
fprintf(stderr, "InnoDB: N heap is wrong %lu %lu\n",
2535
(ulong) page_dir_get_n_heap(page),
2543
mem_heap_free(heap);
2545
if (UNIV_UNLIKELY(ret == FALSE)) {
2548
"InnoDB: Apparent corruption"
2549
" in space %lu page %lu index %s\n",
2550
(ulong) page_get_space_id(page),
2551
(ulong) page_get_page_no(page),
2553
buf_page_print(page, 0);
2559
#ifndef UNIV_HOTBACKUP
2560
/***************************************************************//**
2561
Looks in the page record list for a record with the given heap number.
2562
@return record, NULL if not found */
2565
page_find_rec_with_heap_no(
2566
/*=======================*/
2567
const page_t* page, /*!< in: index page */
2568
ulint heap_no)/*!< in: heap number */
2572
if (page_is_comp(page)) {
2573
rec = page + PAGE_NEW_INFIMUM;
2576
ulint rec_heap_no = rec_get_heap_no_new(rec);
2578
if (rec_heap_no == heap_no) {
2581
} else if (rec_heap_no == PAGE_HEAP_NO_SUPREMUM) {
2586
rec = page + rec_get_next_offs(rec, TRUE);
2589
rec = page + PAGE_OLD_INFIMUM;
2592
ulint rec_heap_no = rec_get_heap_no_old(rec);
2594
if (rec_heap_no == heap_no) {
2597
} else if (rec_heap_no == PAGE_HEAP_NO_SUPREMUM) {
2602
rec = page + rec_get_next_offs(rec, FALSE);
2606
#endif /* !UNIV_HOTBACKUP */