blob: 0bccef930e5a9a0b1589546e88224990e0422a91 [file] [log] [blame]
/*
* Copyright 2017, Data61, CSIRO (ABN 41 687 119 230)
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <allocman/mspace/virtual_pool.h>
#include <allocman/allocman.h>
#include <allocman/util.h>
#include <stdlib.h>
#include <sel4/sel4.h>
#include <sel4utils/mapping.h>
#include <vka/kobject_t.h>
#include <vspace/mapping.h>
#include <string.h>
/* This allocator deliberately does not use the vspace library to manage mappings to prevent
* circular dependencies between the vspace library and the allocator */
static int _add_page(allocman_t *alloc, seL4_CPtr pd, void *vaddr)
{
cspacepath_t frame_path;
seL4_Word frame_cookie;
int error;
error = allocman_cspace_alloc(alloc, &frame_path);
if (error) {
ZF_LOGV("Failed to allocate slot");
return error;
}
frame_cookie = allocman_utspace_alloc(alloc, seL4_PageBits, seL4_ARCH_4KPage, &frame_path, true, &error);
if (error) {
allocman_cspace_free(alloc, &frame_path);
ZF_LOGV("Failed to allocate frame");
return error;
}
while ((error = seL4_ARCH_Page_Map(frame_path.capPtr, pd, (seL4_Word) vaddr, seL4_AllRights,
seL4_ARCH_Default_VMAttributes)) == seL4_FailedLookup) {
cspacepath_t path;
error = allocman_cspace_alloc(alloc, &path);
if (error) {
ZF_LOGV("Failed to allocate slot");
break;
}
seL4_Word failed_bits = seL4_MappingFailedLookupLevel();
vspace_map_obj_t obj;
error = vspace_get_map_obj(failed_bits, &obj);
assert(error == 0);
seL4_Word cookie = allocman_utspace_alloc(alloc, obj.size_bits, obj.type, &path, false, &error);
if (error) {
allocman_cspace_free(alloc, &path);
ZF_LOGV("Failed to allocate paging structure");
break;
}
error = vspace_map_obj(&obj, path.capPtr, pd, (seL4_Word) vaddr, seL4_ARCH_Default_VMAttributes);
if (error != seL4_NoError) {
allocman_utspace_free(alloc, cookie, seL4_PageTableBits);
allocman_cspace_free(alloc, &path);
break;
}
}
if (error != seL4_NoError) {
allocman_cspace_free(alloc, &frame_path);
allocman_utspace_free(alloc, frame_cookie, seL4_PageBits);
return error;
}
/* zero the memory in case we were allocated from a device range */
memset(vaddr, 0, PAGE_SIZE_4K);
return 0;
}
static k_r_malloc_header_t *_morecore(size_t cookie, mspace_k_r_malloc_t *k_r_malloc, size_t new_units)
{
size_t new_size;
k_r_malloc_header_t *new_header;
mspace_virtual_pool_t *virtual_pool = (mspace_virtual_pool_t*)cookie;
new_size = new_units * sizeof(k_r_malloc_header_t);
if (virtual_pool->pool_ptr + new_size > virtual_pool->pool_limit) {
ZF_LOGV("morecore out of virtual pool");
return NULL;
}
while (virtual_pool->pool_ptr + new_size > virtual_pool->pool_top) {
int error;
error = _add_page(virtual_pool->morecore_alloc, virtual_pool->pd, virtual_pool->pool_top);
if (error) {
ZF_LOGV("morecore failed to add page");
return NULL;
}
virtual_pool->pool_top += PAGE_SIZE_4K;
}
new_header = (k_r_malloc_header_t*)virtual_pool->pool_ptr;
virtual_pool->pool_ptr += new_size;
return new_header;
}
void mspace_virtual_pool_create(mspace_virtual_pool_t *virtual_pool, struct mspace_virtual_pool_config config)
{
virtual_pool->pool_ptr = config.vstart;
virtual_pool->pool_top = virtual_pool->pool_ptr;
virtual_pool->pool_limit = config.vstart + config.size;
virtual_pool->morecore_alloc = NULL;
virtual_pool->pd = config.pd;
mspace_k_r_malloc_init(&virtual_pool->k_r_malloc, (size_t)virtual_pool, _morecore);
}
void *_mspace_virtual_pool_alloc(struct allocman *alloc, void *_virtual_pool, size_t bytes, int *error)
{
void *ret;
mspace_virtual_pool_t *virtual_pool = (mspace_virtual_pool_t*)_virtual_pool;
virtual_pool->morecore_alloc = alloc;
ret = mspace_k_r_malloc_alloc(&virtual_pool->k_r_malloc, bytes);
virtual_pool->morecore_alloc = NULL;
SET_ERROR(error, (ret == NULL) ? 1 : 0);
return ret;
}
void _mspace_virtual_pool_free(struct allocman *alloc, void *_virtual_pool, void *ptr, size_t bytes)
{
mspace_virtual_pool_t *virtual_pool = (mspace_virtual_pool_t*)_virtual_pool;
virtual_pool->morecore_alloc = alloc;
mspace_k_r_malloc_free(&virtual_pool->k_r_malloc, ptr);
virtual_pool->morecore_alloc = NULL;
}