// Copyright 2018 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #pragma once #include "android/utils/compiler.h" #include #include #include #include #include #include ANDROID_BEGIN_HEADER #ifdef ADDRESS_SPACE_NAMESPACE namespace ADDRESS_SPACE_NAMESPACE { #endif // This is ported from goldfish_address_space, allowing it to be used for // general sub-allocations of any buffer range. // It is also a pure header library, so there are no compiler tricks needed // to use this in a particular implementation. please don't include this // in a file that is included everywhere else, though. /* Represents a continuous range of addresses and a flag if this block is * available */ struct address_block { uint64_t offset; union { uint64_t size_available; /* VMSTATE_x does not support bit fields */ struct { uint64_t size : 63; uint64_t available : 1; }; }; }; /* A dynamic array of address blocks, with the following invariant: * blocks[i].size > 0 * blocks[i+1].offset = blocks[i].offset + blocks[i].size */ struct address_space_allocator { struct address_block *blocks; int size; int capacity; uint64_t total_bytes; }; #define ANDROID_EMU_ADDRESS_SPACE_BAD_OFFSET (~(uint64_t)0) /* The assert function to abort if something goes wrong. */ static void address_space_assert(bool condition) { #ifdef ANDROID_EMU_ADDRESS_SPACE_ASSERT_FUNC ANDROID_EMU_ADDRESS_SPACE_ASSERT_FUNC(condition); #else (void)condition; assert(condition); #endif } static void* address_space_malloc0(size_t size) { #ifdef ANDROID_EMU_ADDRESS_SPACE_MALLOC0_FUNC return ANDROID_EMU_ADDRESS_SPACE_MALLOC0_FUNC(size); #else void* res = malloc(size); memset(res, 0, size); return res; #endif } static void* address_space_realloc(void* ptr, size_t size) { #ifdef ANDROID_EMU_ADDRESS_SPACE_REALLOC_FUNC return ANDROID_EMU_ADDRESS_SPACE_REALLOC_FUNC(ptr, size); #else void* res = realloc(ptr, size); return res; #endif } static void address_space_free(void* ptr) { #ifdef ANDROID_EMU_ADDRESS_SPACE_FREE_FUNC return ANDROID_EMU_ADDRESS_SPACE_FREE_FUNC(ptr); #else free(ptr); #endif } /* Looks for the smallest (to reduce fragmentation) available block with size to * fit the requested amount and returns its index or -1 if none is available. */ static int address_space_allocator_find_available_block( struct address_block *block, int n_blocks, uint64_t size_at_least) { int index = -1; uint64_t size_at_index = 0; int i; address_space_assert(n_blocks >= 1); for (i = 0; i < n_blocks; ++i, ++block) { uint64_t this_size = block->size; address_space_assert(this_size > 0); if (this_size >= size_at_least && block->available && (index < 0 || this_size < size_at_index)) { index = i; size_at_index = this_size; } } return index; } static int address_space_allocator_grow_capacity(int old_capacity) { address_space_assert(old_capacity >= 1); return old_capacity + old_capacity; } /* Inserts one more address block right after i'th (by borrowing i'th size) and * adjusts sizes: * pre: * size > blocks[i].size * * post: * * might reallocate allocator->blocks if there is no capacity to insert one * * blocks[i].size -= size; * * blocks[i+1].size = size; */ static struct address_block* address_space_allocator_split_block( struct address_space_allocator *allocator, int i, uint64_t size) { address_space_assert(allocator->capacity >= 1); address_space_assert(allocator->size >= 1); address_space_assert(allocator->size <= allocator->capacity); address_space_assert(i >= 0); address_space_assert(i < allocator->size); address_space_assert(size < allocator->blocks[i].size); if (allocator->size == allocator->capacity) { int new_capacity = address_space_allocator_grow_capacity(allocator->capacity); allocator->blocks = (struct address_block*) address_space_realloc( allocator->blocks, sizeof(struct address_block) * new_capacity); address_space_assert(allocator->blocks); allocator->capacity = new_capacity; } struct address_block *blocks = allocator->blocks; /* size = 5, i = 1 * [ 0 | 1 | 2 | 3 | 4 ] => [ 0 | 1 | new | 2 | 3 | 4 ] * i (i+1) (i+2) i (i+1) (i+2) */ memmove(&blocks[i + 2], &blocks[i + 1], sizeof(struct address_block) * (allocator->size - i - 1)); struct address_block *to_borrow_from = &blocks[i]; struct address_block *new_block = to_borrow_from + 1; uint64_t new_size = to_borrow_from->size - size; to_borrow_from->size = new_size; new_block->offset = to_borrow_from->offset + new_size; new_block->size = size; new_block->available = 1; ++allocator->size; return new_block; } /* Marks i'th block as available. If adjacent ((i-1) and (i+1)) blocks are also * available, it merges i'th block with them. * pre: * i < allocator->size * post: * i'th block is merged with adjacent ones if they are available, blocks that * were merged from are removed. allocator->size is updated if blocks were * removed. */ static void address_space_allocator_release_block( struct address_space_allocator *allocator, int i) { struct address_block *blocks = allocator->blocks; int before = i - 1; int after = i + 1; int size = allocator->size; address_space_assert(i >= 0); address_space_assert(i < size); blocks[i].available = 1; if (before >= 0 && blocks[before].available) { if (after < size && blocks[after].available) { // merge (before, i, after) into before blocks[before].size += (blocks[i].size + blocks[after].size); size -= 2; memmove(&blocks[i], &blocks[i + 2], sizeof(struct address_block) * (size - i)); allocator->size = size; } else { // merge (before, i) into before blocks[before].size += blocks[i].size; --size; memmove(&blocks[i], &blocks[i + 1], sizeof(struct address_block) * (size - i)); allocator->size = size; } } else if (after < size && blocks[after].available) { // merge (i, after) into i blocks[i].size += blocks[after].size; --size; memmove(&blocks[after], &blocks[after + 1], sizeof(struct address_block) * (size - after)); allocator->size = size; } } /* Takes a size to allocate an address block and returns an offset where this * block is allocated. This block will not be available for other callers unless * it is explicitly deallocated (see address_space_allocator_deallocate below). */ static uint64_t address_space_allocator_allocate( struct address_space_allocator *allocator, uint64_t size) { int i = address_space_allocator_find_available_block(allocator->blocks, allocator->size, size); if (i < 0) { return ANDROID_EMU_ADDRESS_SPACE_BAD_OFFSET; } else { address_space_assert(i < allocator->size); struct address_block *block = &allocator->blocks[i]; address_space_assert(block->size >= size); if (block->size > size) { block = address_space_allocator_split_block(allocator, i, size); } address_space_assert(block->size == size); block->available = 0; return block->offset; } } /* Takes an offset returned from address_space_allocator_allocate ealier * (see above) and marks this block as available for further allocation. */ static uint32_t address_space_allocator_deallocate( struct address_space_allocator *allocator, uint64_t offset) { struct address_block *block = allocator->blocks; int size = allocator->size; int i; address_space_assert(size >= 1); for (i = 0; i < size; ++i, ++block) { if (block->offset == offset) { if (block->available) { return EINVAL; } else { address_space_allocator_release_block(allocator, i); return 0; } } } return EINVAL; } /* Creates a seed block. */ static void address_space_allocator_init( struct address_space_allocator *allocator, uint64_t size, int initial_capacity) { address_space_assert(initial_capacity >= 1); allocator->blocks = (struct address_block*) malloc(sizeof(struct address_block) * initial_capacity); memset(allocator->blocks, 0, sizeof(struct address_block) * initial_capacity); address_space_assert(allocator->blocks); struct address_block *block = allocator->blocks; block->offset = 0; block->size = size; block->available = 1; allocator->size = 1; allocator->capacity = initial_capacity; allocator->total_bytes = size; } /* At this point there should be no used blocks and all available blocks must * have been merged into one block. */ static void address_space_allocator_destroy( struct address_space_allocator *allocator) { address_space_assert(allocator->size == 1); address_space_assert(allocator->capacity >= allocator->size); address_space_assert(allocator->blocks[0].available); address_space_free(allocator->blocks); } /* Destroy function if we don't care what was previoulsy allocated. * have been merged into one block. */ static void address_space_allocator_destroy_nocleanup( struct address_space_allocator *allocator) { address_space_free(allocator->blocks); } /* Resets the state of the allocator to the initial state without * performing any dynamic memory management. */ static void address_space_allocator_reset( struct address_space_allocator *allocator) { address_space_assert(allocator->size >= 1); allocator->size = 1; struct address_block* block = allocator->blocks; block->offset = 0; block->size = allocator->total_bytes; block->available = 1; } typedef void (*address_block_iter_func_t)(void* context, struct address_block*); typedef void (*address_space_allocator_iter_func_t)(void* context, struct address_space_allocator*); static void address_space_allocator_run( struct address_space_allocator *allocator, void* context, address_space_allocator_iter_func_t allocator_func, address_block_iter_func_t block_func) { struct address_block *block = 0; int size; int i; allocator_func(context, allocator); block = allocator->blocks; size = allocator->size; address_space_assert(size >= 1); for (i = 0; i < size; ++i, ++block) { block_func(context, block); } } #ifdef ADDRESS_SPACE_NAMESPACE } #endif ANDROID_END_HEADER