Patrick Georgi | 11f0079 | 2020-03-04 15:10:45 +0100 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 2 | |
Furquan Shaikh | 76cedd2 | 2020-05-02 10:24:23 -0700 | [diff] [blame] | 3 | #include <acpi/acpi.h> |
Arthur Heymans | 0a635ab | 2022-05-12 20:39:16 +0200 | [diff] [blame] | 4 | #include <bootstate.h> |
Ryan Salsamendi | f511c1f | 2017-06-16 13:15:31 -0700 | [diff] [blame] | 5 | #include <commonlib/endian.h> |
Elyes Haouas | ae1ca82 | 2022-10-07 10:02:38 +0200 | [diff] [blame] | 6 | #include <types.h> |
Arthur Heymans | 0a635ab | 2022-05-12 20:39:16 +0200 | [diff] [blame] | 7 | |
| 8 | #define X86_BDA_SIZE 0x200 |
| 9 | #define X86_BDA_BASE ((void *)0x400) |
| 10 | #define X86_EBDA_SEGMENT ((void *)0x40e) |
| 11 | #define X86_EBDA_LOWMEM ((void *)0x413) |
| 12 | |
| 13 | #define DEFAULT_EBDA_LOWMEM (1024 << 10) |
| 14 | #define DEFAULT_EBDA_SEGMENT 0xF600 |
| 15 | #define DEFAULT_EBDA_SIZE 0x400 |
| 16 | |
Subrata Banik | 19dbffd | 2017-09-10 20:06:22 +0530 | [diff] [blame] | 17 | |
| 18 | static void *get_ebda_start(void) |
| 19 | { |
| 20 | return (void *)((uintptr_t)DEFAULT_EBDA_SEGMENT << 4); |
| 21 | } |
| 22 | |
Subrata Banik | 19dbffd | 2017-09-10 20:06:22 +0530 | [diff] [blame] | 23 | /* |
| 24 | * EBDA area is representing a 1KB memory area just below |
| 25 | * the top of conventional memory (below 1MB) |
| 26 | */ |
| 27 | |
Arthur Heymans | 8b7cd43 | 2019-10-26 20:31:41 +0200 | [diff] [blame] | 28 | static void setup_ebda(u32 low_memory_size, u16 ebda_segment, u16 ebda_size) |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 29 | { |
Ryan Salsamendi | f0b0712 | 2017-06-09 12:01:39 -0700 | [diff] [blame] | 30 | u16 low_memory_kb; |
| 31 | u16 ebda_kb; |
| 32 | void *ebda; |
| 33 | |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 34 | if (!low_memory_size || !ebda_segment || !ebda_size) |
| 35 | return; |
| 36 | |
Ryan Salsamendi | f0b0712 | 2017-06-09 12:01:39 -0700 | [diff] [blame] | 37 | low_memory_kb = low_memory_size >> 10; |
| 38 | ebda_kb = ebda_size >> 10; |
Subrata Banik | 19dbffd | 2017-09-10 20:06:22 +0530 | [diff] [blame] | 39 | ebda = get_ebda_start(); |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 40 | |
Ryan Salsamendi | f0b0712 | 2017-06-09 12:01:39 -0700 | [diff] [blame] | 41 | /* clear BIOS DATA AREA */ |
Ryan Salsamendi | f511c1f | 2017-06-16 13:15:31 -0700 | [diff] [blame] | 42 | zero_n(X86_BDA_BASE, X86_BDA_SIZE); |
Ryan Salsamendi | f0b0712 | 2017-06-09 12:01:39 -0700 | [diff] [blame] | 43 | |
| 44 | /* Avoid unaligned write16() since it's undefined behavior */ |
Ryan Salsamendi | f511c1f | 2017-06-16 13:15:31 -0700 | [diff] [blame] | 45 | write_le16(X86_EBDA_LOWMEM, low_memory_kb); |
| 46 | write_le16(X86_EBDA_SEGMENT, ebda_segment); |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 47 | |
| 48 | /* Set up EBDA */ |
Ryan Salsamendi | f511c1f | 2017-06-16 13:15:31 -0700 | [diff] [blame] | 49 | zero_n(ebda, ebda_size); |
| 50 | write_le16(ebda, ebda_kb); |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 51 | } |
| 52 | |
Arthur Heymans | 0a635ab | 2022-05-12 20:39:16 +0200 | [diff] [blame] | 53 | static void setup_default_ebda(void *unused) |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 54 | { |
Arthur Heymans | 804adaa | 2019-01-09 18:30:33 +0100 | [diff] [blame] | 55 | if (acpi_is_wakeup_s3()) |
| 56 | return; |
| 57 | |
Duncan Laurie | b4aaaa7 | 2012-01-17 09:03:11 -0800 | [diff] [blame] | 58 | setup_ebda(DEFAULT_EBDA_LOWMEM, |
| 59 | DEFAULT_EBDA_SEGMENT, |
| 60 | DEFAULT_EBDA_SIZE); |
| 61 | } |
Arthur Heymans | 0a635ab | 2022-05-12 20:39:16 +0200 | [diff] [blame] | 62 | |
| 63 | /* Ensure EBDA is prepared before Option ROMs. */ |
| 64 | BOOT_STATE_INIT_ENTRY(BS_DEV_INIT, BS_ON_ENTRY, setup_default_ebda, NULL); |