/***************************************************************************** Copyright (c) 1994, 2016, Oracle and/or its affiliates. All Rights Reserved. Copyright (c) 2012, Facebook Inc. Copyright (c) 2018, 2022, MariaDB Corporation. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; version 2 of the License. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA *****************************************************************************/ /********************************************************************//** @file page/page0cur.cc The page cursor Created 10/4/1994 Heikki Tuuri *************************************************************************/ #include "page0cur.h" #include "page0zip.h" #include "btr0btr.h" #include "mtr0log.h" #include "log0recv.h" #include "rem0cmp.h" #include "gis0rtree.h" #include #ifdef BTR_CUR_HASH_ADAPT # ifdef UNIV_SEARCH_PERF_STAT static ulint page_cur_short_succ; # endif /* UNIV_SEARCH_PERF_STAT */ /** Try a search shortcut based on the last insert. @param[in] block index page @param[in] index index tree @param[in] tuple search key @param[in,out] iup_matched_fields already matched fields in the upper limit record @param[in,out] ilow_matched_fields already matched fields in the lower limit record @param[out] cursor page cursor @return true on success */ UNIV_INLINE bool page_cur_try_search_shortcut( const buf_block_t* block, const dict_index_t* index, const dtuple_t* tuple, ulint* iup_matched_fields, ulint* ilow_matched_fields, page_cur_t* cursor) { const rec_t* rec; const rec_t* next_rec; ulint low_match; ulint up_match; ibool success = FALSE; const page_t* page = buf_block_get_frame(block); mem_heap_t* heap = NULL; rec_offs offsets_[REC_OFFS_NORMAL_SIZE]; rec_offs* offsets = offsets_; rec_offs_init(offsets_); ut_ad(dtuple_check_typed(tuple)); ut_ad(page_is_leaf(page)); rec = page_header_get_ptr(page, PAGE_LAST_INSERT); offsets = rec_get_offsets(rec, index, offsets, index->n_core_fields, dtuple_get_n_fields(tuple), &heap); ut_ad(rec); ut_ad(page_rec_is_user_rec(rec)); low_match = up_match = std::min(*ilow_matched_fields, *iup_matched_fields); if (cmp_dtuple_rec_with_match(tuple, rec, index, offsets, &low_match) < 0) { goto exit_func; } if (!(next_rec = page_rec_get_next_const(rec))) { goto exit_func; } if (!page_rec_is_supremum(next_rec)) { offsets = rec_get_offsets(next_rec, index, offsets, index->n_core_fields, dtuple_get_n_fields(tuple), &heap); if (cmp_dtuple_rec_with_match(tuple, next_rec, index, offsets, &up_match) >= 0) { goto exit_func; } *iup_matched_fields = up_match; } page_cur_position(rec, block, cursor); *ilow_matched_fields = low_match; #ifdef UNIV_SEARCH_PERF_STAT page_cur_short_succ++; #endif success = TRUE; exit_func: if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return(success); } /** Try a search shortcut based on the last insert. @param[in] block index page @param[in] index index tree @param[in] tuple search key @param[in,out] iup_matched_fields already matched fields in the upper limit record @param[in,out] iup_matched_bytes already matched bytes in the first partially matched field in the upper limit record @param[in,out] ilow_matched_fields already matched fields in the lower limit record @param[in,out] ilow_matched_bytes already matched bytes in the first partially matched field in the lower limit record @param[out] cursor page cursor @return true on success */ UNIV_INLINE bool page_cur_try_search_shortcut_bytes( const buf_block_t* block, const dict_index_t* index, const dtuple_t* tuple, ulint* iup_matched_fields, ulint* iup_matched_bytes, ulint* ilow_matched_fields, ulint* ilow_matched_bytes, page_cur_t* cursor) { const rec_t* rec; const rec_t* next_rec; ulint low_match; ulint low_bytes; ulint up_match; ulint up_bytes; ibool success = FALSE; const page_t* page = buf_block_get_frame(block); mem_heap_t* heap = NULL; rec_offs offsets_[REC_OFFS_NORMAL_SIZE]; rec_offs* offsets = offsets_; rec_offs_init(offsets_); ut_ad(dtuple_check_typed(tuple)); ut_ad(page_is_leaf(page)); rec = page_header_get_ptr(page, PAGE_LAST_INSERT); offsets = rec_get_offsets(rec, index, offsets, index->n_core_fields, dtuple_get_n_fields(tuple), &heap); ut_ad(rec); ut_ad(page_rec_is_user_rec(rec)); if (ut_pair_cmp(*ilow_matched_fields, *ilow_matched_bytes, *iup_matched_fields, *iup_matched_bytes) < 0) { up_match = low_match = *ilow_matched_fields; up_bytes = low_bytes = *ilow_matched_bytes; } else { up_match = low_match = *iup_matched_fields; up_bytes = low_bytes = *iup_matched_bytes; } if (cmp_dtuple_rec_with_match_bytes( tuple, rec, index, offsets, &low_match, &low_bytes) < 0) { goto exit_func; } if (!(next_rec = page_rec_get_next_const(rec))) { goto exit_func; } if (!page_rec_is_supremum(next_rec)) { offsets = rec_get_offsets(next_rec, index, offsets, index->n_core_fields, dtuple_get_n_fields(tuple), &heap); if (cmp_dtuple_rec_with_match_bytes( tuple, next_rec, index, offsets, &up_match, &up_bytes) >= 0) { goto exit_func; } *iup_matched_fields = up_match; *iup_matched_bytes = up_bytes; } page_cur_position(rec, block, cursor); *ilow_matched_fields = low_match; *ilow_matched_bytes = low_bytes; #ifdef UNIV_SEARCH_PERF_STAT page_cur_short_succ++; #endif success = TRUE; exit_func: if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return(success); } #endif /* BTR_CUR_HASH_ADAPT */ #ifdef PAGE_CUR_LE_OR_EXTENDS /****************************************************************//** Checks if the nth field in a record is a character type field which extends the nth field in tuple, i.e., the field is longer or equal in length and has common first characters. @return TRUE if rec field extends tuple field */ static ibool page_cur_rec_field_extends( /*=======================*/ const dtuple_t* tuple, /*!< in: data tuple */ const rec_t* rec, /*!< in: record */ const rec_offs* offsets,/*!< in: array returned by rec_get_offsets() */ ulint n) /*!< in: compare nth field */ { const dtype_t* type; const dfield_t* dfield; const byte* rec_f; ulint rec_f_len; ut_ad(rec_offs_validate(rec, NULL, offsets)); dfield = dtuple_get_nth_field(tuple, n); type = dfield_get_type(dfield); rec_f = rec_get_nth_field(rec, offsets, n, &rec_f_len); if (type->mtype == DATA_VARCHAR || type->mtype == DATA_CHAR || type->mtype == DATA_FIXBINARY || type->mtype == DATA_BINARY || type->mtype == DATA_BLOB || DATA_GEOMETRY_MTYPE(type->mtype) || type->mtype == DATA_VARMYSQL || type->mtype == DATA_MYSQL) { if (dfield_get_len(dfield) != UNIV_SQL_NULL && rec_f_len != UNIV_SQL_NULL && rec_f_len >= dfield_get_len(dfield) && !cmp_data_data(type->mtype, type->prtype, dfield_get_data(dfield), dfield_get_len(dfield), rec_f, dfield_get_len(dfield))) { return(TRUE); } } return(FALSE); } #endif /* PAGE_CUR_LE_OR_EXTENDS */ /****************************************************************//** Searches the right position for a page cursor. */ bool page_cur_search_with_match( /*=======================*/ const dtuple_t* tuple, /*!< in: data tuple */ page_cur_mode_t mode, /*!< in: PAGE_CUR_L, PAGE_CUR_LE, PAGE_CUR_G, or PAGE_CUR_GE */ ulint* iup_matched_fields, /*!< in/out: already matched fields in upper limit record */ ulint* ilow_matched_fields, /*!< in/out: already matched fields in lower limit record */ page_cur_t* cursor, /*!< out: page cursor */ rtr_info_t* rtr_info)/*!< in/out: rtree search stack */ { ulint up; ulint low; ulint mid; const page_t* page; const rec_t* up_rec; const rec_t* low_rec; const rec_t* mid_rec; ulint up_matched_fields; ulint low_matched_fields; ulint cur_matched_fields; int cmp; const dict_index_t* const index = cursor->index; const buf_block_t* const block = cursor->block; #ifdef UNIV_ZIP_DEBUG const page_zip_des_t* page_zip = buf_block_get_page_zip(block); #endif /* UNIV_ZIP_DEBUG */ mem_heap_t* heap = NULL; rec_offs offsets_[REC_OFFS_NORMAL_SIZE]; rec_offs* offsets = offsets_; rec_offs_init(offsets_); ut_ad(dtuple_validate(tuple)); #ifdef UNIV_DEBUG # ifdef PAGE_CUR_DBG if (mode != PAGE_CUR_DBG) # endif /* PAGE_CUR_DBG */ # ifdef PAGE_CUR_LE_OR_EXTENDS if (mode != PAGE_CUR_LE_OR_EXTENDS) # endif /* PAGE_CUR_LE_OR_EXTENDS */ ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE || mode == PAGE_CUR_G || mode == PAGE_CUR_GE || dict_index_is_spatial(index)); #endif /* UNIV_DEBUG */ page = buf_block_get_frame(block); #ifdef UNIV_ZIP_DEBUG ut_a(!page_zip || page_zip_validate(page_zip, page, index)); #endif /* UNIV_ZIP_DEBUG */ ut_d(page_check_dir(page)); const ulint n_core = page_is_leaf(page) ? index->n_core_fields : 0; #ifdef BTR_CUR_HASH_ADAPT if (n_core && page_get_direction(page) == PAGE_RIGHT && page_header_get_offs(page, PAGE_LAST_INSERT) && mode == PAGE_CUR_LE && !index->is_spatial() && page_header_get_field(page, PAGE_N_DIRECTION) > 3 && page_cur_try_search_shortcut( block, index, tuple, iup_matched_fields, ilow_matched_fields, cursor)) { return false; } # ifdef PAGE_CUR_DBG if (mode == PAGE_CUR_DBG) { mode = PAGE_CUR_LE; } # endif #endif /* BTR_CUR_HASH_ADAPT */ /* If the mode is for R-tree indexes, use the special MBR related compare functions */ if (index->is_spatial() && mode > PAGE_CUR_LE) { /* For leaf level insert, we still use the traditional compare function for now */ if (mode == PAGE_CUR_RTREE_INSERT && n_core) { mode = PAGE_CUR_LE; } else { return rtr_cur_search_with_match( block, (dict_index_t*)index, tuple, mode, cursor, rtr_info); } } /* The following flag does not work for non-latin1 char sets because cmp_full_field does not tell how many bytes matched */ #ifdef PAGE_CUR_LE_OR_EXTENDS ut_a(mode != PAGE_CUR_LE_OR_EXTENDS); #endif /* PAGE_CUR_LE_OR_EXTENDS */ /* If mode PAGE_CUR_G is specified, we are trying to position the cursor to answer a query of the form "tuple < X", where tuple is the input parameter, and X denotes an arbitrary physical record on the page. We want to position the cursor on the first X which satisfies the condition. */ up_matched_fields = *iup_matched_fields; low_matched_fields = *ilow_matched_fields; /* Perform binary search. First the search is done through the page directory, after that as a linear search in the list of records owned by the upper limit directory slot. */ low = 0; up = ulint(page_dir_get_n_slots(page)) - 1; /* Perform binary search until the lower and upper limit directory slots come to the distance 1 of each other */ while (up - low > 1) { mid = (low + up) / 2; const page_dir_slot_t* slot = page_dir_get_nth_slot(page, mid); if (UNIV_UNLIKELY(!(mid_rec = page_dir_slot_get_rec_validate(slot)))) { goto corrupted; } cur_matched_fields = std::min(low_matched_fields, up_matched_fields); offsets = offsets_; offsets = rec_get_offsets( mid_rec, index, offsets, n_core, dtuple_get_n_fields_cmp(tuple), &heap); cmp = cmp_dtuple_rec_with_match( tuple, mid_rec, index, offsets, &cur_matched_fields); if (cmp > 0) { low_slot_match: low = mid; low_matched_fields = cur_matched_fields; } else if (cmp) { #ifdef PAGE_CUR_LE_OR_EXTENDS if (mode == PAGE_CUR_LE_OR_EXTENDS && page_cur_rec_field_extends( tuple, mid_rec, offsets, cur_matched_fields)) { goto low_slot_match; } #endif /* PAGE_CUR_LE_OR_EXTENDS */ up_slot_match: up = mid; up_matched_fields = cur_matched_fields; } else if (mode == PAGE_CUR_G || mode == PAGE_CUR_LE #ifdef PAGE_CUR_LE_OR_EXTENDS || mode == PAGE_CUR_LE_OR_EXTENDS #endif /* PAGE_CUR_LE_OR_EXTENDS */ ) { goto low_slot_match; } else { goto up_slot_match; } } low_rec = page_dir_slot_get_rec_validate( page_dir_get_nth_slot(page, low)); up_rec = page_dir_slot_get_rec_validate( page_dir_get_nth_slot(page, up)); if (UNIV_UNLIKELY(!low_rec || !up_rec)) { corrupted: if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return true; } /* Perform linear search until the upper and lower records come to distance 1 of each other. */ for (;;) { if (const rec_t* next = page_rec_get_next_const(low_rec)) { if (next == up_rec) { break; } mid_rec = next; } else { goto corrupted; } cur_matched_fields = std::min(low_matched_fields, up_matched_fields); offsets = offsets_; offsets = rec_get_offsets( mid_rec, index, offsets, n_core, dtuple_get_n_fields_cmp(tuple), &heap); cmp = cmp_dtuple_rec_with_match( tuple, mid_rec, index, offsets, &cur_matched_fields); if (cmp > 0) { low_rec_match: low_rec = mid_rec; low_matched_fields = cur_matched_fields; } else if (cmp) { #ifdef PAGE_CUR_LE_OR_EXTENDS if (mode == PAGE_CUR_LE_OR_EXTENDS && page_cur_rec_field_extends( tuple, mid_rec, offsets, cur_matched_fields)) { goto low_rec_match; } #endif /* PAGE_CUR_LE_OR_EXTENDS */ up_rec_match: up_rec = mid_rec; up_matched_fields = cur_matched_fields; } else if (mode == PAGE_CUR_G || mode == PAGE_CUR_LE #ifdef PAGE_CUR_LE_OR_EXTENDS || mode == PAGE_CUR_LE_OR_EXTENDS #endif /* PAGE_CUR_LE_OR_EXTENDS */ ) { if (!cmp && !cur_matched_fields) { #ifdef UNIV_DEBUG mtr_t mtr; mtr_start(&mtr); /* We got a match, but cur_matched_fields is 0, it must have REC_INFO_MIN_REC_FLAG */ ulint rec_info = rec_get_info_bits(mid_rec, rec_offs_comp(offsets)); ut_ad(rec_info & REC_INFO_MIN_REC_FLAG); ut_ad(!page_has_prev(page)); mtr_commit(&mtr); #endif cur_matched_fields = dtuple_get_n_fields_cmp(tuple); } goto low_rec_match; } else { goto up_rec_match; } } if (mode <= PAGE_CUR_GE) { page_cur_position(up_rec, block, cursor); } else { page_cur_position(low_rec, block, cursor); } *iup_matched_fields = up_matched_fields; *ilow_matched_fields = low_matched_fields; if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return false; } #ifdef BTR_CUR_HASH_ADAPT /** Search the right position for a page cursor. @param[in] block buffer block @param[in] index index tree @param[in] tuple key to be searched for @param[in] mode search mode @param[in,out] iup_matched_fields already matched fields in the upper limit record @param[in,out] iup_matched_bytes already matched bytes in the first partially matched field in the upper limit record @param[in,out] ilow_matched_fields already matched fields in the lower limit record @param[in,out] ilow_matched_bytes already matched bytes in the first partially matched field in the lower limit record @param[out] cursor page cursor */ bool page_cur_search_with_match_bytes( const dtuple_t* tuple, page_cur_mode_t mode, ulint* iup_matched_fields, ulint* iup_matched_bytes, ulint* ilow_matched_fields, ulint* ilow_matched_bytes, page_cur_t* cursor) { ulint up; ulint low; const page_t* page; const rec_t* up_rec; const rec_t* low_rec; const rec_t* mid_rec; ulint up_matched_fields; ulint up_matched_bytes; ulint low_matched_fields; ulint low_matched_bytes; ulint cur_matched_fields; ulint cur_matched_bytes; int cmp; const dict_index_t* const index = cursor->index; const buf_block_t* const block = cursor->block; #ifdef UNIV_ZIP_DEBUG const page_zip_des_t* page_zip = buf_block_get_page_zip(block); #endif /* UNIV_ZIP_DEBUG */ mem_heap_t* heap = NULL; rec_offs offsets_[REC_OFFS_NORMAL_SIZE]; rec_offs* offsets = offsets_; rec_offs_init(offsets_); ut_ad(dtuple_validate(tuple)); ut_ad(!(tuple->info_bits & REC_INFO_MIN_REC_FLAG)); #ifdef UNIV_DEBUG # ifdef PAGE_CUR_DBG if (mode != PAGE_CUR_DBG) # endif /* PAGE_CUR_DBG */ # ifdef PAGE_CUR_LE_OR_EXTENDS if (mode != PAGE_CUR_LE_OR_EXTENDS) # endif /* PAGE_CUR_LE_OR_EXTENDS */ ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE || mode == PAGE_CUR_G || mode == PAGE_CUR_GE); #endif /* UNIV_DEBUG */ page = buf_block_get_frame(block); #ifdef UNIV_ZIP_DEBUG ut_a(!page_zip || page_zip_validate(page_zip, page, index)); #endif /* UNIV_ZIP_DEBUG */ ut_d(page_check_dir(page)); #ifdef BTR_CUR_HASH_ADAPT if (page_is_leaf(page) && page_get_direction(page) == PAGE_RIGHT && page_header_get_offs(page, PAGE_LAST_INSERT) && mode == PAGE_CUR_LE && page_header_get_field(page, PAGE_N_DIRECTION) > 3 && page_cur_try_search_shortcut_bytes( block, index, tuple, iup_matched_fields, iup_matched_bytes, ilow_matched_fields, ilow_matched_bytes, cursor)) { return false; } # ifdef PAGE_CUR_DBG if (mode == PAGE_CUR_DBG) { mode = PAGE_CUR_LE; } # endif #endif /* BTR_CUR_HASH_ADAPT */ /* The following flag does not work for non-latin1 char sets because cmp_full_field does not tell how many bytes matched */ #ifdef PAGE_CUR_LE_OR_EXTENDS ut_a(mode != PAGE_CUR_LE_OR_EXTENDS); #endif /* PAGE_CUR_LE_OR_EXTENDS */ /* If mode PAGE_CUR_G is specified, we are trying to position the cursor to answer a query of the form "tuple < X", where tuple is the input parameter, and X denotes an arbitrary physical record on the page. We want to position the cursor on the first X which satisfies the condition. */ up_matched_fields = *iup_matched_fields; up_matched_bytes = *iup_matched_bytes; low_matched_fields = *ilow_matched_fields; low_matched_bytes = *ilow_matched_bytes; /* Perform binary search. First the search is done through the page directory, after that as a linear search in the list of records owned by the upper limit directory slot. */ low = 0; up = ulint(page_dir_get_n_slots(page)) - 1; /* Perform binary search until the lower and upper limit directory slots come to the distance 1 of each other */ const ulint n_core = page_is_leaf(page) ? index->n_core_fields : 0; while (up - low > 1) { const ulint mid = (low + up) / 2; mid_rec = page_dir_slot_get_rec_validate( page_dir_get_nth_slot(page, mid)); if (UNIV_UNLIKELY(!mid_rec)) { goto corrupted; } ut_pair_min(&cur_matched_fields, &cur_matched_bytes, low_matched_fields, low_matched_bytes, up_matched_fields, up_matched_bytes); offsets = rec_get_offsets( mid_rec, index, offsets_, n_core, dtuple_get_n_fields_cmp(tuple), &heap); cmp = cmp_dtuple_rec_with_match_bytes( tuple, mid_rec, index, offsets, &cur_matched_fields, &cur_matched_bytes); if (cmp > 0) { low_slot_match: low = mid; low_matched_fields = cur_matched_fields; low_matched_bytes = cur_matched_bytes; } else if (cmp) { #ifdef PAGE_CUR_LE_OR_EXTENDS if (mode == PAGE_CUR_LE_OR_EXTENDS && page_cur_rec_field_extends( tuple, mid_rec, offsets, cur_matched_fields)) { goto low_slot_match; } #endif /* PAGE_CUR_LE_OR_EXTENDS */ up_slot_match: up = mid; up_matched_fields = cur_matched_fields; up_matched_bytes = cur_matched_bytes; } else if (mode == PAGE_CUR_G || mode == PAGE_CUR_LE #ifdef PAGE_CUR_LE_OR_EXTENDS || mode == PAGE_CUR_LE_OR_EXTENDS #endif /* PAGE_CUR_LE_OR_EXTENDS */ ) { goto low_slot_match; } else { goto up_slot_match; } } low_rec = page_dir_slot_get_rec_validate( page_dir_get_nth_slot(page, low)); up_rec = page_dir_slot_get_rec_validate( page_dir_get_nth_slot(page, up)); if (UNIV_UNLIKELY(!low_rec || !up_rec)) { corrupted: if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return true; } /* Perform linear search until the upper and lower records come to distance 1 of each other. */ for (;;) { if (const rec_t* next = page_rec_get_next_const(low_rec)) { if (next == up_rec) { break; } mid_rec = next; } else { goto corrupted; } ut_pair_min(&cur_matched_fields, &cur_matched_bytes, low_matched_fields, low_matched_bytes, up_matched_fields, up_matched_bytes); if (UNIV_UNLIKELY(rec_get_info_bits( mid_rec, dict_table_is_comp(index->table)) & REC_INFO_MIN_REC_FLAG)) { ut_ad(!page_has_prev(page_align(mid_rec))); ut_ad(!page_rec_is_leaf(mid_rec) || rec_is_metadata(mid_rec, *index)); cmp = 1; goto low_rec_match; } offsets = rec_get_offsets( mid_rec, index, offsets_, n_core, dtuple_get_n_fields_cmp(tuple), &heap); cmp = cmp_dtuple_rec_with_match_bytes( tuple, mid_rec, index, offsets, &cur_matched_fields, &cur_matched_bytes); if (cmp > 0) { low_rec_match: low_rec = mid_rec; low_matched_fields = cur_matched_fields; low_matched_bytes = cur_matched_bytes; } else if (cmp) { #ifdef PAGE_CUR_LE_OR_EXTENDS if (mode == PAGE_CUR_LE_OR_EXTENDS && page_cur_rec_field_extends( tuple, mid_rec, offsets, cur_matched_fields)) { goto low_rec_match; } #endif /* PAGE_CUR_LE_OR_EXTENDS */ up_rec_match: up_rec = mid_rec; up_matched_fields = cur_matched_fields; up_matched_bytes = cur_matched_bytes; } else if (mode == PAGE_CUR_G || mode == PAGE_CUR_LE #ifdef PAGE_CUR_LE_OR_EXTENDS || mode == PAGE_CUR_LE_OR_EXTENDS #endif /* PAGE_CUR_LE_OR_EXTENDS */ ) { goto low_rec_match; } else { goto up_rec_match; } } if (mode <= PAGE_CUR_GE) { page_cur_position(up_rec, block, cursor); } else { page_cur_position(low_rec, block, cursor); } *iup_matched_fields = up_matched_fields; *iup_matched_bytes = up_matched_bytes; *ilow_matched_fields = low_matched_fields; *ilow_matched_bytes = low_matched_bytes; if (UNIV_LIKELY_NULL(heap)) { mem_heap_free(heap); } return false; } #endif /* BTR_CUR_HASH_ADAPT */ /***********************************************************//** Positions a page cursor on a randomly chosen user record on a page. If there are no user records, sets the cursor on the infimum record. */ void page_cur_open_on_rnd_user_rec(page_cur_t *cursor) { if (const ulint n_recs= page_get_n_recs(cursor->block->page.frame)) if ((cursor->rec= page_rec_get_nth(cursor->block->page.frame, ut_rnd_interval(n_recs) + 1))) return; cursor->rec= page_get_infimum_rec(cursor->block->page.frame); } /** Set the number of owned records. @param[in,out] rec record in block.frame @param[in] n_owned number of records skipped in the sparse page directory @param[in] comp whether ROW_FORMAT is COMPACT or DYNAMIC */ static void page_rec_set_n_owned(rec_t *rec, ulint n_owned, bool comp) { rec-= comp ? REC_NEW_N_OWNED : REC_OLD_N_OWNED; *rec= static_cast((*rec & ~REC_N_OWNED_MASK) | (n_owned << REC_N_OWNED_SHIFT)); } /** Split a directory slot which owns too many records. @param[in,out] block index page @param[in,out] slot the slot that needs to be split */ static bool page_dir_split_slot(const buf_block_t &block, page_dir_slot_t *slot) { ut_ad(slot <= &block.page.frame[srv_page_size - PAGE_EMPTY_DIR_START]); slot= my_assume_aligned<2>(slot); const ulint n_owned= PAGE_DIR_SLOT_MAX_N_OWNED + 1; ut_ad(page_dir_slot_get_n_owned(slot) == n_owned); static_assert((PAGE_DIR_SLOT_MAX_N_OWNED + 1) / 2 >= PAGE_DIR_SLOT_MIN_N_OWNED, "compatibility"); /* Find a record approximately in the middle. */ const rec_t *rec= page_dir_slot_get_rec_validate(slot + PAGE_DIR_SLOT_SIZE); for (ulint i= n_owned / 2; i--; ) { if (UNIV_UNLIKELY(!rec)) return true; rec= page_rec_get_next_const(rec); } if (UNIV_UNLIKELY(!rec)) return true; /* Add a directory slot immediately below this one. */ constexpr uint16_t n_slots_f= PAGE_N_DIR_SLOTS + PAGE_HEADER; byte *n_slots_p= my_assume_aligned<2>(n_slots_f + block.page.frame); const uint16_t n_slots= mach_read_from_2(n_slots_p); page_dir_slot_t *last_slot= static_cast (block.page.frame + srv_page_size - (PAGE_DIR + PAGE_DIR_SLOT_SIZE) - n_slots * PAGE_DIR_SLOT_SIZE); if (UNIV_UNLIKELY(slot < last_slot)) return true; memmove_aligned<2>(last_slot, last_slot + PAGE_DIR_SLOT_SIZE, slot - last_slot); const ulint half_owned= n_owned / 2; mach_write_to_2(n_slots_p, n_slots + 1); mach_write_to_2(slot, rec - block.page.frame); const bool comp= page_is_comp(block.page.frame) != 0; page_rec_set_n_owned(page_dir_slot_get_rec(slot), half_owned, comp); page_rec_set_n_owned(page_dir_slot_get_rec(slot - PAGE_DIR_SLOT_SIZE), n_owned - half_owned, comp); return false; } /** Split a directory slot which owns too many records. @param[in,out] block index page (ROW_FORMAT=COMPRESSED) @param[in] s the slot that needs to be split @param[in,out] mtr mini-transaction */ static void page_zip_dir_split_slot(buf_block_t *block, ulint s, mtr_t* mtr) { ut_ad(block->page.zip.data); ut_ad(page_is_comp(block->page.frame)); ut_ad(s); page_dir_slot_t *slot= page_dir_get_nth_slot(block->page.frame, s); const ulint n_owned= PAGE_DIR_SLOT_MAX_N_OWNED + 1; ut_ad(page_dir_slot_get_n_owned(slot) == n_owned); static_assert((PAGE_DIR_SLOT_MAX_N_OWNED + 1) / 2 >= PAGE_DIR_SLOT_MIN_N_OWNED, "compatibility"); /* 1. We loop to find a record approximately in the middle of the records owned by the slot. */ const rec_t *rec= page_dir_slot_get_rec(slot + PAGE_DIR_SLOT_SIZE); /* We do not try to prevent crash on corruption here. For ROW_FORMAT=COMPRESSED pages, the next-record links should be validated in page_zip_decompress(). Corruption should only be possible here if the buffer pool was corrupted later. */ for (ulint i= n_owned / 2; i--; ) rec= page_rec_get_next_const(rec); /* Add a directory slot immediately below this one. */ constexpr uint16_t n_slots_f= PAGE_N_DIR_SLOTS + PAGE_HEADER; byte *n_slots_p= my_assume_aligned<2>(n_slots_f + block->page.frame); const uint16_t n_slots= mach_read_from_2(n_slots_p); page_dir_slot_t *last_slot= static_cast (block->page.frame + srv_page_size - (PAGE_DIR + PAGE_DIR_SLOT_SIZE) - n_slots * PAGE_DIR_SLOT_SIZE); memmove_aligned<2>(last_slot, last_slot + PAGE_DIR_SLOT_SIZE, slot - last_slot); const ulint half_owned= n_owned / 2; mtr->write<2>(*block, n_slots_p, 1U + n_slots); /* Log changes to the compressed page header and the dense page directory. */ memcpy_aligned<2>(&block->page.zip.data[n_slots_f], n_slots_p, 2); mach_write_to_2(slot, page_offset(rec)); page_rec_set_n_owned(block, page_dir_slot_get_rec(slot), half_owned, true, mtr); page_rec_set_n_owned(block, page_dir_slot_get_rec(slot - PAGE_DIR_SLOT_SIZE), n_owned - half_owned, true, mtr); } /** Try to balance an underfilled directory slot with an adjacent one, so that there are at least the minimum number of records owned by the slot; this may result in merging the two slots. @param[in,out] block ROW_FORMAT=COMPRESSED page @param[in] s the slot to be balanced @param[in,out] mtr mini-transaction */ static void page_zip_dir_balance_slot(buf_block_t *block, ulint s, mtr_t *mtr) { ut_ad(block->page.zip.data); ut_ad(page_is_comp(block->page.frame)); ut_ad(s > 0); const ulint n_slots = page_dir_get_n_slots(block->page.frame); if (UNIV_UNLIKELY(s + 1 == n_slots)) { /* The last directory slot cannot be balanced. */ return; } ut_ad(s < n_slots); page_dir_slot_t* slot = page_dir_get_nth_slot(block->page.frame, s); rec_t* const up_rec = const_cast (page_dir_slot_get_rec(slot - PAGE_DIR_SLOT_SIZE)); rec_t* const slot_rec = const_cast (page_dir_slot_get_rec(slot)); const ulint up_n_owned = rec_get_n_owned_new(up_rec); ut_ad(rec_get_n_owned_new(page_dir_slot_get_rec(slot)) == PAGE_DIR_SLOT_MIN_N_OWNED - 1); if (up_n_owned <= PAGE_DIR_SLOT_MIN_N_OWNED) { compile_time_assert(2 * PAGE_DIR_SLOT_MIN_N_OWNED - 1 <= PAGE_DIR_SLOT_MAX_N_OWNED); /* Merge the slots. */ page_rec_set_n_owned(block, slot_rec, 0, true, mtr); page_rec_set_n_owned(block, up_rec, up_n_owned + (PAGE_DIR_SLOT_MIN_N_OWNED - 1), true, mtr); /* Shift the slots */ page_dir_slot_t* last_slot = page_dir_get_nth_slot( block->page.frame, n_slots - 1); memmove_aligned<2>(last_slot + PAGE_DIR_SLOT_SIZE, last_slot, slot - last_slot); constexpr uint16_t n_slots_f = PAGE_N_DIR_SLOTS + PAGE_HEADER; byte *n_slots_p= my_assume_aligned<2> (n_slots_f + block->page.frame); mtr->write<2>(*block, n_slots_p, n_slots - 1); memcpy_aligned<2>(n_slots_f + block->page.zip.data, n_slots_p, 2); memset_aligned<2>(last_slot, 0, 2); return; } /* Transfer one record to the underfilled slot */ page_rec_set_n_owned(block, slot_rec, 0, true, mtr); const rec_t* new_rec = page_rec_get_next_low(slot_rec, TRUE); /* We do not try to prevent crash on corruption here. For ROW_FORMAT=COMPRESSED pages, the next-record links should be validated in page_zip_decompress(). Corruption should only be possible here if the buffer pool was corrupted later. */ page_rec_set_n_owned(block, const_cast(new_rec), PAGE_DIR_SLOT_MIN_N_OWNED, true, mtr); mach_write_to_2(slot, page_offset(new_rec)); page_rec_set_n_owned(up_rec, up_n_owned - 1, true); } /** Try to balance an underfilled directory slot with an adjacent one, so that there are at least the minimum number of records owned by the slot; this may result in merging the two slots. @param[in,out] block index page @param[in] s the slot to be balanced */ static void page_dir_balance_slot(const buf_block_t &block, ulint s) { const bool comp= page_is_comp(block.page.frame); ut_ad(!block.page.zip.data); ut_ad(s > 0); const ulint n_slots = page_dir_get_n_slots(block.page.frame); if (UNIV_UNLIKELY(s + 1 == n_slots)) { /* The last directory slot cannot be balanced. */ return; } ut_ad(s < n_slots); page_dir_slot_t* slot = page_dir_get_nth_slot(block.page.frame, s); rec_t* const up_rec = const_cast (page_dir_slot_get_rec(slot - PAGE_DIR_SLOT_SIZE)); rec_t* const slot_rec = const_cast (page_dir_slot_get_rec(slot)); const ulint up_n_owned = comp ? rec_get_n_owned_new(up_rec) : rec_get_n_owned_old(up_rec); ut_ad(page_dir_slot_get_n_owned(slot) == PAGE_DIR_SLOT_MIN_N_OWNED - 1); if (up_n_owned <= PAGE_DIR_SLOT_MIN_N_OWNED) { compile_time_assert(2 * PAGE_DIR_SLOT_MIN_N_OWNED - 1 <= PAGE_DIR_SLOT_MAX_N_OWNED); /* Merge the slots. */ page_rec_set_n_owned(slot_rec, 0, comp); page_rec_set_n_owned(up_rec, up_n_owned + (PAGE_DIR_SLOT_MIN_N_OWNED - 1), comp); /* Shift the slots */ page_dir_slot_t* last_slot = page_dir_get_nth_slot( block.page.frame, n_slots - 1); memmove_aligned<2>(last_slot + PAGE_DIR_SLOT_SIZE, last_slot, slot - last_slot); memset_aligned<2>(last_slot, 0, 2); constexpr uint16_t n_slots_f = PAGE_N_DIR_SLOTS + PAGE_HEADER; byte *n_slots_p= my_assume_aligned<2> (n_slots_f + block.page.frame); mach_write_to_2(n_slots_p, n_slots - 1); return; } /* Transfer one record to the underfilled slot */ const rec_t* new_rec; if (comp) { if (UNIV_UNLIKELY(!(new_rec = page_rec_get_next_low(slot_rec, true)))) { ut_ad("corrupted page" == 0); return; } page_rec_set_n_owned(slot_rec, 0, true); page_rec_set_n_owned(const_cast(new_rec), PAGE_DIR_SLOT_MIN_N_OWNED, true); page_rec_set_n_owned(up_rec, up_n_owned - 1, true); } else { if (UNIV_UNLIKELY(!(new_rec = page_rec_get_next_low(slot_rec, false)))) { ut_ad("corrupted page" == 0); return; } page_rec_set_n_owned(slot_rec, 0, false); page_rec_set_n_owned(const_cast(new_rec), PAGE_DIR_SLOT_MIN_N_OWNED, false); page_rec_set_n_owned(up_rec, up_n_owned - 1, false); } mach_write_to_2(slot, page_offset(new_rec)); } /** Allocate space for inserting an index record. @tparam compressed whether to update the ROW_FORMAT=COMPRESSED @param[in,out] block index page @param[in] need number of bytes needed @param[out] heap_no record heap number @return pointer to the start of the allocated buffer @retval NULL if allocation fails */ template static byte* page_mem_alloc_heap(buf_block_t *block, ulint need, ulint *heap_no) { ut_ad(!compressed || block->page.zip.data); byte *heap_top= my_assume_aligned<2>(PAGE_HEAP_TOP + PAGE_HEADER + block->page.frame); const uint16_t top= mach_read_from_2(heap_top); if (need > page_get_max_insert_size(block->page.frame, 1)) return NULL; byte *n_heap= my_assume_aligned<2> (PAGE_N_HEAP + PAGE_HEADER + block->page.frame); const uint16_t h= mach_read_from_2(n_heap); if (UNIV_UNLIKELY((h + 1) & 0x6000)) { /* At the minimum record size of 5+2 bytes, we can only reach this condition when using innodb_page_size=64k. */ ut_ad((h & 0x7fff) == 8191); ut_ad(srv_page_size == 65536); return NULL; } *heap_no= h & 0x7fff; ut_ad(*heap_no < srv_page_size / REC_N_NEW_EXTRA_BYTES); compile_time_assert(UNIV_PAGE_SIZE_MAX / REC_N_NEW_EXTRA_BYTES < 0x3fff); mach_write_to_2(heap_top, top + need); mach_write_to_2(n_heap, h + 1); if (compressed) { ut_ad(h & 0x8000); memcpy_aligned<4>(&block->page.zip.data[PAGE_HEAP_TOP + PAGE_HEADER], heap_top, 4); } return &block->page.frame[top]; } /** Write log for inserting a B-tree or R-tree record in ROW_FORMAT=REDUNDANT. @param block B-tree or R-tree page @param reuse false=allocate from PAGE_HEAP_TOP; true=reuse PAGE_FREE @param prev_rec byte offset of the predecessor of the record to insert, starting from PAGE_OLD_INFIMUM @param info_bits info_bits of the record @param n_fields_s number of fields << 1 | rec_get_1byte_offs_flag() @param hdr_c number of common record header bytes with prev_rec @param data_c number of common data bytes with prev_rec @param hdr record header bytes to copy to the log @param hdr_l number of copied record header bytes @param data record payload bytes to copy to the log @param data_l number of copied record data bytes */ inline void mtr_t::page_insert(const buf_block_t &block, bool reuse, ulint prev_rec, byte info_bits, ulint n_fields_s, size_t hdr_c, size_t data_c, const byte *hdr, size_t hdr_l, const byte *data, size_t data_l) { ut_ad(!block.page.zip.data); ut_ad(m_log_mode == MTR_LOG_ALL); ut_d(ulint n_slots= page_dir_get_n_slots(block.page.frame)); ut_ad(n_slots >= 2); ut_d(const byte *page_end= page_dir_get_nth_slot(block.page.frame, n_slots - 1)); ut_ad(&block.page.frame[prev_rec + PAGE_OLD_INFIMUM] <= page_end); ut_ad(block.page.frame + page_header_get_offs(block.page.frame, PAGE_HEAP_TOP) <= page_end); ut_ad(fil_page_index_page_check(block.page.frame)); ut_ad(!(~(REC_INFO_MIN_REC_FLAG | REC_INFO_DELETED_FLAG) & info_bits)); ut_ad(n_fields_s >= 2); ut_ad((n_fields_s >> 1) <= REC_MAX_N_FIELDS); ut_ad(data_l + data_c <= REDUNDANT_REC_MAX_DATA_SIZE); set_modified(block); static_assert(REC_INFO_MIN_REC_FLAG == 0x10, "compatibility"); static_assert(REC_INFO_DELETED_FLAG == 0x20, "compatibility"); n_fields_s= (n_fields_s - 2) << 2 | info_bits >> 4; size_t len= prev_rec < MIN_2BYTE ? 2 : prev_rec < MIN_3BYTE ? 3 : 4; static_assert((REC_MAX_N_FIELDS << 1 | 1) <= MIN_3BYTE, "compatibility"); len+= n_fields_s < MIN_2BYTE ? 1 : 2; len+= hdr_c < MIN_2BYTE ? 1 : 2; static_assert(REDUNDANT_REC_MAX_DATA_SIZE <= MIN_3BYTE, "compatibility"); len+= data_c < MIN_2BYTE ? 1 : 2; len+= hdr_l + data_l; const bool small= len < mtr_buf_t::MAX_DATA_SIZE - (1 + 3 + 3 + 5 + 5); byte *l= log_write(block.page.id(), &block.page, len, small); if (UNIV_LIKELY(small)) { ut_d(const byte * const end = l + len); *l++= reuse ? INSERT_REUSE_REDUNDANT : INSERT_HEAP_REDUNDANT; l= mlog_encode_varint(l, prev_rec); l= mlog_encode_varint(l, n_fields_s); l= mlog_encode_varint(l, hdr_c); l= mlog_encode_varint(l, data_c); ::memcpy(l, hdr, hdr_l); l+= hdr_l; ::memcpy(l, data, data_l); l+= data_l; ut_ad(end == l); m_log.close(l); } else { m_log.close(l); l= m_log.open(len - hdr_l - data_l); ut_d(const byte * const end = l + len - hdr_l - data_l); *l++= reuse ? INSERT_REUSE_REDUNDANT : INSERT_HEAP_REDUNDANT; l= mlog_encode_varint(l, prev_rec); l= mlog_encode_varint(l, n_fields_s); l= mlog_encode_varint(l, hdr_c); l= mlog_encode_varint(l, data_c); ut_ad(end == l); m_log.close(l); m_log.push(hdr, static_cast(hdr_l)); m_log.push(data, static_cast(data_l)); } m_last_offset= FIL_PAGE_TYPE; } /** Write log for inserting a B-tree or R-tree record in ROW_FORMAT=COMPACT or ROW_FORMAT=DYNAMIC. @param block B-tree or R-tree page @param reuse false=allocate from PAGE_HEAP_TOP; true=reuse PAGE_FREE @param prev_rec byte offset of the predecessor of the record to insert, starting from PAGE_NEW_INFIMUM @param info_status rec_get_info_and_status_bits() @param shift unless !reuse: number of bytes the PAGE_FREE is moving @param hdr_c number of common record header bytes with prev_rec @param data_c number of common data bytes with prev_rec @param hdr record header bytes to copy to the log @param hdr_l number of copied record header bytes @param data record payload bytes to copy to the log @param data_l number of copied record data bytes */ inline void mtr_t::page_insert(const buf_block_t &block, bool reuse, ulint prev_rec, byte info_status, ssize_t shift, size_t hdr_c, size_t data_c, const byte *hdr, size_t hdr_l, const byte *data, size_t data_l) { ut_ad(!block.page.zip.data); ut_ad(m_log_mode == MTR_LOG_ALL); ut_d(ulint n_slots= page_dir_get_n_slots(block.page.frame)); ut_ad(n_slots >= 2); ut_d(const byte *page_end= page_dir_get_nth_slot(block.page.frame, n_slots - 1)); ut_ad(&block.page.frame[prev_rec + PAGE_NEW_INFIMUM] <= page_end); ut_ad(block.page.frame + page_header_get_offs(block.page.frame, PAGE_HEAP_TOP) <= page_end); ut_ad(fil_page_index_page_check(block.page.frame)); ut_ad(hdr_l + hdr_c + data_l + data_c <= static_cast (page_end - &block.page.frame[PAGE_NEW_SUPREMUM_END])); ut_ad(reuse || shift == 0); #ifdef UNIV_DEBUG switch (~(REC_INFO_MIN_REC_FLAG | REC_INFO_DELETED_FLAG) & info_status) { default: ut_ad(0); break; case REC_STATUS_NODE_PTR: ut_ad(!page_is_leaf(block.page.frame)); break; case REC_STATUS_INSTANT: case REC_STATUS_ORDINARY: ut_ad(page_is_leaf(block.page.frame)); } #endif set_modified(block); static_assert(REC_INFO_MIN_REC_FLAG == 0x10, "compatibility"); static_assert(REC_INFO_DELETED_FLAG == 0x20, "compatibility"); static_assert(REC_STATUS_INSTANT == 4, "compatibility"); const size_t enc_hdr_l= hdr_l << 3 | (info_status & REC_STATUS_INSTANT) | info_status >> 4; size_t len= prev_rec < MIN_2BYTE ? 2 : prev_rec < MIN_3BYTE ? 3 : 4; static_assert(REC_MAX_N_FIELDS * 2 < MIN_3BYTE, "compatibility"); if (reuse) { if (shift < 0) shift= -shift << 1 | 1; else shift<<= 1; len+= static_cast(shift) < MIN_2BYTE ? 1 : static_cast(shift) < MIN_3BYTE ? 2 : 3; } ut_ad(hdr_c + hdr_l <= REC_MAX_N_FIELDS * 2); len+= hdr_c < MIN_2BYTE ? 1 : 2; len+= enc_hdr_l < MIN_2BYTE ? 1 : enc_hdr_l < MIN_3BYTE ? 2 : 3; len+= data_c < MIN_2BYTE ? 1 : data_c < MIN_3BYTE ? 2 : 3; len+= hdr_l + data_l; const bool small= len < mtr_buf_t::MAX_DATA_SIZE - (1 + 3 + 3 + 5 + 5); byte *l= log_write(block.page.id(), &block.page, len, small); if (UNIV_LIKELY(small)) { ut_d(const byte * const end = l + len); *l++= reuse ? INSERT_REUSE_DYNAMIC : INSERT_HEAP_DYNAMIC; l= mlog_encode_varint(l, prev_rec); if (reuse) l= mlog_encode_varint(l, shift); l= mlog_encode_varint(l, enc_hdr_l); l= mlog_encode_varint(l, hdr_c); l= mlog_encode_varint(l, data_c); ::memcpy(l, hdr, hdr_l); l+= hdr_l; ::memcpy(l, data, data_l); l+= data_l; ut_ad(end == l); m_log.close(l); } else { m_log.close(l); l= m_log.open(len - hdr_l - data_l); ut_d(const byte * const end = l + len - hdr_l - data_l); *l++= reuse ? INSERT_REUSE_DYNAMIC : INSERT_HEAP_DYNAMIC; l= mlog_encode_varint(l, prev_rec); if (reuse) l= mlog_encode_varint(l, shift); l= mlog_encode_varint(l, enc_hdr_l); l= mlog_encode_varint(l, hdr_c); l= mlog_encode_varint(l, data_c); ut_ad(end == l); m_log.close(l); m_log.push(hdr, static_cast(hdr_l)); m_log.push(data, static_cast(data_l)); } m_last_offset= FIL_PAGE_TYPE; } /** Report page directory corruption. @param block index page @param index index tree */ ATTRIBUTE_COLD static void page_cur_directory_corrupted(const buf_block_t &block, const dict_index_t &index) { ib::error() << "Directory of " << block.page.id() << " of index " << index.name << " in table " << index.table->name << " is corrupted"; } /***********************************************************//** Inserts a record next to page cursor on an uncompressed page. @return pointer to record @retval nullptr if not enough space was available */ rec_t* page_cur_insert_rec_low( /*====================*/ const page_cur_t*cur, /*!< in: page cursor */ const rec_t* rec, /*!< in: record to insert after cur */ rec_offs* offsets,/*!< in/out: rec_get_offsets(rec, index) */ mtr_t* mtr) /*!< in/out: mini-transaction */ { buf_block_t *block= cur->block; dict_index_t * const index= cur->index; ut_ad(rec_offs_validate(rec, index, offsets)); ut_ad(rec_offs_n_fields(offsets) > 0); ut_ad(index->table->not_redundant() == !!page_is_comp(block->page.frame)); ut_ad(!!page_is_comp(block->page.frame) == !!rec_offs_comp(offsets)); ut_ad(fil_page_index_page_check(block->page.frame)); ut_ad(mach_read_from_8(PAGE_HEADER + PAGE_INDEX_ID + block->page.frame) == index->id || mtr->is_inside_ibuf()); ut_ad(page_dir_get_n_slots(block->page.frame) >= 2); ut_ad(!page_rec_is_supremum(cur->rec)); /* We should not write log for ROW_FORMAT=COMPRESSED pages here. */ ut_ad(!mtr->is_logged() || !(index->table->flags & DICT_TF_MASK_ZIP_SSIZE)); /* 1. Get the size of the physical record in the page */ const ulint rec_size= rec_offs_size(offsets); #ifdef HAVE_MEM_CHECK { const void *rec_start __attribute__((unused))= rec - rec_offs_extra_size(offsets); ulint extra_size __attribute__((unused))= rec_offs_extra_size(offsets) - (page_is_comp(block->page.frame) ? REC_N_NEW_EXTRA_BYTES : REC_N_OLD_EXTRA_BYTES); /* All data bytes of the record must be valid. */ MEM_CHECK_DEFINED(rec, rec_offs_data_size(offsets)); /* The variable-length header must be valid. */ MEM_CHECK_DEFINED(rec_start, extra_size); } #endif /* HAVE_MEM_CHECK */ /* 2. Try to find suitable space from page memory management */ bool reuse= false; ssize_t free_offset= 0; ulint heap_no; byte *insert_buf; const bool comp= page_is_comp(block->page.frame); const ulint extra_size= rec_offs_extra_size(offsets); if (rec_t* free_rec= page_header_get_ptr(block->page.frame, PAGE_FREE)) { /* Try to reuse the head of PAGE_FREE. */ rec_offs foffsets_[REC_OFFS_NORMAL_SIZE]; mem_heap_t *heap= nullptr; rec_offs_init(foffsets_); rec_offs *foffsets= rec_get_offsets(free_rec, index, foffsets_, page_is_leaf(block->page.frame) ? index->n_core_fields : 0, ULINT_UNDEFINED, &heap); const ulint fextra_size= rec_offs_extra_size(foffsets); insert_buf= free_rec - fextra_size; const bool too_small= (fextra_size + rec_offs_data_size(foffsets)) < rec_size; if (UNIV_LIKELY_NULL(heap)) mem_heap_free(heap); if (too_small) goto use_heap; byte *page_free= my_assume_aligned<2>(PAGE_FREE + PAGE_HEADER + block->page.frame); if (comp) { heap_no= rec_get_heap_no_new(free_rec); uint16_t next= mach_read_from_2(free_rec - REC_NEXT); mach_write_to_2(page_free, next ? static_cast(free_rec + next - block->page.frame) : 0); } else { heap_no= rec_get_heap_no_old(free_rec); memcpy(page_free, free_rec - REC_NEXT, 2); } static_assert(PAGE_GARBAGE == PAGE_FREE + 2, "compatibility"); byte *page_garbage= my_assume_aligned<2>(page_free + 2); ut_ad(mach_read_from_2(page_garbage) >= rec_size); mach_write_to_2(page_garbage, mach_read_from_2(page_garbage) - rec_size); reuse= true; free_offset= extra_size - fextra_size; } else { use_heap: insert_buf= page_mem_alloc_heap(block, rec_size, &heap_no); if (UNIV_UNLIKELY(!insert_buf)) return nullptr; } ut_ad(cur->rec != insert_buf + extra_size); rec_t *next_rec= block->page.frame + rec_get_next_offs(cur->rec, comp); ut_ad(next_rec != block->page.frame); /* Update page header fields */ byte *page_last_insert= my_assume_aligned<2>(PAGE_LAST_INSERT + PAGE_HEADER + block->page.frame); const uint16_t last_insert= mach_read_from_2(page_last_insert); ut_ad(!last_insert || !comp || rec_get_node_ptr_flag(block->page.frame + last_insert) == rec_get_node_ptr_flag(rec)); /* Write PAGE_LAST_INSERT */ mach_write_to_2(page_last_insert, page_offset(insert_buf + extra_size)); /* Update PAGE_DIRECTION_B, PAGE_N_DIRECTION if needed */ if (block->page.frame[FIL_PAGE_TYPE + 1] != byte(FIL_PAGE_RTREE)) { byte *dir= &block->page.frame[PAGE_DIRECTION_B + PAGE_HEADER]; byte *n= my_assume_aligned<2> (&block->page.frame[PAGE_N_DIRECTION + PAGE_HEADER]); if (UNIV_UNLIKELY(!last_insert)) { no_direction: *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_NO_DIRECTION); memset(n, 0, 2); } else if (block->page.frame + last_insert == cur->rec && (*dir & ((1U << 3) - 1)) != PAGE_LEFT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_RIGHT); inc_dir: mach_write_to_2(n, mach_read_from_2(n) + 1); } else if (next_rec == block->page.frame + last_insert && (*dir & ((1U << 3) - 1)) != PAGE_RIGHT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_LEFT); goto inc_dir; } else goto no_direction; } /* Update PAGE_N_RECS. */ byte *page_n_recs= my_assume_aligned<2>(PAGE_N_RECS + PAGE_HEADER + block->page.frame); mach_write_to_2(page_n_recs, mach_read_from_2(page_n_recs) + 1); /* Update the preceding record header, the 'owner' record and prepare the record to insert. */ rec_t *insert_rec= insert_buf + extra_size; const ulint data_size= rec_offs_data_size(offsets); memcpy(insert_buf, rec - extra_size, extra_size + data_size); size_t hdr_common= 0; ulint n_owned; const byte info_status= static_cast (rec_get_info_and_status_bits(rec, comp)); ut_ad(!(rec_get_info_bits(rec, comp) & ~(REC_INFO_DELETED_FLAG | REC_INFO_MIN_REC_FLAG))); if (comp) { #ifdef UNIV_DEBUG switch (rec_get_status(cur->rec)) { case REC_STATUS_ORDINARY: case REC_STATUS_NODE_PTR: case REC_STATUS_INSTANT: case REC_STATUS_INFIMUM: break; case REC_STATUS_SUPREMUM: ut_ad("wrong status on cur->rec" == 0); } switch (rec_get_status(rec)) { case REC_STATUS_NODE_PTR: ut_ad(!page_is_leaf(block->page.frame)); break; case REC_STATUS_INSTANT: ut_ad(index->is_instant()); ut_ad(page_is_leaf(block->page.frame)); if (!rec_is_metadata(rec, true)) break; ut_ad(cur->rec == &block->page.frame[PAGE_NEW_INFIMUM]); break; case REC_STATUS_ORDINARY: ut_ad(page_is_leaf(block->page.frame)); ut_ad(!(rec_get_info_bits(rec, true) & ~REC_INFO_DELETED_FLAG)); break; case REC_STATUS_INFIMUM: case REC_STATUS_SUPREMUM: ut_ad("wrong status on rec" == 0); } ut_ad(rec_get_status(next_rec) != REC_STATUS_INFIMUM); #endif rec_set_bit_field_1(insert_rec, 0, REC_NEW_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); insert_rec[-REC_NEW_STATUS]= rec[-REC_NEW_STATUS]; rec_set_bit_field_2(insert_rec, heap_no, REC_NEW_HEAP_NO, REC_HEAP_NO_MASK, REC_HEAP_NO_SHIFT); mach_write_to_2(insert_rec - REC_NEXT, static_cast(next_rec - insert_rec)); mach_write_to_2(cur->rec - REC_NEXT, static_cast(insert_rec - cur->rec)); while (!(n_owned= rec_get_n_owned_new(next_rec))) { next_rec= block->page.frame + rec_get_next_offs(next_rec, true); ut_ad(next_rec != block->page.frame); } rec_set_bit_field_1(next_rec, n_owned + 1, REC_NEW_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); if (!mtr->is_logged()) { mtr->set_modified(*block); goto copied; } const byte * const c_start= cur->rec - extra_size; if (extra_size > REC_N_NEW_EXTRA_BYTES && c_start >= &block->page.frame[PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES]) { /* Find common header bytes with the preceding record. */ const byte *r= rec - (REC_N_NEW_EXTRA_BYTES + 1); for (const byte *c= cur->rec - (REC_N_NEW_EXTRA_BYTES + 1); *r == *c && c-- != c_start; r--); hdr_common= static_cast((rec - (REC_N_NEW_EXTRA_BYTES + 1)) - r); ut_ad(hdr_common <= extra_size - REC_N_NEW_EXTRA_BYTES); } } else { #ifdef UNIV_DEBUG if (!page_is_leaf(block->page.frame)); else if (rec_is_metadata(rec, false)) { ut_ad(index->is_instant()); ut_ad(cur->rec == &block->page.frame[PAGE_OLD_INFIMUM]); } #endif rec_set_bit_field_1(insert_rec, 0, REC_OLD_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); rec_set_bit_field_2(insert_rec, heap_no, REC_OLD_HEAP_NO, REC_HEAP_NO_MASK, REC_HEAP_NO_SHIFT); memcpy(insert_rec - REC_NEXT, cur->rec - REC_NEXT, 2); mach_write_to_2(cur->rec - REC_NEXT, page_offset(insert_rec)); while (!(n_owned= rec_get_n_owned_old(next_rec))) { next_rec= block->page.frame + rec_get_next_offs(next_rec, false); ut_ad(next_rec != block->page.frame); } rec_set_bit_field_1(next_rec, n_owned + 1, REC_OLD_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); if (!mtr->is_logged()) { mtr->set_modified(*block); goto copied; } ut_ad(extra_size > REC_N_OLD_EXTRA_BYTES); const byte * const c_start= cur->rec - extra_size; if (c_start >= &block->page.frame[PAGE_OLD_SUPREMUM_END + REC_N_OLD_EXTRA_BYTES]) { /* Find common header bytes with the preceding record. */ const byte *r= rec - (REC_N_OLD_EXTRA_BYTES + 1); for (const byte *c= cur->rec - (REC_N_OLD_EXTRA_BYTES + 1); *r == *c && c-- != c_start; r--); hdr_common= static_cast((rec - (REC_N_OLD_EXTRA_BYTES + 1)) - r); ut_ad(hdr_common <= extra_size - REC_N_OLD_EXTRA_BYTES); } } /* Insert the record, possibly copying from the preceding record. */ ut_ad(mtr->is_logged()); { const byte *r= rec; const byte *c= cur->rec; const byte *c_end= c + data_size; if (page_rec_is_infimum(c) && data_size > 8) c_end= c + 8; static_assert(REC_N_OLD_EXTRA_BYTES == REC_N_NEW_EXTRA_BYTES + 1, ""); if (c <= insert_buf && c_end > insert_buf) c_end= insert_buf; else if (c_end < next_rec && c_end >= next_rec - REC_N_OLD_EXTRA_BYTES + comp) c_end= next_rec - REC_N_OLD_EXTRA_BYTES + comp; else c_end= std::min(c_end, block->page.frame + srv_page_size - PAGE_DIR - PAGE_DIR_SLOT_SIZE * page_dir_get_n_slots(block->page.frame)); size_t data_common; /* Copy common data bytes of the preceding record. */ for (; c != c_end && *r == *c; c++, r++); data_common= static_cast(r - rec); if (comp) mtr->page_insert(*block, reuse, cur->rec - block->page.frame - PAGE_NEW_INFIMUM, info_status, free_offset, hdr_common, data_common, insert_buf, extra_size - hdr_common - REC_N_NEW_EXTRA_BYTES, r, data_size - data_common); else mtr->page_insert(*block, reuse, cur->rec - block->page.frame - PAGE_OLD_INFIMUM, info_status, rec_get_n_fields_old(insert_rec) << 1 | rec_get_1byte_offs_flag(insert_rec), hdr_common, data_common, insert_buf, extra_size - hdr_common - REC_N_OLD_EXTRA_BYTES, r, data_size - data_common); } copied: ut_ad(!memcmp(insert_buf, rec - extra_size, extra_size - (comp ? REC_N_NEW_EXTRA_BYTES : REC_N_OLD_EXTRA_BYTES))); ut_ad(!memcmp(insert_rec, rec, data_size)); /* We have incremented the n_owned field of the owner record. If the number exceeds PAGE_DIR_SLOT_MAX_N_OWNED, we have to split the corresponding directory slot in two. */ if (UNIV_UNLIKELY(n_owned == PAGE_DIR_SLOT_MAX_N_OWNED)) { const ulint owner= page_dir_find_owner_slot(next_rec); if (UNIV_UNLIKELY(owner == ULINT_UNDEFINED)) { page_cur_directory_corrupted(*block, *index); return nullptr; } if (page_dir_split_slot(*block, page_dir_get_nth_slot(block->page.frame, owner))) return nullptr; } rec_offs_make_valid(insert_buf + extra_size, index, page_is_leaf(block->page.frame), offsets); return insert_buf + extra_size; } /** Add a slot to the dense page directory. @param[in,out] block ROW_FORMAT=COMPRESSED page @param[in] index the index that the page belongs to @param[in,out] mtr mini-transaction */ static inline void page_zip_dir_add_slot(buf_block_t *block, const dict_index_t *index, mtr_t *mtr) { page_zip_des_t *page_zip= &block->page.zip; ut_ad(page_is_comp(page_zip->data)); MEM_CHECK_DEFINED(page_zip->data, page_zip_get_size(page_zip)); /* Read the old n_dense (n_heap has already been incremented). */ ulint n_dense= page_dir_get_n_heap(page_zip->data) - (PAGE_HEAP_NO_USER_LOW + 1U); byte *dir= page_zip->data + page_zip_get_size(page_zip) - PAGE_ZIP_DIR_SLOT_SIZE * n_dense; byte *stored= dir; if (!page_is_leaf(page_zip->data)) { ut_ad(!page_zip->n_blobs); stored-= n_dense * REC_NODE_PTR_SIZE; } else if (index->is_clust()) { /* Move the BLOB pointer array backwards to make space for the columns DB_TRX_ID,DB_ROLL_PTR and the dense directory slot. */ stored-= n_dense * (DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN); byte *externs= stored - page_zip->n_blobs * BTR_EXTERN_FIELD_REF_SIZE; byte *dst= externs - PAGE_ZIP_CLUST_LEAF_SLOT_SIZE; ut_ad(!memcmp(dst, field_ref_zero, PAGE_ZIP_CLUST_LEAF_SLOT_SIZE)); if (const ulint len = ulint(stored - externs)) { memmove(dst, externs, len); mtr->memmove(*block, dst - page_zip->data, externs - page_zip->data, len); } } else { stored-= page_zip->n_blobs * BTR_EXTERN_FIELD_REF_SIZE; ut_ad(!memcmp(stored - PAGE_ZIP_DIR_SLOT_SIZE, field_ref_zero, PAGE_ZIP_DIR_SLOT_SIZE)); } /* Move the uncompressed area backwards to make space for one directory slot. */ if (const ulint len = ulint(dir - stored)) { byte* dst = stored - PAGE_ZIP_DIR_SLOT_SIZE; memmove(dst, stored, len); mtr->memmove(*block, dst - page_zip->data, stored - page_zip->data, len); } } /***********************************************************//** Inserts a record next to page cursor on a compressed and uncompressed page. IMPORTANT: The caller will have to update IBUF_BITMAP_FREE if this is a compressed leaf page in a secondary index. This has to be done either within the same mini-transaction, or by invoking ibuf_reset_free_bits() before mtr_commit(). @return pointer to inserted record @return nullptr on failure */ rec_t* page_cur_insert_rec_zip( /*====================*/ page_cur_t* cursor, /*!< in/out: page cursor, logical position unchanged */ const rec_t* rec, /*!< in: pointer to a physical record */ rec_offs* offsets,/*!< in/out: rec_get_offsets(rec, index) */ mtr_t* mtr) /*!< in/out: mini-transaction */ { page_zip_des_t * const page_zip= page_cur_get_page_zip(cursor); page_t * const page= cursor->block->page.frame; dict_index_t * const index = cursor->index; ut_ad(page_zip); ut_ad(rec_offs_validate(rec, index, offsets)); ut_ad(index->table->not_redundant()); ut_ad(page_is_comp(page)); ut_ad(rec_offs_comp(offsets)); ut_ad(fil_page_get_type(page) == FIL_PAGE_INDEX || fil_page_get_type(page) == FIL_PAGE_RTREE); ut_ad(mach_read_from_8(PAGE_HEADER + PAGE_INDEX_ID + page) == index->id || mtr->is_inside_ibuf()); ut_ad(!page_get_instant(page)); ut_ad(!page_cur_is_after_last(cursor)); #ifdef UNIV_ZIP_DEBUG ut_a(page_zip_validate(page_zip, page, index)); #endif /* UNIV_ZIP_DEBUG */ /* 1. Get the size of the physical record in the page */ const ulint rec_size= rec_offs_size(offsets); #ifdef HAVE_MEM_CHECK { const void *rec_start __attribute__((unused))= rec - rec_offs_extra_size(offsets); ulint extra_size __attribute__((unused))= rec_offs_extra_size(offsets) - REC_N_NEW_EXTRA_BYTES; /* All data bytes of the record must be valid. */ MEM_CHECK_DEFINED(rec, rec_offs_data_size(offsets)); /* The variable-length header must be valid. */ MEM_CHECK_DEFINED(rec_start, extra_size); } #endif /* HAVE_MEM_CHECK */ const bool reorg_before_insert= page_has_garbage(page) && rec_size > page_get_max_insert_size(page, 1) && rec_size <= page_get_max_insert_size_after_reorganize(page, 1); constexpr uint16_t page_free_f= PAGE_FREE + PAGE_HEADER; byte* const page_free = my_assume_aligned<4>(page_free_f + page); uint16_t free_rec= 0; /* 2. Try to find suitable space from page memory management */ ulint heap_no; byte *insert_buf; if (reorg_before_insert || !page_zip_available(page_zip, index->is_clust(), rec_size, 1)) { /* SET GLOBAL might be executed concurrently. Sample the value once. */ ulint level= page_zip_level; #ifdef UNIV_DEBUG const rec_t * const cursor_rec= page_cur_get_rec(cursor); #endif /* UNIV_DEBUG */ if (page_is_empty(page)) { ut_ad(page_cur_is_before_first(cursor)); /* This is an empty page. Recreate to remove the modification log. */ page_create_zip(cursor->block, index, page_header_get_field(page, PAGE_LEVEL), 0, mtr); ut_ad(!page_header_get_ptr(page, PAGE_FREE)); if (page_zip_available(page_zip, index->is_clust(), rec_size, 1)) goto use_heap; /* The cursor should remain on the page infimum. */ return nullptr; } if (page_zip->m_nonempty || page_has_garbage(page)) { ulint pos= page_rec_get_n_recs_before(cursor->rec); if (UNIV_UNLIKELY(pos == ULINT_UNDEFINED)) return nullptr; switch (page_zip_reorganize(cursor->block, index, level, mtr, true)) { case DB_FAIL: ut_ad(cursor->rec == cursor_rec); return nullptr; case DB_SUCCESS: break; default: return nullptr; } if (!pos) ut_ad(cursor->rec == page + PAGE_NEW_INFIMUM); else if (!(cursor->rec= page_rec_get_nth(page, pos))) { cursor->rec= page + PAGE_NEW_SUPREMUM; return nullptr; } ut_ad(!page_header_get_ptr(page, PAGE_FREE)); if (page_zip_available(page_zip, index->is_clust(), rec_size, 1)) goto use_heap; } /* Try compressing the whole page afterwards. */ const mtr_log_t log_mode= mtr->set_log_mode(MTR_LOG_NONE); rec_t *insert_rec= page_cur_insert_rec_low(cursor, rec, offsets, mtr); mtr->set_log_mode(log_mode); if (insert_rec) { ulint pos= page_rec_get_n_recs_before(insert_rec); if (UNIV_UNLIKELY(!pos || pos == ULINT_UNDEFINED)) return nullptr; /* We are writing entire page images to the log. Reduce the redo log volume by reorganizing the page at the same time. */ switch (page_zip_reorganize(cursor->block, index, level, mtr)) { case DB_SUCCESS: /* The page was reorganized: Seek to pos. */ if (pos <= 1) cursor->rec= page + PAGE_NEW_INFIMUM; else if (!(cursor->rec= page_rec_get_nth(page, pos - 1))) { cursor->rec= page + PAGE_NEW_INFIMUM; return nullptr; } insert_rec= page + rec_get_next_offs(cursor->rec, 1); rec_offs_make_valid(insert_rec, index, page_is_leaf(page), offsets); break; case DB_FAIL: /* Theoretically, we could try one last resort of page_zip_reorganize() followed by page_zip_available(), but that would be very unlikely to succeed. (If the full reorganized page failed to compress, why would it succeed to compress the page, plus log the insert of this record?) */ /* Out of space: restore the page */ if (!page_zip_decompress(page_zip, page, false)) ut_error; /* Memory corrupted? */ ut_ad(page_validate(page, index)); /* fall through */ default: insert_rec= nullptr; } } return insert_rec; } free_rec= mach_read_from_2(page_free); if (free_rec) { /* Try to allocate from the head of the free list. */ rec_offs foffsets_[REC_OFFS_NORMAL_SIZE]; mem_heap_t *heap= nullptr; rec_offs_init(foffsets_); rec_offs *foffsets= rec_get_offsets(page + free_rec, index, foffsets_, page_is_leaf(page) ? index->n_core_fields : 0, ULINT_UNDEFINED, &heap); insert_buf= page + free_rec - rec_offs_extra_size(foffsets); if (rec_offs_size(foffsets) < rec_size) { too_small: if (UNIV_LIKELY_NULL(heap)) mem_heap_free(heap); free_rec= 0; goto use_heap; } /* On compressed pages, do not relocate records from the free list. If extra_size would grow, use the heap. */ const ssize_t extra_size_diff= lint(rec_offs_extra_size(offsets) - rec_offs_extra_size(foffsets)); if (UNIV_UNLIKELY(extra_size_diff < 0)) { /* Add an offset to the extra_size. */ if (rec_offs_size(foffsets) < rec_size - ssize_t(extra_size_diff)) goto too_small; insert_buf-= extra_size_diff; } else if (UNIV_UNLIKELY(extra_size_diff)) /* Do not allow extra_size to grow */ goto too_small; byte *const free_rec_ptr= page + free_rec; heap_no= rec_get_heap_no_new(free_rec_ptr); int16_t next_free= mach_read_from_2(free_rec_ptr - REC_NEXT); /* With innodb_page_size=64k, int16_t would be unsafe to use here, but that cannot be used with ROW_FORMAT=COMPRESSED. */ static_assert(UNIV_ZIP_SIZE_SHIFT_MAX == 14, "compatibility"); if (next_free) { next_free= static_cast(next_free + free_rec); if (UNIV_UNLIKELY(int{PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES} > next_free || uint16_t(next_free) >= srv_page_size)) { if (UNIV_LIKELY_NULL(heap)) mem_heap_free(heap); return nullptr; } } byte *hdr= my_assume_aligned<4>(&page_zip->data[page_free_f]); mach_write_to_2(hdr, static_cast(next_free)); const byte *const garbage= my_assume_aligned<2>(page_free + 2); ut_ad(mach_read_from_2(garbage) >= rec_size); mach_write_to_2(my_assume_aligned<2>(hdr + 2), mach_read_from_2(garbage) - rec_size); static_assert(PAGE_GARBAGE == PAGE_FREE + 2, "compatibility"); mtr->memcpy(*cursor->block, page_free, hdr, 4); if (!page_is_leaf(page)) { /* Zero out the node pointer of free_rec, in case it will not be overwritten by insert_rec. */ ut_ad(rec_size > REC_NODE_PTR_SIZE); if (rec_offs_size(foffsets) > rec_size) memset(rec_get_end(free_rec_ptr, foffsets) - REC_NODE_PTR_SIZE, 0, REC_NODE_PTR_SIZE); } else if (index->is_clust()) { /* Zero out DB_TRX_ID,DB_ROLL_PTR in free_rec, in case they will not be overwritten by insert_rec. */ ulint len; ulint trx_id_offs= rec_get_nth_field_offs(foffsets, index->db_trx_id(), &len); ut_ad(len == DATA_TRX_ID_LEN); if (DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN + trx_id_offs + rec_offs_extra_size(foffsets) > rec_size) memset(free_rec_ptr + trx_id_offs, 0, DATA_TRX_ID_LEN + DATA_ROLL_PTR_LEN); ut_ad(free_rec_ptr + trx_id_offs + DATA_TRX_ID_LEN == rec_get_nth_field(free_rec_ptr, foffsets, index->db_roll_ptr(), &len)); ut_ad(len == DATA_ROLL_PTR_LEN); } if (UNIV_LIKELY_NULL(heap)) mem_heap_free(heap); } else { use_heap: ut_ad(!free_rec); insert_buf= page_mem_alloc_heap(cursor->block, rec_size, &heap_no); if (UNIV_UNLIKELY(!insert_buf)) return insert_buf; static_assert(PAGE_N_HEAP == PAGE_HEAP_TOP + 2, "compatibility"); mtr->memcpy(*cursor->block, PAGE_HEAP_TOP + PAGE_HEADER, 4); page_zip_dir_add_slot(cursor->block, index, mtr); } /* next record after current before the insertion */ const rec_t *next_rec = page_rec_get_next_low(cursor->rec, TRUE); if (UNIV_UNLIKELY(!next_rec || rec_get_status(next_rec) == REC_STATUS_INFIMUM || rec_get_status(cursor->rec) > REC_STATUS_INFIMUM)) return nullptr; /* 3. Create the record */ byte *insert_rec= rec_copy(insert_buf, rec, offsets); rec_offs_make_valid(insert_rec, index, page_is_leaf(page), offsets); /* 4. Insert the record in the linked list of records */ ut_ad(cursor->rec != insert_rec); ut_ad(rec_get_status(insert_rec) < REC_STATUS_INFIMUM); mach_write_to_2(insert_rec - REC_NEXT, static_cast (next_rec - insert_rec)); mach_write_to_2(cursor->rec - REC_NEXT, static_cast (insert_rec - cursor->rec)); byte *n_recs= my_assume_aligned<2>(PAGE_N_RECS + PAGE_HEADER + page); mtr->write<2>(*cursor->block, n_recs, 1U + mach_read_from_2(n_recs)); memcpy_aligned<2>(&page_zip->data[PAGE_N_RECS + PAGE_HEADER], n_recs, 2); /* 5. Set the n_owned field in the inserted record to zero, and set the heap_no field */ rec_set_bit_field_1(insert_rec, 0, REC_NEW_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); rec_set_bit_field_2(insert_rec, heap_no, REC_NEW_HEAP_NO, REC_HEAP_NO_MASK, REC_HEAP_NO_SHIFT); MEM_CHECK_DEFINED(rec_get_start(insert_rec, offsets), rec_offs_size(offsets)); /* 6. Update the last insertion info in page header */ byte *last_insert= my_assume_aligned<4>(PAGE_LAST_INSERT + PAGE_HEADER + page_zip->data); const uint16_t last_insert_rec= mach_read_from_2(last_insert); ut_ad(!last_insert_rec || rec_get_node_ptr_flag(page + last_insert_rec) == rec_get_node_ptr_flag(insert_rec)); mach_write_to_2(last_insert, page_offset(insert_rec)); if (!index->is_spatial()) { byte *dir= &page_zip->data[PAGE_HEADER + PAGE_DIRECTION_B]; ut_ad(!(*dir & ~((1U << 3) - 1))); byte *n= my_assume_aligned<2> (&page_zip->data[PAGE_HEADER + PAGE_N_DIRECTION]); if (UNIV_UNLIKELY(!last_insert_rec)) { no_direction: *dir= PAGE_NO_DIRECTION; memset(n, 0, 2); } else if (*dir != PAGE_LEFT && page + last_insert_rec == cursor->rec) { *dir= PAGE_RIGHT; inc_dir: mach_write_to_2(n, mach_read_from_2(n) + 1); } else if (*dir != PAGE_RIGHT && page_rec_get_next(insert_rec) == page + last_insert_rec) { *dir= PAGE_LEFT; goto inc_dir; } else goto no_direction; } /* Write the header fields in one record. */ mtr->memcpy(*cursor->block, my_assume_aligned<8>(PAGE_LAST_INSERT + PAGE_HEADER + page), my_assume_aligned<8>(PAGE_LAST_INSERT + PAGE_HEADER + page_zip->data), PAGE_N_RECS - PAGE_LAST_INSERT + 2); /* 7. It remains to update the owner record. */ ulint n_owned; while (!(n_owned= rec_get_n_owned_new(next_rec))) if (!(next_rec= page_rec_get_next_low(next_rec, true))) return nullptr; rec_set_bit_field_1(const_cast(next_rec), n_owned + 1, REC_NEW_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); page_zip_dir_insert(cursor, free_rec, insert_rec, mtr); /* 8. Now we have incremented the n_owned field of the owner record. If the number exceeds PAGE_DIR_SLOT_MAX_N_OWNED, we have to split the corresponding directory slot in two. */ if (UNIV_UNLIKELY(n_owned == PAGE_DIR_SLOT_MAX_N_OWNED)) { const ulint owner= page_dir_find_owner_slot(next_rec); if (UNIV_UNLIKELY(owner == ULINT_UNDEFINED)) { page_cur_directory_corrupted(*cursor->block, *index); return nullptr; } page_zip_dir_split_slot(cursor->block, owner, mtr); } page_zip_write_rec(cursor->block, insert_rec, index, offsets, 1, mtr); return insert_rec; } /** Prepend a record to the PAGE_FREE list, or shrink PAGE_HEAP_TOP. @param[in,out] block index page @param[in,out] rec record being deleted @param[in] data_size record payload size, in bytes @param[in] extra_size record header size, in bytes */ static void page_mem_free(const buf_block_t &block, rec_t *rec, size_t data_size, size_t extra_size) { ut_ad(page_align(rec) == block.page.frame); ut_ad(!block.page.zip.data); const rec_t *free= page_header_get_ptr(block.page.frame, PAGE_FREE); const uint16_t n_heap= uint16_t(page_header_get_field(block.page.frame, PAGE_N_HEAP) - 1); ut_ad(page_get_n_recs(block.page.frame) < (n_heap & 0x7fff)); const bool deleting_top= n_heap == ((n_heap & 0x8000) ? (rec_get_heap_no_new(rec) | 0x8000) : rec_get_heap_no_old(rec)); if (deleting_top) { byte *page_heap_top= my_assume_aligned<2>(PAGE_HEAP_TOP + PAGE_HEADER + block.page.frame); const uint16_t heap_top= mach_read_from_2(page_heap_top); const size_t extra_savings= heap_top - page_offset(rec + data_size); ut_ad(extra_savings < heap_top); /* When deleting the last record, do not add it to the PAGE_FREE list. Instead, decrement PAGE_HEAP_TOP and PAGE_N_HEAP. */ mach_write_to_2(page_heap_top, page_offset(rec - extra_size)); mach_write_to_2(my_assume_aligned<2>(page_heap_top + 2), n_heap); static_assert(PAGE_N_HEAP == PAGE_HEAP_TOP + 2, "compatibility"); if (extra_savings) { byte *page_garbage= my_assume_aligned<2>(PAGE_GARBAGE + PAGE_HEADER + block.page.frame); uint16_t garbage= mach_read_from_2(page_garbage); ut_ad(garbage >= extra_savings); mach_write_to_2(page_garbage, garbage - extra_savings); } } else { byte *page_free= my_assume_aligned<2>(PAGE_FREE + PAGE_HEADER + block.page.frame); byte *page_garbage= my_assume_aligned<2>(PAGE_GARBAGE + PAGE_HEADER + block.page.frame); mach_write_to_2(page_free, page_offset(rec)); mach_write_to_2(page_garbage, mach_read_from_2(page_garbage) + extra_size + data_size); } memset_aligned<2>(PAGE_LAST_INSERT + PAGE_HEADER + block.page.frame, 0, 2); byte *page_n_recs= my_assume_aligned<2>(PAGE_N_RECS + PAGE_HEADER + block.page.frame); mach_write_to_2(page_n_recs, mach_read_from_2(page_n_recs) - 1); const byte* const end= rec + data_size; if (!deleting_top) { uint16_t next= free ? ((n_heap & 0x8000) ? static_cast(free - rec) : static_cast(free - block.page.frame)) : uint16_t{0}; mach_write_to_2(rec - REC_NEXT, next); } else rec-= extra_size; memset(rec, 0, end - rec); } /***********************************************************//** Deletes a record at the page cursor. The cursor is moved to the next record after the deleted one. */ void page_cur_delete_rec( /*================*/ page_cur_t* cursor, /*!< in/out: a page cursor */ const rec_offs* offsets,/*!< in: rec_get_offsets( cursor->rec, index) */ mtr_t* mtr) /*!< in/out: mini-transaction */ { page_dir_slot_t* cur_dir_slot; rec_t* current_rec; rec_t* prev_rec = NULL; rec_t* next_rec; ulint cur_n_owned; rec_t* rec; /* page_zip_validate() will fail here when btr_cur_pessimistic_delete() invokes btr_set_min_rec_mark(). Then, both "page_zip" and "block->page.frame" would have the min-rec-mark set on the smallest user record, but "block->page.frame" would additionally have it set on the smallest-but-one record. Because sloppy page_zip_validate_low() only ignores min-rec-flag differences in the smallest user record, it cannot be used here either. */ current_rec = cursor->rec; const dict_index_t* const index = cursor->index; buf_block_t* const block = cursor->block; ut_ad(rec_offs_validate(current_rec, index, offsets)); ut_ad(!!page_is_comp(block->page.frame) == index->table->not_redundant()); ut_ad(fil_page_index_page_check(block->page.frame)); ut_ad(mach_read_from_8(PAGE_HEADER + PAGE_INDEX_ID + block->page.frame) == index->id || mtr->is_inside_ibuf()); ut_ad(mtr->is_named_space(index->table->space)); /* The record must not be the supremum or infimum record. */ ut_ad(page_rec_is_user_rec(current_rec)); if (page_get_n_recs(block->page.frame) == 1 && !rec_is_alter_metadata(current_rec, *index)) { /* Empty the page. */ ut_ad(page_is_leaf(block->page.frame)); /* Usually, this should be the root page, and the whole index tree should become empty. However, this could also be a call in btr_cur_pessimistic_update() to delete the only record in the page and to insert another one. */ ut_ad(page_rec_is_supremum(page_rec_get_next(cursor->rec))); page_cur_set_after_last(block, cursor); page_create_empty(page_cur_get_block(cursor), const_cast(index), mtr); return; } /* Save to local variables some data associated with current_rec */ ulint cur_slot_no = page_dir_find_owner_slot(current_rec); if (UNIV_UNLIKELY(!cur_slot_no || cur_slot_no == ULINT_UNDEFINED)) { /* Avoid crashing due to a corrupted page. */ page_cur_directory_corrupted(*block, *index); return; } cur_dir_slot = page_dir_get_nth_slot(block->page.frame, cur_slot_no); cur_n_owned = page_dir_slot_get_n_owned(cur_dir_slot); /* The page gets invalid for btr_pcur_restore_pos(). We avoid invoking buf_block_modify_clock_inc(block) because its consistency checks would fail for the dummy block that is being used during IMPORT TABLESPACE. */ block->modify_clock++; /* Find the next and the previous record. Note that the cursor is left at the next record. */ rec = const_cast (page_dir_slot_get_rec(cur_dir_slot + PAGE_DIR_SLOT_SIZE)); /* rec now points to the record of the previous directory slot. Look for the immediate predecessor of current_rec in a loop. */ while (current_rec != rec) { prev_rec = rec; if (!(rec = page_rec_get_next(rec))) { /* Avoid crashing due to a corrupted page. */ return; } } if (!(next_rec = page_cur_move_to_next(cursor))) { /* Avoid crashing due to a corrupted page. */ return; } /* Remove the record from the linked list of records */ /* If the deleted record is pointed to by a dir slot, update the record pointer in slot. In the following if-clause we assume that prev_rec is owned by the same slot, i.e., PAGE_DIR_SLOT_MIN_N_OWNED >= 2. */ /* Update the number of owned records of the slot */ compile_time_assert(PAGE_DIR_SLOT_MIN_N_OWNED >= 2); ut_ad(cur_n_owned > 1); rec_t* slot_rec = const_cast (page_dir_slot_get_rec(cur_dir_slot)); if (UNIV_LIKELY_NULL(block->page.zip.data)) { ut_ad(page_is_comp(block->page.frame)); if (current_rec == slot_rec) { page_zip_rec_set_owned(block, prev_rec, 1, mtr); page_zip_rec_set_owned(block, slot_rec, 0, mtr); slot_rec = prev_rec; mach_write_to_2(cur_dir_slot, page_offset(slot_rec)); } else if (cur_n_owned == 1 && !page_rec_is_supremum(slot_rec)) { page_zip_rec_set_owned(block, slot_rec, 0, mtr); } mach_write_to_2(prev_rec - REC_NEXT, static_cast (next_rec - prev_rec)); slot_rec[-REC_NEW_N_OWNED] = static_cast( (slot_rec[-REC_NEW_N_OWNED] & ~REC_N_OWNED_MASK) | (cur_n_owned - 1) << REC_N_OWNED_SHIFT); page_header_reset_last_insert(block, mtr); page_zip_dir_delete(block, rec, index, offsets, page_header_get_ptr(block->page.frame, PAGE_FREE), mtr); if (cur_n_owned <= PAGE_DIR_SLOT_MIN_N_OWNED) { page_zip_dir_balance_slot(block, cur_slot_no, mtr); } return; } if (current_rec == slot_rec) { slot_rec = prev_rec; mach_write_to_2(cur_dir_slot, page_offset(slot_rec)); } const size_t data_size = rec_offs_data_size(offsets); const size_t extra_size = rec_offs_extra_size(offsets); if (page_is_comp(block->page.frame)) { mtr->page_delete(*block, page_offset(prev_rec) - PAGE_NEW_INFIMUM, extra_size - REC_N_NEW_EXTRA_BYTES, data_size); mach_write_to_2(prev_rec - REC_NEXT, static_cast (next_rec - prev_rec)); slot_rec[-REC_NEW_N_OWNED] = static_cast( (slot_rec[-REC_NEW_N_OWNED] & ~REC_N_OWNED_MASK) | (cur_n_owned - 1) << REC_N_OWNED_SHIFT); } else { mtr->page_delete(*block, page_offset(prev_rec) - PAGE_OLD_INFIMUM); memcpy(prev_rec - REC_NEXT, current_rec - REC_NEXT, 2); slot_rec[-REC_OLD_N_OWNED] = static_cast( (slot_rec[-REC_OLD_N_OWNED] & ~REC_N_OWNED_MASK) | (cur_n_owned - 1) << REC_N_OWNED_SHIFT); } page_mem_free(*block, current_rec, data_size, extra_size); /* Now we have decremented the number of owned records of the slot. If the number drops below PAGE_DIR_SLOT_MIN_N_OWNED, we balance the slots. */ if (cur_n_owned <= PAGE_DIR_SLOT_MIN_N_OWNED) { page_dir_balance_slot(*block, cur_slot_no); } ut_ad(page_is_comp(block->page.frame) ? page_simple_validate_new(block->page.frame) : page_simple_validate_old(block->page.frame)); } /** Apply a INSERT_HEAP_REDUNDANT or INSERT_REUSE_REDUNDANT record that was written by page_cur_insert_rec_low() for a ROW_FORMAT=REDUNDANT page. @param block B-tree or R-tree page in ROW_FORMAT=COMPACT or DYNAMIC @param reuse false=allocate from PAGE_HEAP_TOP; true=reuse PAGE_FREE @param prev byte offset of the predecessor, relative to PAGE_OLD_INFIMUM @param enc_hdr encoded fixed-size header bits @param hdr_c number of common record header bytes with prev @param data_c number of common data bytes with prev @param data literal header and data bytes @param data_len length of the literal data, in bytes @return whether the operation failed (inconcistency was noticed) */ bool page_apply_insert_redundant(const buf_block_t &block, bool reuse, ulint prev, ulint enc_hdr, size_t hdr_c, size_t data_c, const void *data, size_t data_len) { page_t * const page= block.page.frame; const uint16_t n_slots= page_dir_get_n_slots(page); byte *page_n_heap= my_assume_aligned<2>(PAGE_N_HEAP + PAGE_HEADER + page); const uint16_t h= mach_read_from_2(page_n_heap); const page_id_t id(block.page.id()); if (UNIV_UNLIKELY(n_slots < 2 || h < n_slots || h < PAGE_HEAP_NO_USER_LOW || h >= srv_page_size / REC_N_OLD_EXTRA_BYTES || !fil_page_index_page_check(page) || page_get_page_no(page) != id.page_no() || mach_read_from_2(my_assume_aligned<2> (PAGE_OLD_SUPREMUM - REC_NEXT + page)))) { corrupted: ib::error() << (reuse ? "Not applying INSERT_REUSE_REDUNDANT" " due to corruption on " : "Not applying INSERT_HEAP_REDUNDANT" " due to corruption on ") << id; return true; } byte * const last_slot= page_dir_get_nth_slot(page, n_slots - 1); byte * const page_heap_top= my_assume_aligned<2> (PAGE_HEAP_TOP + PAGE_HEADER + page); const byte *const heap_bot= &page[PAGE_OLD_SUPREMUM_END]; byte *heap_top= page + mach_read_from_2(page_heap_top); if (UNIV_UNLIKELY(heap_bot > heap_top || heap_top > last_slot)) goto corrupted; if (UNIV_UNLIKELY(mach_read_from_2(last_slot) != PAGE_OLD_SUPREMUM)) goto corrupted; if (UNIV_UNLIKELY(mach_read_from_2(page_dir_get_nth_slot(page, 0)) != PAGE_OLD_INFIMUM)) goto corrupted; rec_t * const prev_rec= page + PAGE_OLD_INFIMUM + prev; if (!prev); else if (UNIV_UNLIKELY(heap_bot + (REC_N_OLD_EXTRA_BYTES + 1) > prev_rec || prev_rec > heap_top)) goto corrupted; const ulint pn_fields= rec_get_bit_field_2(prev_rec, REC_OLD_N_FIELDS, REC_OLD_N_FIELDS_MASK, REC_OLD_N_FIELDS_SHIFT); if (UNIV_UNLIKELY(pn_fields == 0 || pn_fields > REC_MAX_N_FIELDS)) goto corrupted; const ulint pextra_size= REC_N_OLD_EXTRA_BYTES + (rec_get_1byte_offs_flag(prev_rec) ? pn_fields : pn_fields * 2); if (prev_rec == &page[PAGE_OLD_INFIMUM]); else if (UNIV_UNLIKELY(prev_rec - pextra_size < heap_bot)) goto corrupted; if (UNIV_UNLIKELY(hdr_c && prev_rec - hdr_c < heap_bot)) goto corrupted; const ulint pdata_size= rec_get_data_size_old(prev_rec); if (UNIV_UNLIKELY(prev_rec + pdata_size > heap_top)) goto corrupted; rec_t * const next_rec= page + mach_read_from_2(prev_rec - REC_NEXT); if (next_rec == page + PAGE_OLD_SUPREMUM); else if (UNIV_UNLIKELY(heap_bot + REC_N_OLD_EXTRA_BYTES > next_rec || next_rec > heap_top)) goto corrupted; const bool is_short= (enc_hdr >> 2) & 1; const ulint n_fields= (enc_hdr >> 3) + 1; if (UNIV_UNLIKELY(n_fields > REC_MAX_N_FIELDS)) goto corrupted; const ulint extra_size= REC_N_OLD_EXTRA_BYTES + (is_short ? n_fields : n_fields * 2); hdr_c+= REC_N_OLD_EXTRA_BYTES; if (UNIV_UNLIKELY(hdr_c > extra_size)) goto corrupted; if (UNIV_UNLIKELY(extra_size - hdr_c > data_len)) goto corrupted; /* We buffer all changes to the record header locally, so that we will avoid modifying the page before all consistency checks have been fulfilled. */ alignas(2) byte insert_buf[REC_N_OLD_EXTRA_BYTES + REC_MAX_N_FIELDS * 2]; ulint n_owned; rec_t *owner_rec= next_rec; for (ulint ns= PAGE_DIR_SLOT_MAX_N_OWNED; !(n_owned= rec_get_n_owned_old(owner_rec)); ) { owner_rec= page + mach_read_from_2(owner_rec - REC_NEXT); if (owner_rec == &page[PAGE_OLD_SUPREMUM]); else if (UNIV_UNLIKELY(heap_bot + REC_N_OLD_EXTRA_BYTES > owner_rec || owner_rec > heap_top)) goto corrupted; if (!ns--) goto corrupted; /* Corrupted (cyclic?) next-record list */ } page_dir_slot_t *owner_slot= last_slot; if (n_owned > PAGE_DIR_SLOT_MAX_N_OWNED) goto corrupted; else { mach_write_to_2(insert_buf, owner_rec - page); static_assert(PAGE_DIR_SLOT_SIZE == 2, "compatibility"); const page_dir_slot_t * const first_slot= page_dir_get_nth_slot(page, 0); while (memcmp_aligned<2>(owner_slot, insert_buf, 2)) if ((owner_slot+= 2) == first_slot) goto corrupted; } memcpy(insert_buf, data, extra_size - hdr_c); byte *insert_rec= &insert_buf[extra_size]; memcpy(insert_rec - hdr_c, prev_rec - hdr_c, hdr_c); rec_set_bit_field_1(insert_rec, (enc_hdr & 3) << 4, REC_OLD_INFO_BITS, REC_INFO_BITS_MASK, REC_INFO_BITS_SHIFT); rec_set_1byte_offs_flag(insert_rec, is_short); rec_set_n_fields_old(insert_rec, n_fields); rec_set_bit_field_1(insert_rec, 0, REC_OLD_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); const ulint data_size= rec_get_data_size_old(insert_rec); if (UNIV_UNLIKELY(data_c > data_size)) goto corrupted; if (UNIV_UNLIKELY(extra_size - hdr_c + data_size - data_c != data_len)) goto corrupted; /* Perform final consistency checks and then apply the change to the page. */ byte *buf; if (reuse) { byte *page_free= my_assume_aligned<2>(PAGE_FREE + PAGE_HEADER + page); rec_t *free_rec= page + mach_read_from_2(page_free); if (UNIV_UNLIKELY(heap_bot + REC_N_OLD_EXTRA_BYTES > free_rec || free_rec > heap_top)) goto corrupted; const ulint fn_fields= rec_get_n_fields_old(free_rec); const ulint fextra_size= REC_N_OLD_EXTRA_BYTES + (rec_get_1byte_offs_flag(free_rec) ? fn_fields : fn_fields * 2); if (UNIV_UNLIKELY(free_rec - fextra_size < heap_bot)) goto corrupted; const ulint fdata_size= rec_get_data_size_old(free_rec); if (UNIV_UNLIKELY(free_rec + fdata_size > heap_top)) goto corrupted; if (UNIV_UNLIKELY(extra_size + data_size > fextra_size + fdata_size)) goto corrupted; byte *page_garbage= my_assume_aligned<2>(page_free + 2); if (UNIV_UNLIKELY(mach_read_from_2(page_garbage) < fextra_size + fdata_size)) goto corrupted; buf= free_rec - fextra_size; const rec_t *const next_free= page + mach_read_from_2(free_rec - REC_NEXT); if (next_free == page); else if (UNIV_UNLIKELY(next_free < &heap_bot[REC_N_OLD_EXTRA_BYTES + 1] || heap_top < next_free)) goto corrupted; mach_write_to_2(page_garbage, mach_read_from_2(page_garbage) - extra_size - data_size); rec_set_bit_field_2(insert_rec, rec_get_heap_no_old(free_rec), REC_OLD_HEAP_NO, REC_HEAP_NO_MASK, REC_HEAP_NO_SHIFT); memcpy(page_free, free_rec - REC_NEXT, 2); } else { if (UNIV_UNLIKELY(heap_top + extra_size + data_size > last_slot)) goto corrupted; rec_set_bit_field_2(insert_rec, h, REC_OLD_HEAP_NO, REC_HEAP_NO_MASK, REC_HEAP_NO_SHIFT); mach_write_to_2(page_n_heap, h + 1); mach_write_to_2(page_heap_top, mach_read_from_2(page_heap_top) + extra_size + data_size); buf= heap_top; } ut_ad(data_size - data_c == data_len - (extra_size - hdr_c)); byte *page_last_insert= my_assume_aligned<2>(PAGE_LAST_INSERT + PAGE_HEADER + page); const uint16_t last_insert= mach_read_from_2(page_last_insert); memcpy(buf, insert_buf, extra_size); buf+= extra_size; mach_write_to_2(page_last_insert, buf - page); memcpy(prev_rec - REC_NEXT, page_last_insert, 2); memcpy(buf, prev_rec, data_c); memcpy(buf + data_c, static_cast(data) + (extra_size - hdr_c), data_len - (extra_size - hdr_c)); rec_set_bit_field_1(owner_rec, n_owned + 1, REC_OLD_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); /* Update PAGE_DIRECTION_B, PAGE_N_DIRECTION if needed */ if (page[FIL_PAGE_TYPE + 1] != byte(FIL_PAGE_RTREE)) { byte *dir= &page[PAGE_DIRECTION_B + PAGE_HEADER]; byte *n_dir= my_assume_aligned<2> (&page[PAGE_N_DIRECTION + PAGE_HEADER]); if (UNIV_UNLIKELY(!last_insert)) { no_direction: *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_NO_DIRECTION); memset(n_dir, 0, 2); } else if (page + last_insert == prev_rec && (*dir & ((1U << 3) - 1)) != PAGE_LEFT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_RIGHT); inc_dir: mach_write_to_2(n_dir, mach_read_from_2(n_dir) + 1); } else if (next_rec == page + last_insert && (*dir & ((1U << 3) - 1)) != PAGE_RIGHT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_LEFT); goto inc_dir; } else goto no_direction; } /* Update PAGE_N_RECS. */ byte *page_n_recs= my_assume_aligned<2>(PAGE_N_RECS + PAGE_HEADER + page); mach_write_to_2(page_n_recs, mach_read_from_2(page_n_recs) + 1); if (UNIV_UNLIKELY(n_owned == PAGE_DIR_SLOT_MAX_N_OWNED)) return page_dir_split_slot(block, owner_slot); ut_ad(page_simple_validate_old(page)); return false; } /** Apply a INSERT_HEAP_DYNAMIC or INSERT_REUSE_DYNAMIC record that was written by page_cur_insert_rec_low() for a ROW_FORMAT=COMPACT or DYNAMIC page. @param block B-tree or R-tree page in ROW_FORMAT=COMPACT or DYNAMIC @param reuse false=allocate from PAGE_HEAP_TOP; true=reuse PAGE_FREE @param prev byte offset of the predecessor, relative to PAGE_NEW_INFIMUM @param shift unless !reuse: number of bytes the PAGE_FREE is moving @param enc_hdr_l number of copied record header bytes, plus record type bits @param hdr_c number of common record header bytes with prev @param data_c number of common data bytes with prev @param data literal header and data bytes @param data_len length of the literal data, in bytes @return whether the operation failed (inconcistency was noticed) */ bool page_apply_insert_dynamic(const buf_block_t &block, bool reuse, ulint prev, ulint shift, ulint enc_hdr_l, size_t hdr_c, size_t data_c, const void *data, size_t data_len) { page_t * const page= block.page.frame; const uint16_t n_slots= page_dir_get_n_slots(page); byte *page_n_heap= my_assume_aligned<2>(PAGE_N_HEAP + PAGE_HEADER + page); ulint h= mach_read_from_2(page_n_heap); const page_id_t id(block.page.id()); if (UNIV_UNLIKELY(n_slots < 2 || h < (PAGE_HEAP_NO_USER_LOW | 0x8000) || (h & 0x7fff) >= srv_page_size / REC_N_NEW_EXTRA_BYTES || (h & 0x7fff) < n_slots || !fil_page_index_page_check(page) || page_get_page_no(page) != id.page_no() || mach_read_from_2(my_assume_aligned<2> (PAGE_NEW_SUPREMUM - REC_NEXT + page)) || ((enc_hdr_l & REC_STATUS_INSTANT) && !page_is_leaf(page)) || (enc_hdr_l >> 3) > data_len)) { corrupted: ib::error() << (reuse ? "Not applying INSERT_REUSE_DYNAMIC" " due to corruption on " : "Not applying INSERT_HEAP_DYNAMIC" " due to corruption on ") << id; return true; } byte * const last_slot= page_dir_get_nth_slot(page, n_slots - 1); byte * const page_heap_top= my_assume_aligned<2> (PAGE_HEAP_TOP + PAGE_HEADER + page); const byte *const heap_bot= &page[PAGE_NEW_SUPREMUM_END]; byte *heap_top= page + mach_read_from_2(page_heap_top); if (UNIV_UNLIKELY(heap_bot > heap_top || heap_top > last_slot)) goto corrupted; if (UNIV_UNLIKELY(mach_read_from_2(last_slot) != PAGE_NEW_SUPREMUM)) goto corrupted; if (UNIV_UNLIKELY(mach_read_from_2(page_dir_get_nth_slot(page, 0)) != PAGE_NEW_INFIMUM)) goto corrupted; uint16_t n= static_cast(PAGE_NEW_INFIMUM + prev); rec_t *prev_rec= page + n; n= static_cast(n + mach_read_from_2(prev_rec - REC_NEXT)); if (!prev); else if (UNIV_UNLIKELY(heap_bot + REC_N_NEW_EXTRA_BYTES > prev_rec || prev_rec > heap_top)) goto corrupted; rec_t * const next_rec= page + n; if (next_rec == page + PAGE_NEW_SUPREMUM); else if (UNIV_UNLIKELY(heap_bot + REC_N_NEW_EXTRA_BYTES > next_rec || next_rec > heap_top)) goto corrupted; ulint n_owned; rec_t *owner_rec= next_rec; n= static_cast(next_rec - page); for (ulint ns= PAGE_DIR_SLOT_MAX_N_OWNED; !(n_owned= rec_get_n_owned_new(owner_rec)); ) { n= static_cast(n + mach_read_from_2(owner_rec - REC_NEXT)); owner_rec= page + n; if (n == PAGE_NEW_SUPREMUM); else if (UNIV_UNLIKELY(heap_bot + REC_N_NEW_EXTRA_BYTES > owner_rec || owner_rec > heap_top)) goto corrupted; if (!ns--) goto corrupted; /* Corrupted (cyclic?) next-record list */ } page_dir_slot_t* owner_slot= last_slot; if (n_owned > PAGE_DIR_SLOT_MAX_N_OWNED) goto corrupted; else { static_assert(PAGE_DIR_SLOT_SIZE == 2, "compatibility"); alignas(2) byte slot_buf[2]; mach_write_to_2(slot_buf, owner_rec - page); const page_dir_slot_t * const first_slot= page_dir_get_nth_slot(page, 0); while (memcmp_aligned<2>(owner_slot, slot_buf, 2)) if ((owner_slot+= 2) == first_slot) goto corrupted; } const ulint extra_size= REC_N_NEW_EXTRA_BYTES + hdr_c + (enc_hdr_l >> 3); const ulint data_size= data_c + data_len - (enc_hdr_l >> 3); /* Perform final consistency checks and then apply the change to the page. */ byte *buf; if (reuse) { byte *page_free= my_assume_aligned<2>(PAGE_FREE + PAGE_HEADER + page); rec_t *free_rec= page + mach_read_from_2(page_free); if (UNIV_UNLIKELY(heap_bot + REC_N_NEW_EXTRA_BYTES > free_rec || free_rec > heap_top)) goto corrupted; buf= free_rec - extra_size; if (shift & 1) buf-= shift >> 1; else buf+= shift >> 1; if (UNIV_UNLIKELY(heap_bot > buf || &buf[extra_size + data_size] > heap_top)) goto corrupted; byte *page_garbage= my_assume_aligned<2>(page_free + 2); if (UNIV_UNLIKELY(mach_read_from_2(page_garbage) < extra_size + data_size)) goto corrupted; if ((n= mach_read_from_2(free_rec - REC_NEXT)) != 0) { n= static_cast(n + free_rec - page); if (UNIV_UNLIKELY(n < PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES || heap_top < page + n)) goto corrupted; } mach_write_to_2(page_free, n); mach_write_to_2(page_garbage, mach_read_from_2(page_garbage) - (extra_size + data_size)); h= rec_get_heap_no_new(free_rec); } else { if (UNIV_UNLIKELY(heap_top + extra_size + data_size > last_slot)) goto corrupted; mach_write_to_2(page_n_heap, h + 1); h&= 0x7fff; mach_write_to_2(page_heap_top, mach_read_from_2(page_heap_top) + extra_size + data_size); buf= heap_top; } memcpy(buf, data, (enc_hdr_l >> 3)); buf+= enc_hdr_l >> 3; data_len-= enc_hdr_l >> 3; data= &static_cast(data)[enc_hdr_l >> 3]; memcpy(buf, prev_rec - REC_N_NEW_EXTRA_BYTES - hdr_c, hdr_c); buf+= hdr_c; *buf++= static_cast((enc_hdr_l & 3) << 4); /* info_bits; n_owned=0 */ *buf++= static_cast(h >> 5); /* MSB of heap number */ h= (h & ((1U << 5) - 1)) << 3; static_assert(REC_STATUS_ORDINARY == 0, "compatibility"); static_assert(REC_STATUS_INSTANT == 4, "compatibility"); if (page_is_leaf(page)) h|= enc_hdr_l & REC_STATUS_INSTANT; else { ut_ad(!(enc_hdr_l & REC_STATUS_INSTANT)); /* Checked at the start */ h|= REC_STATUS_NODE_PTR; } *buf++= static_cast(h); /* LSB of heap number, and status */ static_assert(REC_NEXT == 2, "compatibility"); buf+= REC_NEXT; mach_write_to_2(buf - REC_NEXT, static_cast(next_rec - buf)); byte *page_last_insert= my_assume_aligned<2>(PAGE_LAST_INSERT + PAGE_HEADER + page); const uint16_t last_insert= mach_read_from_2(page_last_insert); mach_write_to_2(page_last_insert, buf - page); mach_write_to_2(prev_rec - REC_NEXT, static_cast(buf - prev_rec)); memcpy(buf, prev_rec, data_c); buf+= data_c; memcpy(buf, data, data_len); rec_set_bit_field_1(owner_rec, n_owned + 1, REC_NEW_N_OWNED, REC_N_OWNED_MASK, REC_N_OWNED_SHIFT); /* Update PAGE_DIRECTION_B, PAGE_N_DIRECTION if needed */ if (page[FIL_PAGE_TYPE + 1] != byte(FIL_PAGE_RTREE)) { byte *dir= &page[PAGE_DIRECTION_B + PAGE_HEADER]; byte *n_dir= my_assume_aligned<2>(&page[PAGE_N_DIRECTION + PAGE_HEADER]); if (UNIV_UNLIKELY(!last_insert)) { no_direction: *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_NO_DIRECTION); memset(n_dir, 0, 2); } else if (page + last_insert == prev_rec && (*dir & ((1U << 3) - 1)) != PAGE_LEFT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_RIGHT); inc_dir: mach_write_to_2(n_dir, mach_read_from_2(n_dir) + 1); } else if (next_rec == page + last_insert && (*dir & ((1U << 3) - 1)) != PAGE_RIGHT) { *dir= static_cast((*dir & ~((1U << 3) - 1)) | PAGE_LEFT); goto inc_dir; } else goto no_direction; } /* Update PAGE_N_RECS. */ byte *page_n_recs= my_assume_aligned<2>(PAGE_N_RECS + PAGE_HEADER + page); mach_write_to_2(page_n_recs, mach_read_from_2(page_n_recs) + 1); if (UNIV_UNLIKELY(n_owned == PAGE_DIR_SLOT_MAX_N_OWNED)) return page_dir_split_slot(block, owner_slot); ut_ad(page_simple_validate_new(page)); return false; } /** Apply a DELETE_ROW_FORMAT_REDUNDANT record that was written by page_cur_delete_rec() for a ROW_FORMAT=REDUNDANT page. @param block B-tree or R-tree page in ROW_FORMAT=REDUNDANT @param prev byte offset of the predecessor, relative to PAGE_OLD_INFIMUM @return whether the operation failed (inconcistency was noticed) */ bool page_apply_delete_redundant(const buf_block_t &block, ulint prev) { page_t * const page= block.page.frame; const uint16_t n_slots= page_dir_get_n_slots(page); ulint n_recs= page_get_n_recs(page); const page_id_t id(block.page.id()); if (UNIV_UNLIKELY(!n_recs || n_slots < 2 || !fil_page_index_page_check(page) || page_get_page_no(page) != id.page_no() || mach_read_from_2(my_assume_aligned<2> (PAGE_OLD_SUPREMUM - REC_NEXT + page)) || page_is_comp(page))) { corrupted: ib::error() << "Not applying DELETE_ROW_FORMAT_REDUNDANT" " due to corruption on " << id; return true; } byte *slot= page_dir_get_nth_slot(page, n_slots - 1); rec_t *prev_rec= page + PAGE_OLD_INFIMUM + prev; if (UNIV_UNLIKELY(prev_rec > slot)) goto corrupted; uint16_t n= mach_read_from_2(prev_rec - REC_NEXT); rec_t *rec= page + n; if (UNIV_UNLIKELY(n < PAGE_OLD_SUPREMUM_END + REC_N_OLD_EXTRA_BYTES || slot < rec)) goto corrupted; const ulint extra_size= REC_N_OLD_EXTRA_BYTES + rec_get_n_fields_old(rec) * (rec_get_1byte_offs_flag(rec) ? 1 : 2); const ulint data_size= rec_get_data_size_old(rec); if (UNIV_UNLIKELY(n < PAGE_OLD_SUPREMUM_END + extra_size || slot < rec + data_size)) goto corrupted; n= mach_read_from_2(rec - REC_NEXT); rec_t *next= page + n; if (n == PAGE_OLD_SUPREMUM); else if (UNIV_UNLIKELY(n < PAGE_OLD_SUPREMUM_END + REC_N_OLD_EXTRA_BYTES || slot < next)) goto corrupted; rec_t *s= rec; ulint slot_owned; for (ulint i= n_recs; !(slot_owned= rec_get_n_owned_old(s)); ) { n= mach_read_from_2(s - REC_NEXT); s= page + n; if (n == PAGE_OLD_SUPREMUM); else if (UNIV_UNLIKELY(n < PAGE_OLD_SUPREMUM_END + REC_N_OLD_EXTRA_BYTES || slot < s)) goto corrupted; if (UNIV_UNLIKELY(!i--)) /* Corrupted (cyclic?) next-record list */ goto corrupted; } slot_owned--; /* The first slot is always pointing to the infimum record. Find the directory slot pointing to s. */ const byte * const first_slot= page + srv_page_size - (PAGE_DIR + 2); alignas(2) byte slot_offs[2]; mach_write_to_2(slot_offs, s - page); static_assert(PAGE_DIR_SLOT_SIZE == 2, "compatibility"); while (memcmp_aligned<2>(slot, slot_offs, 2)) if ((slot+= 2) == first_slot) goto corrupted; if (rec == s) { s= prev_rec; mach_write_to_2(slot, s - page); } memcpy(prev_rec - REC_NEXT, rec - REC_NEXT, 2); s-= REC_OLD_N_OWNED; *s= static_cast((*s & ~REC_N_OWNED_MASK) | slot_owned << REC_N_OWNED_SHIFT); page_mem_free(block, rec, data_size, extra_size); if (slot_owned < PAGE_DIR_SLOT_MIN_N_OWNED) page_dir_balance_slot(block, (first_slot - slot) / 2); ut_ad(page_simple_validate_old(page)); return false; } /** Apply a DELETE_ROW_FORMAT_DYNAMIC record that was written by page_cur_delete_rec() for a ROW_FORMAT=COMPACT or DYNAMIC page. @param block B-tree or R-tree page in ROW_FORMAT=COMPACT or DYNAMIC @param prev byte offset of the predecessor, relative to PAGE_NEW_INFIMUM @param hdr_size record header size, excluding REC_N_NEW_EXTRA_BYTES @param data_size data payload size, in bytes @return whether the operation failed (inconcistency was noticed) */ bool page_apply_delete_dynamic(const buf_block_t &block, ulint prev, size_t hdr_size, size_t data_size) { page_t * const page= block.page.frame; const uint16_t n_slots= page_dir_get_n_slots(page); ulint n_recs= page_get_n_recs(page); const page_id_t id(block.page.id()); if (UNIV_UNLIKELY(!n_recs || n_slots < 2 || !fil_page_index_page_check(page) || page_get_page_no(page) != id.page_no() || mach_read_from_2(my_assume_aligned<2> (PAGE_NEW_SUPREMUM - REC_NEXT + page)) || !page_is_comp(page))) { corrupted: ib::error() << "Not applying DELETE_ROW_FORMAT_DYNAMIC" " due to corruption on " << id; return true; } byte *slot= page_dir_get_nth_slot(page, n_slots - 1); uint16_t n= static_cast(PAGE_NEW_INFIMUM + prev); rec_t *prev_rec= page + n; if (UNIV_UNLIKELY(prev_rec > slot)) goto corrupted; n= static_cast(n + mach_read_from_2(prev_rec - REC_NEXT)); rec_t *rec= page + n; if (UNIV_UNLIKELY(n < PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES || slot < rec)) goto corrupted; const ulint extra_size= REC_N_NEW_EXTRA_BYTES + hdr_size; if (UNIV_UNLIKELY(n < PAGE_NEW_SUPREMUM_END + extra_size || slot < rec + data_size)) goto corrupted; n= static_cast(n + mach_read_from_2(rec - REC_NEXT)); rec_t *next= page + n; if (n == PAGE_NEW_SUPREMUM); else if (UNIV_UNLIKELY(n < PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES || slot < next)) goto corrupted; rec_t *s= rec; n= static_cast(rec - page); ulint slot_owned; for (ulint i= n_recs; !(slot_owned= rec_get_n_owned_new(s)); ) { const uint16_t next= mach_read_from_2(s - REC_NEXT); if (UNIV_UNLIKELY(next < REC_N_NEW_EXTRA_BYTES || next > static_cast(-REC_N_NEW_EXTRA_BYTES))) goto corrupted; n= static_cast(n + next); s= page + n; if (n == PAGE_NEW_SUPREMUM); else if (UNIV_UNLIKELY(n < PAGE_NEW_SUPREMUM_END + REC_N_NEW_EXTRA_BYTES || slot < s)) goto corrupted; if (UNIV_UNLIKELY(!i--)) /* Corrupted (cyclic?) next-record list */ goto corrupted; } slot_owned--; /* The first slot is always pointing to the infimum record. Find the directory slot pointing to s. */ const byte * const first_slot= page + srv_page_size - (PAGE_DIR + 2); alignas(2) byte slot_offs[2]; mach_write_to_2(slot_offs, s - page); static_assert(PAGE_DIR_SLOT_SIZE == 2, "compatibility"); while (memcmp_aligned<2>(slot, slot_offs, 2)) if ((slot+= 2) == first_slot) goto corrupted; if (rec == s) { s= prev_rec; mach_write_to_2(slot, s - page); } mach_write_to_2(prev_rec - REC_NEXT, static_cast(next - prev_rec)); s-= REC_NEW_N_OWNED; *s= static_cast((*s & ~REC_N_OWNED_MASK) | slot_owned << REC_N_OWNED_SHIFT); page_mem_free(block, rec, data_size, extra_size); if (slot_owned < PAGE_DIR_SLOT_MIN_N_OWNED) page_dir_balance_slot(block, (first_slot - slot) / 2); ut_ad(page_simple_validate_new(page)); return false; } #ifdef UNIV_COMPILE_TEST_FUNCS /*******************************************************************//** Print the first n numbers, generated by ut_rnd_gen() to make sure (visually) that it works properly. */ void test_ut_rnd_gen( int n) /*!< in: print first n numbers */ { int i; unsigned long long rnd; for (i = 0; i < n; i++) { rnd = ut_rnd_gen(); printf("%llu\t%%2=%llu %%3=%llu %%5=%llu %%7=%llu %%11=%llu\n", rnd, rnd % 2, rnd % 3, rnd % 5, rnd % 7, rnd % 11); } } #endif /* UNIV_COMPILE_TEST_FUNCS */