You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
4.6 KiB
202 lines
4.6 KiB
/* |
|
* EEZ Modular Firmware |
|
* Copyright (C) 2021-present, Envox d.o.o. |
|
* |
|
* 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, either version 3 of the License, or |
|
* (at your option) any later version. |
|
|
|
* 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 this program. If not, see <http://www.gnu.org/licenses/>. |
|
*/ |
|
|
|
#include <stdio.h> |
|
#include <math.h> |
|
#include <assert.h> |
|
#include <memory.h> |
|
#include <cstring> |
|
|
|
#include <eez/core/alloc.h> |
|
#include <eez/core/os.h> |
|
|
|
namespace eez { |
|
|
|
static const size_t ALIGNMENT = 8; |
|
static const size_t MIN_BLOCK_SIZE = 8; |
|
|
|
struct AllocBlock { |
|
AllocBlock *next; |
|
int free; |
|
size_t size; |
|
uint32_t id; |
|
}; |
|
|
|
static uint8_t *g_heap; |
|
|
|
#if defined(EEZ_PLATFORM_STM32) |
|
#pragma GCC diagnostic push |
|
#pragma GCC diagnostic ignored "-Wparentheses" |
|
#endif |
|
|
|
EEZ_MUTEX_DECLARE(alloc); |
|
|
|
#if defined(EEZ_PLATFORM_STM32) |
|
#pragma GCC diagnostic pop |
|
#endif |
|
|
|
void initAllocHeap(uint8_t *heap, size_t heapSize) { |
|
g_heap = heap; |
|
|
|
AllocBlock *first = (AllocBlock *)g_heap; |
|
first->next = 0; |
|
first->free = 1; |
|
first->size = heapSize - sizeof(AllocBlock); |
|
|
|
EEZ_MUTEX_CREATE(alloc); |
|
} |
|
|
|
void *alloc(size_t size, uint32_t id) { |
|
if (size == 0) { |
|
return nullptr; |
|
} |
|
|
|
if (EEZ_MUTEX_WAIT(alloc, 0)) { |
|
AllocBlock *firstBlock = (AllocBlock *)g_heap; |
|
|
|
AllocBlock *block = firstBlock; |
|
size = ((size + ALIGNMENT - 1) / ALIGNMENT) * ALIGNMENT; |
|
|
|
// find free block with enough size |
|
// TODO merge multiple free consecutive blocks into one that has enough size |
|
while (block) { |
|
if (block->free && block->size >= size) { |
|
break; |
|
} |
|
block = block->next; |
|
} |
|
|
|
if (!block) { |
|
EEZ_MUTEX_RELEASE(alloc); |
|
return nullptr; |
|
} |
|
|
|
int remainingSize = block->size - size - sizeof(AllocBlock); |
|
if (remainingSize >= (int)MIN_BLOCK_SIZE) { |
|
// remainingSize is enough to create a new block |
|
auto newBlock = (AllocBlock *)((uint8_t *)block + sizeof(AllocBlock) + size); |
|
newBlock->next = block->next; |
|
newBlock->free = 1; |
|
newBlock->size = remainingSize; |
|
|
|
block->next = newBlock; |
|
block->size = size; |
|
} |
|
|
|
block->free = 0; |
|
block->id = id; |
|
|
|
EEZ_MUTEX_RELEASE(alloc); |
|
|
|
return block + 1; |
|
} |
|
|
|
return nullptr; |
|
} |
|
|
|
void free(void *ptr) { |
|
if (ptr == 0) { |
|
return; |
|
} |
|
|
|
if (EEZ_MUTEX_WAIT(alloc, 0)) { |
|
AllocBlock *firstBlock = (AllocBlock *)g_heap; |
|
|
|
AllocBlock *prevBlock = nullptr; |
|
AllocBlock *block = firstBlock; |
|
|
|
while (block && block + 1 < ptr) { |
|
prevBlock = block; |
|
block = block->next; |
|
} |
|
|
|
if (!block || block + 1 != ptr || block->free) { |
|
assert(false); |
|
EEZ_MUTEX_RELEASE(alloc); |
|
return; |
|
} |
|
|
|
// reset memory to catch errors when memory is used after free is called |
|
memset(ptr, 0xCC, block->size); |
|
|
|
auto nextBlock = block->next; |
|
if (nextBlock && nextBlock->free) { |
|
if (prevBlock && prevBlock->free) { |
|
// both next and prev blocks are free, merge 3 blocks into one |
|
prevBlock->next = nextBlock->next; |
|
prevBlock->size += sizeof(AllocBlock) + block->size + sizeof(AllocBlock) + nextBlock->size; |
|
} else { |
|
// next block is free, merge 2 blocks into one |
|
block->next = nextBlock->next; |
|
block->size += sizeof(AllocBlock) + nextBlock->size; |
|
block->free = 1; |
|
} |
|
} else if (prevBlock && prevBlock->free) { |
|
// prev block is free, merge 2 blocks into one |
|
prevBlock->next = nextBlock; |
|
prevBlock->size += sizeof(AllocBlock) + block->size; |
|
} else { |
|
// just free |
|
block->free = 1; |
|
} |
|
|
|
EEZ_MUTEX_RELEASE(alloc); |
|
} |
|
} |
|
|
|
template<typename T> void freeObject(T *ptr) { |
|
ptr->~T(); |
|
free(ptr); |
|
} |
|
|
|
#if OPTION_SCPI |
|
void dumpAlloc(scpi_t *context) { |
|
AllocBlock *first = (AllocBlock *)g_heap; |
|
AllocBlock *block = first; |
|
while (block) { |
|
char buffer[100]; |
|
if (block->free) { |
|
snprintf(buffer, sizeof(buffer), "FREE: %d", (int)block->size); |
|
} else { |
|
snprintf(buffer, sizeof(buffer), "ALOC (0x%08x): %d", (unsigned int)block->id, (int)block->size); |
|
} |
|
SCPI_ResultText(context, buffer); |
|
block = block->next; |
|
} |
|
} |
|
#endif |
|
|
|
void getAllocInfo(uint32_t &free, uint32_t &alloc) { |
|
free = 0; |
|
alloc = 0; |
|
if (EEZ_MUTEX_WAIT(alloc, 0)) { |
|
AllocBlock *first = (AllocBlock *)g_heap; |
|
AllocBlock *block = first; |
|
while (block) { |
|
if (block->free) { |
|
free += block->size; |
|
} else { |
|
alloc += block->size; |
|
} |
|
block = block->next; |
|
} |
|
EEZ_MUTEX_RELEASE(alloc); |
|
} |
|
} |
|
|
|
} // eez
|
|
|