|  | // 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; | 
|  | } |