Commit 7716bfc4 authored by Bradley C. Kuszmaul's avatar Bradley C. Kuszmaul Committed by Yoni Fogel

close[t:5054] Fix #5054. Improve spec for block_allocator.

git-svn-id: file:///svn/toku/tokudb@44417 c7de825b-a66e-492c-adef-691d508d4ae1
parent 810b6b64
...@@ -94,12 +94,6 @@ grow_blocks_array (BLOCK_ALLOCATOR ba) { ...@@ -94,12 +94,6 @@ grow_blocks_array (BLOCK_ALLOCATOR ba) {
void void
block_allocator_merge_blockpairs_into (u_int64_t d, struct block_allocator_blockpair dst[/*d*/], block_allocator_merge_blockpairs_into (u_int64_t d, struct block_allocator_blockpair dst[/*d*/],
u_int64_t s, const struct block_allocator_blockpair src[/*s*/]) u_int64_t s, const struct block_allocator_blockpair src[/*s*/])
// Effect: Merge dst[d] and src[s] into dst[d+s], merging in place.
// Initially dst and src hold sorted arrays (sorted by increasing offset).
// Finally dst contains all d+s elements sorted in order.
// dst must be large enough.
// Requires no overlaps.
// This is not static so that we can write a unit test for it. (Otherwise, this is static only to be used from inside the block allocator)
{ {
u_int64_t tail = d+s; u_int64_t tail = d+s;
while (d>0 && s>0) { while (d>0 && s>0) {
......
...@@ -22,9 +22,8 @@ extern "C" { ...@@ -22,9 +22,8 @@ extern "C" {
#error #error
#endif #endif
#define BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE (2*BLOCK_ALLOCATOR_HEADER_RESERVE) // Block allocator.
// Overview: A block allocator manages the allocation of variable-sized blocks.
// A block allocator manages the allocation of variable-sized blocks.
// The translation of block numbers to addresses is handled elsewhere. // The translation of block numbers to addresses is handled elsewhere.
// The allocation of block numbers is handled elsewhere. // The allocation of block numbers is handled elsewhere.
...@@ -38,10 +37,12 @@ extern "C" { ...@@ -38,10 +37,12 @@ extern "C" {
// We can free blocks. // We can free blocks.
// We can determine the size of a block. // We can determine the size of a block.
#define BLOCK_ALLOCATOR_TOTAL_HEADER_RESERVE (2*BLOCK_ALLOCATOR_HEADER_RESERVE)
typedef struct block_allocator *BLOCK_ALLOCATOR; typedef struct block_allocator *BLOCK_ALLOCATOR;
void void create_block_allocator (BLOCK_ALLOCATOR * ba, u_int64_t reserve_at_beginning, u_int64_t alignment);
create_block_allocator (BLOCK_ALLOCATOR * ba, u_int64_t reserve_at_beginning, u_int64_t alignment);
// Effect: Create a block allocator, in which the first RESERVE_AT_BEGINNING bytes are not put into a block. // Effect: Create a block allocator, in which the first RESERVE_AT_BEGINNING bytes are not put into a block.
// All blocks be start on a multiple of ALIGNMENT. // All blocks be start on a multiple of ALIGNMENT.
// Aborts if we run out of memory. // Aborts if we run out of memory.
...@@ -50,8 +51,7 @@ create_block_allocator (BLOCK_ALLOCATOR * ba, u_int64_t reserve_at_beginning, u_ ...@@ -50,8 +51,7 @@ create_block_allocator (BLOCK_ALLOCATOR * ba, u_int64_t reserve_at_beginning, u_
// reserve_at_beginning (IN) Size of reserved block at beginning. This size does not have to be aligned. // reserve_at_beginning (IN) Size of reserved block at beginning. This size does not have to be aligned.
// alignment (IN) Block alignment. // alignment (IN) Block alignment.
void void destroy_block_allocator (BLOCK_ALLOCATOR *ba);
destroy_block_allocator (BLOCK_ALLOCATOR *ba);
// Effect: Destroy a block allocator at *ba. // Effect: Destroy a block allocator at *ba.
// Also, set *ba=NULL. // Also, set *ba=NULL.
// Rationale: If there was only one copy of the pointer, this kills that copy too. // Rationale: If there was only one copy of the pointer, this kills that copy too.
...@@ -59,8 +59,7 @@ destroy_block_allocator (BLOCK_ALLOCATOR *ba); ...@@ -59,8 +59,7 @@ destroy_block_allocator (BLOCK_ALLOCATOR *ba);
// ba (IN/OUT): // ba (IN/OUT):
void void block_allocator_alloc_block_at (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t offset);
block_allocator_alloc_block_at (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t offset);
// Effect: Allocate a block of the specified size at a particular offset. // Effect: Allocate a block of the specified size at a particular offset.
// Aborts if anything goes wrong. // Aborts if anything goes wrong.
// The performance of this function may be as bad as Theta(N), where N is the number of blocks currently in use. // The performance of this function may be as bad as Theta(N), where N is the number of blocks currently in use.
...@@ -77,14 +76,12 @@ struct block_allocator_blockpair { ...@@ -77,14 +76,12 @@ struct block_allocator_blockpair {
u_int64_t offset; u_int64_t offset;
u_int64_t size; u_int64_t size;
}; };
void void block_allocator_alloc_blocks_at (BLOCK_ALLOCATOR ba, u_int64_t n_blocks, struct block_allocator_blockpair *pairs);
block_allocator_alloc_blocks_at (BLOCK_ALLOCATOR ba, u_int64_t n_blocks, struct block_allocator_blockpair *pairs);
// Effect: Take pairs in any order, and add them all, as if we did block_allocator_alloc_block() on each pair. // Effect: Take pairs in any order, and add them all, as if we did block_allocator_alloc_block() on each pair.
// This should run in time O(N + M log M) where N is the number of blocks in ba, and M is the number of new blocks. // This should run in time O(N + M log M) where N is the number of blocks in ba, and M is the number of new blocks.
// Modifies: pairs (sorts them). // Modifies: pairs (sorts them).
void void block_allocator_alloc_block (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t *offset);
block_allocator_alloc_block (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t *offset);
// Effect: Allocate a block of the specified size at an address chosen by the allocator. // Effect: Allocate a block of the specified size at an address chosen by the allocator.
// Aborts if anything goes wrong. // Aborts if anything goes wrong.
// The block address will be a multiple of the alignment. // The block address will be a multiple of the alignment.
...@@ -93,8 +90,7 @@ block_allocator_alloc_block (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t *offs ...@@ -93,8 +90,7 @@ block_allocator_alloc_block (BLOCK_ALLOCATOR ba, u_int64_t size, u_int64_t *offs
// size (IN): The size of the block. (The size does not have to be aligned.) // size (IN): The size of the block. (The size does not have to be aligned.)
// offset (OUT): The location of the block. // offset (OUT): The location of the block.
void void block_allocator_free_block (BLOCK_ALLOCATOR ba, u_int64_t offset);
block_allocator_free_block (BLOCK_ALLOCATOR ba, u_int64_t offset);
// Effect: Free the block at offset. // Effect: Free the block at offset.
// Requires: There must be a block currently allocated at that offset. // Requires: There must be a block currently allocated at that offset.
// Parameters: // Parameters:
...@@ -102,47 +98,51 @@ block_allocator_free_block (BLOCK_ALLOCATOR ba, u_int64_t offset); ...@@ -102,47 +98,51 @@ block_allocator_free_block (BLOCK_ALLOCATOR ba, u_int64_t offset);
// offset (IN): The offset of the block. // offset (IN): The offset of the block.
u_int64_t u_int64_t block_allocator_block_size (BLOCK_ALLOCATOR ba, u_int64_t offset);
block_allocator_block_size (BLOCK_ALLOCATOR ba, u_int64_t offset);
// Effect: Return the size of the block that starts at offset. // Effect: Return the size of the block that starts at offset.
// Requires: There must be a block currently allocated at that offset. // Requires: There must be a block currently allocated at that offset.
// Parameters: // Parameters:
// ba (IN/OUT): The block allocator. (Modifies ba.) // ba (IN/OUT): The block allocator. (Modifies ba.)
// offset (IN): The offset of the block. // offset (IN): The offset of the block.
void void block_allocator_validate (BLOCK_ALLOCATOR ba);
block_allocator_validate (BLOCK_ALLOCATOR ba);
// Effect: Check to see if the block allocator is OK. This may take a long time. // Effect: Check to see if the block allocator is OK. This may take a long time.
// Usage Hints: Probably only use this for unit tests. // Usage Hints: Probably only use this for unit tests.
void void block_allocator_print (BLOCK_ALLOCATOR ba);
block_allocator_print (BLOCK_ALLOCATOR ba); // Effect: Print information about the block allocator.
// Rationale: This is probably useful only for debugging.
u_int64_t u_int64_t block_allocator_allocated_limit (BLOCK_ALLOCATOR ba);
block_allocator_allocated_limit (BLOCK_ALLOCATOR ba);
// Effect: Return the unallocated block address of "infinite" size. // Effect: Return the unallocated block address of "infinite" size.
// That is, return the smallest address that is above all the allocated blocks. // That is, return the smallest address that is above all the allocated blocks.
// Rationale: When writing the root FIFO we don't know how big the block is. // Rationale: When writing the root FIFO we don't know how big the block is.
// So we start at the "infinite" block, write the fifo, and then // So we start at the "infinite" block, write the fifo, and then
// allocate_block_at of the correct size and offset to account for the root FIFO. // allocate_block_at of the correct size and offset to account for the root FIFO.
int int block_allocator_get_nth_block_in_layout_order (BLOCK_ALLOCATOR ba, u_int64_t b, u_int64_t *offset, u_int64_t *size);
block_allocator_get_nth_block_in_layout_order (BLOCK_ALLOCATOR ba, u_int64_t b, u_int64_t *offset, u_int64_t *size);
// Effect: Consider the blocks in sorted order. The reserved block at the beginning is number 0. The next one is number 1 and so forth. // Effect: Consider the blocks in sorted order. The reserved block at the beginning is number 0. The next one is number 1 and so forth.
// Return the offset and size of the block with that number. // Return the offset and size of the block with that number.
// Return 0 if there is a block that big, return nonzero if b is too big. // Return 0 if there is a block that big, return nonzero if b is too big.
// This is probably only useful for tests. // Rationale: This is probably useful only for tests.
void void block_allocator_get_unused_statistics(BLOCK_ALLOCATOR ba, TOKU_DB_FRAGMENTATION report);
block_allocator_get_unused_statistics(BLOCK_ALLOCATOR ba, TOKU_DB_FRAGMENTATION report); // Effect: Fill in report to indicate how the file is used.
//Requires: report->file_size_bytes is filled in // Requires:
//Requires: report->data_bytes is filled in // report->file_size_bytes is filled in
//Requires: report->checkpoint_bytes_additional is filled in // report->data_bytes is filled in
// report->checkpoint_bytes_additional is filled in
void void block_allocator_merge_blockpairs_into (u_int64_t d, struct block_allocator_blockpair dst[/*d*/],
block_allocator_merge_blockpairs_into (u_int64_t d, struct block_allocator_blockpair dst[/*d*/],
u_int64_t s, const struct block_allocator_blockpair src[/*s*/]); u_int64_t s, const struct block_allocator_blockpair src[/*s*/]);
// This is exposed so it can be tested. // Effect: Merge dst[d] and src[s] into dst[d+s], merging in place.
// Initially dst and src hold sorted arrays (sorted by increasing offset).
// Finally dst contains all d+s elements sorted in order.
// Requires:
// dst and src are sorted.
// dst must be large enough.
// No blocks may overlap.
// Rationale: This is exposed so it can be tested by a glass box tester. Otherwise it would be static (file-scope) function inside block_allocator.c
#if defined(__cplusplus) || defined(__cilkplusplus) #if defined(__cplusplus) || defined(__cilkplusplus)
}; };
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment