mariadb/storage/xtradb/include/buf0flu.ic

168 lines
5.2 KiB
Text
Raw Normal View History

2009-03-25 23:11:11 -07:00
/*****************************************************************************
2013-12-16 15:38:05 +01:00
Copyright (c) 1995, 2009, Oracle and/or its affiliates. All Rights Reserved.
2009-03-25 23:11:11 -07:00
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
2013-12-16 15:38:05 +01:00
this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA
2009-03-25 23:11:11 -07:00
*****************************************************************************/
2009-09-07 10:22:53 +00:00
/**************************************************//**
@file include/buf0flu.ic
The database buffer pool flush algorithm
Created 11/5/1995 Heikki Tuuri
*******************************************************/
2009-09-07 10:22:53 +00:00
#ifndef UNIV_HOTBACKUP
#include "buf0buf.h"
#include "mtr0mtr.h"
2013-12-16 15:38:05 +01:00
#include "srv0srv.h"
2009-09-07 10:22:53 +00:00
/********************************************************************//**
2009-03-25 23:11:11 -07:00
Inserts a modified block into the flush list. */
UNIV_INTERN
void
buf_flush_insert_into_flush_list(
/*=============================*/
2011-07-14 21:22:41 +02:00
buf_pool_t* buf_pool, /*!< buffer pool instance */
buf_block_t* block, /*!< in/out: block which is modified */
2013-12-16 15:38:05 +01:00
lsn_t lsn); /*!< in: oldest modification */
2009-09-07 10:22:53 +00:00
/********************************************************************//**
Inserts a modified block into the flush list in the right sorted position.
This function is used by recovery, because there the modifications do not
necessarily come in the order of lsn's. */
UNIV_INTERN
void
buf_flush_insert_sorted_into_flush_list(
/*====================================*/
2011-07-14 21:22:41 +02:00
buf_pool_t* buf_pool, /*!< buffer pool instance */
buf_block_t* block, /*!< in/out: block which is modified */
2013-12-16 15:38:05 +01:00
lsn_t lsn); /*!< in: oldest modification */
2009-09-07 10:22:53 +00:00
/********************************************************************//**
This function should be called at a mini-transaction commit, if a page was
modified in it. Puts the block to the list of modified blocks, if it is not
already in it. */
UNIV_INLINE
void
buf_flush_note_modification(
/*========================*/
2009-09-07 10:22:53 +00:00
buf_block_t* block, /*!< in: block which is modified */
mtr_t* mtr) /*!< in: mtr */
{
2011-07-14 21:22:41 +02:00
buf_pool_t* buf_pool = buf_pool_from_block(block);
2009-06-24 18:43:25 -07:00
2013-12-16 15:38:05 +01:00
ut_ad(!srv_read_only_mode);
ut_ad(buf_block_get_state(block) == BUF_BLOCK_FILE_PAGE);
ut_ad(block->page.buf_fix_count > 0);
#ifdef UNIV_SYNC_DEBUG
ut_ad(rw_lock_own(&(block->lock), RW_LOCK_EX));
#endif /* UNIV_SYNC_DEBUG */
2011-07-14 21:22:41 +02:00
ut_ad(!buf_flush_list_mutex_own(buf_pool));
2013-05-08 09:52:54 +02:00
ut_ad(!mtr->made_dirty || log_flush_order_mutex_own());
ut_ad(mtr->start_lsn != 0);
ut_ad(mtr->modifications);
2011-07-14 21:22:41 +02:00
mutex_enter(&block->mutex);
ut_ad(block->page.newest_modification <= mtr->end_lsn);
block->page.newest_modification = mtr->end_lsn;
if (!block->page.oldest_modification) {
2013-12-16 15:38:05 +01:00
ut_a(mtr->made_dirty);
ut_ad(log_flush_order_mutex_own());
2011-07-14 21:22:41 +02:00
buf_flush_insert_into_flush_list(
buf_pool, block, mtr->start_lsn);
} else {
ut_ad(block->page.oldest_modification <= mtr->start_lsn);
}
2009-06-24 18:43:25 -07:00
mutex_exit(&block->mutex);
2013-12-16 15:38:05 +01:00
srv_stats.buf_pool_write_requests.inc();
}
2009-09-07 10:22:53 +00:00
/********************************************************************//**
This function should be called when recovery has modified a buffer page. */
UNIV_INLINE
void
buf_flush_recv_note_modification(
/*=============================*/
2009-09-07 10:22:53 +00:00
buf_block_t* block, /*!< in: block which is modified */
2013-12-16 15:38:05 +01:00
lsn_t start_lsn, /*!< in: start lsn of the first mtr in a
set of mtr's */
2013-12-16 15:38:05 +01:00
lsn_t end_lsn) /*!< in: end lsn of the last mtr in the
set of mtr's */
{
2011-07-14 21:22:41 +02:00
buf_pool_t* buf_pool = buf_pool_from_block(block);
2009-06-24 18:43:25 -07:00
2013-12-16 15:38:05 +01:00
ut_ad(!srv_read_only_mode);
ut_ad(buf_block_get_state(block) == BUF_BLOCK_FILE_PAGE);
ut_ad(block->page.buf_fix_count > 0);
#ifdef UNIV_SYNC_DEBUG
ut_ad(rw_lock_own(&(block->lock), RW_LOCK_EX));
#endif /* UNIV_SYNC_DEBUG */
2011-07-14 21:22:41 +02:00
ut_ad(!buf_flush_list_mutex_own(buf_pool));
ut_ad(log_flush_order_mutex_own());
2011-07-14 21:22:41 +02:00
ut_ad(start_lsn != 0);
ut_ad(block->page.newest_modification <= end_lsn);
2011-07-14 21:22:41 +02:00
mutex_enter(&block->mutex);
block->page.newest_modification = end_lsn;
if (!block->page.oldest_modification) {
2011-07-14 21:22:41 +02:00
buf_flush_insert_sorted_into_flush_list(
buf_pool, block, start_lsn);
} else {
ut_ad(block->page.oldest_modification <= start_lsn);
}
2011-07-14 21:22:41 +02:00
mutex_exit(&block->mutex);
}
2009-09-07 10:22:53 +00:00
#endif /* !UNIV_HOTBACKUP */
2013-12-16 15:38:05 +01:00
/******************************************************************//**
2014-02-26 16:25:11 +01:00
Check if a flush list flush is in progress for any buffer pool instance, or if
all the instances are clean, for heuristic purposes.
@return true if flush list flush is in progress or buffer pool is clean */
2013-12-16 15:38:05 +01:00
UNIV_INLINE
bool
buf_flush_flush_list_in_progress(void)
/*==================================*/
{
2014-02-26 16:25:11 +01:00
bool all_clean = true;
2013-12-16 15:38:05 +01:00
for (ulint i = 0; i < srv_buf_pool_instances; i++) {
const buf_pool_t* buf_pool = buf_pool_from_array(i);
if (buf_pool->init_flush[BUF_FLUSH_LIST]
|| buf_pool->n_flush[BUF_FLUSH_LIST]) {
return(true);
}
2014-02-26 16:25:11 +01:00
if (all_clean) {
all_clean = (UT_LIST_GET_LEN(buf_pool->flush_list)
== 0);
}
2013-12-16 15:38:05 +01:00
}
2014-02-26 16:25:11 +01:00
return(all_clean);
2013-12-16 15:38:05 +01:00
}