2006-12-07 13:29:04 +00:00
|
|
|
/******************************************************
|
|
|
|
Binary buddy allocator for compressed pages
|
|
|
|
|
|
|
|
(c) 2006 Innobase Oy
|
|
|
|
|
|
|
|
Created December 2006 by Marko Makela
|
|
|
|
*******************************************************/
|
|
|
|
|
|
|
|
#define THIS_MODULE
|
|
|
|
#include "buf0buddy.h"
|
|
|
|
#ifdef UNIV_NONINL
|
|
|
|
# include "buf0buddy.ic"
|
|
|
|
#endif
|
|
|
|
#undef THIS_MODULE
|
|
|
|
#include "buf0buf.h"
|
2006-12-11 09:54:13 +00:00
|
|
|
#include "buf0lru.h"
|
2006-12-08 14:44:28 +00:00
|
|
|
#include "buf0flu.h"
|
2006-12-11 14:27:43 +00:00
|
|
|
#include "page0zip.h"
|
2006-12-07 13:29:04 +00:00
|
|
|
|
2007-02-16 09:22:50 +00:00
|
|
|
/* Statistic counters */
|
2007-02-13 12:41:02 +00:00
|
|
|
|
2007-02-16 09:22:50 +00:00
|
|
|
/** Number of frames allocated from the buffer pool to the buddy system.
|
|
|
|
Protected by buf_pool->mutex. */
|
|
|
|
ulint buf_buddy_n_frames;
|
|
|
|
/** Counts of blocks allocated from the buddy system.
|
|
|
|
Protected by buf_pool->mutex. */
|
2007-03-12 12:01:45 +00:00
|
|
|
ulint buf_buddy_used[BUF_BUDDY_SIZES + 1];
|
2007-02-16 09:22:50 +00:00
|
|
|
/** Counts of blocks relocated by the buddy system.
|
|
|
|
Protected by buf_pool->mutex. */
|
2007-03-12 12:01:45 +00:00
|
|
|
ib_uint64_t buf_buddy_relocated[BUF_BUDDY_SIZES + 1];
|
2007-02-13 12:41:02 +00:00
|
|
|
|
|
|
|
/** Preferred minimum number of frames allocated from the buffer pool
|
|
|
|
to the buddy system. When this number is exceeded, the buddy allocator
|
|
|
|
will not try to free clean compressed-only pages in order to satisfy
|
|
|
|
an allocation request. Protected by buf_pool->mutex. */
|
2007-02-16 09:22:50 +00:00
|
|
|
ulint buf_buddy_min_n_frames = ULINT_UNDEFINED;
|
2007-02-13 12:41:02 +00:00
|
|
|
|
2006-12-29 09:13:57 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Get the offset of the buddy of a compressed page frame. */
|
|
|
|
UNIV_INLINE
|
|
|
|
byte*
|
|
|
|
buf_buddy_get(
|
|
|
|
/*==========*/
|
|
|
|
/* out: the buddy relative of page */
|
|
|
|
byte* page, /* in: compressed page */
|
|
|
|
ulint size) /* in: page size in bytes */
|
|
|
|
{
|
|
|
|
ut_ad(ut_is_2pow(size));
|
|
|
|
ut_ad(size >= BUF_BUDDY_LOW);
|
|
|
|
ut_ad(size < BUF_BUDDY_HIGH);
|
|
|
|
ut_ad(!ut_align_offset(page, size));
|
|
|
|
|
|
|
|
if (((ulint) page) & size) {
|
|
|
|
return(page - size);
|
|
|
|
} else {
|
|
|
|
return(page + size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-12 14:18:22 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Add a block to the head of the appropriate buddy free list. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
buf_buddy_add_to_free(
|
|
|
|
/*==================*/
|
|
|
|
buf_page_t* bpage, /* in,own: block to be freed */
|
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
|
|
|
{
|
|
|
|
#ifdef UNIV_DEBUG_VALGRIND
|
|
|
|
buf_page_t* b = UT_LIST_GET_FIRST(buf_pool->zip_free[i]);
|
|
|
|
|
|
|
|
if (b) UNIV_MEM_VALID(b, BUF_BUDDY_LOW << i);
|
|
|
|
#endif /* UNIV_DEBUG_VALGRIND */
|
|
|
|
|
|
|
|
ut_ad(buf_pool->zip_free[i].start != bpage);
|
|
|
|
UT_LIST_ADD_FIRST(list, buf_pool->zip_free[i], bpage);
|
|
|
|
|
|
|
|
#ifdef UNIV_DEBUG_VALGRIND
|
|
|
|
if (b) UNIV_MEM_FREE(b, BUF_BUDDY_LOW << i);
|
2007-09-06 11:36:35 +00:00
|
|
|
UNIV_MEM_ASSERT_AND_FREE(bpage, BUF_BUDDY_LOW << i);
|
2007-01-12 14:18:22 +00:00
|
|
|
#endif /* UNIV_DEBUG_VALGRIND */
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
Remove a block from the appropriate buddy free list. */
|
|
|
|
UNIV_INLINE
|
|
|
|
void
|
|
|
|
buf_buddy_remove_from_free(
|
|
|
|
/*=======================*/
|
|
|
|
buf_page_t* bpage, /* in: block to be removed */
|
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
|
|
|
{
|
|
|
|
#ifdef UNIV_DEBUG_VALGRIND
|
|
|
|
buf_page_t* prev = UT_LIST_GET_PREV(list, bpage);
|
|
|
|
buf_page_t* next = UT_LIST_GET_NEXT(list, bpage);
|
|
|
|
|
|
|
|
if (prev) UNIV_MEM_VALID(prev, BUF_BUDDY_LOW << i);
|
|
|
|
if (next) UNIV_MEM_VALID(next, BUF_BUDDY_LOW << i);
|
2007-01-12 22:23:12 +00:00
|
|
|
|
|
|
|
ut_ad(!prev || buf_page_get_state(prev) == BUF_BLOCK_ZIP_FREE);
|
|
|
|
ut_ad(!next || buf_page_get_state(next) == BUF_BLOCK_ZIP_FREE);
|
2007-01-12 14:18:22 +00:00
|
|
|
#endif /* UNIV_DEBUG_VALGRIND */
|
|
|
|
|
|
|
|
ut_ad(buf_page_get_state(bpage) == BUF_BLOCK_ZIP_FREE);
|
|
|
|
UT_LIST_REMOVE(list, buf_pool->zip_free[i], bpage);
|
|
|
|
|
|
|
|
#ifdef UNIV_DEBUG_VALGRIND
|
|
|
|
if (prev) UNIV_MEM_FREE(prev, BUF_BUDDY_LOW << i);
|
|
|
|
if (next) UNIV_MEM_FREE(next, BUF_BUDDY_LOW << i);
|
|
|
|
#endif /* UNIV_DEBUG_VALGRIND */
|
|
|
|
}
|
|
|
|
|
2006-12-07 13:29:04 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Try to allocate a block from buf_pool->zip_free[]. */
|
2006-12-11 14:27:43 +00:00
|
|
|
static
|
2006-12-07 13:29:04 +00:00
|
|
|
void*
|
2006-12-11 14:27:43 +00:00
|
|
|
buf_buddy_alloc_zip(
|
2006-12-07 15:05:06 +00:00
|
|
|
/*================*/
|
2006-12-07 13:29:04 +00:00
|
|
|
/* out: allocated block, or NULL
|
|
|
|
if buf_pool->zip_free[] was empty */
|
2006-12-11 14:27:43 +00:00
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
2006-12-07 13:29:04 +00:00
|
|
|
{
|
|
|
|
buf_page_t* bpage;
|
|
|
|
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
2006-12-07 13:29:04 +00:00
|
|
|
ut_a(i < BUF_BUDDY_SIZES);
|
|
|
|
|
2007-01-12 12:36:40 +00:00
|
|
|
#if defined UNIV_DEBUG && !defined UNIV_DEBUG_VALGRIND
|
|
|
|
/* Valgrind would complain about accessing free memory. */
|
|
|
|
UT_LIST_VALIDATE(list, buf_page_t, buf_pool->zip_free[i]);
|
|
|
|
#endif /* UNIV_DEBUG && !UNIV_DEBUG_VALGRIND */
|
2006-12-07 13:29:04 +00:00
|
|
|
bpage = UT_LIST_GET_FIRST(buf_pool->zip_free[i]);
|
|
|
|
|
|
|
|
if (bpage) {
|
2007-01-12 12:36:40 +00:00
|
|
|
UNIV_MEM_VALID(bpage, BUF_BUDDY_LOW << i);
|
2006-12-07 13:29:04 +00:00
|
|
|
ut_a(buf_page_get_state(bpage) == BUF_BLOCK_ZIP_FREE);
|
|
|
|
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_remove_from_free(bpage, i);
|
2006-12-11 14:27:43 +00:00
|
|
|
} else if (i + 1 < BUF_BUDDY_SIZES) {
|
|
|
|
/* Attempt to split. */
|
|
|
|
bpage = buf_buddy_alloc_zip(i + 1);
|
2006-12-07 14:58:58 +00:00
|
|
|
|
|
|
|
if (bpage) {
|
2006-12-18 11:23:58 +00:00
|
|
|
buf_page_t* buddy = (buf_page_t*)
|
|
|
|
(((char*) bpage) + (BUF_BUDDY_LOW << i));
|
2006-12-07 14:58:58 +00:00
|
|
|
|
2006-12-20 11:38:19 +00:00
|
|
|
ut_ad(!buf_pool_contains_zip(buddy));
|
2006-12-19 12:38:18 +00:00
|
|
|
ut_d(memset(buddy, i, BUF_BUDDY_LOW << i));
|
|
|
|
buddy->state = BUF_BLOCK_ZIP_FREE;
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_add_to_free(buddy, i);
|
2006-12-07 14:58:58 +00:00
|
|
|
}
|
2006-12-07 13:29:04 +00:00
|
|
|
}
|
|
|
|
|
2006-12-19 12:38:18 +00:00
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
if (bpage) {
|
|
|
|
memset(bpage, ~i, BUF_BUDDY_LOW << i);
|
|
|
|
}
|
|
|
|
#endif /* UNIV_DEBUG */
|
|
|
|
|
2007-01-12 12:36:40 +00:00
|
|
|
UNIV_MEM_ALLOC(bpage, BUF_BUDDY_SIZES << i);
|
|
|
|
|
2006-12-07 13:29:04 +00:00
|
|
|
return(bpage);
|
|
|
|
}
|
2006-12-07 15:22:44 +00:00
|
|
|
|
2006-12-11 09:54:13 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Deallocate a buffer frame of UNIV_PAGE_SIZE. */
|
|
|
|
static
|
|
|
|
void
|
2006-12-11 14:27:43 +00:00
|
|
|
buf_buddy_block_free(
|
2006-12-11 09:54:13 +00:00
|
|
|
/*=================*/
|
|
|
|
void* buf) /* in: buffer frame to deallocate */
|
|
|
|
{
|
2007-01-05 10:49:56 +00:00
|
|
|
const ulint fold = BUF_POOL_ZIP_FOLD_PTR(buf);
|
2006-12-11 09:54:13 +00:00
|
|
|
buf_page_t* bpage;
|
|
|
|
buf_block_t* block;
|
|
|
|
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
2007-01-05 10:49:56 +00:00
|
|
|
ut_a(!ut_align_offset(buf, UNIV_PAGE_SIZE));
|
2006-12-11 09:54:13 +00:00
|
|
|
|
2007-08-01 08:13:22 +00:00
|
|
|
HASH_SEARCH(hash, buf_pool->zip_hash, fold, buf_page_t*, bpage,
|
2006-12-11 09:54:13 +00:00
|
|
|
((buf_block_t*) bpage)->frame == buf);
|
|
|
|
ut_a(bpage);
|
|
|
|
ut_a(buf_page_get_state(bpage) == BUF_BLOCK_MEMORY);
|
2007-01-24 10:36:05 +00:00
|
|
|
ut_ad(!bpage->in_page_hash);
|
|
|
|
ut_ad(bpage->in_zip_hash);
|
|
|
|
ut_d(bpage->in_zip_hash = FALSE);
|
2006-12-21 14:59:41 +00:00
|
|
|
HASH_DELETE(buf_page_t, hash, buf_pool->zip_hash, fold, bpage);
|
|
|
|
|
2006-12-19 12:38:18 +00:00
|
|
|
ut_d(memset(buf, 0, UNIV_PAGE_SIZE));
|
2007-01-12 09:14:15 +00:00
|
|
|
UNIV_MEM_INVALID(buf, UNIV_PAGE_SIZE);
|
2006-12-11 09:54:13 +00:00
|
|
|
|
|
|
|
block = (buf_block_t*) bpage;
|
|
|
|
mutex_enter(&block->mutex);
|
|
|
|
buf_LRU_block_free_non_file_page(block);
|
|
|
|
mutex_exit(&block->mutex);
|
2007-02-13 12:41:02 +00:00
|
|
|
|
|
|
|
ut_ad(buf_buddy_n_frames > 0);
|
|
|
|
buf_buddy_n_frames--;
|
2006-12-11 09:54:13 +00:00
|
|
|
}
|
|
|
|
|
2006-12-11 14:27:43 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Allocate a buffer block to the buddy allocator. */
|
|
|
|
static
|
|
|
|
void
|
|
|
|
buf_buddy_block_register(
|
|
|
|
/*=====================*/
|
|
|
|
buf_block_t* block) /* in: buffer frame to allocate */
|
|
|
|
{
|
2007-01-05 10:49:56 +00:00
|
|
|
const ulint fold = BUF_POOL_ZIP_FOLD(block);
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
|
|
|
|
2006-12-13 14:58:54 +00:00
|
|
|
buf_block_set_state(block, BUF_BLOCK_MEMORY);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
ut_a(block->frame);
|
2007-01-05 10:49:56 +00:00
|
|
|
ut_a(!ut_align_offset(block->frame, UNIV_PAGE_SIZE));
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2007-01-24 10:36:05 +00:00
|
|
|
ut_ad(!block->page.in_page_hash);
|
|
|
|
ut_ad(!block->page.in_zip_hash);
|
|
|
|
ut_d(block->page.in_zip_hash = TRUE);
|
2006-12-11 14:27:43 +00:00
|
|
|
HASH_INSERT(buf_page_t, hash, buf_pool->zip_hash, fold, &block->page);
|
2007-02-13 12:41:02 +00:00
|
|
|
|
|
|
|
buf_buddy_n_frames++;
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
|
|
|
Allocate a block from a bigger object. */
|
|
|
|
static
|
|
|
|
void*
|
|
|
|
buf_buddy_alloc_from(
|
|
|
|
/*=================*/
|
|
|
|
/* out: allocated block */
|
|
|
|
void* buf, /* in: a block that is free to use */
|
|
|
|
ulint i, /* in: index of buf_pool->zip_free[] */
|
|
|
|
ulint j) /* in: size of buf as an index
|
|
|
|
of buf_pool->zip_free[] */
|
|
|
|
{
|
|
|
|
ulint offs = BUF_BUDDY_LOW << j;
|
2007-02-13 12:41:02 +00:00
|
|
|
ut_ad(j <= BUF_BUDDY_SIZES);
|
|
|
|
ut_ad(j > i);
|
|
|
|
ut_ad(!ut_align_offset(buf, offs));
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
/* Add the unused parts of the block to the free lists. */
|
|
|
|
while (j > i) {
|
|
|
|
buf_page_t* bpage;
|
|
|
|
|
|
|
|
offs >>= 1;
|
|
|
|
j--;
|
|
|
|
|
|
|
|
bpage = (buf_page_t*) ((byte*) buf + offs);
|
2006-12-19 12:38:18 +00:00
|
|
|
ut_d(memset(bpage, j, BUF_BUDDY_LOW << j));
|
2006-12-11 14:27:43 +00:00
|
|
|
bpage->state = BUF_BLOCK_ZIP_FREE;
|
2007-01-12 12:36:40 +00:00
|
|
|
#if defined UNIV_DEBUG && !defined UNIV_DEBUG_VALGRIND
|
|
|
|
/* Valgrind would complain about accessing free memory. */
|
|
|
|
UT_LIST_VALIDATE(list, buf_page_t, buf_pool->zip_free[j]);
|
|
|
|
#endif /* UNIV_DEBUG && !UNIV_DEBUG_VALGRIND */
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_add_to_free(bpage, j);
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2006-12-12 16:28:22 +00:00
|
|
|
Try to allocate a block by freeing an unmodified page. */
|
2006-12-11 14:27:43 +00:00
|
|
|
static
|
|
|
|
void*
|
2006-12-12 16:28:22 +00:00
|
|
|
buf_buddy_alloc_clean(
|
|
|
|
/*==================*/
|
2006-12-11 14:27:43 +00:00
|
|
|
/* out: allocated block, or NULL */
|
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
|
|
|
{
|
|
|
|
buf_page_t* bpage;
|
|
|
|
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2007-02-13 12:41:02 +00:00
|
|
|
if (buf_buddy_n_frames > buf_buddy_min_n_frames) {
|
|
|
|
|
|
|
|
goto free_LRU;
|
|
|
|
}
|
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (BUF_BUDDY_LOW << i >= PAGE_ZIP_MIN_SIZE
|
|
|
|
&& i < BUF_BUDDY_SIZES) {
|
2006-12-12 16:28:22 +00:00
|
|
|
/* Try to find a clean compressed-only page
|
|
|
|
of the same size. */
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
page_zip_des_t dummy_zip;
|
|
|
|
ulint j;
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
page_zip_set_size(&dummy_zip, BUF_BUDDY_LOW << i);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
j = ut_min(UT_LIST_GET_LEN(buf_pool->zip_clean), 100);
|
|
|
|
bpage = UT_LIST_GET_FIRST(buf_pool->zip_clean);
|
|
|
|
|
|
|
|
mutex_enter(&buf_pool->zip_mutex);
|
|
|
|
|
|
|
|
for (; j--; bpage = UT_LIST_GET_NEXT(list, bpage)) {
|
|
|
|
if (bpage->zip.ssize != dummy_zip.ssize
|
2006-12-15 15:05:18 +00:00
|
|
|
|| !buf_LRU_free_block(bpage, FALSE)) {
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2007-01-16 08:59:39 +00:00
|
|
|
/* Reuse the block. */
|
2006-12-12 16:28:22 +00:00
|
|
|
|
|
|
|
mutex_exit(&buf_pool->zip_mutex);
|
|
|
|
bpage = buf_buddy_alloc_zip(i);
|
2007-01-16 08:59:39 +00:00
|
|
|
|
|
|
|
/* bpage may be NULL if buf_buddy_free()
|
|
|
|
[invoked by buf_LRU_free_block() via
|
|
|
|
buf_LRU_block_remove_hashed_page()]
|
|
|
|
recombines blocks and invokes
|
|
|
|
buf_buddy_block_free(). Because
|
|
|
|
buf_pool->mutex will not be released
|
|
|
|
after buf_buddy_block_free(), there will
|
|
|
|
be at least one block available in the
|
|
|
|
buffer pool, and thus it does not make sense
|
|
|
|
to deallocate any further compressed blocks. */
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
return(bpage);
|
|
|
|
}
|
2006-12-12 16:28:22 +00:00
|
|
|
|
|
|
|
mutex_exit(&buf_pool->zip_mutex);
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
/* Free blocks from the end of the LRU list until enough space
|
|
|
|
is available. */
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2007-01-09 12:39:50 +00:00
|
|
|
free_LRU:
|
2006-12-12 16:28:22 +00:00
|
|
|
for (bpage = UT_LIST_GET_LAST(buf_pool->LRU); bpage;
|
|
|
|
bpage = UT_LIST_GET_PREV(LRU, bpage)) {
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-18 11:23:58 +00:00
|
|
|
void* ret;
|
|
|
|
mutex_t* block_mutex = buf_page_get_mutex(bpage);
|
|
|
|
|
2007-02-13 12:41:02 +00:00
|
|
|
if (UNIV_UNLIKELY(!buf_page_in_file(bpage))) {
|
2006-12-21 14:59:41 +00:00
|
|
|
|
|
|
|
/* This is most likely BUF_BLOCK_REMOVE_HASH,
|
|
|
|
that is, the block is already being freed. */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2006-12-18 11:23:58 +00:00
|
|
|
mutex_enter(block_mutex);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-15 15:05:18 +00:00
|
|
|
/* Keep the compressed pages of uncompressed blocks. */
|
|
|
|
if (!buf_LRU_free_block(bpage, FALSE)) {
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-18 11:23:58 +00:00
|
|
|
mutex_exit(block_mutex);
|
2006-12-12 16:28:22 +00:00
|
|
|
continue;
|
|
|
|
}
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-18 11:23:58 +00:00
|
|
|
mutex_exit(block_mutex);
|
|
|
|
|
2007-01-09 12:39:50 +00:00
|
|
|
/* The block was successfully freed.
|
|
|
|
Attempt to allocate memory. */
|
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (i < BUF_BUDDY_SIZES) {
|
2006-12-12 16:28:22 +00:00
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
ret = buf_buddy_alloc_zip(i);
|
2006-12-12 16:28:22 +00:00
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (ret) {
|
|
|
|
|
|
|
|
return(ret);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
buf_block_t* block = buf_LRU_get_free_only();
|
|
|
|
|
|
|
|
if (block) {
|
|
|
|
buf_buddy_block_register(block);
|
|
|
|
return(block->frame);
|
|
|
|
}
|
2006-12-12 16:28:22 +00:00
|
|
|
}
|
2007-01-09 12:39:50 +00:00
|
|
|
|
|
|
|
/* A successful buf_LRU_free_block() may release and
|
|
|
|
reacquire buf_pool->mutex, and thus bpage->LRU of
|
|
|
|
an uncompressed page may point to garbage. Furthermore,
|
|
|
|
if bpage were a compressed page descriptor, it would
|
|
|
|
have been deallocated by buf_LRU_free_block().
|
|
|
|
|
|
|
|
Thus, we must restart the traversal of the LRU list. */
|
|
|
|
|
|
|
|
goto free_LRU;
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************
|
2007-02-08 14:19:55 +00:00
|
|
|
Allocate a block. The thread calling this function must hold
|
|
|
|
buf_pool->mutex and must not hold buf_pool->zip_mutex or any block->mutex.
|
|
|
|
The buf_pool->mutex may only be released and reacquired if lru != NULL. */
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
void*
|
|
|
|
buf_buddy_alloc_low(
|
|
|
|
/*================*/
|
2007-02-08 14:19:55 +00:00
|
|
|
/* out: allocated block,
|
|
|
|
possibly NULL if lru==NULL */
|
2006-12-20 15:56:58 +00:00
|
|
|
ulint i, /* in: index of buf_pool->zip_free[],
|
|
|
|
or BUF_BUDDY_SIZES */
|
2007-02-08 14:19:55 +00:00
|
|
|
ibool* lru) /* in: pointer to a variable that will be assigned
|
|
|
|
TRUE if storage was allocated from the LRU list
|
|
|
|
and buf_pool->mutex was temporarily released,
|
|
|
|
or NULL if the LRU list should not be used */
|
2006-12-11 14:27:43 +00:00
|
|
|
{
|
|
|
|
buf_block_t* block;
|
|
|
|
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (i < BUF_BUDDY_SIZES) {
|
|
|
|
/* Try to allocate from the buddy system. */
|
|
|
|
block = buf_buddy_alloc_zip(i);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (block) {
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2007-02-13 12:41:02 +00:00
|
|
|
goto func_exit;
|
2006-12-20 15:56:58 +00:00
|
|
|
}
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Try allocating from the buf_pool->free list. */
|
|
|
|
block = buf_LRU_get_free_only();
|
|
|
|
|
|
|
|
if (block) {
|
|
|
|
|
|
|
|
goto alloc_big;
|
|
|
|
}
|
|
|
|
|
2006-12-12 12:14:58 +00:00
|
|
|
if (!lru) {
|
|
|
|
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
/* Try replacing a clean page in the buffer pool. */
|
2006-12-11 14:27:43 +00:00
|
|
|
|
2006-12-12 16:28:22 +00:00
|
|
|
block = buf_buddy_alloc_clean(i);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
if (block) {
|
|
|
|
|
2007-02-13 12:41:02 +00:00
|
|
|
goto func_exit;
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Try replacing an uncompressed page in the buffer pool. */
|
2006-12-12 12:14:58 +00:00
|
|
|
mutex_exit(&buf_pool->mutex);
|
2006-12-11 14:27:43 +00:00
|
|
|
block = buf_LRU_get_free_block(0);
|
2007-02-08 14:19:55 +00:00
|
|
|
*lru = TRUE;
|
2006-12-12 12:14:58 +00:00
|
|
|
mutex_enter(&buf_pool->mutex);
|
2006-12-11 14:27:43 +00:00
|
|
|
|
|
|
|
alloc_big:
|
|
|
|
buf_buddy_block_register(block);
|
|
|
|
|
2007-02-13 12:41:02 +00:00
|
|
|
block = buf_buddy_alloc_from(block->frame, i, BUF_BUDDY_SIZES);
|
|
|
|
|
|
|
|
func_exit:
|
|
|
|
buf_buddy_used[i]++;
|
|
|
|
return(block);
|
2006-12-11 14:27:43 +00:00
|
|
|
}
|
|
|
|
|
2007-01-19 12:27:47 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Try to relocate the control block of a compressed page. */
|
|
|
|
static
|
|
|
|
ibool
|
|
|
|
buf_buddy_relocate_block(
|
|
|
|
/*=====================*/
|
|
|
|
/* out: TRUE if relocated */
|
|
|
|
buf_page_t* bpage, /* in: block to relocate */
|
|
|
|
buf_page_t* dpage) /* in: free block to relocate to */
|
|
|
|
{
|
|
|
|
buf_page_t* b;
|
|
|
|
|
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
|
|
|
|
switch (buf_page_get_state(bpage)) {
|
|
|
|
case BUF_BLOCK_ZIP_FREE:
|
|
|
|
case BUF_BLOCK_NOT_USED:
|
|
|
|
case BUF_BLOCK_READY_FOR_USE:
|
|
|
|
case BUF_BLOCK_FILE_PAGE:
|
|
|
|
case BUF_BLOCK_MEMORY:
|
|
|
|
case BUF_BLOCK_REMOVE_HASH:
|
|
|
|
ut_error;
|
|
|
|
case BUF_BLOCK_ZIP_DIRTY:
|
|
|
|
/* Cannot relocate dirty pages. */
|
|
|
|
return(FALSE);
|
|
|
|
|
|
|
|
case BUF_BLOCK_ZIP_PAGE:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_enter(&buf_pool->zip_mutex);
|
|
|
|
|
|
|
|
if (!buf_page_can_relocate(bpage)) {
|
|
|
|
mutex_exit(&buf_pool->zip_mutex);
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
buf_relocate(bpage, dpage);
|
|
|
|
ut_d(bpage->state = BUF_BLOCK_ZIP_FREE);
|
|
|
|
|
|
|
|
/* relocate buf_pool->zip_clean */
|
|
|
|
b = UT_LIST_GET_PREV(list, dpage);
|
|
|
|
UT_LIST_REMOVE(list, buf_pool->zip_clean, dpage);
|
|
|
|
|
|
|
|
if (b) {
|
|
|
|
UT_LIST_INSERT_AFTER(list, buf_pool->zip_clean, b, dpage);
|
|
|
|
} else {
|
|
|
|
UT_LIST_ADD_FIRST(list, buf_pool->zip_clean, dpage);
|
|
|
|
}
|
|
|
|
|
|
|
|
mutex_exit(&buf_pool->zip_mutex);
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
/**************************************************************************
|
|
|
|
Try to relocate a block. */
|
|
|
|
static
|
|
|
|
ibool
|
|
|
|
buf_buddy_relocate(
|
|
|
|
/*===============*/
|
2007-01-19 12:27:47 +00:00
|
|
|
/* out: TRUE if relocated */
|
|
|
|
void* src, /* in: block to relocate */
|
|
|
|
void* dst, /* in: free block to relocate to */
|
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
2006-12-08 14:44:28 +00:00
|
|
|
{
|
|
|
|
buf_page_t* bpage;
|
|
|
|
const ulint size = BUF_BUDDY_LOW << i;
|
|
|
|
|
2007-01-18 18:29:12 +00:00
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
2006-12-11 15:48:37 +00:00
|
|
|
ut_ad(!ut_align_offset(src, size));
|
|
|
|
ut_ad(!ut_align_offset(dst, size));
|
2007-10-30 09:27:09 +00:00
|
|
|
UNIV_MEM_ASSERT_W(dst, size);
|
2006-12-08 14:44:28 +00:00
|
|
|
|
|
|
|
/* We assume that all memory from buf_buddy_alloc()
|
|
|
|
is used for either compressed pages or buf_page_t
|
|
|
|
objects covering compressed pages. */
|
|
|
|
|
|
|
|
if (size >= PAGE_ZIP_MIN_SIZE) {
|
|
|
|
/* This is a compressed page. */
|
|
|
|
mutex_t* mutex;
|
|
|
|
|
2007-01-12 22:23:12 +00:00
|
|
|
/* The src block may be split into smaller blocks,
|
|
|
|
some of which may be free. Thus, the
|
|
|
|
mach_read_from_4() calls below may attempt to read
|
|
|
|
from free memory. The memory is "owned" by the buddy
|
|
|
|
allocator (and it has been allocated from the buffer
|
|
|
|
pool), so there is nothing wrong about this. The
|
|
|
|
mach_read_from_4() calls here will only trigger bogus
|
|
|
|
Valgrind memcheck warnings in UNIV_DEBUG_VALGRIND builds. */
|
2006-12-18 11:23:58 +00:00
|
|
|
bpage = buf_page_hash_get(
|
2007-10-25 07:03:02 +00:00
|
|
|
mach_read_from_4((const byte*) src
|
2006-12-18 11:23:58 +00:00
|
|
|
+ FIL_PAGE_ARCH_LOG_NO_OR_SPACE_ID),
|
2007-10-25 07:03:02 +00:00
|
|
|
mach_read_from_4((const byte*) src
|
2006-12-18 11:23:58 +00:00
|
|
|
+ FIL_PAGE_OFFSET));
|
2006-12-19 12:38:18 +00:00
|
|
|
|
|
|
|
if (!bpage || bpage->zip.data != src) {
|
|
|
|
/* The block has probably been freshly
|
|
|
|
allocated by buf_LRU_get_free_block() but not
|
|
|
|
added to buf_pool->page_hash yet. Obviously,
|
|
|
|
it cannot be relocated. */
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2006-12-20 11:38:19 +00:00
|
|
|
if (page_zip_get_size(&bpage->zip) != size) {
|
|
|
|
/* The block is of different size. We would
|
|
|
|
have to relocate all blocks covered by src.
|
|
|
|
For the sake of simplicity, give up. */
|
|
|
|
ut_ad(page_zip_get_size(&bpage->zip) < size);
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2007-01-12 22:23:12 +00:00
|
|
|
/* The block must have been allocated, but it may
|
|
|
|
contain uninitialized data. */
|
2007-10-31 09:00:08 +00:00
|
|
|
UNIV_MEM_ASSERT_W(src, size);
|
2007-01-12 22:23:12 +00:00
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
mutex = buf_page_get_mutex(bpage);
|
|
|
|
|
|
|
|
mutex_enter(mutex);
|
|
|
|
|
2007-01-04 21:40:10 +00:00
|
|
|
if (buf_page_can_relocate(bpage)) {
|
|
|
|
/* Relocate the compressed page. */
|
|
|
|
ut_a(bpage->zip.data == src);
|
|
|
|
memcpy(dst, src, size);
|
|
|
|
bpage->zip.data = dst;
|
|
|
|
mutex_exit(mutex);
|
2007-02-13 12:41:02 +00:00
|
|
|
success:
|
|
|
|
UNIV_MEM_INVALID(src, size);
|
|
|
|
buf_buddy_relocated[i]++;
|
2007-01-04 21:40:10 +00:00
|
|
|
return(TRUE);
|
2006-12-08 14:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mutex_exit(mutex);
|
2007-01-04 15:45:42 +00:00
|
|
|
} else if (i == buf_buddy_get_slot(sizeof(buf_page_t))) {
|
2006-12-08 14:44:28 +00:00
|
|
|
/* This must be a buf_page_t object. */
|
2007-10-31 09:00:08 +00:00
|
|
|
UNIV_MEM_ASSERT_RW(src, size);
|
2007-02-13 12:41:02 +00:00
|
|
|
if (buf_buddy_relocate_block(src, dst)) {
|
|
|
|
|
|
|
|
goto success;
|
|
|
|
}
|
2006-12-08 14:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
|
2006-12-07 15:22:44 +00:00
|
|
|
/**************************************************************************
|
2006-12-11 09:54:13 +00:00
|
|
|
Deallocate a block. */
|
2006-12-07 15:22:44 +00:00
|
|
|
|
2006-12-11 09:54:13 +00:00
|
|
|
void
|
2006-12-07 15:22:44 +00:00
|
|
|
buf_buddy_free_low(
|
|
|
|
/*===============*/
|
2006-12-20 11:38:19 +00:00
|
|
|
void* buf, /* in: block to be freed, must not be
|
|
|
|
pointed to by the buffer pool */
|
2006-12-07 15:22:44 +00:00
|
|
|
ulint i) /* in: index of buf_pool->zip_free[] */
|
|
|
|
{
|
|
|
|
buf_page_t* bpage;
|
|
|
|
buf_page_t* buddy;
|
2007-01-18 18:29:12 +00:00
|
|
|
|
|
|
|
ut_ad(mutex_own(&buf_pool->mutex));
|
|
|
|
ut_ad(!mutex_own(&buf_pool->zip_mutex));
|
2007-02-13 12:41:02 +00:00
|
|
|
ut_ad(i <= BUF_BUDDY_SIZES);
|
|
|
|
ut_ad(buf_buddy_used[i] > 0);
|
|
|
|
|
|
|
|
buf_buddy_used[i]--;
|
2006-12-07 15:22:44 +00:00
|
|
|
recombine:
|
2007-10-31 09:00:08 +00:00
|
|
|
UNIV_MEM_ASSERT_AND_ALLOC(buf, BUF_BUDDY_LOW << i);
|
2007-01-09 16:11:11 +00:00
|
|
|
ut_d(((buf_page_t*) buf)->state = BUF_BLOCK_ZIP_FREE);
|
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
if (i == BUF_BUDDY_SIZES) {
|
|
|
|
buf_buddy_block_free(buf);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2006-12-07 15:22:44 +00:00
|
|
|
ut_ad(i < BUF_BUDDY_SIZES);
|
|
|
|
ut_ad(buf == ut_align_down(buf, BUF_BUDDY_LOW << i));
|
2006-12-20 11:38:19 +00:00
|
|
|
ut_ad(!buf_pool_contains_zip(buf));
|
2006-12-07 15:22:44 +00:00
|
|
|
|
|
|
|
/* Try to combine adjacent blocks. */
|
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
buddy = (buf_page_t*) buf_buddy_get(((byte*) buf), BUF_BUDDY_LOW << i);
|
2006-12-07 15:22:44 +00:00
|
|
|
|
2007-01-12 12:36:40 +00:00
|
|
|
#ifndef UNIV_DEBUG_VALGRIND
|
|
|
|
/* Valgrind would complain about accessing free memory. */
|
|
|
|
|
2006-12-18 12:55:04 +00:00
|
|
|
if (buddy->state != BUF_BLOCK_ZIP_FREE) {
|
|
|
|
|
|
|
|
goto buddy_nonfree;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The field buddy->state can only be trusted for free blocks.
|
|
|
|
If buddy->state == BUF_BLOCK_ZIP_FREE, the block is free if
|
|
|
|
it is in the free list. */
|
2007-01-12 12:36:40 +00:00
|
|
|
#endif /* !UNIV_DEBUG_VALGRIND */
|
2006-12-18 12:55:04 +00:00
|
|
|
|
2007-01-12 14:18:22 +00:00
|
|
|
for (bpage = UT_LIST_GET_FIRST(buf_pool->zip_free[i]); bpage; ) {
|
2007-01-12 12:36:40 +00:00
|
|
|
UNIV_MEM_VALID(bpage, BUF_BUDDY_LOW << i);
|
2006-12-18 11:23:58 +00:00
|
|
|
ut_ad(buf_page_get_state(bpage) == BUF_BLOCK_ZIP_FREE);
|
2006-12-08 14:44:28 +00:00
|
|
|
|
2006-12-07 15:22:44 +00:00
|
|
|
if (bpage == buddy) {
|
2006-12-08 14:44:28 +00:00
|
|
|
buddy_free:
|
2006-12-07 15:22:44 +00:00
|
|
|
/* The buddy is free: recombine */
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_remove_from_free(bpage, i);
|
2006-12-20 13:29:02 +00:00
|
|
|
buddy_free2:
|
2007-01-09 16:11:11 +00:00
|
|
|
ut_ad(buf_page_get_state(buddy) == BUF_BLOCK_ZIP_FREE);
|
2006-12-20 13:29:02 +00:00
|
|
|
ut_ad(!buf_pool_contains_zip(buddy));
|
2006-12-20 15:56:58 +00:00
|
|
|
i++;
|
|
|
|
buf = ut_align_down(buf, BUF_BUDDY_LOW << i);
|
2006-12-07 15:22:44 +00:00
|
|
|
|
2006-12-20 15:56:58 +00:00
|
|
|
goto recombine;
|
2006-12-07 15:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ut_a(bpage != buf);
|
2007-01-12 14:18:22 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
buf_page_t* next = UT_LIST_GET_NEXT(list, bpage);
|
2007-09-06 11:36:35 +00:00
|
|
|
UNIV_MEM_ASSERT_AND_FREE(bpage, BUF_BUDDY_LOW << i);
|
2007-01-12 14:18:22 +00:00
|
|
|
bpage = next;
|
|
|
|
}
|
2006-12-07 15:22:44 +00:00
|
|
|
}
|
|
|
|
|
2007-01-12 12:36:40 +00:00
|
|
|
#ifndef UNIV_DEBUG_VALGRIND
|
2006-12-18 12:55:04 +00:00
|
|
|
buddy_nonfree:
|
2007-01-12 12:36:40 +00:00
|
|
|
/* Valgrind would complain about accessing free memory. */
|
2006-12-20 11:38:19 +00:00
|
|
|
ut_d(UT_LIST_VALIDATE(list, buf_page_t, buf_pool->zip_free[i]));
|
2007-01-12 12:36:40 +00:00
|
|
|
#endif /* UNIV_DEBUG_VALGRIND */
|
2006-12-20 11:38:19 +00:00
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
/* The buddy is not free. Is there a free block of this size? */
|
|
|
|
bpage = UT_LIST_GET_FIRST(buf_pool->zip_free[i]);
|
|
|
|
|
|
|
|
if (bpage) {
|
2006-12-20 13:29:02 +00:00
|
|
|
/* Remove the block from the free list, because a successful
|
|
|
|
buf_buddy_relocate() will overwrite bpage->list. */
|
|
|
|
|
2007-01-12 22:23:12 +00:00
|
|
|
UNIV_MEM_VALID(bpage, BUF_BUDDY_LOW << i);
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_remove_from_free(bpage, i);
|
2006-12-20 13:29:02 +00:00
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
/* Try to relocate the buddy of buf to the free block. */
|
|
|
|
if (buf_buddy_relocate(buddy, bpage, i)) {
|
|
|
|
|
2007-01-09 16:11:11 +00:00
|
|
|
ut_d(buddy->state = BUF_BLOCK_ZIP_FREE);
|
2006-12-20 13:29:02 +00:00
|
|
|
goto buddy_free2;
|
2006-12-08 14:44:28 +00:00
|
|
|
}
|
|
|
|
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_add_to_free(bpage, i);
|
2006-12-20 13:29:02 +00:00
|
|
|
|
2006-12-08 14:44:28 +00:00
|
|
|
/* Try to relocate the buddy of the free block to buf. */
|
2006-12-18 12:39:39 +00:00
|
|
|
buddy = (buf_page_t*) buf_buddy_get(((byte*) bpage),
|
|
|
|
BUF_BUDDY_LOW << i);
|
2006-12-08 14:44:28 +00:00
|
|
|
|
2007-01-12 22:23:12 +00:00
|
|
|
#if defined UNIV_DEBUG && !defined UNIV_DEBUG_VALGRIND
|
2006-12-08 14:44:28 +00:00
|
|
|
{
|
|
|
|
const buf_page_t* b;
|
|
|
|
|
2007-01-12 22:23:12 +00:00
|
|
|
/* The buddy must not be (completely) free, because
|
|
|
|
we always recombine adjacent free blocks.
|
|
|
|
(Parts of the buddy can be free in
|
|
|
|
buf_pool->zip_free[j] with j < i.)*/
|
2006-12-08 14:44:28 +00:00
|
|
|
for (b = UT_LIST_GET_FIRST(buf_pool->zip_free[i]);
|
|
|
|
b; b = UT_LIST_GET_NEXT(list, b)) {
|
|
|
|
|
|
|
|
ut_a(b != buddy);
|
|
|
|
}
|
|
|
|
}
|
2007-01-12 12:36:40 +00:00
|
|
|
#endif /* UNIV_DEBUG && !UNIV_DEBUG_VALGRIND */
|
2006-12-08 14:44:28 +00:00
|
|
|
|
|
|
|
if (buf_buddy_relocate(buddy, buf, i)) {
|
|
|
|
|
2006-12-19 12:38:18 +00:00
|
|
|
buf = bpage;
|
2007-01-12 22:23:12 +00:00
|
|
|
UNIV_MEM_VALID(bpage, BUF_BUDDY_LOW << i);
|
2007-01-09 16:11:11 +00:00
|
|
|
ut_d(buddy->state = BUF_BLOCK_ZIP_FREE);
|
2006-12-08 14:44:28 +00:00
|
|
|
goto buddy_free;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-12-07 15:22:44 +00:00
|
|
|
/* Free the block to the buddy list. */
|
|
|
|
bpage = buf;
|
2007-01-09 16:11:11 +00:00
|
|
|
#ifdef UNIV_DEBUG
|
|
|
|
if (i < buf_buddy_get_slot(PAGE_ZIP_MIN_SIZE)) {
|
|
|
|
/* This area has most likely been allocated for at
|
|
|
|
least one compressed-only block descriptor. Check
|
|
|
|
that there are no live objects in the area. This is
|
|
|
|
not a complete check: it may yield false positives as
|
|
|
|
well as false negatives. Also, due to buddy blocks
|
|
|
|
being recombined, it is possible (although unlikely)
|
|
|
|
that this branch is never reached. */
|
|
|
|
|
|
|
|
char* c;
|
|
|
|
|
2007-01-12 12:36:40 +00:00
|
|
|
# ifndef UNIV_DEBUG_VALGRIND
|
|
|
|
/* Valgrind would complain about accessing
|
|
|
|
uninitialized memory. Besides, Valgrind performs a
|
|
|
|
more exhaustive check, at every memory access. */
|
2007-01-09 16:11:11 +00:00
|
|
|
const buf_page_t* b = buf;
|
|
|
|
const buf_page_t* const b_end = (buf_page_t*)
|
|
|
|
((char*) b + (BUF_BUDDY_LOW << i));
|
|
|
|
|
|
|
|
for (; b < b_end; b++) {
|
|
|
|
/* Avoid false positives (and cause false
|
|
|
|
negatives) by checking for b->space < 1000. */
|
|
|
|
|
|
|
|
if ((b->state == BUF_BLOCK_ZIP_PAGE
|
2007-01-19 12:27:47 +00:00
|
|
|
|| b->state == BUF_BLOCK_ZIP_DIRTY)
|
|
|
|
&& b->space > 0 && b->space < 1000) {
|
2007-01-09 16:11:11 +00:00
|
|
|
fprintf(stderr,
|
|
|
|
"buddy dirty %p %u (%u,%u) %p,%lu\n",
|
|
|
|
(void*) b,
|
|
|
|
b->state, b->space, b->offset,
|
|
|
|
buf, i);
|
|
|
|
}
|
|
|
|
}
|
2007-01-12 12:36:40 +00:00
|
|
|
# endif /* !UNIV_DEBUG_VALGRIND */
|
2007-01-09 16:11:11 +00:00
|
|
|
|
|
|
|
/* Scramble the block. This should make any pointers
|
|
|
|
invalid and trigger a segmentation violation. Because
|
|
|
|
the scrambling can be reversed, it may be possible to
|
|
|
|
track down the object pointing to the freed data by
|
|
|
|
dereferencing the unscrambled bpage->LRU or
|
|
|
|
bpage->list pointers. */
|
|
|
|
for (c = (char*) buf + (BUF_BUDDY_LOW << i);
|
|
|
|
c-- > (char*) buf; ) {
|
|
|
|
*c = ~*c ^ i;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Fill large blocks with a constant pattern. */
|
|
|
|
memset(bpage, i, BUF_BUDDY_LOW << i);
|
|
|
|
}
|
|
|
|
#endif /* UNIV_DEBUG */
|
2006-12-07 15:22:44 +00:00
|
|
|
bpage->state = BUF_BLOCK_ZIP_FREE;
|
2007-01-12 14:18:22 +00:00
|
|
|
buf_buddy_add_to_free(bpage, i);
|
2006-12-07 15:22:44 +00:00
|
|
|
}
|