Skip to content

Commit

Permalink
Dummy allocator for heap
Browse files Browse the repository at this point in the history
  • Loading branch information
JayFoxRox committed Jun 6, 2019
1 parent dc97da6 commit 7efaa1d
Show file tree
Hide file tree
Showing 6 changed files with 159 additions and 17 deletions.
3 changes: 3 additions & 0 deletions .gitmodules
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
[submodule "tinyalloc"]
path = tinyalloc
url = https://github.com/thi-ng/tinyalloc
1 change: 1 addition & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ add_executable(openswe1r
emulation.c
export.c
shader.c
alloc.c

dll/kernel32.c

Expand Down
138 changes: 138 additions & 0 deletions alloc.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
// Copyright 2019 OpenSWE1R Maintainers
// Licensed under GPLv2 or any later version
// Refer to the included LICENSE.txt file.

// This is a temporary placeholder for a memory allocator.

#include "alloc.h"

#include <stdint.h>
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>

static unsigned int block_count = 0;
static uint32_t block_size = 0;
static uint8_t* block_usage = NULL;

static void mark_used(unsigned int block, bool used) {
uint8_t mask = 1 << (block % 8);
if (used) {
block_usage[block / 8] |= mask;
} else {
block_usage[block / 8] &= ~mask;
}
}

static bool is_used(unsigned int block) {
uint8_t mask = 1 << (block % 8);
return block_usage[block / 8] & mask;
}

static unsigned int get_block_count(uint32_t size) {
return (size + block_size - 1) / block_size;
}

static unsigned int count_blocks(unsigned int block, bool used) {
unsigned int count = 0;
while(block < block_count) {
if (is_used(block) != used) {
break;
}
block++;
count++;
}
return count;
}

static unsigned int find_free_blocks(unsigned int needed_blocks) {

unsigned int best_count = 0;
unsigned int best_block = 0;

unsigned int block = 0;
while(block < block_count) {

bool block_used = is_used(block);
unsigned int available_blocks = count_blocks(block, block_used);

// Only look at unused blocks that are large enough
if (!block_used && (available_blocks >= needed_blocks)) {

// Check if we don't have a result yet, or if this is a better fit
if ((best_count == 0) || (available_blocks < best_count)) {

// Update result
best_block = block;
best_count = available_blocks;

// If we have an exact fit, use that
if (best_count == needed_blocks) {
break;
}

}
}

block += available_blocks;
}

// Assert that we have a result
assert(best_count != 0);

// Returns best block address
return best_block;
}

void alloc_initialize(uint32_t size, uint32_t _block_size) {

// Sanity check arguments, we expect proper alignment
assert((size % _block_size) == 0);

// Store settings
block_size = _block_size;
block_count = get_block_count(size);

// Allocate flags which tell us wether a block is used
size_t used_flag_size = (block_count + 7) / 8;
block_usage = malloc(used_flag_size);
memset(block_usage, 0x00, used_flag_size);

}

uint32_t alloc_allocate(uint32_t size) {

// Don't allow zero-size allocations (breaks algorithm)
assert(size > 0);

unsigned int needed_blocks = get_block_count(size);

//FIXME: `+ 2` is a hack, so we can recognize block splits
unsigned int block = find_free_blocks(needed_blocks + 2) + 1;

printf("Allocated %u blocks (0x%X / 0x%X bytes)\n", needed_blocks, needed_blocks * block_size, size);

for(unsigned int i = 0; i < needed_blocks; i++) {
mark_used(block + i, true);
}

return block * block_size;
}

void alloc_free(uint32_t offset) {

// Only free addresses that are block-aligned
assert(offset % block_size == 0);
unsigned int block = offset / block_size;

unsigned int used_blocks = count_blocks(block, true);

// Avoid double-free
assert(used_blocks > 0);

for(unsigned int i = 0; i < used_blocks; i++) {
mark_used(block + i, false);
}

}
5 changes: 5 additions & 0 deletions alloc.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
#include <stdint.h>

void alloc_initialize(uint32_t size, uint32_t _block_size);
uint32_t alloc_allocate(uint32_t size);
void alloc_free(uint32_t offset);
28 changes: 11 additions & 17 deletions emulation.c
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,8 @@
#include <sys/mman.h>
#include <errno.h>
#endif
#else
#include "alloc.h"
#endif

//FIXME: These are hacks (register when mapping instead!)!
Expand Down Expand Up @@ -341,22 +343,14 @@ Address Allocate(Size size) {
#ifdef UC_NATIVE
Address ret = aligned_malloc(0x1000, size);
#else
static uint32_t address = HEAP_ADDRESS;

#if 1
//FIXME: This is a hack to fix alignment + to avoid too small allocations
address += 0xFFF;
address &= 0xFFFFF000;
#endif

Address ret = address;

address += size;
//FIXME: Proper allocator

assert(address < (HEAP_ADDRESS + heapSize));
int use = (address - HEAP_ADDRESS);
printf("%u / %u = %u percent\n", use, heapSize, (use * 100) / heapSize);
static bool initialized = false;
if (!initialized) {
alloc_initialize(heapSize, 0x1000);
initialized = true;
}
uint32_t offset = alloc_allocate(size);
Address ret = heapAddress + offset;
printf("0x%X + 0x%X = 0x%X\n", heapAddress, offset, ret);
#endif

assert(ret != 0);
Expand All @@ -373,7 +367,7 @@ void Free(Address address) {
#ifdef UC_NATIVE
aligned_free(address);
#else
//FIXME!
alloc_free(address - heapAddress);
#endif
}

Expand Down
1 change: 1 addition & 0 deletions tinyalloc
Submodule tinyalloc added at 9ebd4a

0 comments on commit 7efaa1d

Please sign in to comment.