blob: b3a632648176f0ebec932f4aea9af0a11df23f5e [file] [log] [blame] [edit]
// Copyright 2016 The Fuchsia Authors
// Copyright (c) 2014 Travis Geiselbrecht
//
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file or at
// https://opensource.org/licenses/MIT
#include "vm/bootalloc.h"
#include <align.h>
#include <lib/instrumentation/asan.h>
#include <stdint.h>
#include <stdlib.h>
#include <sys/types.h>
#include <trace.h>
#include <vm/physmap.h>
#include <vm/pmm.h>
#include <vm/vm.h>
#include "vm_priv.h"
#define LOCAL_TRACE VM_GLOBAL_TRACE(0)
// Simple boot time allocator that starts by allocating physical memory off
// the end of wherever the kernel is loaded in physical space.
//
// Pointers are returned from the kernel's physmap
// store the start and current pointer to the boot allocator in physical address
paddr_t boot_alloc_start;
paddr_t boot_alloc_end;
// run in physical space without the mmu set up, so by computing the address of _end
// and saving it, we've effectively computed the physical address of the end of the
// kernel.
// We can't allow asan to check the globals here as it happens on a different
// aspace where asan shadow isn't mapped.
NO_ASAN __NO_SAFESTACK void boot_alloc_init() {
boot_alloc_start = reinterpret_cast<paddr_t>(_end);
// TODO(fxbug.dev/32414): This is a compile-time no-op that defeats any compiler
// optimizations based on its knowledge/assumption that `&_end` is a
// constant here that equals the `&_end` constant as computed elsewhere.
// Without this, the compiler can see that boot_alloc_start is never set to
// any other value and replace code that uses the boot_alloc_start value
// with code that computes `&_end` on the spot. What the compiler doesn't
// know is that this `&_end` is crucially a PC-relative computation when
// the PC is a (low) physical address. Later code that uses
// boot_alloc_start will be running at a kernel (high) virtual address and
// so its `&_end` will be nowhere near the same value. The compiler isn't
// wrong to do this sort of optimization when it can and other such cases
// will eventually arise. So long-term we'll need more thorough
// compile-time separation of the early boot code that runs in physical
// space from normal kernel code. For now, this asm generates no
// additional code but tells the compiler that it has no idea what value
// boot_alloc_start might take, so it has to compute the `&_end` value now.
__asm__("" : "=g"(boot_alloc_start) : "0"(boot_alloc_start));
boot_alloc_end = reinterpret_cast<paddr_t>(_end);
}
void boot_alloc_reserve(paddr_t start, size_t len) {
uintptr_t end = ALIGN((start + len), PAGE_SIZE);
if (end >= boot_alloc_start) {
if ((start > boot_alloc_start) && ((start - boot_alloc_start) > (128 * 1024 * 1024))) {
// if we've got 128MB of space, that's good enough
// it's possible that the start may be *way* far up
// (gigabytes) and there may not be space after it...
return;
}
boot_alloc_start = boot_alloc_end = end;
}
}
void* boot_alloc_mem(size_t len) {
uintptr_t ptr;
ptr = ALIGN(boot_alloc_end, 8);
boot_alloc_end = (ptr + ALIGN(len, 8));
LTRACEF("len %zu, phys ptr %#" PRIxPTR " ptr %p\n", len, ptr, paddr_to_physmap(ptr));
return paddr_to_physmap(ptr);
}
// called from arch start.S
// run in physical space without the mmu set up, so stick to basic, relocatable code
__NO_SAFESTACK
paddr_t boot_alloc_page_phys() {
paddr_t ptr = ALIGN(boot_alloc_end, PAGE_SIZE);
boot_alloc_end = ptr + PAGE_SIZE;
return ptr;
}