mirror of
https://github.com/MariaDB/server.git
synced 2025-01-15 19:42:28 +01:00
MDEV-35049: Use CRC-32C and avoid allocating heap
For the adaptive hash index, dtuple_fold() and rec_fold() were employing a slow rolling hash algorithm, computing hash values ("fold") for one field and one byte at a time, while depending on calls to rec_get_offsets(). We already have optimized implementations of CRC-32C and have been successfully using that function in some other InnoDB tables, but not yet in the adaptive hash index. Any linear function such as any CRC will fail the avalanche test that any cryptographically secure hash function is expected to pass: any single-bit change in the input key should affect on average half the bits in the output. But we always were happy with less than cryptographically secure: in fact, ut_fold_ulint_pair() or ut_fold_binary() are just about as linear as any CRC, using a combination of multiplication and addition, partly carry-less. It is worth noting that exclusive-or corresponds to carry-less subtraction or addition in a binary Galois field, or GF(2). We only need some way of reducing key prefixes into hash values. The CRC-32C should be better than a Rabin–Karp rolling hash algorithm. Compared to the old hash algorithm, it has the drawback that there will be only 32 bits of entropy before we choose the hash table cell by a modulus operation. The size of each adaptive hash index array is (innodb_buffer_pool_size / 512) / innodb_adaptive_hash_index_parts. With the maximum number of partitions (512), we would not exceed 1<<32 elements per array until the buffer pool size exceeds 1<<50 bytes (1 PiB). We would hit other limits before that: the virtual address space on many contemporary 64-bit processor implementations is only 48 bits (256 TiB). So, we can simply go for the SIMD accelerated CRC-32C. rec_fold(): Take a combined parameter n_bytes_fields. Determine the length of each field on the fly, and compute CRC-32C over a single contiguous range of bytes, from the start of the record payload area to the end of the last full or partial field. For secondary index records in ROW_FORMAT=REDUNDANT, also the data area that is reserved for NULL values (to facilitate in-place updates between NULL and NOT NULL values) will be included in the count. Luckily, InnoDB always zero-initialized such unused area; refer to data_write_sql_null() in rec_convert_dtuple_to_rec_old(). For other than ROW_FORMAT=REDUNDANT, no space is allocated for NULL values, and therefore the CRC-32C will only cover the actual payload of the key prefix. dtuple_fold(): For ROW_FORMAT=REDUNDANT, include the dummy NULL values in the CRC-32C, so that the values will be comparable with rec_fold(). innodb_ahi-t: A unit test for rec_fold() and dtuple_fold(). btr_search_build_page_hash_index(), btr_search_drop_page_hash_index(): Use a fixed-size stack buffer for computing the fold values, to avoid dynamic memory allocation. btr_search_drop_page_hash_index(): Do not release part.latch if we need to invoke multiple batches of rec_fold(). dtuple_t: Allocate fewer bits for the fields. The maximum number of data fields is about 1023, so uint16_t will be fine for them. The info_bits is stored in less than 1 byte. ut_pair_min(), ut_pair_cmp(): Remove. We can actually combine and compare int(n_fields << 16 | n_bytes). PAGE_CUR_LE_OR_EXTENDS, PAGE_CUR_DBG: Remove. These were never defined, because they would only work with latin1_swedish_ci if at all. btr_cur_t::check_mismatch(): Replaces !btr_search_check_guess(). cmp_dtuple_rec_bytes(): Replaces cmp_dtuple_rec_with_match_bytes(). Determine the offsets of fields on the fly. page_cur_try_search_shortcut_bytes(): This caller of cmp_dtuple_rec_bytes() will not be invoked on the change buffer tree. cmp_dtuple_rec_leaf(): Replaces cmp_dtuple_rec_with_match() for comparing leaf-page records. buf_block_t::ahi_left_bytes_fields: Consolidated Atomic_relaxed<uint32_t> of curr_left_side << 31 | curr_n_bytes << 16 | curr_n_fields. The other set of parameters (n_fields, n_bytes, left_side) was removed as redundant. btr_search_update_hash_node_on_insert(): Merged to btr_search_update_hash_on_insert(). btr_search_build_page_hash_index(): Take combined left_bytes_fields instead of n_fields, n_bytes, left_side. btr_search_update_block_hash_info(), btr_search_update_hash_ref(): Merged to btr_search_info_update_hash(). btr_cur_t::n_bytes_fields: Replaces n_bytes << 16 | n_fields. We also remove many redundant checks of btr_search.enabled. If we are holding any btr_sea::partition::latch, then a nonnull pointer in buf_block_t::index must imply that the adaptive hash index is enabled. Reviewed by: Vladislav Lesin
This commit is contained in:
parent
9c8bdc6c15
commit
4dcb1b575b
43 changed files with 2193 additions and 2701 deletions
|
@ -362,7 +362,6 @@ SET(INNOBASE_SOURCES
|
|||
include/ut0sort.h
|
||||
include/ut0stage.h
|
||||
include/ut0ut.h
|
||||
include/ut0ut.inl
|
||||
include/ut0vec.h
|
||||
include/ut0vec.inl
|
||||
include/ut0wqueue.h
|
||||
|
|
|
@ -558,8 +558,8 @@ buf_block_t *btr_root_block_sx(dict_index_t *index, mtr_t *mtr, dberr_t *err)
|
|||
return root;
|
||||
}
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
else
|
||||
ut_ad(!root->index || !root->index->freed());
|
||||
ut_d(else if (dict_index_t *index= root->index))
|
||||
ut_ad(!index->freed());
|
||||
#endif
|
||||
return root;
|
||||
}
|
||||
|
@ -772,7 +772,7 @@ static rec_offs *btr_page_get_parent(rec_offs *offsets, mem_heap_t *heap,
|
|||
{
|
||||
ut_ad(block->page.lock.have_u_or_x() ||
|
||||
(!block->page.lock.have_s() && index->lock.have_x()));
|
||||
ulint up_match= 0, low_match= 0;
|
||||
uint16_t up_match= 0, low_match= 0;
|
||||
cursor->page_cur.block= block;
|
||||
if (page_cur_search_with_match(tuple, PAGE_CUR_LE, &up_match,
|
||||
&low_match, &cursor->page_cur,
|
||||
|
@ -1976,7 +1976,7 @@ btr_root_raise_and_insert(
|
|||
|
||||
ut_ad(dtuple_check_typed(tuple));
|
||||
/* Reposition the cursor to the child node */
|
||||
ulint low_match = 0, up_match = 0;
|
||||
uint16_t low_match = 0, up_match = 0;
|
||||
|
||||
if (page_cur_search_with_match(tuple, PAGE_CUR_LE,
|
||||
&up_match, &low_match,
|
||||
|
@ -2660,7 +2660,7 @@ btr_insert_into_right_sibling(
|
|||
return nullptr;
|
||||
}
|
||||
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
if (page_cur_search_with_match(tuple,
|
||||
PAGE_CUR_LE, &up_match, &low_match,
|
||||
|
@ -3142,7 +3142,7 @@ insert_empty:
|
|||
page_cursor = btr_cur_get_page_cur(cursor);
|
||||
page_cursor->block = insert_block;
|
||||
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
if (page_cur_search_with_match(tuple,
|
||||
PAGE_CUR_LE, &up_match, &low_match,
|
||||
|
|
|
@ -1100,8 +1100,7 @@ dberr_t btr_cur_t::search_leaf(const dtuple_t *tuple, page_cur_mode_t mode,
|
|||
MEM_UNDEFINED(&up_bytes, sizeof up_bytes);
|
||||
MEM_UNDEFINED(&low_match, sizeof low_match);
|
||||
MEM_UNDEFINED(&low_bytes, sizeof low_bytes);
|
||||
ut_d(up_match= ULINT_UNDEFINED);
|
||||
ut_d(low_match= ULINT_UNDEFINED);
|
||||
ut_d(up_match= low_match= uint16_t(~0u));
|
||||
|
||||
ut_ad(!(latch_mode & BTR_ALREADY_S_LATCHED) ||
|
||||
mtr->memo_contains_flagged(&index()->lock,
|
||||
|
@ -1118,7 +1117,6 @@ dberr_t btr_cur_t::search_leaf(const dtuple_t *tuple, page_cur_mode_t mode,
|
|||
|| latch_mode == BTR_MODIFY_TREE
|
||||
|| latch_mode == BTR_MODIFY_ROOT_AND_LEAF);
|
||||
|
||||
flag= BTR_CUR_BINARY;
|
||||
#ifndef BTR_CUR_ADAPT
|
||||
guess= nullptr;
|
||||
#else
|
||||
|
@ -1126,6 +1124,7 @@ dberr_t btr_cur_t::search_leaf(const dtuple_t *tuple, page_cur_mode_t mode,
|
|||
guess= info->root_guess;
|
||||
|
||||
# ifdef BTR_CUR_HASH_ADAPT
|
||||
flag= BTR_CUR_BINARY;
|
||||
# ifdef UNIV_SEARCH_PERF_STAT
|
||||
info->n_searches++;
|
||||
# endif
|
||||
|
@ -1138,9 +1137,9 @@ dberr_t btr_cur_t::search_leaf(const dtuple_t *tuple, page_cur_mode_t mode,
|
|||
latch_mode, this, mtr))
|
||||
{
|
||||
/* Search using the hash index succeeded */
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
++btr_cur_n_sea;
|
||||
|
||||
return DB_SUCCESS;
|
||||
|
@ -1349,9 +1348,9 @@ dberr_t btr_cur_t::search_leaf(const dtuple_t *tuple, page_cur_mode_t mode,
|
|||
if (page_cur_search_with_match(tuple, mode, &up_match, &low_match,
|
||||
&page_cur, nullptr))
|
||||
goto corrupted;
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
goto func_exit;
|
||||
}
|
||||
|
||||
|
@ -1398,22 +1397,21 @@ release_tree:
|
|||
|
||||
reached_latched_leaf:
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
if (btr_search.enabled && !(tuple->info_bits & REC_INFO_MIN_REC_FLAG))
|
||||
if (!(tuple->info_bits & REC_INFO_MIN_REC_FLAG) && btr_search.enabled)
|
||||
{
|
||||
if (page_cur_search_with_match_bytes(tuple, mode,
|
||||
&up_match, &up_bytes,
|
||||
&low_match, &low_bytes, &page_cur))
|
||||
if (page_cur_search_with_match_bytes(*tuple, mode, &up_match, &low_match,
|
||||
&page_cur, &up_bytes, &low_bytes))
|
||||
goto corrupted;
|
||||
}
|
||||
else
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
if (page_cur_search_with_match(tuple, mode, &up_match, &low_match,
|
||||
&page_cur, nullptr))
|
||||
goto corrupted;
|
||||
if (page_cur_search_with_match(tuple, mode, &up_match, &low_match,
|
||||
&page_cur, nullptr))
|
||||
goto corrupted;
|
||||
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
|
||||
if (latch_mode == BTR_MODIFY_TREE &&
|
||||
btr_cur_need_opposite_intention(block->page, index()->is_clust(),
|
||||
|
@ -1656,9 +1654,9 @@ dberr_t btr_cur_t::pessimistic_search_leaf(const dtuple_t *tuple,
|
|||
err= DB_CORRUPTION;
|
||||
else
|
||||
{
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
/* We do a dirty read of btr_search.enabled here. We will recheck in
|
||||
|
@ -1770,8 +1768,9 @@ dberr_t btr_cur_search_to_nth_level(ulint level,
|
|||
MEM_UNDEFINED(&cursor->low_bytes, sizeof cursor->low_bytes);
|
||||
cursor->up_match= 0;
|
||||
cursor->low_match= 0;
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
cursor->flag= BTR_CUR_BINARY;
|
||||
|
||||
#endif
|
||||
#ifndef BTR_CUR_ADAPT
|
||||
buf_block_t *block= nullptr;
|
||||
#else
|
||||
|
@ -2518,13 +2517,8 @@ fail_err:
|
|||
ut_ad(entry->is_metadata());
|
||||
ut_ad(index->is_instant());
|
||||
ut_ad(flags == BTR_NO_LOCKING_FLAG);
|
||||
} else if (index->table->is_temporary()) {
|
||||
} else {
|
||||
if (!reorg && cursor->flag == BTR_CUR_HASH) {
|
||||
btr_search_update_hash_node_on_insert(cursor);
|
||||
} else {
|
||||
btr_search_update_hash_on_insert(cursor);
|
||||
}
|
||||
} else if (!index->table->is_temporary()) {
|
||||
btr_search_update_hash_on_insert(cursor, reorg);
|
||||
}
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
|
||||
|
@ -2588,7 +2582,9 @@ btr_cur_pessimistic_insert(
|
|||
|| dict_index_is_clust(index)
|
||||
|| (flags & BTR_CREATE_FLAG));
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
cursor->flag = BTR_CUR_BINARY;
|
||||
#endif
|
||||
|
||||
/* Check locks and write to undo log, if specified */
|
||||
|
||||
|
@ -2694,9 +2690,8 @@ btr_cur_pessimistic_insert(
|
|||
ut_ad(index->is_instant());
|
||||
ut_ad(flags & BTR_NO_LOCKING_FLAG);
|
||||
ut_ad(!(flags & BTR_CREATE_FLAG));
|
||||
} else if (index->table->is_temporary()) {
|
||||
} else {
|
||||
btr_search_update_hash_on_insert(cursor);
|
||||
} else if (!index->table->is_temporary()) {
|
||||
btr_search_update_hash_on_insert(cursor, false);
|
||||
}
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
if (inherit && !(flags & BTR_NO_LOCKING_FLAG)) {
|
||||
|
@ -3325,7 +3320,7 @@ static void btr_cur_trim_alter_metadata(dtuple_t* entry,
|
|||
if (n_fields != index->n_uniq) {
|
||||
ut_ad(n_fields
|
||||
>= index->n_core_fields);
|
||||
entry->n_fields = n_fields;
|
||||
entry->n_fields = uint16_t(n_fields);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -3361,7 +3356,7 @@ static void btr_cur_trim_alter_metadata(dtuple_t* entry,
|
|||
ut_ad(n_fields >= index->n_core_fields);
|
||||
|
||||
mtr.commit();
|
||||
entry->n_fields = n_fields + 1;
|
||||
entry->n_fields = uint16_t(n_fields + 1);
|
||||
}
|
||||
|
||||
/** Trim an update tuple due to instant ADD COLUMN, if needed.
|
||||
|
@ -3417,7 +3412,7 @@ btr_cur_trim(
|
|||
ulint n_fields = upd_get_nth_field(update, 0)
|
||||
->field_no;
|
||||
ut_ad(n_fields + 1 >= entry->n_fields);
|
||||
entry->n_fields = n_fields;
|
||||
entry->n_fields = uint16_t(n_fields);
|
||||
}
|
||||
} else {
|
||||
entry->trim(*index);
|
||||
|
@ -4817,10 +4812,10 @@ class btr_est_cur_t
|
|||
|
||||
/** Matched fields and bytes which are used for on-page search, see
|
||||
btr_cur_t::(up|low)_(match|bytes) comments for details */
|
||||
ulint m_up_match= 0;
|
||||
ulint m_up_bytes= 0;
|
||||
ulint m_low_match= 0;
|
||||
ulint m_low_bytes= 0;
|
||||
uint16_t m_up_match= 0;
|
||||
uint16_t m_up_bytes= 0;
|
||||
uint16_t m_low_match= 0;
|
||||
uint16_t m_low_bytes= 0;
|
||||
|
||||
public:
|
||||
btr_est_cur_t(dict_index_t *index, const dtuple_t &tuple,
|
||||
|
@ -4844,12 +4839,7 @@ public:
|
|||
m_page_mode= PAGE_CUR_LE;
|
||||
break;
|
||||
default:
|
||||
#ifdef PAGE_CUR_LE_OR_EXTENDS
|
||||
ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE ||
|
||||
mode == PAGE_CUR_LE_OR_EXTENDS);
|
||||
#else /* PAGE_CUR_LE_OR_EXTENDS */
|
||||
ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE);
|
||||
#endif /* PAGE_CUR_LE_OR_EXTENDS */
|
||||
m_page_mode= mode;
|
||||
break;
|
||||
}
|
||||
|
|
|
@ -435,7 +435,7 @@ btr_pcur_t::restore_position(btr_latch_mode restore_latch_mode, mtr_t *mtr)
|
|||
rec_offs_init(offsets);
|
||||
restore_status ret_val= restore_status::NOT_SAME;
|
||||
if (rel_pos == BTR_PCUR_ON && btr_pcur_is_on_user_rec(this)) {
|
||||
ulint n_matched_fields= 0;
|
||||
uint16_t n_matched_fields= 0;
|
||||
if (!cmp_dtuple_rec_with_match(
|
||||
tuple, btr_pcur_get_rec(this), index,
|
||||
rec_get_offsets(btr_pcur_get_rec(this), index, offsets,
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -1039,6 +1039,26 @@ void buf_page_print(const byte *read_buf, ulint zip_size)
|
|||
#endif
|
||||
}
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
|
||||
/** Ensure that some adaptive hash index fields are initialized */
|
||||
static void buf_block_init_low(buf_block_t *block) noexcept
|
||||
{
|
||||
/* No adaptive hash index entries may point to a previously unused
|
||||
(and now freshly allocated) block. */
|
||||
MEM_MAKE_DEFINED(&block->index, sizeof block->index);
|
||||
MEM_MAKE_DEFINED(&block->n_pointers, sizeof block->n_pointers);
|
||||
MEM_MAKE_DEFINED(&block->n_hash_helps, sizeof block->n_hash_helps);
|
||||
# if defined UNIV_AHI_DEBUG || defined UNIV_DEBUG
|
||||
ut_a(!block->index);
|
||||
ut_a(!block->n_pointers);
|
||||
ut_a(!block->n_hash_helps);
|
||||
# endif
|
||||
}
|
||||
#else /* BTR_CUR_HASH_ADAPT */
|
||||
inline void buf_block_init_low(buf_block_t*) {}
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
|
||||
/** Initialize a buffer page descriptor.
|
||||
@param[in,out] block buffer page descriptor
|
||||
@param[in] frame buffer page frame */
|
||||
|
@ -1048,8 +1068,7 @@ buf_block_init(buf_block_t* block, byte* frame)
|
|||
{
|
||||
/* This function should only be executed at database startup or by
|
||||
buf_pool.resize(). Either way, adaptive hash index must not exist. */
|
||||
assert_block_ahi_empty_on_init(block);
|
||||
|
||||
buf_block_init_low(block);
|
||||
block->page.frame = frame;
|
||||
|
||||
MEM_MAKE_DEFINED(&block->modify_clock, sizeof block->modify_clock);
|
||||
|
@ -1057,10 +1076,6 @@ buf_block_init(buf_block_t* block, byte* frame)
|
|||
MEM_MAKE_DEFINED(&block->page.lock, sizeof block->page.lock);
|
||||
block->page.lock.init();
|
||||
block->page.init(buf_page_t::NOT_USED, page_id_t(~0ULL));
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
MEM_MAKE_DEFINED(&block->index, sizeof block->index);
|
||||
ut_ad(!block->index);
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
ut_d(block->in_unzip_LRU_list = false);
|
||||
ut_d(block->in_withdraw_list = false);
|
||||
|
||||
|
@ -1490,19 +1505,7 @@ inline bool buf_pool_t::realloc(buf_block_t *block)
|
|||
mysql_mutex_unlock(&buf_pool.flush_list_mutex);
|
||||
block->page.set_corrupt_id();
|
||||
|
||||
/* set other flags of buf_block_t */
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
/* This code should only be executed by resize(),
|
||||
while the adaptive hash index is disabled. */
|
||||
assert_block_ahi_empty(block);
|
||||
assert_block_ahi_empty_on_init(new_block);
|
||||
ut_ad(!block->index);
|
||||
new_block->index = NULL;
|
||||
new_block->n_hash_helps = 0;
|
||||
new_block->n_fields = 1;
|
||||
new_block->left_side = TRUE;
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
buf_block_init_low(new_block);
|
||||
ut_d(block->page.set_state(buf_page_t::MEMORY));
|
||||
/* free block */
|
||||
new_block = block;
|
||||
|
@ -2453,27 +2456,6 @@ buf_page_t *buf_page_get_zip(const page_id_t page_id)
|
|||
return bpage;
|
||||
}
|
||||
|
||||
/********************************************************************//**
|
||||
Initialize some fields of a control block. */
|
||||
UNIV_INLINE
|
||||
void
|
||||
buf_block_init_low(
|
||||
/*===============*/
|
||||
buf_block_t* block) /*!< in: block to init */
|
||||
{
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
/* No adaptive hash index entries may point to a previously
|
||||
unused (and now freshly allocated) block. */
|
||||
assert_block_ahi_empty_on_init(block);
|
||||
block->index = NULL;
|
||||
|
||||
block->n_hash_helps = 0;
|
||||
block->n_fields = 1;
|
||||
block->n_bytes = 0;
|
||||
block->left_side = TRUE;
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
}
|
||||
|
||||
/********************************************************************//**
|
||||
Decompress a block.
|
||||
@return true if successful */
|
||||
|
|
|
@ -76,20 +76,7 @@ void dtuple_t::trim(const dict_index_t& index)
|
|||
}
|
||||
}
|
||||
|
||||
n_fields = i;
|
||||
}
|
||||
|
||||
/*********************************************************************//**
|
||||
Sets number of fields used in a tuple. Normally this is set in
|
||||
dtuple_create, but if you want later to set it smaller, you can use this. */
|
||||
void
|
||||
dtuple_set_n_fields(
|
||||
/*================*/
|
||||
dtuple_t* tuple, /*!< in: tuple */
|
||||
ulint n_fields) /*!< in: number of fields */
|
||||
{
|
||||
tuple->n_fields = n_fields;
|
||||
tuple->n_fields_cmp = n_fields;
|
||||
n_fields = uint16_t(i);
|
||||
}
|
||||
|
||||
/**********************************************************//**
|
||||
|
@ -587,8 +574,8 @@ dtuple_convert_big_rec(
|
|||
ulint longest;
|
||||
|
||||
const bool mblob = entry->is_alter_metadata();
|
||||
ut_ad(entry->n_fields - mblob >= index->first_user_field());
|
||||
ut_ad(entry->n_fields - mblob <= index->n_fields);
|
||||
ut_ad(unsigned(entry->n_fields - mblob) >= index->first_user_field());
|
||||
ut_ad(unsigned(entry->n_fields - mblob) <= index->n_fields);
|
||||
|
||||
if (mblob) {
|
||||
longest_i = index->first_user_field();
|
||||
|
|
|
@ -2327,8 +2327,8 @@ dict_table_copy_v_types(
|
|||
/* tuple could have more virtual columns than existing table,
|
||||
if we are calling this for creating index along with adding
|
||||
virtual columns */
|
||||
ulint n_fields = ut_min(dtuple_get_n_v_fields(tuple),
|
||||
static_cast<ulint>(table->n_v_def));
|
||||
ulint n_fields = std::min<ulint>(dtuple_get_n_v_fields(tuple),
|
||||
table->n_v_def);
|
||||
|
||||
for (ulint i = 0; i < n_fields; i++) {
|
||||
|
||||
|
@ -3670,7 +3670,7 @@ dict_index_build_node_ptr(
|
|||
dtuple_t* tuple;
|
||||
dfield_t* field;
|
||||
byte* buf;
|
||||
ulint n_unique = dict_index_get_n_unique_in_tree_nonleaf(index);
|
||||
uint16_t n_unique = dict_index_get_n_unique_in_tree_nonleaf(index);
|
||||
|
||||
tuple = dtuple_create(heap, n_unique + 1);
|
||||
|
||||
|
@ -3721,7 +3721,7 @@ dict_index_build_data_tuple(
|
|||
{
|
||||
ut_ad(!index->is_clust());
|
||||
|
||||
dtuple_t* tuple = dtuple_create(heap, n_fields);
|
||||
dtuple_t* tuple = dtuple_create(heap, uint16_t(n_fields));
|
||||
|
||||
dict_index_copy_types(tuple, index, n_fields);
|
||||
|
||||
|
|
|
@ -1319,7 +1319,7 @@ static dberr_t dict_load_columns(dict_table_t *table, unsigned use_uncommitted,
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -1446,7 +1446,7 @@ dict_load_virtual_col(dict_table_t *table, bool uncommitted, ulint nth_v_col)
|
|||
|
||||
dfield_t dfield[2];
|
||||
dtuple_t tuple{
|
||||
0,2,2,dfield,0,nullptr
|
||||
0,2,2,0,dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -1690,7 +1690,7 @@ static dberr_t dict_load_fields(dict_index_t *index, bool uncommitted,
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -1949,7 +1949,7 @@ dberr_t dict_load_indexes(dict_table_t *table, bool uncommitted,
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -2338,7 +2338,7 @@ static dict_table_t *dict_load_table_one(const span<const char> &name,
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -2590,7 +2590,7 @@ dict_load_table_on_id(
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -2699,7 +2699,7 @@ static dberr_t dict_load_foreign_cols(dict_foreign_t *foreign, trx_id_t trx_id)
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -2875,7 +2875,7 @@ dict_load_foreign(
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
@ -3086,7 +3086,7 @@ dict_load_foreigns(
|
|||
bool check_recursive = !trx_id;
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
|
|
@ -3412,7 +3412,7 @@ fts_add_doc_by_id(
|
|||
|
||||
/* Search based on Doc ID. Here, we'll need to consider the case
|
||||
when there is no primary index on Doc ID */
|
||||
const ulint n_uniq = table->fts_n_uniq();
|
||||
const auto n_uniq = table->fts_n_uniq();
|
||||
tuple = dtuple_create(heap, n_uniq);
|
||||
dfield = dtuple_get_nth_field(tuple, 0);
|
||||
dfield->type.mtype = DATA_INT;
|
||||
|
@ -3463,9 +3463,7 @@ fts_add_doc_by_id(
|
|||
doc_pcur = &pcur;
|
||||
} else {
|
||||
dtuple_t* clust_ref;
|
||||
ulint n_fields;
|
||||
|
||||
n_fields = dict_index_get_n_unique(clust_index);
|
||||
auto n_fields = dict_index_get_n_unique(clust_index);
|
||||
|
||||
clust_ref = dtuple_create(heap, n_fields);
|
||||
dict_index_copy_types(clust_ref, clust_index, n_fields);
|
||||
|
|
|
@ -140,12 +140,11 @@ rtr_index_build_node_ptr(
|
|||
dtuple_t* tuple;
|
||||
dfield_t* field;
|
||||
byte* buf;
|
||||
ulint n_unique;
|
||||
ulint info_bits;
|
||||
|
||||
ut_ad(dict_index_is_spatial(index));
|
||||
|
||||
n_unique = DICT_INDEX_SPATIAL_NODEPTR_SIZE;
|
||||
uint16_t n_unique = DICT_INDEX_SPATIAL_NODEPTR_SIZE;
|
||||
|
||||
tuple = dtuple_create(heap, n_unique + 1);
|
||||
|
||||
|
@ -210,8 +209,8 @@ rtr_update_mbr_field(
|
|||
big_rec_t* dummy_big_rec;
|
||||
buf_block_t* block;
|
||||
rec_t* child_rec;
|
||||
ulint up_match = 0;
|
||||
ulint low_match = 0;
|
||||
uint16_t up_match = 0;
|
||||
uint16_t low_match = 0;
|
||||
ulint child;
|
||||
ulint rec_info;
|
||||
bool ins_suc = true;
|
||||
|
@ -607,7 +606,7 @@ rtr_adjust_upper_level(
|
|||
/* Insert the node for the new page. */
|
||||
node_ptr_upper = rtr_index_build_node_ptr(
|
||||
sea_cur->index(), new_mbr, first, new_page_no, heap);
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
err = page_cur_search_with_match(node_ptr_upper,
|
||||
PAGE_CUR_LE,
|
||||
&up_match, &low_match,
|
||||
|
@ -1106,7 +1105,7 @@ corrupted:
|
|||
page_cursor->block = cur_split_node->n_node != first_rec_group
|
||||
? new_block : block;
|
||||
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
if (page_cur_search_with_match(tuple,
|
||||
PAGE_CUR_LE, &up_match, &low_match,
|
||||
|
@ -1421,7 +1420,7 @@ rtr_root_raise_and_insert(
|
|||
if (tuple) {
|
||||
ut_ad(dtuple_check_typed(tuple));
|
||||
/* Reposition the cursor to the child node */
|
||||
ulint low_match = 0, up_match = 0;
|
||||
uint16_t low_match = 0, up_match = 0;
|
||||
|
||||
if (page_cur_search_with_match(tuple, PAGE_CUR_LE,
|
||||
&up_match, &low_match,
|
||||
|
|
|
@ -337,7 +337,7 @@ rtr_pcur_getnext_from_path(
|
|||
|
||||
if (mode == PAGE_CUR_RTREE_LOCATE) {
|
||||
if (target_level == 0 && level == 0) {
|
||||
ulint low_match = 0, up_match = 0;
|
||||
uint16_t low_match = 0, up_match = 0;
|
||||
|
||||
found = false;
|
||||
|
||||
|
@ -564,8 +564,8 @@ dberr_t rtr_search_to_nth_level(btr_cur_t *cur, que_thr_t *thr,
|
|||
MEM_UNDEFINED(&cur->up_bytes, sizeof cur->up_bytes);
|
||||
MEM_UNDEFINED(&cur->low_match, sizeof cur->low_match);
|
||||
MEM_UNDEFINED(&cur->low_bytes, sizeof cur->low_bytes);
|
||||
ut_d(cur->up_match= ULINT_UNDEFINED);
|
||||
ut_d(cur->low_match= ULINT_UNDEFINED);
|
||||
ut_d(cur->up_match= uint16_t(~0U));
|
||||
ut_d(cur->low_match= uint16_t(~0U));
|
||||
|
||||
const bool latch_by_caller= latch_mode & BTR_ALREADY_S_LATCHED;
|
||||
|
||||
|
@ -577,8 +577,9 @@ dberr_t rtr_search_to_nth_level(btr_cur_t *cur, que_thr_t *thr,
|
|||
ut_ad(!latch_by_caller || latch_mode == BTR_SEARCH_LEAF ||
|
||||
latch_mode == BTR_MODIFY_LEAF);
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
cur->flag= BTR_CUR_BINARY;
|
||||
|
||||
#endif
|
||||
#ifndef BTR_CUR_ADAPT
|
||||
buf_block_t *guess= nullptr;
|
||||
#else
|
||||
|
@ -617,7 +618,7 @@ dberr_t rtr_search_to_nth_level(btr_cur_t *cur, que_thr_t *thr,
|
|||
/* Start with the root page. */
|
||||
page_id_t page_id(index->table->space_id, index->page);
|
||||
|
||||
ulint up_match= 0, up_bytes= 0, low_match= 0, low_bytes= 0;
|
||||
uint16_t up_match= 0, up_bytes= 0, low_match= 0, low_bytes= 0;
|
||||
ulint height= ULINT_UNDEFINED;
|
||||
|
||||
/* We use these modified search modes on non-leaf levels of the
|
||||
|
@ -632,14 +633,8 @@ dberr_t rtr_search_to_nth_level(btr_cur_t *cur, que_thr_t *thr,
|
|||
page_mode= PAGE_CUR_LE;
|
||||
break;
|
||||
default:
|
||||
#ifdef PAGE_CUR_LE_OR_EXTENDS
|
||||
ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE
|
||||
|| RTREE_SEARCH_MODE(mode)
|
||||
|| mode == PAGE_CUR_LE_OR_EXTENDS);
|
||||
#else /* PAGE_CUR_LE_OR_EXTENDS */
|
||||
ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE
|
||||
|| RTREE_SEARCH_MODE(mode));
|
||||
#endif /* PAGE_CUR_LE_OR_EXTENDS */
|
||||
ut_ad(mode == PAGE_CUR_L || mode == PAGE_CUR_LE ||
|
||||
RTREE_SEARCH_MODE(mode));
|
||||
page_mode= mode;
|
||||
break;
|
||||
}
|
||||
|
@ -973,9 +968,9 @@ dberr_t rtr_search_to_nth_level(btr_cur_t *cur, que_thr_t *thr,
|
|||
cur->up_match= up_match;
|
||||
cur->up_bytes= up_bytes;
|
||||
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != ULINT_UNDEFINED || mode != PAGE_CUR_LE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_GE);
|
||||
ut_ad(up_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
ut_ad(low_match != uint16_t(~0U) || mode != PAGE_CUR_LE);
|
||||
}
|
||||
|
||||
goto func_exit;
|
||||
|
@ -1685,7 +1680,7 @@ rtr_cur_restore_position(
|
|||
ut_ad(r_cursor == node->cursor);
|
||||
|
||||
search_again:
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
page_cursor->block = buf_page_get_gen(
|
||||
page_id_t(index->table->space_id, page_no),
|
||||
|
|
|
@ -1490,7 +1490,7 @@ static void innodb_drop_database(handlerton*, char *path)
|
|||
|
||||
dfield_t dfield;
|
||||
dtuple_t tuple{
|
||||
0,1,1,&dfield,0,nullptr
|
||||
0,1,1,0,&dfield,nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif
|
||||
|
|
|
@ -3611,9 +3611,9 @@ innobase_row_to_mysql(
|
|||
|
||||
/* The InnoDB row may contain an extra FTS_DOC_ID column at the end. */
|
||||
ut_ad(row->n_fields == dict_table_get_n_cols(itab));
|
||||
ut_ad(n_fields == row->n_fields - DATA_N_SYS_COLS
|
||||
+ dict_table_get_n_v_cols(itab)
|
||||
- !!(DICT_TF2_FLAG_IS_SET(itab, DICT_TF2_FTS_HAS_DOC_ID)));
|
||||
ut_ad(row->n_fields == n_fields + DATA_N_SYS_COLS
|
||||
- dict_table_get_n_v_cols(itab)
|
||||
+ !!(DICT_TF2_FLAG_IS_SET(itab, DICT_TF2_FTS_HAS_DOC_ID)));
|
||||
|
||||
for (uint i = 0; i < n_fields; i++) {
|
||||
Field* field = table->field[i];
|
||||
|
@ -4679,11 +4679,9 @@ innobase_build_col_map(
|
|||
DBUG_ENTER("innobase_build_col_map");
|
||||
DBUG_ASSERT(altered_table != table);
|
||||
DBUG_ASSERT(new_table != old_table);
|
||||
DBUG_ASSERT(dict_table_get_n_cols(new_table)
|
||||
+ dict_table_get_n_v_cols(new_table)
|
||||
DBUG_ASSERT(unsigned(new_table->n_cols + new_table->n_v_cols)
|
||||
>= altered_table->s->fields + DATA_N_SYS_COLS);
|
||||
DBUG_ASSERT(dict_table_get_n_cols(old_table)
|
||||
+ dict_table_get_n_v_cols(old_table)
|
||||
DBUG_ASSERT(unsigned(old_table->n_cols + old_table->n_v_cols)
|
||||
>= table->s->fields + DATA_N_SYS_COLS
|
||||
|| ha_innobase::omits_virtual_cols(*table->s));
|
||||
DBUG_ASSERT(!!defaults == !!(ha_alter_info->handler_flags
|
||||
|
|
|
@ -188,7 +188,7 @@ static dtuple_t *ibuf_entry_build(const rec_t *ibuf_rec, ulint not_redundant,
|
|||
const byte* data;
|
||||
ulint len;
|
||||
|
||||
tuple = dtuple_create(heap, n_fields);
|
||||
tuple = dtuple_create(heap, uint16_t(n_fields));
|
||||
|
||||
index = dict_mem_index_create(
|
||||
dict_table_t::create({C_STRING_WITH_LEN("")}, nullptr,
|
||||
|
@ -393,7 +393,7 @@ ibuf_insert_to_index_page(
|
|||
return DB_CORRUPTION;
|
||||
}
|
||||
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
page_cur.index = index;
|
||||
page_cur.block = block;
|
||||
|
||||
|
@ -517,7 +517,7 @@ ibuf_set_del_mark(
|
|||
page_cur_t page_cur;
|
||||
page_cur.block = block;
|
||||
page_cur.index = index;
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
ut_ad(dtuple_check_typed(entry));
|
||||
|
||||
|
@ -575,7 +575,7 @@ ibuf_delete(
|
|||
page_cur_t page_cur;
|
||||
page_cur.block = block;
|
||||
page_cur.index = index;
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
|
||||
ut_ad(dtuple_check_typed(entry));
|
||||
ut_ad(!index->is_spatial());
|
||||
|
|
|
@ -645,14 +645,16 @@ struct btr_path_t {
|
|||
|
||||
/** Values for the flag documenting the used search method */
|
||||
enum btr_cur_method {
|
||||
BTR_CUR_HASH = 1, /*!< successful shortcut using
|
||||
BTR_CUR_BINARY, /*!< success using the binary search */
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
BTR_CUR_HASH, /*!< successful shortcut using
|
||||
the hash index */
|
||||
BTR_CUR_HASH_FAIL, /*!< failure using hash, success using
|
||||
binary search: the misleading hash
|
||||
reference is stored in the field
|
||||
hash_node, and might be necessary to
|
||||
update */
|
||||
BTR_CUR_BINARY /*!< success using the binary search */
|
||||
#endif
|
||||
};
|
||||
|
||||
/** The tree cursor: the definition appears here only for the compiler
|
||||
|
@ -663,11 +665,15 @@ struct btr_cur_t {
|
|||
/** The following fields are used in
|
||||
search_leaf() to pass information: */
|
||||
/* @{ */
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
enum btr_cur_method flag; /*!< Search method used */
|
||||
#else
|
||||
static constexpr btr_cur_method flag{BTR_CUR_BINARY};
|
||||
#endif
|
||||
ulint tree_height; /*!< Tree height if the search is done
|
||||
for a pessimistic insert or update
|
||||
operation */
|
||||
ulint up_match; /*!< If the search mode was PAGE_CUR_LE,
|
||||
uint16_t up_match; /*!< If the search mode was PAGE_CUR_LE,
|
||||
the number of matched fields to the
|
||||
the first user record to the right of
|
||||
the cursor record after search_leaf();
|
||||
|
@ -680,27 +686,26 @@ struct btr_cur_t {
|
|||
record if that record is on a
|
||||
different leaf page! (See the note in
|
||||
row_ins_duplicate_error_in_clust.) */
|
||||
ulint up_bytes; /*!< number of matched bytes to the
|
||||
uint16_t up_bytes; /*!< number of matched bytes to the
|
||||
right at the time cursor positioned;
|
||||
only used internally in searches: not
|
||||
defined after the search */
|
||||
ulint low_match; /*!< if search mode was PAGE_CUR_LE,
|
||||
uint16_t low_match; /*!< if search mode was PAGE_CUR_LE,
|
||||
the number of matched fields to the
|
||||
first user record AT THE CURSOR or
|
||||
to the left of it after search_leaf();
|
||||
NOT defined for PAGE_CUR_GE or any
|
||||
other search modes; see also the NOTE
|
||||
in up_match! */
|
||||
ulint low_bytes; /*!< number of matched bytes to the
|
||||
uint16_t low_bytes; /*!< number of matched bytes to the
|
||||
left at the time cursor positioned;
|
||||
only used internally in searches: not
|
||||
defined after the search */
|
||||
ulint n_fields; /*!< prefix length used in a hash
|
||||
search if hash_node != NULL */
|
||||
ulint n_bytes; /*!< hash prefix bytes if hash_node !=
|
||||
NULL */
|
||||
ulint fold; /*!< fold value used in the search if
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
uint32_t n_bytes_fields; /*!< prefix used in a hash search */
|
||||
uint32_t fold; /*!< fold value used in the search if
|
||||
flag is BTR_CUR_HASH */
|
||||
#endif
|
||||
/* @} */
|
||||
rtr_info_t* rtr_info; /*!< rtree search info */
|
||||
btr_cur_t() { memset((void*) this, 0, sizeof *this); }
|
||||
|
@ -745,6 +750,16 @@ struct btr_cur_t {
|
|||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
void search_info_update() const noexcept;
|
||||
|
||||
/** Check if a guessed position for a tree cursor is correct.
|
||||
@param tuple search key
|
||||
@param mode PAGE_CUR_L, PAGE_CUR_LE, PAGE_CUR_G, PAGE_CUR_GE
|
||||
@param comp nonzero if ROW_FORMAT=REDUNDANT is not being used
|
||||
@retval true on mismatch or corruption
|
||||
@retval false on a match; if mode=PAGE_CUR_LE, then up_match,low_match
|
||||
will be set correctly. */
|
||||
bool check_mismatch(const dtuple_t& tuple, page_cur_mode_t mode, ulint comp)
|
||||
noexcept;
|
||||
#endif
|
||||
};
|
||||
|
||||
|
|
|
@ -59,7 +59,7 @@ btr_pcur_get_up_match(
|
|||
|
||||
btr_cursor = btr_pcur_get_btr_cur(cursor);
|
||||
|
||||
ut_ad(btr_cursor->up_match != ULINT_UNDEFINED);
|
||||
ut_ad(btr_cursor->up_match != uint16_t(~0U));
|
||||
|
||||
return(btr_cursor->up_match);
|
||||
}
|
||||
|
@ -80,7 +80,7 @@ btr_pcur_get_low_match(
|
|||
|| (cursor->pos_state == BTR_PCUR_IS_POSITIONED));
|
||||
|
||||
btr_cursor = btr_pcur_get_btr_cur(cursor);
|
||||
ut_ad(btr_cursor->low_match != ULINT_UNDEFINED);
|
||||
ut_ad(btr_cursor->low_match != uint16_t(~0U));
|
||||
|
||||
return(btr_cursor->low_match);
|
||||
}
|
||||
|
|
|
@ -28,14 +28,14 @@ Created 2/17/1996 Heikki Tuuri
|
|||
|
||||
#include "dict0dict.h"
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
#include "buf0buf.h"
|
||||
# include "buf0buf.h"
|
||||
|
||||
#ifdef UNIV_PFS_RWLOCK
|
||||
# ifdef UNIV_PFS_RWLOCK
|
||||
extern mysql_pfs_key_t btr_search_latch_key;
|
||||
#endif /* UNIV_PFS_RWLOCK */
|
||||
# endif /* UNIV_PFS_RWLOCK */
|
||||
|
||||
#define btr_search_sys_create() btr_search.create()
|
||||
#define btr_search_sys_free() btr_search.free()
|
||||
# define btr_search_sys_create() btr_search.create()
|
||||
# define btr_search_sys_free() btr_search.free()
|
||||
|
||||
/** Tries to guess the right search position based on the hash search info
|
||||
of the index. Note that if mode is PAGE_CUR_LE, which is used in inserts,
|
||||
|
@ -52,8 +52,8 @@ bool
|
|||
btr_search_guess_on_hash(
|
||||
dict_index_t* index,
|
||||
const dtuple_t* tuple,
|
||||
ulint mode,
|
||||
ulint latch_mode,
|
||||
page_cur_mode_t mode,
|
||||
btr_latch_mode latch_mode,
|
||||
btr_cur_t* cursor,
|
||||
mtr_t* mtr) noexcept;
|
||||
|
||||
|
@ -61,34 +61,27 @@ btr_search_guess_on_hash(
|
|||
If new_block is already hashed, then any hash index for block is dropped.
|
||||
If new_block is not hashed, and block is hashed, then a new hash index is
|
||||
built to new_block with the same parameters as block.
|
||||
@param[in,out] new_block destination page
|
||||
@param[in,out] block source page (subject to deletion later) */
|
||||
@param new_block destination page
|
||||
@param block source page (subject to deletion later) */
|
||||
void btr_search_move_or_delete_hash_entries(buf_block_t *new_block,
|
||||
buf_block_t *block) noexcept;
|
||||
|
||||
/** Drop any adaptive hash index entries that point to an index page.
|
||||
@param[in,out] block block containing index page, s- or x-latched, or an
|
||||
index page for which we know that
|
||||
block->buf_fix_count == 0 or it is an index page which
|
||||
has already been removed from the buf_pool.page_hash
|
||||
i.e.: it is in state BUF_BLOCK_REMOVE_HASH
|
||||
@param[in] garbage_collect drop ahi only if the index is marked
|
||||
as freed */
|
||||
@param block latched block containing index page, or a buffer-unfixed
|
||||
index page or a block in state BUF_BLOCK_REMOVE_HASH
|
||||
@param garbage_collect drop ahi only if the index is marked as freed */
|
||||
void btr_search_drop_page_hash_index(buf_block_t* block,
|
||||
bool garbage_collect) noexcept;
|
||||
bool garbage_collect) noexcept;
|
||||
|
||||
/** Drop possible adaptive hash index entries when a page is evicted
|
||||
from the buffer pool or freed in a file, or the index is being dropped.
|
||||
@param[in] page_id page id */
|
||||
@param page_id page identifier of the being-dropped page */
|
||||
void btr_search_drop_page_hash_when_freed(const page_id_t page_id) noexcept;
|
||||
|
||||
/** Update the page hash index after a single record is inserted on a page.
|
||||
@param cursor cursor which was positioned before the inserted record */
|
||||
void btr_search_update_hash_node_on_insert(btr_cur_t *cursor) noexcept;
|
||||
|
||||
/** Update the page hash index after a single record is inserted on a page.
|
||||
@param cursor cursor which was positioned before the inserted record */
|
||||
void btr_search_update_hash_on_insert(btr_cur_t *cursor) noexcept;
|
||||
@param cursor cursor which was positioned before the inserted record
|
||||
@param reorg whether the page was reorganized */
|
||||
void btr_search_update_hash_on_insert(btr_cur_t *cursor, bool reorg) noexcept;
|
||||
|
||||
/** Updates the page hash index before a single record is deleted from a page.
|
||||
@param cursor cursor positioned on the to-be-deleted record */
|
||||
|
@ -109,7 +102,9 @@ struct ahi_node;
|
|||
/** The hash index system */
|
||||
struct btr_sea
|
||||
{
|
||||
/** the actual value of innodb_adaptive_hash_index */
|
||||
/** the actual value of innodb_adaptive_hash_index, protected by
|
||||
all partition::latch. Note that if buf_block_t::index is not nullptr
|
||||
while a thread is holding a partition::latch, then also this must hold. */
|
||||
Atomic_relaxed<bool> enabled;
|
||||
|
||||
/** Disable the adaptive hash search system and empty the index. */
|
||||
|
@ -122,9 +117,9 @@ struct btr_sea
|
|||
/** Partition of the hash table */
|
||||
struct partition
|
||||
{
|
||||
/** latch protecting the hash table */
|
||||
/** latch protecting table */
|
||||
alignas(CPU_LEVEL1_DCACHE_LINESIZE) srw_spin_lock latch;
|
||||
/** map of dtuple_fold() or rec_fold() to rec_t* in buf_page_t::frame */
|
||||
/** map of CRC-32C of rec prefix to rec_t* in buf_page_t::frame */
|
||||
hash_table_t table;
|
||||
/** latch protecting blocks, spare; may be acquired while holding latch */
|
||||
srw_mutex blocks_mutex;
|
||||
|
@ -153,22 +148,22 @@ struct btr_sea
|
|||
__attribute__((nonnull))
|
||||
# if defined UNIV_AHI_DEBUG || defined UNIV_DEBUG
|
||||
/** Insert or replace an entry into the hash table.
|
||||
@param fold hash value of rec
|
||||
@param fold CRC-32C of rec prefix
|
||||
@param rec B-tree leaf page record
|
||||
@param block the buffer block that contains rec */
|
||||
void insert(ulint fold, const rec_t *rec, buf_block_t *block) noexcept;
|
||||
void insert(uint32_t fold, const rec_t *rec, buf_block_t *block) noexcept;
|
||||
# else
|
||||
/** Insert or replace an entry into the hash table.
|
||||
@param fold hash value of data
|
||||
@param fold CRC-32C of rec prefix
|
||||
@param rec B-tree leaf page record */
|
||||
void insert(ulint fold, const rec_t *rec) noexcept;
|
||||
void insert(uint32_t fold, const rec_t *rec) noexcept;
|
||||
# endif
|
||||
|
||||
/** Delete a pointer to a record if it exists.
|
||||
@param fold hash value of rec
|
||||
@param rec B-tree leaf page record
|
||||
@param fold CRC-32C of rec prefix
|
||||
@param rec B-tree leaf page record
|
||||
@return whether a record existed and was removed */
|
||||
inline bool erase(ulint fold, const rec_t *rec) noexcept;
|
||||
inline bool erase(uint32_t fold, const rec_t *rec) noexcept;
|
||||
};
|
||||
|
||||
/** innodb_adaptive_hash_index_parts */
|
||||
|
@ -200,9 +195,9 @@ extern btr_sea btr_search;
|
|||
|
||||
# ifdef UNIV_SEARCH_PERF_STAT
|
||||
/** Number of successful adaptive hash index lookups */
|
||||
extern ulint btr_search_n_succ;
|
||||
extern ulint btr_search_n_succ;
|
||||
/** Number of failed adaptive hash index lookups */
|
||||
extern ulint btr_search_n_hash_fail;
|
||||
extern ulint btr_search_n_hash_fail;
|
||||
# endif /* UNIV_SEARCH_PERF_STAT */
|
||||
#else /* BTR_CUR_HASH_ADAPT */
|
||||
# define btr_search_sys_create()
|
||||
|
|
|
@ -857,91 +857,37 @@ struct buf_block_t{
|
|||
x-latch on the block */
|
||||
/* @} */
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
/** @name Hash search fields (unprotected)
|
||||
NOTE that these fields are NOT protected by any semaphore! */
|
||||
/* @{ */
|
||||
/** @name Hash search fields */
|
||||
/* @{ */
|
||||
/** flag: (true=first, false=last) identical-prefix key is included */
|
||||
static constexpr uint32_t LEFT_SIDE= 1U << 31;
|
||||
|
||||
volatile uint16_t n_bytes; /*!< recommended prefix length for hash
|
||||
search: number of bytes in
|
||||
an incomplete last field */
|
||||
volatile uint16_t n_fields; /*!< recommended prefix length for hash
|
||||
search: number of full fields */
|
||||
uint16_t n_hash_helps; /*!< counter which controls building
|
||||
of a new hash index for the page */
|
||||
volatile bool left_side; /*!< true or false, depending on
|
||||
whether the leftmost record of several
|
||||
records with the same prefix should be
|
||||
indexed in the hash index */
|
||||
/* @} */
|
||||
|
||||
/** @name Hash search fields
|
||||
These 5 fields may only be modified when:
|
||||
we are holding the appropriate x-latch in btr_search_latches[], and
|
||||
one of the following holds:
|
||||
(1) in_file(), and we are holding lock in any mode, or
|
||||
(2) !is_read_fixed()&&(state()>=UNFIXED||state()==REMOVE_HASH).
|
||||
|
||||
An exception to this is when we init or create a page
|
||||
in the buffer pool in buf0buf.cc.
|
||||
|
||||
Another exception for buf_pool_t::clear_hash_index() is that
|
||||
assigning block->index = NULL (and block->n_pointers = 0)
|
||||
is allowed whenever all AHI latches are exclusively locked.
|
||||
|
||||
Another exception is that ha_insert_for_fold() may
|
||||
decrement n_pointers without holding the appropriate latch
|
||||
in btr_search.parts. Thus, n_pointers must be
|
||||
protected by atomic memory access.
|
||||
|
||||
This implies that the fields may be read without race
|
||||
condition whenever any of the following hold:
|
||||
- the btr_search.parts.latch is being held, or
|
||||
- state() == NOT_USED || state() == MEMORY,
|
||||
and holding some latch prevents the state from changing to that.
|
||||
|
||||
Some use of assert_block_ahi_empty() or assert_block_ahi_valid()
|
||||
is prone to race conditions while buf_pool_t::clear_hash_index() is
|
||||
executing (the adaptive hash index is being disabled). Such use
|
||||
is explicitly commented. */
|
||||
|
||||
/* @{ */
|
||||
/** AHI parameters: LEFT_SIDE | prefix_bytes << 16 | prefix_fields.
|
||||
Protected by the btr_sea::partition::latch and
|
||||
(1) in_file(), and we are holding lock in any mode, or
|
||||
(2) !is_read_fixed()&&(state()>=UNFIXED||state()==REMOVE_HASH). */
|
||||
Atomic_relaxed<uint32_t> ahi_left_bytes_fields;
|
||||
|
||||
/** counter which controls building of a new hash index for the page;
|
||||
may be nonzero even if !index */
|
||||
Atomic_relaxed<uint16_t> n_hash_helps;
|
||||
# if defined UNIV_AHI_DEBUG || defined UNIV_DEBUG
|
||||
Atomic_counter<ulint>
|
||||
n_pointers; /*!< used in debugging: the number of
|
||||
pointers in the adaptive hash index
|
||||
pointing to this frame */
|
||||
# define assert_block_ahi_empty(block) \
|
||||
ut_a((block)->n_pointers == 0)
|
||||
# define assert_block_ahi_empty_on_init(block) do { \
|
||||
MEM_MAKE_DEFINED(&(block)->n_pointers, sizeof (block)->n_pointers); \
|
||||
assert_block_ahi_empty(block); \
|
||||
} while (0)
|
||||
# define assert_block_ahi_valid(block) \
|
||||
ut_a((block)->index || (block)->n_pointers == 0)
|
||||
/** number of pointers from the btr_sea::partition::table;
|
||||
!n_pointers == !index */
|
||||
Atomic_counter<uint16_t> n_pointers;
|
||||
# define assert_block_ahi_empty(block) ut_a(!(block)->n_pointers)
|
||||
# define assert_block_ahi_valid(b) ut_a((b)->index || !(b)->n_pointers)
|
||||
# else /* UNIV_AHI_DEBUG || UNIV_DEBUG */
|
||||
# define assert_block_ahi_empty(block) /* nothing */
|
||||
# define assert_block_ahi_empty_on_init(block) /* nothing */
|
||||
# define assert_block_ahi_valid(block) /* nothing */
|
||||
# endif /* UNIV_AHI_DEBUG || UNIV_DEBUG */
|
||||
unsigned curr_n_fields:10;/*!< prefix length for hash indexing:
|
||||
number of full fields */
|
||||
unsigned curr_n_bytes:15;/*!< number of bytes in hash
|
||||
indexing */
|
||||
unsigned curr_left_side:1;/*!< TRUE or FALSE in hash indexing */
|
||||
dict_index_t* index; /*!< Index for which the
|
||||
adaptive hash index has been
|
||||
created, or NULL if the page
|
||||
does not exist in the
|
||||
index. Note that it does not
|
||||
guarantee that the index is
|
||||
complete, though: there may
|
||||
have been hash collisions,
|
||||
record deletions, etc. */
|
||||
/* @} */
|
||||
/** index for which the adaptive hash index has been created,
|
||||
or nullptr if the page does not exist in the index.
|
||||
Protected by btr_sea::partition::latch. */
|
||||
Atomic_relaxed<dict_index_t*> index;
|
||||
/* @} */
|
||||
#else /* BTR_CUR_HASH_ADAPT */
|
||||
# define assert_block_ahi_empty(block) /* nothing */
|
||||
# define assert_block_ahi_empty_on_init(block) /* nothing */
|
||||
# define assert_block_ahi_valid(block) /* nothing */
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
void fix() { page.fix(); }
|
||||
|
|
|
@ -193,7 +193,7 @@ dtuple_set_info_bits(
|
|||
Gets number of fields used in record comparisons.
|
||||
@return number of fields used in comparisons in rem0cmp.* */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
dtuple_get_n_fields_cmp(
|
||||
/*====================*/
|
||||
const dtuple_t* tuple) /*!< in: tuple */
|
||||
|
@ -270,6 +270,7 @@ dtuple_create_with_vcol(
|
|||
/*********************************************************************//**
|
||||
Sets number of fields used in a tuple. Normally this is set in
|
||||
dtuple_create, but if you want later to set it smaller, you can use this. */
|
||||
inline
|
||||
void
|
||||
dtuple_set_n_fields(
|
||||
/*================*/
|
||||
|
@ -316,20 +317,6 @@ dtuple_get_n_ext(
|
|||
/*=============*/
|
||||
const dtuple_t* tuple) /*!< in: tuple */
|
||||
MY_ATTRIBUTE((nonnull));
|
||||
/** Fold a prefix given as the number of fields of a tuple.
|
||||
@param[in] tuple index record
|
||||
@param[in] n_fields number of complete fields to fold
|
||||
@param[in] n_bytes number of bytes to fold in the last field
|
||||
@param[in] index_id index tree ID
|
||||
@return the folded value */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
dtuple_fold(
|
||||
const dtuple_t* tuple,
|
||||
ulint n_fields,
|
||||
ulint n_bytes,
|
||||
index_id_t tree_id)
|
||||
MY_ATTRIBUTE((warn_unused_result));
|
||||
/*******************************************************************//**
|
||||
Sets types of fields binary in a tuple. */
|
||||
UNIV_INLINE
|
||||
|
@ -499,20 +486,20 @@ struct dfield_t{
|
|||
|
||||
/** Structure for an SQL data tuple of fields (logical record) */
|
||||
struct dtuple_t {
|
||||
ulint info_bits; /*!< info bits of an index record:
|
||||
byte info_bits; /*!< info bits of an index record:
|
||||
the default is 0; this field is used
|
||||
if an index record is built from
|
||||
a data tuple */
|
||||
ulint n_fields; /*!< number of fields in dtuple */
|
||||
ulint n_fields_cmp; /*!< number of fields which should
|
||||
uint16_t n_fields; /*!< number of fields in dtuple */
|
||||
uint16_t n_fields_cmp; /*!< number of fields which should
|
||||
be used in comparison services
|
||||
of rem0cmp.*; the index search
|
||||
is performed by comparing only these
|
||||
fields, others are ignored; the
|
||||
default value in dtuple creation is
|
||||
the same value as n_fields */
|
||||
uint16_t n_v_fields; /*!< number of virtual fields */
|
||||
dfield_t* fields; /*!< fields */
|
||||
ulint n_v_fields; /*!< number of virtual fields */
|
||||
dfield_t* v_fields; /*!< fields on virtual column */
|
||||
#ifdef UNIV_DEBUG
|
||||
ulint magic_n; /*!< magic number, used in
|
||||
|
@ -574,7 +561,7 @@ struct dtuple_t {
|
|||
inline void copy_field_types(const dict_index_t &index);
|
||||
};
|
||||
|
||||
inline ulint dtuple_get_n_fields(const dtuple_t* tuple)
|
||||
inline uint16_t dtuple_get_n_fields(const dtuple_t* tuple)
|
||||
{ return tuple->n_fields; }
|
||||
inline dtype_t* dfield_get_type(dfield_t* field) { return &field->type; }
|
||||
inline const dtype_t* dfield_get_type(const dfield_t* field)
|
||||
|
@ -608,7 +595,7 @@ inline void dfield_set_ext(dfield_t* field) { field->ext = 1; }
|
|||
/** Gets number of virtual fields in a data tuple.
|
||||
@param[in] tuple dtuple to check
|
||||
@return number of fields */
|
||||
inline ulint
|
||||
inline uint16_t
|
||||
dtuple_get_n_v_fields(const dtuple_t* tuple) { return tuple->n_v_fields; }
|
||||
|
||||
inline const dfield_t* dtuple_get_nth_field(const dtuple_t* tuple, ulint n)
|
||||
|
|
|
@ -238,14 +238,14 @@ dtuple_set_info_bits(
|
|||
dtuple_t* tuple, /*!< in: tuple */
|
||||
ulint info_bits) /*!< in: info bits */
|
||||
{
|
||||
tuple->info_bits = info_bits;
|
||||
tuple->info_bits = byte(info_bits);
|
||||
}
|
||||
|
||||
/*********************************************************************//**
|
||||
Gets number of fields used in record comparisons.
|
||||
@return number of fields used in comparisons in rem0cmp.* */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
dtuple_get_n_fields_cmp(
|
||||
/*====================*/
|
||||
const dtuple_t* tuple) /*!< in: tuple */
|
||||
|
@ -264,7 +264,7 @@ dtuple_set_n_fields_cmp(
|
|||
comparisons in rem0cmp.* */
|
||||
{
|
||||
ut_ad(n_fields_cmp <= tuple->n_fields);
|
||||
tuple->n_fields_cmp = n_fields_cmp;
|
||||
tuple->n_fields_cmp = uint16_t(n_fields_cmp);
|
||||
}
|
||||
|
||||
/** Creates a data tuple from an already allocated chunk of memory.
|
||||
|
@ -291,9 +291,9 @@ dtuple_create_from_mem(
|
|||
|
||||
tuple = (dtuple_t*) buf;
|
||||
tuple->info_bits = 0;
|
||||
tuple->n_fields = n_fields;
|
||||
tuple->n_v_fields = n_v_fields;
|
||||
tuple->n_fields_cmp = n_fields;
|
||||
tuple->n_fields = uint16_t(n_fields);
|
||||
tuple->n_v_fields = uint16_t(n_v_fields);
|
||||
tuple->n_fields_cmp = uint16_t(n_fields);
|
||||
tuple->fields = (dfield_t*) &tuple[1];
|
||||
if (n_v_fields > 0) {
|
||||
tuple->v_fields = &tuple->fields[n_fields];
|
||||
|
@ -398,6 +398,12 @@ dtuple_create_with_vcol(
|
|||
return(tuple);
|
||||
}
|
||||
|
||||
inline void dtuple_set_n_fields(dtuple_t *tuple, ulint n_fields)
|
||||
{
|
||||
tuple->n_fields= uint16_t(n_fields);
|
||||
tuple->n_fields_cmp= uint16_t(n_fields);
|
||||
}
|
||||
|
||||
/** Copies a data tuple's virtual fields to another. This is a shallow copy;
|
||||
@param[in,out] d_tuple destination tuple
|
||||
@param[in] s_tuple source tuple */
|
||||
|
@ -432,7 +438,7 @@ dtuple_copy(
|
|||
ulint n_fields = dtuple_get_n_fields(tuple);
|
||||
ulint n_v_fields = dtuple_get_n_v_fields(tuple);
|
||||
dtuple_t* new_tuple = dtuple_create_with_vcol(
|
||||
heap, n_fields, n_v_fields);
|
||||
heap, tuple->n_fields, tuple->n_v_fields);
|
||||
ulint i;
|
||||
|
||||
for (i = 0; i < n_fields; i++) {
|
||||
|
@ -527,63 +533,6 @@ dtuple_set_types_binary(
|
|||
}
|
||||
}
|
||||
|
||||
/** Fold a prefix given as the number of fields of a tuple.
|
||||
@param[in] tuple index record
|
||||
@param[in] n_fields number of complete fields to fold
|
||||
@param[in] n_bytes number of bytes to fold in the last field
|
||||
@param[in] index_id index tree ID
|
||||
@return the folded value */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
dtuple_fold(
|
||||
const dtuple_t* tuple,
|
||||
ulint n_fields,
|
||||
ulint n_bytes,
|
||||
index_id_t tree_id)
|
||||
{
|
||||
const dfield_t* field;
|
||||
ulint i;
|
||||
const byte* data;
|
||||
ulint len;
|
||||
ulint fold;
|
||||
|
||||
ut_ad(tuple);
|
||||
ut_ad(tuple->magic_n == DATA_TUPLE_MAGIC_N);
|
||||
ut_ad(dtuple_check_typed(tuple));
|
||||
|
||||
fold = ut_fold_ull(tree_id);
|
||||
|
||||
for (i = 0; i < n_fields; i++) {
|
||||
field = dtuple_get_nth_field(tuple, i);
|
||||
|
||||
data = (const byte*) dfield_get_data(field);
|
||||
len = dfield_get_len(field);
|
||||
|
||||
if (len != UNIV_SQL_NULL) {
|
||||
fold = ut_fold_ulint_pair(fold,
|
||||
ut_fold_binary(data, len));
|
||||
}
|
||||
}
|
||||
|
||||
if (n_bytes > 0) {
|
||||
field = dtuple_get_nth_field(tuple, i);
|
||||
|
||||
data = (const byte*) dfield_get_data(field);
|
||||
len = dfield_get_len(field);
|
||||
|
||||
if (len != UNIV_SQL_NULL) {
|
||||
if (len > n_bytes) {
|
||||
len = n_bytes;
|
||||
}
|
||||
|
||||
fold = ut_fold_ulint_pair(fold,
|
||||
ut_fold_binary(data, len));
|
||||
}
|
||||
}
|
||||
|
||||
return(fold);
|
||||
}
|
||||
|
||||
/**********************************************************************//**
|
||||
Writes an SQL null field full of zeros. */
|
||||
UNIV_INLINE
|
||||
|
|
|
@ -661,7 +661,7 @@ Gets the number of all non-virtual columns (also system) in a table
|
|||
in the dictionary cache.
|
||||
@return number of columns of a table */
|
||||
UNIV_INLINE
|
||||
unsigned
|
||||
uint16_t
|
||||
dict_table_get_n_cols(
|
||||
/*==================*/
|
||||
const dict_table_t* table) /*!< in: table */
|
||||
|
@ -671,7 +671,7 @@ dict_table_get_n_cols(
|
|||
@param[in] table the table to check
|
||||
@return number of virtual columns of a table */
|
||||
UNIV_INLINE
|
||||
unsigned
|
||||
uint16_t
|
||||
dict_table_get_n_v_cols(
|
||||
const dict_table_t* table);
|
||||
|
||||
|
|
|
@ -264,7 +264,7 @@ Gets the number of all non-virtual columns (also system) in a table
|
|||
in the dictionary cache.
|
||||
@return number of non-virtual columns of a table */
|
||||
UNIV_INLINE
|
||||
unsigned
|
||||
uint16_t
|
||||
dict_table_get_n_cols(
|
||||
/*==================*/
|
||||
const dict_table_t* table) /*!< in: table */
|
||||
|
@ -277,7 +277,7 @@ dict_table_get_n_cols(
|
|||
@param[in] table the table to check
|
||||
@return number of virtual columns of a table */
|
||||
UNIV_INLINE
|
||||
unsigned
|
||||
uint16_t
|
||||
dict_table_get_n_v_cols(
|
||||
const dict_table_t* table)
|
||||
{
|
||||
|
|
|
@ -1080,7 +1080,7 @@ struct dict_index_t {
|
|||
|
||||
/** number of consecutive searches which would have succeeded, or
|
||||
did succeed, using the hash index; the range is 0
|
||||
.. BTR_SEARCH_BUILD_LIMIT + 5 */
|
||||
.. BTR_SEARCH_BUILD_LIMIT */
|
||||
Atomic_relaxed<uint8_t> n_hash_potential{0};
|
||||
|
||||
/** whether the last search would have succeeded, or
|
||||
|
@ -1089,15 +1089,8 @@ struct dict_index_t {
|
|||
search, and the calculation itself is not always accurate! */
|
||||
Atomic_relaxed<bool> last_hash_succ{false};
|
||||
|
||||
/** recommended full field prefix */
|
||||
uint16_t n_fields= 1;
|
||||
/** recommended number of bytes in an incomplete field */
|
||||
uint16_t n_bytes= 0;
|
||||
|
||||
/** whether the leftmost record of several records with the same
|
||||
prefix should be indexed */
|
||||
bool left_side= true;
|
||||
|
||||
/** recommended parameters; @see buf_block_t::left_bytes_fields */
|
||||
Atomic_relaxed<uint32_t> left_bytes_fields{buf_block_t::LEFT_SIDE | 1};
|
||||
/** number of buf_block_t::index pointers to this index */
|
||||
Atomic_counter<size_t> ref_count{0};
|
||||
|
||||
|
@ -2578,7 +2571,7 @@ public:
|
|||
bool is_stats_table() const;
|
||||
|
||||
/** @return number of unique columns in FTS_DOC_ID index */
|
||||
unsigned fts_n_uniq() const { return versioned() ? 2 : 1; }
|
||||
uint16_t fts_n_uniq() const { return versioned() ? 2 : 1; }
|
||||
|
||||
/** @return the index for that starts with a specific column */
|
||||
dict_index_t *get_index(const dict_col_t &col) const;
|
||||
|
|
|
@ -221,38 +221,34 @@ page_cur_search_with_match(
|
|||
page_cur_mode_t mode, /*!< in: PAGE_CUR_L,
|
||||
PAGE_CUR_LE, PAGE_CUR_G, or
|
||||
PAGE_CUR_GE */
|
||||
ulint* iup_matched_fields,
|
||||
uint16_t* iup_matched_fields,
|
||||
/*!< in/out: already matched
|
||||
fields in upper limit record */
|
||||
ulint* ilow_matched_fields,
|
||||
uint16_t* ilow_matched_fields,
|
||||
/*!< in/out: already matched
|
||||
fields in lower limit record */
|
||||
page_cur_t* cursor, /*!< in/out: page cursor */
|
||||
rtr_info_t* rtr_info);/*!< in/out: rtree search stack */
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
MY_ATTRIBUTE((warn_unused_result))
|
||||
|
||||
/** Search the right position for a page cursor.
|
||||
@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[in,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);
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
@param tuple search key
|
||||
@param mode search mode
|
||||
@param iup_fields matched fields in the upper limit record
|
||||
@param ilow_fields matched fields in the low limit record
|
||||
@param cursor page cursor
|
||||
@param iup_bytes matched bytes after iup_fields
|
||||
@param ilow_bytes matched bytes after ilow_fields
|
||||
@return whether the first partially matched field in the lower limit record,
|
||||
or the page is corrupted */
|
||||
bool page_cur_search_with_match_bytes(const dtuple_t &tuple,
|
||||
page_cur_mode_t mode,
|
||||
uint16_t *iup_fields,
|
||||
uint16_t *ilow_fields,
|
||||
page_cur_t *cursor,
|
||||
uint16_t *iup_bytes,
|
||||
uint16_t *ilow_bytes)
|
||||
noexcept;
|
||||
|
||||
/***********************************************************//**
|
||||
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. */
|
||||
|
|
|
@ -72,11 +72,6 @@ enum page_cur_mode_t {
|
|||
PAGE_CUR_L = 3,
|
||||
PAGE_CUR_LE = 4,
|
||||
|
||||
/* PAGE_CUR_LE_OR_EXTENDS = 5,*/ /* This is a search mode used in
|
||||
"column LIKE 'abc%' ORDER BY column DESC";
|
||||
we have to find strings which are <= 'abc' or
|
||||
which extend it */
|
||||
|
||||
/* These search mode is for search R-tree index. */
|
||||
PAGE_CUR_CONTAIN = 7,
|
||||
PAGE_CUR_INTERSECT = 8,
|
||||
|
|
|
@ -55,6 +55,7 @@ cmp_cols_are_equal(
|
|||
@retval positive if data1 is greater than data2 */
|
||||
int cmp_data(ulint mtype, ulint prtype, bool descending,
|
||||
const byte *data1, size_t len1, const byte *data2, size_t len2)
|
||||
noexcept
|
||||
MY_ATTRIBUTE((warn_unused_result));
|
||||
|
||||
/** Compare two data fields.
|
||||
|
@ -148,33 +149,12 @@ inline int cmp_geometry_field(const void *a, const void *b)
|
|||
int cmp_dtuple_rec_with_match_low(const dtuple_t *dtuple, const rec_t *rec,
|
||||
const dict_index_t *index,
|
||||
const rec_offs *offsets,
|
||||
ulint n_cmp, ulint *matched_fields)
|
||||
ulint n_cmp, uint16_t *matched_fields)
|
||||
MY_ATTRIBUTE((nonnull));
|
||||
#define cmp_dtuple_rec_with_match(tuple,rec,index,offsets,fields) \
|
||||
cmp_dtuple_rec_with_match_low( \
|
||||
tuple,rec,index,offsets,dtuple_get_n_fields_cmp(tuple),fields)
|
||||
/** Compare a data tuple to a physical record.
|
||||
@param[in] dtuple data tuple
|
||||
@param[in] rec B-tree or R-tree index record
|
||||
@param[in] index index tree
|
||||
@param[in] offsets rec_get_offsets(rec)
|
||||
@param[in,out] matched_fields number of completely matched fields
|
||||
@param[in,out] matched_bytes number of matched bytes in the first
|
||||
field that is not matched
|
||||
@return the comparison result of dtuple and rec
|
||||
@retval 0 if dtuple is equal to rec
|
||||
@retval negative if dtuple is less than rec
|
||||
@retval positive if dtuple is greater than rec */
|
||||
int
|
||||
cmp_dtuple_rec_with_match_bytes(
|
||||
const dtuple_t* dtuple,
|
||||
const rec_t* rec,
|
||||
const dict_index_t* index,
|
||||
const rec_offs* offsets,
|
||||
ulint* matched_fields,
|
||||
ulint* matched_bytes)
|
||||
MY_ATTRIBUTE((warn_unused_result));
|
||||
/** Compare a data tuple to a physical record.
|
||||
@see cmp_dtuple_rec_with_match
|
||||
@param dtuple data tuple
|
||||
@param rec index record
|
||||
|
@ -187,7 +167,7 @@ cmp_dtuple_rec_with_match_bytes(
|
|||
inline int cmp_dtuple_rec(const dtuple_t *dtuple, const rec_t *rec,
|
||||
const dict_index_t *index, const rec_offs *offsets)
|
||||
{
|
||||
ulint matched= 0;
|
||||
uint16_t matched= 0;
|
||||
return cmp_dtuple_rec_with_match(dtuple, rec, index, offsets, &matched);
|
||||
}
|
||||
|
||||
|
|
|
@ -178,7 +178,7 @@ The following function is used to get the number of fields
|
|||
in an old-style record.
|
||||
@return number of data fields */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_n_fields_old(
|
||||
/*=================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
|
@ -353,7 +353,7 @@ The following function is used to get the order number
|
|||
of an old-style record in the heap of the index page.
|
||||
@return heap order number */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_heap_no_old(
|
||||
/*================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
|
@ -363,7 +363,7 @@ The following function is used to get the order number
|
|||
of a new-style record in the heap of the index page.
|
||||
@return heap order number */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_heap_no_new(
|
||||
/*================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
|
|
|
@ -164,7 +164,7 @@ rec_set_bit_field_1(
|
|||
/******************************************************//**
|
||||
Gets a bit field from within 2 bytes. */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_bit_field_2(
|
||||
/*================*/
|
||||
const rec_t* rec, /*!< in: pointer to record origin */
|
||||
|
@ -174,7 +174,7 @@ rec_get_bit_field_2(
|
|||
{
|
||||
ut_ad(rec);
|
||||
|
||||
return((mach_read_from_2(rec - offs) & mask) >> shift);
|
||||
return uint16_t((mach_read_from_2(rec - offs) & mask) >> shift);
|
||||
}
|
||||
|
||||
/******************************************************//**
|
||||
|
@ -307,18 +307,14 @@ The following function is used to get the number of fields
|
|||
in an old-style record.
|
||||
@return number of data fields */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_n_fields_old(
|
||||
/*=================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
{
|
||||
ulint ret;
|
||||
|
||||
ut_ad(rec);
|
||||
|
||||
ret = rec_get_bit_field_2(rec, REC_OLD_N_FIELDS,
|
||||
REC_OLD_N_FIELDS_MASK,
|
||||
REC_OLD_N_FIELDS_SHIFT);
|
||||
uint16_t ret = rec_get_bit_field_2(rec, REC_OLD_N_FIELDS,
|
||||
REC_OLD_N_FIELDS_MASK,
|
||||
REC_OLD_N_FIELDS_SHIFT);
|
||||
ut_ad(ret <= REC_MAX_N_FIELDS);
|
||||
ut_ad(ret > 0);
|
||||
|
||||
|
@ -397,7 +393,7 @@ rec_n_fields_is_sane(
|
|||
/* a record for older SYS_INDEXES table
|
||||
(missing merge_threshold column) is acceptable. */
|
||||
|| (index->table->id == DICT_INDEXES_ID
|
||||
&& n_fields == dtuple_get_n_fields(entry) - 1));
|
||||
&& n_fields + 1 == dtuple_get_n_fields(entry)));
|
||||
}
|
||||
|
||||
/******************************************************//**
|
||||
|
@ -518,7 +514,7 @@ The following function is used to get the order number
|
|||
of an old-style record in the heap of the index page.
|
||||
@return heap order number */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_heap_no_old(
|
||||
/*================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
|
@ -532,7 +528,7 @@ The following function is used to get the order number
|
|||
of a new-style record in the heap of the index page.
|
||||
@return heap order number */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
uint16_t
|
||||
rec_get_heap_no_new(
|
||||
/*================*/
|
||||
const rec_t* rec) /*!< in: physical record */
|
||||
|
|
|
@ -53,49 +53,6 @@ Created 1/20/1994 Heikki Tuuri
|
|||
#define ut_max std::max
|
||||
#define ut_min std::min
|
||||
|
||||
/** Calculate the minimum of two pairs.
|
||||
@param[out] min_hi MSB of the minimum pair
|
||||
@param[out] min_lo LSB of the minimum pair
|
||||
@param[in] a_hi MSB of the first pair
|
||||
@param[in] a_lo LSB of the first pair
|
||||
@param[in] b_hi MSB of the second pair
|
||||
@param[in] b_lo LSB of the second pair */
|
||||
UNIV_INLINE
|
||||
void
|
||||
ut_pair_min(
|
||||
ulint* min_hi,
|
||||
ulint* min_lo,
|
||||
ulint a_hi,
|
||||
ulint a_lo,
|
||||
ulint b_hi,
|
||||
ulint b_lo);
|
||||
/******************************************************//**
|
||||
Compares two ulints.
|
||||
@return 1 if a > b, 0 if a == b, -1 if a < b */
|
||||
UNIV_INLINE
|
||||
int
|
||||
ut_ulint_cmp(
|
||||
/*=========*/
|
||||
ulint a, /*!< in: ulint */
|
||||
ulint b); /*!< in: ulint */
|
||||
/** Compare two pairs of integers.
|
||||
@param[in] a_h more significant part of first pair
|
||||
@param[in] a_l less significant part of first pair
|
||||
@param[in] b_h more significant part of second pair
|
||||
@param[in] b_l less significant part of second pair
|
||||
@return comparison result of (a_h,a_l) and (b_h,b_l)
|
||||
@retval -1 if (a_h,a_l) is less than (b_h,b_l)
|
||||
@retval 0 if (a_h,a_l) is equal to (b_h,b_l)
|
||||
@retval 1 if (a_h,a_l) is greater than (b_h,b_l) */
|
||||
UNIV_INLINE
|
||||
int
|
||||
ut_pair_cmp(
|
||||
ulint a_h,
|
||||
ulint a_l,
|
||||
ulint b_h,
|
||||
ulint b_l)
|
||||
MY_ATTRIBUTE((warn_unused_result));
|
||||
|
||||
/*************************************************************//**
|
||||
Calculates fast the remainder of n/m when m is a power of two.
|
||||
@param n in: numerator
|
||||
|
@ -119,24 +76,6 @@ when m is a power of two. In other words, rounds n up to m * k.
|
|||
template <typename T> inline T ut_calc_align(T n, T m)
|
||||
{ return static_cast<T>(UT_CALC_ALIGN(n, m)); }
|
||||
|
||||
/*************************************************************//**
|
||||
Calculates fast the 2-logarithm of a number, rounded upward to an
|
||||
integer.
|
||||
@return logarithm in the base 2, rounded upward */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
ut_2_log(
|
||||
/*=====*/
|
||||
ulint n); /*!< in: number */
|
||||
/*************************************************************//**
|
||||
Calculates 2 to power n.
|
||||
@return 2 to power n */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
ut_2_exp(
|
||||
/*=====*/
|
||||
ulint n); /*!< in: number */
|
||||
|
||||
/**********************************************************//**
|
||||
Returns the number of milliseconds since some epoch. The
|
||||
value may wrap around. It should only be used for heuristic
|
||||
|
@ -424,7 +363,5 @@ private:
|
|||
|
||||
} // namespace ib
|
||||
|
||||
#include "ut0ut.inl"
|
||||
|
||||
#endif
|
||||
|
||||
|
|
|
@ -1,143 +0,0 @@
|
|||
/*****************************************************************************
|
||||
|
||||
Copyright (c) 1994, 2015, Oracle and/or its affiliates. All Rights Reserved.
|
||||
|
||||
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 include/ut0ut.ic
|
||||
Various utilities
|
||||
|
||||
Created 5/30/1994 Heikki Tuuri
|
||||
*******************************************************************/
|
||||
|
||||
#include <algorithm>
|
||||
|
||||
/** Calculate the minimum of two pairs.
|
||||
@param[out] min_hi MSB of the minimum pair
|
||||
@param[out] min_lo LSB of the minimum pair
|
||||
@param[in] a_hi MSB of the first pair
|
||||
@param[in] a_lo LSB of the first pair
|
||||
@param[in] b_hi MSB of the second pair
|
||||
@param[in] b_lo LSB of the second pair */
|
||||
UNIV_INLINE
|
||||
void
|
||||
ut_pair_min(
|
||||
ulint* min_hi,
|
||||
ulint* min_lo,
|
||||
ulint a_hi,
|
||||
ulint a_lo,
|
||||
ulint b_hi,
|
||||
ulint b_lo)
|
||||
{
|
||||
if (a_hi == b_hi) {
|
||||
*min_hi = a_hi;
|
||||
*min_lo = std::min(a_lo, b_lo);
|
||||
} else if (a_hi < b_hi) {
|
||||
*min_hi = a_hi;
|
||||
*min_lo = a_lo;
|
||||
} else {
|
||||
*min_hi = b_hi;
|
||||
*min_lo = b_lo;
|
||||
}
|
||||
}
|
||||
|
||||
/******************************************************//**
|
||||
Compares two ulints.
|
||||
@return 1 if a > b, 0 if a == b, -1 if a < b */
|
||||
UNIV_INLINE
|
||||
int
|
||||
ut_ulint_cmp(
|
||||
/*=========*/
|
||||
ulint a, /*!< in: ulint */
|
||||
ulint b) /*!< in: ulint */
|
||||
{
|
||||
if (a < b) {
|
||||
return(-1);
|
||||
} else if (a == b) {
|
||||
return(0);
|
||||
} else {
|
||||
return(1);
|
||||
}
|
||||
}
|
||||
|
||||
/** Compare two pairs of integers.
|
||||
@param[in] a_h more significant part of first pair
|
||||
@param[in] a_l less significant part of first pair
|
||||
@param[in] b_h more significant part of second pair
|
||||
@param[in] b_l less significant part of second pair
|
||||
@return comparison result of (a_h,a_l) and (b_h,b_l)
|
||||
@retval -1 if (a_h,a_l) is less than (b_h,b_l)
|
||||
@retval 0 if (a_h,a_l) is equal to (b_h,b_l)
|
||||
@retval 1 if (a_h,a_l) is greater than (b_h,b_l) */
|
||||
UNIV_INLINE
|
||||
int
|
||||
ut_pair_cmp(
|
||||
ulint a_h,
|
||||
ulint a_l,
|
||||
ulint b_h,
|
||||
ulint b_l)
|
||||
{
|
||||
if (a_h < b_h) {
|
||||
return(-1);
|
||||
}
|
||||
if (a_h > b_h) {
|
||||
return(1);
|
||||
}
|
||||
return(ut_ulint_cmp(a_l, b_l));
|
||||
}
|
||||
|
||||
/*************************************************************//**
|
||||
Calculates fast the 2-logarithm of a number, rounded upward to an
|
||||
integer.
|
||||
@return logarithm in the base 2, rounded upward */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
ut_2_log(
|
||||
/*=====*/
|
||||
ulint n) /*!< in: number != 0 */
|
||||
{
|
||||
ulint res;
|
||||
|
||||
res = 0;
|
||||
|
||||
ut_ad(n > 0);
|
||||
|
||||
n = n - 1;
|
||||
|
||||
for (;;) {
|
||||
n = n / 2;
|
||||
|
||||
if (n == 0) {
|
||||
break;
|
||||
}
|
||||
|
||||
res++;
|
||||
}
|
||||
|
||||
return(res + 1);
|
||||
}
|
||||
|
||||
/*************************************************************//**
|
||||
Calculates 2 to power n.
|
||||
@return 2 to power n */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
ut_2_exp(
|
||||
/*=====*/
|
||||
ulint n) /*!< in: number */
|
||||
{
|
||||
return((ulint) 1 << n);
|
||||
}
|
|
@ -68,6 +68,7 @@ Note that if write operation is very fast, a) or b) can be fine as alternative.
|
|||
#include <sys/syscall.h>
|
||||
#endif
|
||||
|
||||
#include <algorithm>
|
||||
#include <atomic>
|
||||
#include <thread>
|
||||
#include <mutex>
|
||||
|
|
|
@ -1451,7 +1451,8 @@ void mtr_t::page_lock_upgrade(const buf_block_t &block)
|
|||
(MTR_MEMO_PAGE_SX_FIX | MTR_MEMO_PAGE_X_FIX));
|
||||
|
||||
#ifdef BTR_CUR_HASH_ADAPT
|
||||
ut_ad(!block.index || !block.index->freed());
|
||||
ut_d(if (dict_index_t *index= block.index))
|
||||
ut_ad(!index->freed());
|
||||
#endif /* BTR_CUR_HASH_ADAPT */
|
||||
}
|
||||
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -203,6 +203,7 @@ cmp_decimal(const byte* a, ulint a_length, const byte* b, ulint b_length)
|
|||
@retval positive if data1 is greater than data2 */
|
||||
int cmp_data(ulint mtype, ulint prtype, bool descending,
|
||||
const byte *data1, size_t len1, const byte *data2, size_t len2)
|
||||
noexcept
|
||||
{
|
||||
ut_ad(len1 != UNIV_SQL_DEFAULT);
|
||||
ut_ad(len2 != UNIV_SQL_DEFAULT);
|
||||
|
@ -341,15 +342,14 @@ int cmp_data(ulint mtype, ulint prtype, bool descending,
|
|||
int cmp_dtuple_rec_with_match_low(const dtuple_t *dtuple, const rec_t *rec,
|
||||
const dict_index_t *index,
|
||||
const rec_offs *offsets,
|
||||
ulint n_cmp, ulint *matched_fields)
|
||||
ulint n_cmp, uint16_t *matched_fields)
|
||||
{
|
||||
ulint cur_field; /* current field number */
|
||||
int ret = 0; /* return value */
|
||||
|
||||
ut_ad(dtuple_check_typed(dtuple));
|
||||
ut_ad(rec_offs_validate(rec, index, offsets));
|
||||
|
||||
cur_field = *matched_fields;
|
||||
auto cur_field = *matched_fields;
|
||||
|
||||
ut_ad(n_cmp > 0);
|
||||
ut_ad(n_cmp <= dtuple_get_n_fields(dtuple));
|
||||
|
@ -363,11 +363,9 @@ int cmp_dtuple_rec_with_match_low(const dtuple_t *dtuple, const rec_t *rec,
|
|||
|
||||
/* The "infimum node pointer" is always first. */
|
||||
if (UNIV_UNLIKELY(rec_info & REC_INFO_MIN_REC_FLAG)) {
|
||||
ret = !(tup_info & REC_INFO_MIN_REC_FLAG);
|
||||
goto order_resolved;
|
||||
return !(tup_info & REC_INFO_MIN_REC_FLAG);
|
||||
} else if (UNIV_UNLIKELY(tup_info & REC_INFO_MIN_REC_FLAG)) {
|
||||
ret = -1;
|
||||
goto order_resolved;
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -407,223 +405,12 @@ int cmp_dtuple_rec_with_match_low(const dtuple_t *dtuple, const rec_t *rec,
|
|||
dtuple_b_ptr, dtuple_f_len,
|
||||
rec_b_ptr, rec_f_len);
|
||||
if (ret) {
|
||||
goto order_resolved;
|
||||
}
|
||||
}
|
||||
|
||||
order_resolved:
|
||||
*matched_fields = cur_field;
|
||||
return(ret);
|
||||
}
|
||||
|
||||
/** Get the pad character code point for a type.
|
||||
@param[in] type
|
||||
@return pad character code point
|
||||
@retval ULINT_UNDEFINED if no padding is specified */
|
||||
UNIV_INLINE
|
||||
ulint
|
||||
cmp_get_pad_char(
|
||||
const dtype_t* type)
|
||||
{
|
||||
switch (type->mtype) {
|
||||
case DATA_FIXBINARY:
|
||||
case DATA_BINARY:
|
||||
if (dtype_get_charset_coll(type->prtype)
|
||||
== DATA_MYSQL_BINARY_CHARSET_COLL) {
|
||||
/* Starting from 5.0.18, do not pad
|
||||
VARBINARY or BINARY columns. */
|
||||
return(ULINT_UNDEFINED);
|
||||
}
|
||||
/* Fall through */
|
||||
case DATA_CHAR:
|
||||
case DATA_VARCHAR:
|
||||
case DATA_MYSQL:
|
||||
case DATA_VARMYSQL:
|
||||
/* Space is the padding character for all char and binary
|
||||
strings, and starting from 5.0.3, also for TEXT strings. */
|
||||
return(0x20);
|
||||
case DATA_GEOMETRY:
|
||||
/* DATA_GEOMETRY is binary data, not ASCII-based. */
|
||||
return(ULINT_UNDEFINED);
|
||||
case DATA_BLOB:
|
||||
if (!(type->prtype & DATA_BINARY_TYPE)) {
|
||||
return(0x20);
|
||||
}
|
||||
/* Fall through */
|
||||
default:
|
||||
/* No padding specified */
|
||||
return(ULINT_UNDEFINED);
|
||||
}
|
||||
}
|
||||
|
||||
/** Compare a data tuple to a physical record.
|
||||
@param[in] dtuple data tuple
|
||||
@param[in] rec B-tree or R-tree index record
|
||||
@param[in] index index tree
|
||||
@param[in] offsets rec_get_offsets(rec)
|
||||
@param[in,out] matched_fields number of completely matched fields
|
||||
@param[in,out] matched_bytes number of matched bytes in the first
|
||||
field that is not matched
|
||||
@return the comparison result of dtuple and rec
|
||||
@retval 0 if dtuple is equal to rec
|
||||
@retval negative if dtuple is less than rec
|
||||
@retval positive if dtuple is greater than rec */
|
||||
int
|
||||
cmp_dtuple_rec_with_match_bytes(
|
||||
const dtuple_t* dtuple,
|
||||
const rec_t* rec,
|
||||
const dict_index_t* index,
|
||||
const rec_offs* offsets,
|
||||
ulint* matched_fields,
|
||||
ulint* matched_bytes)
|
||||
{
|
||||
ut_ad(dtuple_check_typed(dtuple));
|
||||
ut_ad(rec_offs_validate(rec, index, offsets));
|
||||
ut_ad(!(REC_INFO_MIN_REC_FLAG
|
||||
& dtuple_get_info_bits(dtuple)));
|
||||
|
||||
if (UNIV_UNLIKELY(REC_INFO_MIN_REC_FLAG
|
||||
& rec_get_info_bits(rec, rec_offs_comp(offsets)))) {
|
||||
ut_ad(page_rec_is_first(rec, page_align(rec)));
|
||||
ut_ad(!page_has_prev(page_align(rec)));
|
||||
ut_ad(rec_is_metadata(rec, *index));
|
||||
return 1;
|
||||
}
|
||||
|
||||
ulint cur_field = *matched_fields;
|
||||
ulint cur_bytes = *matched_bytes;
|
||||
ulint n_cmp = dtuple_get_n_fields_cmp(dtuple);
|
||||
int ret = 0;
|
||||
|
||||
ut_ad(n_cmp <= dtuple_get_n_fields(dtuple));
|
||||
ut_ad(cur_field <= n_cmp);
|
||||
ut_ad(cur_field + (cur_bytes > 0) <= rec_offs_n_fields(offsets));
|
||||
|
||||
/* Match fields in a loop; stop if we run out of fields in dtuple
|
||||
or find an externally stored field */
|
||||
|
||||
while (cur_field < n_cmp) {
|
||||
const dfield_t* dfield = dtuple_get_nth_field(
|
||||
dtuple, cur_field);
|
||||
const dtype_t* type = dfield_get_type(dfield);
|
||||
ulint dtuple_f_len = dfield_get_len(dfield);
|
||||
const byte* dtuple_b_ptr;
|
||||
const byte* rec_b_ptr;
|
||||
ulint rec_f_len;
|
||||
|
||||
dtuple_b_ptr = static_cast<const byte*>(
|
||||
dfield_get_data(dfield));
|
||||
|
||||
ut_ad(!rec_offs_nth_default(offsets, cur_field));
|
||||
rec_b_ptr = rec_get_nth_field(rec, offsets,
|
||||
cur_field, &rec_f_len);
|
||||
ut_ad(!rec_offs_nth_extern(offsets, cur_field));
|
||||
|
||||
/* If we have matched yet 0 bytes, it may be that one or
|
||||
both the fields are SQL null, or the record or dtuple may be
|
||||
the predefined minimum record. */
|
||||
if (cur_bytes == 0) {
|
||||
if (dtuple_f_len == UNIV_SQL_NULL) {
|
||||
if (rec_f_len == UNIV_SQL_NULL) {
|
||||
|
||||
goto next_field;
|
||||
}
|
||||
|
||||
ret = -1;
|
||||
goto order_resolved;
|
||||
} else if (rec_f_len == UNIV_SQL_NULL) {
|
||||
/* We define the SQL null to be the
|
||||
smallest possible value of a field
|
||||
in the alphabetical order */
|
||||
|
||||
ret = 1;
|
||||
goto order_resolved;
|
||||
}
|
||||
}
|
||||
|
||||
switch (type->mtype) {
|
||||
case DATA_FIXBINARY:
|
||||
case DATA_BINARY:
|
||||
case DATA_INT:
|
||||
case DATA_SYS_CHILD:
|
||||
case DATA_SYS:
|
||||
break;
|
||||
case DATA_BLOB:
|
||||
if (type->prtype & DATA_BINARY_TYPE) {
|
||||
break;
|
||||
}
|
||||
/* fall through */
|
||||
default:
|
||||
ret = cmp_data(type->mtype, type->prtype, false,
|
||||
dtuple_b_ptr, dtuple_f_len,
|
||||
rec_b_ptr, rec_f_len);
|
||||
|
||||
if (!ret) {
|
||||
goto next_field;
|
||||
}
|
||||
|
||||
cur_bytes = 0;
|
||||
goto order_resolved;
|
||||
}
|
||||
|
||||
/* Set the pointers at the current byte */
|
||||
|
||||
rec_b_ptr += cur_bytes;
|
||||
dtuple_b_ptr += cur_bytes;
|
||||
/* Compare then the fields */
|
||||
|
||||
for (const ulint pad = cmp_get_pad_char(type);;
|
||||
cur_bytes++) {
|
||||
ulint rec_byte = pad;
|
||||
ulint dtuple_byte = pad;
|
||||
|
||||
if (rec_f_len <= cur_bytes) {
|
||||
if (dtuple_f_len <= cur_bytes) {
|
||||
|
||||
goto next_field;
|
||||
}
|
||||
|
||||
if (rec_byte == ULINT_UNDEFINED) {
|
||||
ret = 1;
|
||||
|
||||
goto order_resolved;
|
||||
}
|
||||
} else {
|
||||
rec_byte = *rec_b_ptr++;
|
||||
}
|
||||
|
||||
if (dtuple_f_len <= cur_bytes) {
|
||||
if (dtuple_byte == ULINT_UNDEFINED) {
|
||||
ret = -1;
|
||||
|
||||
goto order_resolved;
|
||||
}
|
||||
} else {
|
||||
dtuple_byte = *dtuple_b_ptr++;
|
||||
}
|
||||
|
||||
if (dtuple_byte < rec_byte) {
|
||||
ret = -1;
|
||||
goto order_resolved;
|
||||
} else if (dtuple_byte > rec_byte) {
|
||||
ret = 1;
|
||||
goto order_resolved;
|
||||
}
|
||||
}
|
||||
|
||||
next_field:
|
||||
cur_field++;
|
||||
cur_bytes = 0;
|
||||
}
|
||||
|
||||
ut_ad(cur_bytes == 0);
|
||||
|
||||
order_resolved:
|
||||
*matched_fields = cur_field;
|
||||
*matched_bytes = cur_bytes;
|
||||
|
||||
return !ret || UNIV_LIKELY(!index->fields[cur_field].descending)
|
||||
? ret : -ret;
|
||||
return ret;
|
||||
}
|
||||
|
||||
/** Check if a dtuple is a prefix of a record.
|
||||
|
@ -636,8 +423,7 @@ bool cmp_dtuple_is_prefix_of_rec(const dtuple_t *dtuple, const rec_t *rec,
|
|||
const dict_index_t *index,
|
||||
const rec_offs *offsets)
|
||||
{
|
||||
ulint matched_fields= 0;
|
||||
ulint n_fields= dtuple_get_n_fields(dtuple);
|
||||
uint16_t matched_fields= 0, n_fields= dtuple_get_n_fields(dtuple);
|
||||
ut_ad(n_fields <= rec_offs_n_fields(offsets));
|
||||
cmp_dtuple_rec_with_match(dtuple, rec, index, offsets, &matched_fields);
|
||||
return matched_fields == n_fields;
|
||||
|
|
|
@ -1980,15 +1980,13 @@ row_ins_dupl_error_with_rec(
|
|||
dict_index_t* index, /*!< in: index */
|
||||
const rec_offs* offsets)/*!< in: rec_get_offsets(rec, index) */
|
||||
{
|
||||
ulint matched_fields;
|
||||
ulint n_unique;
|
||||
ulint i;
|
||||
|
||||
ut_ad(rec_offs_validate(rec, index, offsets));
|
||||
|
||||
n_unique = dict_index_get_n_unique(index);
|
||||
const auto n_unique = dict_index_get_n_unique(index);
|
||||
|
||||
matched_fields = 0;
|
||||
uint16_t matched_fields = 0;
|
||||
|
||||
cmp_dtuple_rec_with_match(entry, rec, index, offsets, &matched_fields);
|
||||
|
||||
|
@ -2085,7 +2083,6 @@ row_ins_scan_sec_index_for_duplicate(
|
|||
mem_heap_t* offsets_heap)
|
||||
/*!< in/out: memory heap that can be emptied */
|
||||
{
|
||||
ulint n_unique;
|
||||
int cmp;
|
||||
ulint n_fields_cmp;
|
||||
btr_pcur_t pcur;
|
||||
|
@ -2097,7 +2094,7 @@ row_ins_scan_sec_index_for_duplicate(
|
|||
|
||||
ut_ad(!index->lock.have_any());
|
||||
|
||||
n_unique = dict_index_get_n_unique(index);
|
||||
const auto n_unique = dict_index_get_n_unique(index);
|
||||
|
||||
/* If the secondary index is unique, but one of the fields in the
|
||||
n_unique first fields is NULL, a unique key violation cannot occur,
|
||||
|
@ -2227,7 +2224,7 @@ row_ins_duplicate_online(ulint n_uniq, const dtuple_t *entry,
|
|||
const rec_t *rec, const dict_index_t *index,
|
||||
rec_offs *offsets)
|
||||
{
|
||||
ulint fields = 0;
|
||||
uint16_t fields = 0;
|
||||
|
||||
/* During rebuild, there should not be any delete-marked rows
|
||||
in the new table. */
|
||||
|
@ -3267,7 +3264,7 @@ row_ins_clust_index_entry(
|
|||
: index->table->is_temporary()
|
||||
? BTR_NO_LOCKING_FLAG : 0;
|
||||
#endif /* WITH_WSREP */
|
||||
const ulint orig_n_fields = entry->n_fields;
|
||||
const auto orig_n_fields = entry->n_fields;
|
||||
|
||||
/* For intermediate table during copy alter table,
|
||||
skip the undo log and record lock checking for
|
||||
|
|
|
@ -744,7 +744,7 @@ row_log_table_low_redundant(
|
|||
ulint avail_size;
|
||||
mem_heap_t* heap = NULL;
|
||||
dtuple_t* tuple;
|
||||
const ulint n_fields = rec_get_n_fields_old(rec);
|
||||
const auto n_fields = rec_get_n_fields_old(rec);
|
||||
|
||||
ut_ad(index->n_fields >= n_fields);
|
||||
ut_ad(index->n_fields == n_fields || index->is_instant());
|
||||
|
@ -1701,6 +1701,7 @@ err_exit:
|
|||
if (error) {
|
||||
goto err_exit;
|
||||
}
|
||||
ut_ad(pcur->btr_cur.flag == BTR_CUR_BINARY);
|
||||
|
||||
if (page_rec_is_infimum(btr_pcur_get_rec(pcur))
|
||||
|| btr_pcur_get_low_match(pcur) < index->n_uniq) {
|
||||
|
@ -1770,6 +1771,8 @@ row_log_table_apply_delete(
|
|||
goto all_done;
|
||||
}
|
||||
|
||||
ut_ad(btr_pcur_get_btr_cur(&pcur)->flag == BTR_CUR_BINARY);
|
||||
|
||||
if (page_rec_is_infimum(btr_pcur_get_rec(&pcur))
|
||||
|| btr_pcur_get_low_match(&pcur) < index->n_uniq) {
|
||||
all_done:
|
||||
|
@ -1903,6 +1906,8 @@ func_exit_committed:
|
|||
return error;
|
||||
}
|
||||
|
||||
ut_ad(btr_pcur_get_btr_cur(&pcur)->flag == BTR_CUR_BINARY);
|
||||
|
||||
ut_ad(!page_rec_is_infimum(btr_pcur_get_rec(&pcur))
|
||||
&& btr_pcur_get_low_match(&pcur) >= index->n_uniq);
|
||||
|
||||
|
|
|
@ -764,7 +764,7 @@ row_rec_to_index_entry_impl(
|
|||
ut_ad(info_bits == 0);
|
||||
ut_ad(!pad);
|
||||
}
|
||||
dtuple_t* entry = dtuple_create(heap, rec_len);
|
||||
dtuple_t* entry = dtuple_create(heap, uint16_t(rec_len));
|
||||
dfield_t* dfield = entry->fields;
|
||||
|
||||
dtuple_set_n_fields_cmp(entry,
|
||||
|
@ -861,7 +861,7 @@ copy_user_fields:
|
|||
}
|
||||
|
||||
if (mblob == 2) {
|
||||
ulint n_fields = ulint(dfield - entry->fields);
|
||||
uint16_t n_fields = uint16_t(dfield - entry->fields);
|
||||
ut_ad(entry->n_fields >= n_fields);
|
||||
entry->n_fields = n_fields;
|
||||
}
|
||||
|
|
|
@ -2612,9 +2612,9 @@ row_sel_convert_mysql_key_to_innobase(
|
|||
|
||||
key_end = key_ptr + key_len;
|
||||
|
||||
/* Permit us to access any field in the tuple (ULINT_MAX): */
|
||||
/* Permit us to access any field in the tuple: */
|
||||
|
||||
dtuple_set_n_fields(tuple, ULINT_MAX);
|
||||
ut_d(dtuple_set_n_fields(tuple, uint16_t(~0)));
|
||||
|
||||
dfield = dtuple_get_nth_field(tuple, 0);
|
||||
field = dict_index_get_nth_field(index, 0);
|
||||
|
@ -2781,7 +2781,7 @@ row_sel_convert_mysql_key_to_innobase(
|
|||
/* We set the length of tuple to n_fields: we assume that the memory
|
||||
area allocated for it is big enough (usually bigger than n_fields). */
|
||||
|
||||
dtuple_set_n_fields(tuple, n_fields);
|
||||
dtuple_set_n_fields(tuple, uint16_t(n_fields));
|
||||
}
|
||||
|
||||
/**************************************************************//**
|
||||
|
@ -3451,7 +3451,7 @@ Row_sel_get_clust_rec_for_mysql::operator()(
|
|||
page_cur_t page_cursor;
|
||||
page_cursor.block = block;
|
||||
page_cursor.index = sec_index;
|
||||
ulint up_match = 0, low_match = 0;
|
||||
uint16_t up_match = 0, low_match = 0;
|
||||
ut_ad(!page_cur_search_with_match(tuple, PAGE_CUR_LE,
|
||||
&up_match,
|
||||
&low_match,
|
||||
|
@ -4171,8 +4171,7 @@ row_sel_fill_vrow(
|
|||
offsets = rec_get_offsets(rec, index, offsets, index->n_core_fields,
|
||||
ULINT_UNDEFINED, &heap);
|
||||
|
||||
*vrow = dtuple_create_with_vcol(
|
||||
heap, 0, dict_table_get_n_v_cols(index->table));
|
||||
*vrow = dtuple_create_with_vcol(heap, 0, index->table->n_v_cols);
|
||||
|
||||
/* Initialize all virtual row's mtype to DATA_MISSING */
|
||||
dtuple_init_v_fld(*vrow);
|
||||
|
@ -6003,7 +6002,6 @@ row_count_rtree_recs(
|
|||
mem_heap_t* heap;
|
||||
dtuple_t* entry;
|
||||
dtuple_t* search_entry = prebuilt->search_tuple;
|
||||
ulint entry_len;
|
||||
ulint i;
|
||||
byte* buf;
|
||||
|
||||
|
@ -6014,10 +6012,9 @@ row_count_rtree_recs(
|
|||
heap = mem_heap_create(256);
|
||||
|
||||
/* Build a search tuple. */
|
||||
entry_len = dict_index_get_n_fields(index);
|
||||
entry = dtuple_create(heap, entry_len);
|
||||
entry = dtuple_create(heap, index->n_fields);
|
||||
|
||||
for (i = 0; i < entry_len; i++) {
|
||||
for (i = 0; i < index->n_fields; i++) {
|
||||
const dict_field_t* ind_field
|
||||
= dict_index_get_nth_field(index, i);
|
||||
const dict_col_t* col
|
||||
|
@ -6795,9 +6792,9 @@ count_row:
|
|||
|
||||
if (prev_entry)
|
||||
{
|
||||
ulint matched_fields= 0;
|
||||
uint16_t matched= 0;
|
||||
int cmp= cmp_dtuple_rec_with_match(prev_entry, rec, index, offsets,
|
||||
&matched_fields);
|
||||
&matched);
|
||||
const char* msg;
|
||||
|
||||
if (UNIV_LIKELY(cmp < 0));
|
||||
|
@ -6810,7 +6807,7 @@ not_ok:
|
|||
<< ": " << *prev_entry << ", "
|
||||
<< rec_offsets_print(rec, offsets);
|
||||
}
|
||||
else if (index->is_unique() && matched_fields >=
|
||||
else if (index->is_unique() && matched >=
|
||||
dict_index_get_n_ordering_defined_by_user(index))
|
||||
{
|
||||
/* NULL values in unique indexes are considered not to be duplicates */
|
||||
|
|
|
@ -522,7 +522,7 @@ row_vers_build_cur_vrow_low(
|
|||
const rec_t* version;
|
||||
rec_t* prev_version;
|
||||
mem_heap_t* heap = NULL;
|
||||
ulint num_v = dict_table_get_n_v_cols(index->table);
|
||||
const auto num_v = dict_table_get_n_v_cols(index->table);
|
||||
const dfield_t* field;
|
||||
ulint i;
|
||||
bool all_filled = false;
|
||||
|
|
|
@ -45,8 +45,7 @@ Created 3/26/1996 Heikki Tuuri
|
|||
const dtuple_t trx_undo_metadata = {
|
||||
/* This also works for REC_INFO_METADATA_ALTER, because the
|
||||
delete-mark (REC_INFO_DELETED_FLAG) is ignored when searching. */
|
||||
REC_INFO_METADATA_ADD, 0, 0,
|
||||
NULL, 0, NULL
|
||||
REC_INFO_METADATA_ADD, 0, 0, 0, nullptr, nullptr
|
||||
#ifdef UNIV_DEBUG
|
||||
, DATA_TUPLE_MAGIC_N
|
||||
#endif /* UNIV_DEBUG */
|
||||
|
@ -580,7 +579,7 @@ trx_undo_rec_get_row_ref(
|
|||
{
|
||||
ut_ad(index->is_primary());
|
||||
|
||||
const ulint ref_len = dict_index_get_n_unique(index);
|
||||
const uint16_t ref_len = dict_index_get_n_unique(index);
|
||||
|
||||
dtuple_t* tuple = dtuple_create(heap, ref_len);
|
||||
*ref = tuple;
|
||||
|
|
|
@ -25,6 +25,12 @@ ADD_EXECUTABLE(innodb_fts-t innodb_fts-t.cc)
|
|||
TARGET_LINK_LIBRARIES(innodb_fts-t mysys mytap)
|
||||
ADD_DEPENDENCIES(innodb_fts-t GenError)
|
||||
MY_ADD_TEST(innodb_fts)
|
||||
IF (WITH_INNODB_AHI)
|
||||
ADD_EXECUTABLE(innodb_ahi-t innodb_ahi-t.cc)
|
||||
TARGET_LINK_LIBRARIES(innodb_ahi-t mysys mytap)
|
||||
ADD_DEPENDENCIES(innodb_ahi-t GenError)
|
||||
MY_ADD_TEST(innodb_ahi)
|
||||
ENDIF()
|
||||
# See explanation in innobase/CmakeLists.txt
|
||||
IF(CMAKE_SYSTEM_PROCESSOR MATCHES "ppc64|powerpc64|s390x")
|
||||
ADD_COMPILE_FLAGS(
|
||||
|
|
221
storage/innobase/unittest/innodb_ahi-t.cc
Normal file
221
storage/innobase/unittest/innodb_ahi-t.cc
Normal file
|
@ -0,0 +1,221 @@
|
|||
#include "tap.h"
|
||||
|
||||
#define SUX_LOCK_GENERIC
|
||||
#define NO_ELISION
|
||||
#define thd_kill_level(thd) 0
|
||||
#define srv0mon_h
|
||||
#define MONITOR_INC(x)
|
||||
#define MONITOR_INC_VALUE(x,y)
|
||||
#include "../btr/btr0sea.cc"
|
||||
const size_t alloc_max_retries= 0;
|
||||
const byte zero[16384]= { '\0', };
|
||||
const byte *field_ref_zero= zero;
|
||||
ulint srv_buf_pool_curr_size, srv_buf_pool_old_size, srv_buf_pool_size;
|
||||
uint32_t srv_page_size_shift= 14;
|
||||
ulong srv_page_size= 1 << 14;
|
||||
dict_sys_t dict_sys;
|
||||
buf_pool_t buf_pool;
|
||||
buf_pool_t::chunk_t::map *buf_pool_t::chunk_t::map_reg;
|
||||
buf_pool_t::chunk_t::map *buf_pool_t::chunk_t::map_ref;
|
||||
|
||||
void buf_pool_t::free_block(buf_block_t*) noexcept {}
|
||||
void dict_mem_table_free(dict_table_t*) {}
|
||||
void dict_mem_index_free(dict_index_t*) {}
|
||||
buf_block_t *buf_LRU_get_free_block(buf_LRU_get) { return nullptr; }
|
||||
ibool dtuple_check_typed(const dtuple_t*) { return true; }
|
||||
bool btr_cur_t::check_mismatch(const dtuple_t&,page_cur_mode_t,ulint) noexcept
|
||||
{ return false; }
|
||||
buf_block_t *buf_page_get_gen(const page_id_t, ulint, rw_lock_type_t,
|
||||
buf_block_t*,ulint,mtr_t*,dberr_t*)
|
||||
{ return nullptr; }
|
||||
bool buf_page_make_young_if_needed(buf_page_t*) { return false; }
|
||||
|
||||
mtr_t::mtr_t()= default;
|
||||
mtr_t::~mtr_t()= default;
|
||||
void mtr_t::start() {}
|
||||
void mtr_t::commit() {}
|
||||
void mtr_t::rollback_to_savepoint(ulint, ulint) {}
|
||||
void small_vector_base::grow_by_1(void *, size_t) {}
|
||||
|
||||
void sql_print_error(const char*, ...) {}
|
||||
ulint ut_find_prime(ulint n) { return n; }
|
||||
void mem_heap_block_free(mem_block_info_t*, mem_block_info_t*){}
|
||||
namespace ib { error::~error() {} fatal_or_error::~fatal_or_error() {} }
|
||||
std::ostream &operator<<(std::ostream &out, const page_id_t) { return out; }
|
||||
|
||||
#ifdef UNIV_DEBUG
|
||||
byte data_error;
|
||||
bool srw_lock_debug::have_wr() const noexcept { return false; }
|
||||
void srw_lock_debug::rd_unlock() noexcept {}
|
||||
void srw_lock_debug::rd_lock(SRW_LOCK_ARGS(const char*,unsigned)) noexcept {}
|
||||
#endif
|
||||
|
||||
void page_hash_latch::read_lock_wait() {}
|
||||
# ifndef PTHREAD_ADAPTIVE_MUTEX_INITIALIZER_NP
|
||||
template<> void pthread_mutex_wrapper<true>::wr_wait() noexcept {}
|
||||
# endif
|
||||
template<> void srw_lock_<true>::rd_wait() noexcept {}
|
||||
template<> void srw_lock_<true>::wr_wait() noexcept {}
|
||||
template<bool spin> void ssux_lock_impl<spin>::wake() noexcept {}
|
||||
template<bool spin> void srw_mutex_impl<spin>::wake() noexcept {}
|
||||
|
||||
#ifdef UNIV_PFS_MEMORY
|
||||
PSI_memory_key ut_new_get_key_by_file(uint32_t){ return PSI_NOT_INSTRUMENTED; }
|
||||
PSI_memory_key mem_key_other, mem_key_std;
|
||||
#endif
|
||||
|
||||
#ifdef UNIV_PFS_RWLOCK
|
||||
template<bool spin>
|
||||
void srw_lock_impl<spin>::psi_wr_lock(const char*, unsigned) noexcept {}
|
||||
template<bool spin>
|
||||
void srw_lock_impl<spin>::psi_rd_lock(const char*, unsigned) noexcept {}
|
||||
|
||||
void dict_sys_t::unlock() noexcept {}
|
||||
void dict_sys_t::freeze(const char *, unsigned) noexcept {}
|
||||
void dict_sys_t::unfreeze() noexcept {}
|
||||
#endif /* UNIV_PFS_RWLOCK */
|
||||
|
||||
void ut_dbg_assertion_failed(const char *e, const char *file, unsigned line)
|
||||
{
|
||||
fprintf(stderr, "%s:%u: Assertion %s failed\n", file, line, e ? e : "");
|
||||
abort();
|
||||
}
|
||||
|
||||
int main(int, char **argv)
|
||||
{
|
||||
MY_INIT(*argv);
|
||||
plan(42);
|
||||
|
||||
btr_search.create();
|
||||
btr_search.free();
|
||||
|
||||
dfield_t fields[2]= {{nullptr,0,0,UNIV_SQL_NULL,{0,DATA_VARCHAR,3,1,1}},
|
||||
{(char*)"42",0,0,2,{0,DATA_CHAR,2,1,1}}};
|
||||
dtuple_t tuple2{0,2,2,0,fields,nullptr, ut_d(DATA_TUPLE_MAGIC_N) };
|
||||
dict_col_t cols[]={{}, {}, {DATA_NOT_NULL,DATA_CHAR,2,1,1,1,0,0,{nullptr,0}}};
|
||||
dict_field_t ifields[3]= {{}, {}, {}};
|
||||
dict_table_t table{};
|
||||
dict_index_t index{};
|
||||
index.table= &table;
|
||||
index.n_uniq= 3;
|
||||
index.n_nullable= 3;
|
||||
index.n_fields= 3;
|
||||
index.n_core_fields= 3;
|
||||
index.n_core_null_bytes= 1;
|
||||
index.fields= ifields;
|
||||
|
||||
ifields[0].col= &cols[0];
|
||||
ifields[1].col= &cols[2];
|
||||
ifields[2].col= &cols[2];
|
||||
ifields[1].fixed_len= 2;
|
||||
ifields[2].fixed_len= 2;
|
||||
|
||||
constexpr uint32_t crc42= 0x2e7d3dcb, crc3z42= 0x9a6e3c2c,
|
||||
crc2z= 0xf16177d2, crc3z= 0x6064a37a;
|
||||
|
||||
{
|
||||
btr_cur_t cursor;
|
||||
cursor.page_cur.index= &index;
|
||||
cursor.n_bytes_fields= 2;
|
||||
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc42, "dtuple_fold(NULL,'42')");
|
||||
table.flags= DICT_TF_COMPACT;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc42, "dtuple_fold(NULL,'42')");
|
||||
fields[0].type.mtype= DATA_CHAR;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc42, "dtuple_fold(NULL,'42')");
|
||||
table.flags= 0;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc3z42, "dtuple_fold('\\0\\0\\0','42')");
|
||||
fields[0].type.mtype= DATA_VARCHAR;
|
||||
|
||||
cursor.n_bytes_fields= 1;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
table.flags= DICT_TF_COMPACT;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
fields[0].type.mtype= DATA_CHAR;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
table.flags= 0;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc3z, "dtuple_fold('\\0\\0\\0')");
|
||||
fields[0].type.mtype= DATA_VARCHAR;
|
||||
|
||||
cursor.n_bytes_fields= 2 << 16;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
table.flags= DICT_TF_COMPACT;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
fields[0].type.mtype= DATA_CHAR;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == 0, "dtuple_fold(NULL)");
|
||||
table.flags= 0;
|
||||
ok(dtuple_fold(&tuple2, &cursor) == crc2z, "dtuple_fold('\\0\\0')");
|
||||
fields[0].type.mtype= DATA_VARCHAR;
|
||||
}
|
||||
|
||||
byte *page= static_cast<byte*>(aligned_malloc(16384, 16384));
|
||||
memset_aligned<16384>(page, 0, 16384);
|
||||
byte *rec= &page[256];
|
||||
page[PAGE_HEADER + PAGE_HEAP_TOP]= 1;
|
||||
page[PAGE_HEADER + PAGE_HEAP_TOP + 1]= 4 + 2;
|
||||
const byte r1_varchar[]= {2,0x80,0,0,0,2<<1|1,0,0, '4','2'};
|
||||
const byte r2_varchar[]= {0,2,0x80,0,0,0,0,2<<1,0,0, '4','2'};
|
||||
const byte r1_var3[]= {2,0x80,0x80,0,0,0,3<<1|1,0,0, '4','2'};
|
||||
const byte r2_var3[]= {0,2,0x80,0,0x80,0,0,0,0,3<<1,0,0, '4','2'};
|
||||
const byte r1_char[]={2+3,0x83,0,0,0,2<<1|1,0,0, 0,0,0,'4','2'};
|
||||
const byte r2_char[]= {0,2+3,0x80,3,0,0,0,2<<1,0,0, 0,0,0,'4','2'};
|
||||
const byte c[]= { 0,1,0,0,0,0,0, '4','2'};
|
||||
const byte c3[]= { 0,3,0,0,0,0,0, '4','2'};
|
||||
|
||||
memcpy(rec - sizeof r1_varchar + 2, r1_varchar, sizeof r1_varchar);
|
||||
ok(rec_fold(rec, index, 2, false) == crc42, "rec_fold(NULL, '42')");
|
||||
ok(rec_fold(rec, index, 1, false) == 0, "rec_fold(NULL)");
|
||||
ok(rec_fold(rec, index, 2 << 16, false) == 0, "rec_fold(NULL)");
|
||||
memcpy(rec - sizeof r2_varchar + 2, r2_varchar, sizeof r2_varchar);
|
||||
ok(rec_fold(rec, index, 2, false) == crc42, "rec_fold(NULL, '42')");
|
||||
ok(rec_fold(rec, index, 1, false) == 0, "rec_fold(NULL)");
|
||||
ok(rec_fold(rec, index, 2 << 16, false) == 0, "rec_fold(NULL)");
|
||||
|
||||
memcpy(rec - sizeof r1_var3 + 2, r1_var3, sizeof r1_var3);
|
||||
ok(rec_fold(rec, index, 3, false) == crc42, "rec_fold(NULL, NULL, '42')");
|
||||
ok(rec_fold(rec, index, 2, false) == 0, "rec_fold(NULL,NULL)");
|
||||
ok(rec_fold(rec, index, 1 | 2 << 16, false) == 0, "rec_fold(NULL,NULL)");
|
||||
memcpy(rec - sizeof r2_var3 + 2, r2_var3, sizeof r2_var3);
|
||||
ok(rec_fold(rec, index, 3, false) == crc42, "rec_fold(NULL, NULL, '42')");
|
||||
ok(rec_fold(rec, index, 2, false) == 0, "rec_fold(NULL,NULL)");
|
||||
ok(rec_fold(rec, index, 1 | 2 << 16, false) == 0, "rec_fold(NULL,NULL)");
|
||||
|
||||
fields[0].type.mtype= DATA_CHAR;
|
||||
memcpy(rec - sizeof r1_char + 3 + 2, r1_char, sizeof r1_char);
|
||||
ok(rec_fold(rec, index, 2, false) == crc3z42, "rec_fold('\\0\\0\\0', '42')");
|
||||
ok(rec_fold(rec, index, 1, false) == crc3z, "rec_fold('\\0\\0\\0')");
|
||||
ok(rec_fold(rec, index, 2 << 16, false) == crc2z, "rec_fold('\\0\\0')");
|
||||
memcpy(rec - sizeof r2_char + 3 + 2, r2_char, sizeof r2_char);
|
||||
ok(rec_fold(rec, index, 2, false) == crc3z42, "rec_fold('\\0\\0\\0', '42')");
|
||||
ok(rec_fold(rec, index, 1, false) == crc3z, "rec_fold('\\0\\0\\0')");
|
||||
ok(rec_fold(rec, index, 2 << 16, false) == crc2z, "rec_fold('\\0\\0')");
|
||||
|
||||
page[PAGE_HEADER + PAGE_N_HEAP]= 0x80;
|
||||
table.flags= DICT_TF_COMPACT;
|
||||
memcpy(rec - sizeof c + 2, c, sizeof c);
|
||||
ok(rec_fold(rec, index, 2, true) == crc42, "rec_fold(NULL, '42')");
|
||||
ok(rec_fold(rec, index, 1, true) == 0, "rec_fold(NULL)");
|
||||
ok(rec_fold(rec, index, 2 << 16, true) == 0, "rec_fold(NULL)");
|
||||
fields[0].type.mtype= DATA_VARCHAR;
|
||||
ok(rec_fold(rec, index, 2, true) == crc42, "rec_fold(NULL, '42')");
|
||||
ok(rec_fold(rec, index, 1, true) == 0, "rec_fold(NULL)");
|
||||
ok(rec_fold(rec, index, 2 << 16, true) == 0, "rec_fold(NULL)");
|
||||
|
||||
memcpy(rec - sizeof c3 + 2, c3, sizeof c3);
|
||||
fields[0].type.mtype= DATA_CHAR;
|
||||
|
||||
ifields[1].col= &cols[1];
|
||||
ifields[1].fixed_len= 0;
|
||||
|
||||
ok(rec_fold(rec, index, 3, true) == crc42, "rec_fold(NULL, NULL, '42')");
|
||||
ok(rec_fold(rec, index, 2, true) == 0, "rec_fold(NULL, NULL)");
|
||||
ok(rec_fold(rec, index, 1 | 2 << 16, true) == 0, "rec_fold(NULL, NULL)");
|
||||
fields[0].type.mtype= DATA_VARCHAR;
|
||||
ok(rec_fold(rec, index, 3, true) == crc42, "rec_fold(NULL, NULL, '42')");
|
||||
ok(rec_fold(rec, index, 2, true) == 0, "rec_fold(NULL, NULL)");
|
||||
ok(rec_fold(rec, index, 1 | 2 << 16, true) == 0, "rec_fold(NULL, NULL)");
|
||||
aligned_free(page);
|
||||
|
||||
my_end(MY_CHECK_ERROR);
|
||||
return exit_status();
|
||||
}
|
Loading…
Reference in a new issue