mirror of
https://github.com/MariaDB/server.git
synced 2026-05-15 19:37:16 +02:00
MDEV-26200 buf_pool.flush_list corrupted by buffer pool resizing or ROW_FORMAT=COMPRESSED
The lazy deletion of clean blocks from buf_pool.flush_list that was
introduced in commit 6441bc614a (MDEV-25113)
introduced a race condition around the function
buf_flush_relocate_on_flush_list().
The test innodb_zip.wl5522_debug_zip as well as the buffer pool
resizing tests would occasionally fail in debug builds due to
buf_pool.flush_list.count disagreeing with the actual length of the
doubly-linked list.
The safe procedure for relocating a block in buf_pool.flush_list should be
as follows, now that we implement lazy deletion from buf_pool.flush_list:
1. Acquire buf_pool.mutex.
2. Acquire the exclusive buf_pool.page_hash.latch.
3. Acquire buf_pool.flush_list_mutex.
4. Copy the block descriptor.
5. Invoke buf_flush_relocate_on_flush_list().
6. Release buf_pool.flush_list_mutex.
buf_flush_relocate_on_flush_list(): Assert that
buf_pool.flush_list_mutex is being held. Invoke
buf_page_t::oldest_modification() only once, using
std::memory_order_relaxed, now that the mutex protects us.
buf_LRU_free_page(), buf_LRU_block_remove_hashed(): Avoid
an unlock-lock cycle on hash_lock. (We must not acquire hash_lock
while already holding buf_pool.flush_list_mutex, because that
could lead to a deadlock due to latching order violation.)
This commit is contained in:
parent
316a8cebf5
commit
bf435a3f4d
3 changed files with 25 additions and 26 deletions
|
|
@ -1566,6 +1566,7 @@ inline bool buf_pool_t::realloc(buf_block_t *block)
|
|||
if (block->page.can_relocate()) {
|
||||
memcpy_aligned<OS_FILE_LOG_BLOCK_SIZE>(
|
||||
new_block->frame, block->frame, srv_page_size);
|
||||
mysql_mutex_lock(&buf_pool.flush_list_mutex);
|
||||
new (&new_block->page) buf_page_t(block->page);
|
||||
|
||||
/* relocate LRU list */
|
||||
|
|
@ -1625,6 +1626,7 @@ inline bool buf_pool_t::realloc(buf_block_t *block)
|
|||
buf_flush_relocate_on_flush_list(&block->page,
|
||||
&new_block->page);
|
||||
}
|
||||
mysql_mutex_unlock(&buf_pool.flush_list_mutex);
|
||||
block->page.set_corrupt_id();
|
||||
|
||||
/* set other flags of buf_block_t */
|
||||
|
|
@ -3131,12 +3133,14 @@ evict_from_pool:
|
|||
/* Note: this is the uncompressed block and it is not
|
||||
accessible by other threads yet because it is not in
|
||||
any list or hash table */
|
||||
mysql_mutex_lock(&buf_pool.flush_list_mutex);
|
||||
buf_relocate(bpage, &block->page);
|
||||
|
||||
/* Set after buf_relocate(). */
|
||||
block->page.set_buf_fix_count(1);
|
||||
|
||||
buf_flush_relocate_on_flush_list(bpage, &block->page);
|
||||
mysql_mutex_unlock(&buf_pool.flush_list_mutex);
|
||||
|
||||
/* Buffer-fix, I/O-fix, and X-latch the block
|
||||
for the duration of the decompression.
|
||||
|
|
@ -3646,8 +3650,10 @@ loop:
|
|||
}
|
||||
|
||||
rw_lock_x_lock(&free_block->lock);
|
||||
mysql_mutex_lock(&buf_pool.flush_list_mutex);
|
||||
buf_relocate(&block->page, &free_block->page);
|
||||
buf_flush_relocate_on_flush_list(&block->page, &free_block->page);
|
||||
mysql_mutex_unlock(&buf_pool.flush_list_mutex);
|
||||
|
||||
free_block->page.set_state(BUF_BLOCK_FILE_PAGE);
|
||||
buf_unzip_LRU_add_block(free_block, FALSE);
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue