blob: 6b7b96fa575780d422d3e65f2e0eed3a3ef4f823 [file] [log] [blame]
// Copyright 2017 The Fuchsia Authors
//
// 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
#ifndef ZIRCON_KERNEL_INCLUDE_MEXEC_H_
#define ZIRCON_KERNEL_INCLUDE_MEXEC_H_
#define MEMMOV_OPS_DST_OFFSET (0)
#define MEMMOV_OPS_SRC_OFFSET (8)
#define MEMMOV_OPS_LEN_OFFSET (16)
#define MEMMOV_OPS_STRUCT_LEN (24)
#define MAX_OPS_PER_PAGE (169) // Calculated below
#ifndef __ASSEMBLER__
#include <lib/zx/status.h>
#include <stddef.h>
#include <stdint.h>
#include <zircon/compiler.h>
#include <zircon/types.h>
#include <fbl/array.h>
#include <ktl/byte.h>
#include <ktl/span.h>
#include <vm/vm_object.h>
// Warning: The geometry of this struct is depended upon by the mexec assembly
// function. Do not modify without also updating mexec.S.
typedef struct __PACKED {
void* dst;
void* src;
size_t len;
} memmov_ops_t;
static_assert(sizeof(memmov_ops_t) == MEMMOV_OPS_STRUCT_LEN,
"sizeof memmov_ops_t must match MEMMOV_OPS_STRUCT_LEN");
#define MAX_OPS_PER_PAGE_DEF ((PAGE_SIZE / MEMMOV_OPS_STRUCT_LEN) - 1)
static_assert(MAX_OPS_PER_PAGE_DEF == MAX_OPS_PER_PAGE,
"Calculated max ops per page must match literal value");
// Implemented in assembly. Copies the new kernel into place and branches to it.
typedef void (*mexec_asm_func)(uint64_t arg0, uint64_t arg1, uint64_t arg2, uint64_t aux,
memmov_ops_t* ops, void* new_kernel_addr);
// Writes an mexec data ZBI into the provided buffer and returns the size of
// that ZBI if successful.
zx::status<size_t> WriteMexecData(ktl::span<ktl::byte> buffer);
/* This function is called at the beginning of mexec. Interrupts are not yet
* disabled, but only one CPU is running.
*/
void platform_mexec_prep(uintptr_t final_bootimage_addr, size_t final_bootimage_len);
/* Ask the platform to mexec into the next kernel.
* This function is called after platform_mexec_prep(), with interrupts disabled.
*/
void platform_mexec(mexec_asm_func mexec_assembly, memmov_ops_t* ops, uintptr_t new_bootimage_addr,
size_t new_bootimage_len, uintptr_t entry64_addr);
/* Allocate |count| pages where no page has a physical address less than
* |lower_bound|.
* Results are returned via the array pointed to by |paddrs| with the
* assumption there is enough storage to contain |count| results.
* |limit| defines the highest address to search before giving up.
*/
zx_status_t alloc_pages_greater_than(paddr_t lower_bound, size_t count, size_t limit,
paddr_t* paddrs);
static_assert(__offsetof(memmov_ops_t, dst) == MEMMOV_OPS_DST_OFFSET, "");
static_assert(__offsetof(memmov_ops_t, src) == MEMMOV_OPS_SRC_OFFSET, "");
static_assert(__offsetof(memmov_ops_t, len) == MEMMOV_OPS_LEN_OFFSET, "");
#endif // __ASSEMBLER__
#endif // ZIRCON_KERNEL_INCLUDE_MEXEC_H_