2021-02-28 02:18:39 +01:00
|
|
|
/* SPDX-License-Identifier: GPL-3.0-or-later */
|
|
|
|
/* See the end of this file for copyright, license, and warranty information. */
|
2020-10-11 19:08:20 +02:00
|
|
|
|
2020-12-08 12:55:23 +01:00
|
|
|
#include <ardix/atomic.h>
|
2020-10-11 19:08:20 +02:00
|
|
|
#include <ardix/list.h>
|
2020-10-12 17:54:07 +02:00
|
|
|
#include <ardix/malloc.h>
|
2020-10-11 19:08:20 +02:00
|
|
|
#include <ardix/string.h>
|
|
|
|
#include <ardix/types.h>
|
|
|
|
|
2021-01-05 13:59:44 +01:00
|
|
|
#include <toolchain.h>
|
|
|
|
|
2020-10-11 19:08:20 +02:00
|
|
|
/*
|
|
|
|
* Stupid memory allocator implementation.
|
|
|
|
*
|
|
|
|
* This design is heavily inspired by (read: stolen from) Doug Lea's Malloc
|
|
|
|
* <http://gee.cs.oswego.edu/dl/html/malloc.html>. It basically works like this:
|
|
|
|
*
|
|
|
|
* Memory is divided into individual blocks of dynamic size. Every block has a header
|
|
|
|
* containing its size w/out overhead; free blocks additionally have a
|
|
|
|
* `struct list_head` after that in order to keep track of where the free blocks are.
|
|
|
|
* This list is ordered by size ascendingly, so we can directly take the first
|
|
|
|
* sufficiently-sized block when iterating over the list in `malloc()`.
|
|
|
|
* Additionally, the effective block size is copied to the very end of the block
|
|
|
|
* (directly after the last usable address) in order to detect two contiguous free
|
|
|
|
* blocks when `free()`ing. How? By (ab)using the LSB of the at-least-4-byte-aligned
|
|
|
|
* size value as a flag for whether the block is currently in use. `free()` can then
|
|
|
|
* just check the size values of the neighboring blocks by doing a simple pointer
|
|
|
|
* calculation, and merge the two blocks into a big one if possible. This minimizes
|
|
|
|
* fragmentation with only slight overhead.
|
|
|
|
*
|
|
|
|
* On 32-bit systems, a free block in memory followed by an allocated one might look
|
2020-10-11 19:35:30 +02:00
|
|
|
* something along the lines of this:
|
2020-10-11 19:08:20 +02:00
|
|
|
*
|
|
|
|
* -----------------------------------------------------------------------------
|
|
|
|
* 0x20010000 | usable size in bytes (238)
|
|
|
|
* 0x20010004 | ptr to next-smaller free block \
|
|
|
|
* 0x20010008 | ptr to next-bigger free block | Usable memory area. If this
|
|
|
|
* : | | was allocated, the returned
|
|
|
|
* : | <unused garbage data> | ptr would be 0x20010004.
|
|
|
|
* : | /
|
|
|
|
* 0x200100EE | usable size in bytes (238)
|
|
|
|
* -----------------------------------------------------------------------------
|
|
|
|
* 0x200100F2 | usable size in bytes (32) + 1 for the "used" bit
|
|
|
|
* : | \
|
|
|
|
* : | <user-defined data> | 32 bytes
|
|
|
|
* : | /
|
|
|
|
* 0x20010112 | usable size in bytes (32 + 1)
|
|
|
|
* -----------------------------------------------------------------------------
|
|
|
|
*
|
2020-10-11 19:35:30 +02:00
|
|
|
* That makes the minimal allocation size `sizeof(struct list_head *)`, because we need to
|
|
|
|
* store those pointers for the linked list when `free()`ing a block.
|
2020-10-11 19:08:20 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Memory block header.
|
|
|
|
* This sits at the beginning of every memory block (duh).
|
|
|
|
*/
|
|
|
|
struct memblk {
|
|
|
|
/**
|
|
|
|
* The block's effectively usable size, i.e. the total block size minus
|
2020-10-12 17:54:07 +02:00
|
|
|
* `2 * MEMBLK_SIZE_LENGTH`.
|
2020-10-11 19:08:20 +02:00
|
|
|
*
|
|
|
|
* This size will also be written to the very end of the block, just after
|
|
|
|
* the last usable address. Additionally, since blocks are always aligned
|
|
|
|
* to at least 4 bytes anyways, we can use the LSB of this size as a flag
|
|
|
|
* for whether the block is currently allocated (1) or not (0). This is
|
|
|
|
* going to make it much easier to detect two free neighboring blocks when
|
|
|
|
* `free()`ing one.
|
|
|
|
*/
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
/** If the block is allocated, this will be overwritten */
|
|
|
|
struct list_head list;
|
|
|
|
|
|
|
|
/* ... void ... */
|
|
|
|
|
|
|
|
/* Here, at the end of this block, would be a copy of `size`. */
|
|
|
|
};
|
|
|
|
|
|
|
|
/** The length of the `size` member in `struct memblk`. */
|
2020-10-14 13:47:42 +02:00
|
|
|
#define MEMBLK_SIZE_LENGTH SIZEOF_MEMBER(struct memblk, size)
|
2020-10-12 17:54:07 +02:00
|
|
|
/** Total overhead per allocated block in bytes (2 * size_t). */
|
|
|
|
#define MEMBLK_OVERHEAD (2 * MEMBLK_SIZE_LENGTH)
|
2020-10-11 19:08:20 +02:00
|
|
|
|
2020-10-12 17:54:07 +02:00
|
|
|
/** Minimum effective allocation size (and all sizes must be a multiple of this one). */
|
2021-02-03 04:04:08 +01:00
|
|
|
#define MIN_BLKSZ (sizeof(struct memblk) - MEMBLK_OVERHEAD)
|
2020-10-12 17:54:07 +02:00
|
|
|
|
|
|
|
/** The list of free blocks, ordered by ascending size. */
|
2020-10-11 19:08:20 +02:00
|
|
|
LIST_HEAD(memblk_free_list);
|
|
|
|
|
|
|
|
static void memblk_set_size(struct memblk *block, size_t size)
|
|
|
|
{
|
|
|
|
block->size = size;
|
|
|
|
void *endptr = block;
|
2020-10-12 17:54:07 +02:00
|
|
|
endptr += MEMBLK_SIZE_LENGTH;
|
2021-02-03 04:04:08 +01:00
|
|
|
endptr += size & ~1u; /* discard the allocated bit */
|
2021-02-28 18:11:00 +01:00
|
|
|
*(size_t *)endptr = size;
|
2020-10-11 19:08:20 +02:00
|
|
|
}
|
|
|
|
|
2020-10-12 15:35:10 +02:00
|
|
|
/**
|
|
|
|
* Split a single free memory block up into two individual blocks such that the block
|
|
|
|
* passed to this function will contain `size` bytes and the newly-created block has
|
|
|
|
* the rest minus overhead. The new block is inserted into the list of free blocks;
|
|
|
|
* however, the original block will *not* be re-sorted.
|
|
|
|
*
|
|
|
|
* @param blk The block to split up.
|
|
|
|
* @param size The new (at least by `MEMBLK_OFFSET + n` bytes smaller) size of the block.
|
|
|
|
* @return The newly created block.
|
|
|
|
*/
|
|
|
|
static struct memblk *memblk_split(struct memblk *blk, size_t size)
|
|
|
|
{
|
|
|
|
struct memblk *cursor;
|
2021-02-03 04:04:08 +01:00
|
|
|
struct memblk *newblk = (void *)blk + MEMBLK_OVERHEAD + (size & ~1u);
|
2020-10-12 15:35:10 +02:00
|
|
|
|
2021-02-03 04:04:08 +01:00
|
|
|
memblk_set_size(newblk, blk->size - MEMBLK_OVERHEAD - (size & ~1u));
|
2020-10-12 15:35:10 +02:00
|
|
|
memblk_set_size(blk, size);
|
|
|
|
|
|
|
|
list_for_each_entry_reverse(&blk->list, cursor, list) {
|
|
|
|
if (cursor->size >= newblk->size || &cursor->list == &memblk_free_list) {
|
|
|
|
list_insert(&cursor->list, &newblk->list);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return newblk;
|
|
|
|
}
|
|
|
|
|
2020-10-11 19:08:20 +02:00
|
|
|
void malloc_init(void *heap, size_t size)
|
|
|
|
{
|
|
|
|
struct memblk *blk = heap;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TODO: This check will prevent accidentally calling the method twice, but should
|
|
|
|
* ideally cause an error of some sort if it fails. Once we have proper error
|
|
|
|
* dispatching/handling routines, we should do that here.
|
|
|
|
*/
|
|
|
|
if (list_is_empty(&memblk_free_list)) {
|
2020-10-12 17:54:07 +02:00
|
|
|
memset(heap, 0, size);
|
2020-10-12 15:35:10 +02:00
|
|
|
memblk_set_size(blk, size - MEMBLK_OVERHEAD);
|
2020-10-11 19:08:20 +02:00
|
|
|
list_insert(&memblk_free_list, &blk->list);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-05 13:59:44 +01:00
|
|
|
__shared __attribute__((malloc))
|
2020-10-11 19:08:20 +02:00
|
|
|
void *malloc(size_t size)
|
|
|
|
{
|
2020-10-12 15:35:10 +02:00
|
|
|
struct memblk *blk;
|
2020-10-11 19:08:20 +02:00
|
|
|
size_t remaining_blksz;
|
|
|
|
|
2020-10-24 17:44:18 +02:00
|
|
|
if (list_is_empty(&memblk_free_list))
|
|
|
|
return NULL;
|
|
|
|
|
2020-10-11 19:08:20 +02:00
|
|
|
if (size == 0)
|
|
|
|
return NULL; /* as per POSIX.1-2008 */
|
|
|
|
|
|
|
|
/* round up to the next multiple of `MIN_BLKSZ` */
|
2021-02-28 18:11:00 +01:00
|
|
|
size = (size / MIN_BLKSZ) * MIN_BLKSZ;
|
2020-10-11 19:08:20 +02:00
|
|
|
size += MIN_BLKSZ;
|
|
|
|
|
2020-12-02 00:00:05 +01:00
|
|
|
atomic_enter();
|
|
|
|
|
2020-10-11 19:08:20 +02:00
|
|
|
list_for_each_entry(&memblk_free_list, blk, list) {
|
|
|
|
/* blocks are sorted by size */
|
|
|
|
if (blk->size >= size)
|
|
|
|
break;
|
|
|
|
}
|
2020-12-02 00:00:05 +01:00
|
|
|
if (blk->size < size) {
|
|
|
|
atomic_leave();
|
2020-10-12 15:35:10 +02:00
|
|
|
return NULL; /* TODO: set errno to ENOMEM once we have it */
|
2020-12-02 00:00:05 +01:00
|
|
|
}
|
2020-10-11 19:08:20 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If we've made it to here, we have found a sufficiently big block,
|
|
|
|
* meaning we can't possibly fail anymore. Since that block is likely
|
|
|
|
* larger than the requested size, we are going to check if it is
|
|
|
|
* possible to create a new, smaller block right at the end of the
|
2020-10-12 15:35:10 +02:00
|
|
|
* allocated area. If it isn't, we just hand out the entire block.
|
2020-10-11 19:08:20 +02:00
|
|
|
*/
|
|
|
|
remaining_blksz = blk->size - size;
|
2020-10-12 15:35:10 +02:00
|
|
|
if (remaining_blksz >= MIN_BLKSZ + MEMBLK_OVERHEAD)
|
|
|
|
memblk_split(blk, size | 0x1u /* allocated bit */);
|
|
|
|
else
|
|
|
|
memblk_set_size(blk, blk->size | 0x1u /* allocated bit */);
|
2020-10-11 19:08:20 +02:00
|
|
|
|
|
|
|
list_delete(&blk->list);
|
|
|
|
|
2020-12-02 00:00:05 +01:00
|
|
|
atomic_leave();
|
|
|
|
|
2020-10-11 19:08:20 +02:00
|
|
|
/* Keep the size field intact */
|
2020-10-12 17:54:07 +02:00
|
|
|
return ((void *)blk) + MEMBLK_SIZE_LENGTH;
|
2020-10-11 19:08:20 +02:00
|
|
|
}
|
|
|
|
|
2021-01-05 13:59:44 +01:00
|
|
|
__shared __attribute__((malloc))
|
2020-10-11 19:08:20 +02:00
|
|
|
void *calloc(size_t nmemb, size_t size)
|
|
|
|
{
|
|
|
|
void *ptr = malloc(nmemb * size);
|
|
|
|
|
|
|
|
if (ptr != NULL)
|
|
|
|
memset(ptr, 0, nmemb * size);
|
|
|
|
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Merge two neighboring free blocks to one big block */
|
|
|
|
static void memblk_merge(struct memblk *lblk, struct memblk *hblk)
|
|
|
|
{
|
2020-10-12 17:54:07 +02:00
|
|
|
size_t *endsz = (void *)hblk + hblk->size + MEMBLK_SIZE_LENGTH;
|
2020-10-11 19:08:20 +02:00
|
|
|
lblk->size = lblk->size + hblk->size + MEMBLK_OVERHEAD;
|
|
|
|
*endsz = lblk->size;
|
|
|
|
}
|
|
|
|
|
2021-01-05 13:59:44 +01:00
|
|
|
__shared
|
2020-10-11 19:08:20 +02:00
|
|
|
void free(void *ptr)
|
|
|
|
{
|
|
|
|
struct memblk *tmp;
|
2020-10-12 17:54:07 +02:00
|
|
|
struct memblk *blk = ptr - MEMBLK_SIZE_LENGTH;
|
2020-10-11 19:08:20 +02:00
|
|
|
size_t *neighsz;
|
|
|
|
|
|
|
|
if (ptr == NULL)
|
|
|
|
return; /* as per POSIX.1-2008 */
|
|
|
|
|
2020-10-12 15:35:10 +02:00
|
|
|
if ((blk->size & 0x1u) == 0)
|
2020-10-12 12:47:20 +02:00
|
|
|
return; /* TODO: Raise exception on double-free */
|
|
|
|
|
2020-12-02 00:00:05 +01:00
|
|
|
atomic_enter();
|
|
|
|
|
2021-02-03 04:04:08 +01:00
|
|
|
memblk_set_size(blk, blk->size & ~1u);
|
2020-10-11 19:08:20 +02:00
|
|
|
|
|
|
|
/* check if our higher/right neighbor is allocated and merge if it is not */
|
|
|
|
neighsz = (void *)blk + MEMBLK_OVERHEAD + blk->size;
|
2020-10-12 15:35:10 +02:00
|
|
|
if ((*neighsz & 0x1u) == 0) {
|
|
|
|
tmp = container_of(neighsz, struct memblk, size);
|
2020-10-11 19:08:20 +02:00
|
|
|
memblk_merge(blk, tmp);
|
|
|
|
list_delete(&tmp->list);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* same thing for the lower/left block */
|
2020-10-12 17:54:07 +02:00
|
|
|
neighsz = (void *)blk - MEMBLK_SIZE_LENGTH;
|
2020-10-12 15:35:10 +02:00
|
|
|
if ((*neighsz & 0x1u) == 0) {
|
2020-10-12 17:54:07 +02:00
|
|
|
tmp = (void *)neighsz - *neighsz - MEMBLK_SIZE_LENGTH;
|
2020-10-11 19:08:20 +02:00
|
|
|
memblk_merge(tmp, blk);
|
|
|
|
list_delete(&tmp->list);
|
|
|
|
blk = tmp; /* discard the higher (now partial) block */
|
|
|
|
}
|
|
|
|
|
|
|
|
list_for_each_entry(&memblk_free_list, tmp, list) {
|
2020-10-12 15:35:10 +02:00
|
|
|
if (tmp->size >= blk->size)
|
2020-10-11 19:08:20 +02:00
|
|
|
break;
|
|
|
|
}
|
2020-10-12 15:35:10 +02:00
|
|
|
list_insert_before(&tmp->list, &blk->list);
|
2020-12-02 00:00:05 +01:00
|
|
|
|
|
|
|
atomic_leave();
|
2020-10-11 19:08:20 +02:00
|
|
|
}
|
2020-10-11 19:35:30 +02:00
|
|
|
|
|
|
|
/*
|
2021-02-28 02:18:39 +01:00
|
|
|
* This file is part of Ardix.
|
|
|
|
* Copyright (c) 2020, 2021 Felix Kopp <owo@fef.moe>.
|
2020-10-11 19:35:30 +02:00
|
|
|
*
|
2021-02-28 02:18:39 +01:00
|
|
|
* Ardix 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.
|
2020-10-11 19:35:30 +02:00
|
|
|
*
|
2021-02-28 02:18:39 +01:00
|
|
|
* Ardix 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.
|
2020-10-11 19:35:30 +02:00
|
|
|
*
|
2021-02-28 02:18:39 +01:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2020-10-11 19:35:30 +02:00
|
|
|
*/
|