| // SPDX-License-Identifier: GPL-2.0+ |
| /* |
| * Copyright (C) 2015, Bin Meng <bmeng.cn@gmail.com> |
| */ |
| |
| #define LOG_CATEGORY LOGC_ARCH |
| |
| #include <efi_loader.h> |
| #include <lmb.h> |
| #include <log.h> |
| #include <asm/e820.h> |
| #include <asm/global_data.h> |
| |
| DECLARE_GLOBAL_DATA_PTR; |
| |
| static const char *const e820_type_name[E820_COUNT] = { |
| [E820_RAM] = "RAM", |
| [E820_RESERVED] = "Reserved", |
| [E820_ACPI] = "ACPI", |
| [E820_NVS] = "ACPI NVS", |
| [E820_UNUSABLE] = "Unusable", |
| }; |
| |
| void e820_dump(struct e820_entry *entries, uint count) |
| { |
| int i; |
| |
| printf("%12s %10s %s\n", "Addr", "Size", "Type"); |
| for (i = 0; i < count; i++) { |
| struct e820_entry *entry = &entries[i]; |
| |
| printf("%12llx %10llx %s\n", entry->addr, entry->size, |
| entry->type < E820_COUNT ? |
| e820_type_name[entry->type] : |
| simple_itoa(entry->type)); |
| } |
| } |
| |
| /* |
| * Install a default e820 table with 4 entries as follows: |
| * |
| * 0x000000-0x0a0000 Useable RAM |
| * 0x0a0000-0x100000 Reserved for ISA |
| * 0x100000-gd->ram_size Useable RAM |
| * CONFIG_PCIE_ECAM_BASE PCIe ECAM |
| */ |
| __weak unsigned int install_e820_map(unsigned int max_entries, |
| struct e820_entry *entries) |
| { |
| entries[0].addr = 0; |
| entries[0].size = ISA_START_ADDRESS; |
| entries[0].type = E820_RAM; |
| entries[1].addr = ISA_START_ADDRESS; |
| entries[1].size = ISA_END_ADDRESS - ISA_START_ADDRESS; |
| entries[1].type = E820_RESERVED; |
| entries[2].addr = ISA_END_ADDRESS; |
| entries[2].size = gd->ram_size - ISA_END_ADDRESS; |
| entries[2].type = E820_RAM; |
| entries[3].addr = CONFIG_PCIE_ECAM_BASE; |
| entries[3].size = CONFIG_PCIE_ECAM_SIZE; |
| entries[3].type = E820_RESERVED; |
| |
| return 4; |
| } |
| |
| void e820_init(struct e820_ctx *ctx, struct e820_entry *entries, |
| int max_entries) |
| { |
| memset(ctx, '\0', sizeof(*ctx)); |
| ctx->entries = entries; |
| ctx->max_entries = max_entries; |
| } |
| |
| void e820_add(struct e820_ctx *ctx, enum e820_type type, u64 addr, u64 size) |
| { |
| struct e820_entry *entry = &ctx->entries[ctx->count++]; |
| |
| if (ctx->count <= ctx->max_entries) { |
| entry->addr = addr; |
| entry->size = size; |
| entry->type = type; |
| } |
| ctx->addr = addr + size; |
| } |
| |
| void e820_next(struct e820_ctx *ctx, enum e820_type type, u64 size) |
| { |
| e820_add(ctx, type, ctx->addr, size); |
| } |
| |
| void e820_to_addr(struct e820_ctx *ctx, enum e820_type type, u64 addr) |
| { |
| e820_next(ctx, type, addr - ctx->addr); |
| } |
| |
| int e820_finish(struct e820_ctx *ctx) |
| { |
| if (ctx->count > ctx->max_entries) { |
| printf("e820 has %d entries but room for only %d\n", ctx->count, |
| ctx->max_entries); |
| panic("e820 table too large"); |
| } |
| log_debug("e820 map installed, n=%d\n", ctx->count); |
| if (_DEBUG) |
| e820_dump(ctx->entries, ctx->count); |
| |
| return ctx->count; |
| } |
| |
| #if CONFIG_IS_ENABLED(EFI_LOADER) |
| void efi_add_known_memory(void) |
| { |
| struct e820_entry e820[E820MAX]; |
| unsigned int i, num; |
| u64 start; |
| int type; |
| |
| num = install_e820_map(ARRAY_SIZE(e820), e820); |
| |
| for (i = 0; i < num; ++i) { |
| start = e820[i].addr; |
| |
| switch (e820[i].type) { |
| case E820_RAM: |
| type = EFI_CONVENTIONAL_MEMORY; |
| break; |
| case E820_RESERVED: |
| type = EFI_RESERVED_MEMORY_TYPE; |
| break; |
| case E820_ACPI: |
| type = EFI_ACPI_RECLAIM_MEMORY; |
| break; |
| case E820_NVS: |
| type = EFI_ACPI_MEMORY_NVS; |
| break; |
| case E820_UNUSABLE: |
| default: |
| type = EFI_UNUSABLE_MEMORY; |
| break; |
| } |
| |
| if (type != EFI_CONVENTIONAL_MEMORY) |
| efi_add_memory_map(start, e820[i].size, type); |
| } |
| } |
| #endif /* CONFIG_IS_ENABLED(EFI_LOADER) */ |
| |
| #if CONFIG_IS_ENABLED(LMB_ARCH_MEM_MAP) |
| void lmb_arch_add_memory(void) |
| { |
| struct e820_entry e820[E820MAX]; |
| unsigned int i, num; |
| u64 ram_top; |
| |
| num = install_e820_map(ARRAY_SIZE(e820), e820); |
| |
| ram_top = (u64)gd->ram_top & ~EFI_PAGE_MASK; |
| if (!ram_top) |
| ram_top = 0x100000000ULL; |
| |
| for (i = 0; i < num; ++i) { |
| if (e820[i].type == E820_RAM) { |
| u64 start, size, rgn_top; |
| |
| start = e820[i].addr; |
| size = e820[i].size; |
| rgn_top = start + size; |
| |
| if (start > ram_top) |
| continue; |
| |
| if (rgn_top > ram_top) |
| size -= rgn_top - ram_top; |
| |
| lmb_add(start, size); |
| } |
| } |
| } |
| #endif /* CONFIG_IS_ENABLED(LMB_ARCH_MEM_MAP) */ |