94 lines
3.0 KiB
C
94 lines
3.0 KiB
C
/*
|
|
* Copyright (C) 2020 Intel Corporation.
|
|
*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
#include <types.h>
|
|
#include <errno.h>
|
|
#include <asm/pgtable.h>
|
|
#include <boot.h>
|
|
#include "multiboot_priv.h"
|
|
|
|
/**
|
|
* @pre abi != NULL
|
|
*/
|
|
int32_t multiboot_to_acrn_bi(struct acrn_boot_info *abi, void *mb_info) {
|
|
uint32_t i;
|
|
struct multiboot_info *mbi = (struct multiboot_info *)(hpa2hva_early((uint64_t)mb_info));
|
|
struct multiboot_mmap *mmap = (struct multiboot_mmap *)hpa2hva_early((uint64_t)mbi->mi_mmap_addr);
|
|
struct multiboot_module *mods = (struct multiboot_module *)hpa2hva_early((uint64_t)mbi->mi_mods_addr);
|
|
|
|
(void)strncpy_s((void *)(abi->cmdline), MAX_BOOTARGS_SIZE, (char *)hpa2hva_early((uint64_t)mbi->mi_cmdline),
|
|
strnlen_s((char *)hpa2hva_early((uint64_t)mbi->mi_cmdline), (MAX_BOOTARGS_SIZE - 1U)));
|
|
|
|
(void)strncpy_s((void *)(abi->loader_name), MAX_LOADER_NAME_SIZE,
|
|
(char *)hpa2hva_early((uint64_t)mbi->mi_loader_name),
|
|
strnlen_s((char *)hpa2hva_early((uint64_t)mbi->mi_loader_name), (MAX_LOADER_NAME_SIZE - 1U)));
|
|
|
|
abi->mmap_entries = mbi->mi_mmap_length / sizeof(struct multiboot_mmap);
|
|
|
|
if (((mbi->mi_flags & MULTIBOOT_INFO_HAS_MMAP) != 0U) && (abi->mmap_entries != 0U) && (mmap != NULL)) {
|
|
|
|
if (abi->mmap_entries > MAX_MMAP_ENTRIES) {
|
|
abi->mmap_entries = MAX_MMAP_ENTRIES;
|
|
}
|
|
|
|
for (i = 0U; i < abi->mmap_entries; i++) {
|
|
abi->mmap_entry[i].baseaddr = (mmap + i)->baseaddr;
|
|
abi->mmap_entry[i].length = (mmap + i)->length;
|
|
abi->mmap_entry[i].type = (mmap + i)->type;
|
|
}
|
|
} else {
|
|
abi->mmap_entries = 0U;
|
|
}
|
|
|
|
abi->mods_count = mbi->mi_mods_count;
|
|
if (((mbi->mi_flags & MULTIBOOT_INFO_HAS_MODS) != 0U) && (mbi->mi_mods_count != 0U) && (mods != NULL)) {
|
|
if (abi->mods_count > MAX_MODULE_NUM) {
|
|
abi->mods_count = MAX_MODULE_NUM;
|
|
}
|
|
|
|
for (i = 0U; i < abi->mods_count; i++) {
|
|
abi->mods[i].start = hpa2hva_early((uint64_t)(mods + i)->mm_mod_start);
|
|
if ((mods + i)->mm_mod_end > (mods + i)->mm_mod_start) {
|
|
abi->mods[i].size = (mods + i)->mm_mod_end - (mods + i)->mm_mod_start;
|
|
}
|
|
(void)strncpy_s((void *)(abi->mods[i].string), MAX_MOD_STRING_SIZE,
|
|
(char *)hpa2hva_early((uint64_t)(mods + i)->mm_string),
|
|
strnlen_s((char *)hpa2hva_early((uint64_t)(mods + i)->mm_string), MAX_BOOTARGS_SIZE));
|
|
}
|
|
} else {
|
|
abi->mods_count = 0U;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int32_t init_multiboot_info(uint32_t *registers)
|
|
{
|
|
int32_t ret = -ENODEV;
|
|
uint32_t magic = registers[0];
|
|
uint32_t info = registers[1];
|
|
struct acrn_boot_info *abi = get_acrn_boot_info();
|
|
|
|
if (boot_from_multiboot(magic, info)) {
|
|
if (multiboot_to_acrn_bi(abi, hpa2hva_early((uint64_t)info)) == 0) {
|
|
strncpy_s(abi->protocol_name, MAX_PROTOCOL_NAME_SIZE,
|
|
"Multiboot", (MAX_PROTOCOL_NAME_SIZE - 1U));
|
|
ret = 0;
|
|
}
|
|
#ifdef CONFIG_MULTIBOOT2
|
|
} else if (boot_from_multiboot2(magic)) {
|
|
if (multiboot2_to_acrn_bi(abi, hpa2hva_early((uint64_t)info)) == 0) {
|
|
strncpy_s(abi->protocol_name, MAX_PROTOCOL_NAME_SIZE,
|
|
"Multiboot2", (MAX_PROTOCOL_NAME_SIZE - 1U));
|
|
ret = 0;
|
|
}
|
|
#endif
|
|
} else {
|
|
/* Currently there are only multiboot and multiboot2 */
|
|
}
|
|
return ret;
|
|
}
|