Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 1 | /* |
| 2 | * This file is part of the coreboot project. |
| 3 | * |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 4 | * Copyright (C) 2003 Eric Biederman |
| 5 | * Copyright (C) 2005 Steve Magnani |
Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 6 | * Copyright (C) 2008-2009 coresystems GmbH |
| 7 | * |
| 8 | * This program is free software; you can redistribute it and/or modify |
| 9 | * it under the terms of the GNU General Public License as published by |
| 10 | * the Free Software Foundation; version 2 of the License. |
| 11 | * |
| 12 | * This program is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 15 | * GNU General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU General Public License |
| 18 | * along with this program; if not, write to the Free Software |
Paul Menzel | a46a712 | 2013-02-23 18:37:27 +0100 | [diff] [blame] | 19 | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 20 | */ |
Stefan Reinauer | 4f1cb23 | 2006-07-19 15:32:49 +0000 | [diff] [blame] | 21 | |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 22 | #include <console/console.h> |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 23 | #include <cpu/cpu.h> |
| 24 | #include <boot/tables.h> |
Stefan Reinauer | ca374d4 | 2008-01-18 16:16:45 +0000 | [diff] [blame] | 25 | #include <boot/coreboot_tables.h> |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 26 | #include <arch/pirq_routing.h> |
| 27 | #include <arch/smp/mpspec.h> |
Stefan Reinauer | 688b385 | 2004-01-28 16:56:14 +0000 | [diff] [blame] | 28 | #include <arch/acpi.h> |
Stefan Reinauer | 0dff6e3 | 2007-10-23 22:17:45 +0000 | [diff] [blame] | 29 | #include <string.h> |
Robert Millan | 81af3d4 | 2008-11-11 20:20:54 +0000 | [diff] [blame] | 30 | #include <cpu/x86/multiboot.h> |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 31 | #include <cbmem.h> |
Myles Watson | 5817078 | 2009-10-28 16:13:28 +0000 | [diff] [blame] | 32 | #include <lib.h> |
Sven Schnelle | 164bcfd | 2011-08-14 20:56:34 +0200 | [diff] [blame] | 33 | #include <smbios.h> |
Steven J. Magnani | d94e1d6 | 2005-09-12 18:41:30 +0000 | [diff] [blame] | 34 | |
Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 35 | |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 36 | void cbmem_arch_init(void) |
| 37 | { |
| 38 | /* defined in gdt.c */ |
| 39 | move_gdt(); |
| 40 | } |
| 41 | |
Eric Biederman | b78c197 | 2004-10-14 20:54:17 +0000 | [diff] [blame] | 42 | struct lb_memory *write_tables(void) |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 43 | { |
Stefan Reinauer | df77f34 | 2009-04-06 14:00:53 +0000 | [diff] [blame] | 44 | unsigned long low_table_start, low_table_end; |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 45 | unsigned long rom_table_start, rom_table_end; |
| 46 | |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 47 | /* Even if high tables are configured, some tables are copied both to |
| 48 | * the low and the high area, so payloads and OSes don't need to know |
| 49 | * about the high tables. |
Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 50 | */ |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 51 | unsigned long high_table_pointer; |
Stefan Reinauer | 3c7f46b | 2009-02-27 23:09:55 +0000 | [diff] [blame] | 52 | |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 53 | rom_table_start = 0xf0000; |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 54 | rom_table_end = 0xf0000; |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 55 | |
| 56 | /* Start low addr at 0x500, so we don't run into conflicts with the BDA |
Martin Roth | 7b5f8ef | 2013-07-08 16:22:10 -0600 | [diff] [blame] | 57 | * in case our data structures grow beyond 0x400. Only multiboot, GDT |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 58 | * and the coreboot table use low_tables. |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 59 | */ |
| 60 | low_table_start = 0; |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 61 | low_table_end = 0x500; |
| 62 | |
Patrick Georgi | e166782 | 2012-05-05 15:29:32 +0200 | [diff] [blame] | 63 | #if CONFIG_GENERATE_PIRQ_TABLE |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 64 | #define MAX_PIRQ_TABLE_SIZE (4 * 1024) |
| 65 | post_code(0x9a); |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 66 | |
| 67 | /* This table must be between 0x0f0000 and 0x100000 */ |
| 68 | rom_table_end = write_pirq_routing_table(rom_table_end); |
| 69 | rom_table_end = ALIGN(rom_table_end, 1024); |
| 70 | |
| 71 | /* And add a high table version for those payloads that |
| 72 | * want to live in the F segment |
| 73 | */ |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 74 | high_table_pointer = (unsigned long)cbmem_add(CBMEM_ID_PIRQ, MAX_PIRQ_TABLE_SIZE); |
| 75 | if (high_table_pointer) { |
| 76 | unsigned long new_high_table_pointer; |
| 77 | new_high_table_pointer = write_pirq_routing_table(high_table_pointer); |
| 78 | // FIXME make pirq table code intelligent enough to know how |
| 79 | // much space it's going to need. |
| 80 | if (new_high_table_pointer > (high_table_pointer + MAX_PIRQ_TABLE_SIZE)) { |
| 81 | printk(BIOS_ERR, "ERROR: Increase PIRQ size.\n"); |
| 82 | } |
| 83 | printk(BIOS_DEBUG, "PIRQ table: %ld bytes.\n", |
| 84 | new_high_table_pointer - high_table_pointer); |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 85 | } |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 86 | |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 87 | #endif |
| 88 | |
Patrick Georgi | e166782 | 2012-05-05 15:29:32 +0200 | [diff] [blame] | 89 | #if CONFIG_GENERATE_MP_TABLE |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 90 | #define MAX_MP_TABLE_SIZE (4 * 1024) |
| 91 | post_code(0x9b); |
| 92 | |
| 93 | /* The smp table must be in 0-1K, 639K-640K, or 960K-1M */ |
| 94 | rom_table_end = write_smp_table(rom_table_end); |
| 95 | rom_table_end = ALIGN(rom_table_end, 1024); |
| 96 | |
| 97 | high_table_pointer = (unsigned long)cbmem_add(CBMEM_ID_MPTABLE, MAX_MP_TABLE_SIZE); |
| 98 | if (high_table_pointer) { |
| 99 | unsigned long new_high_table_pointer; |
| 100 | new_high_table_pointer = write_smp_table(high_table_pointer); |
| 101 | // FIXME make mp table code intelligent enough to know how |
| 102 | // much space it's going to need. |
| 103 | if (new_high_table_pointer > (high_table_pointer + MAX_MP_TABLE_SIZE)) { |
| 104 | printk(BIOS_ERR, "ERROR: Increase MP table size.\n"); |
| 105 | } |
| 106 | |
| 107 | printk(BIOS_DEBUG, "MP table: %ld bytes.\n", |
| 108 | new_high_table_pointer - high_table_pointer); |
| 109 | } |
| 110 | #endif /* CONFIG_GENERATE_MP_TABLE */ |
| 111 | |
Patrick Georgi | e166782 | 2012-05-05 15:29:32 +0200 | [diff] [blame] | 112 | #if CONFIG_GENERATE_ACPI_TABLES |
Sven Schnelle | 164bcfd | 2011-08-14 20:56:34 +0200 | [diff] [blame] | 113 | #define MAX_ACPI_SIZE (45 * 1024) |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 114 | post_code(0x9c); |
Li-Ta Lo | 8e79fc3 | 2004-04-15 17:33:21 +0000 | [diff] [blame] | 115 | |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 116 | /* Write ACPI tables to F segment and high tables area */ |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 117 | |
| 118 | /* Ok, this is a bit hacky still, because some day we want to have this |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 119 | * completely dynamic. But right now we are setting fixed sizes. |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 120 | * It's probably still better than the old high_table_base code because |
| 121 | * now at least we know when we have an overflow in the area. |
| 122 | * |
| 123 | * We want to use 1MB - 64K for Resume backup. We use 512B for TOC and |
| 124 | * 512 byte for GDT, 4K for PIRQ and 4K for MP table and 8KB for the |
| 125 | * coreboot table. This leaves us with 47KB for all of ACPI. Let's see |
| 126 | * how far we get. |
| 127 | */ |
| 128 | high_table_pointer = (unsigned long)cbmem_add(CBMEM_ID_ACPI, MAX_ACPI_SIZE); |
| 129 | if (high_table_pointer) { |
| 130 | unsigned long acpi_start = high_table_pointer; |
| 131 | unsigned long new_high_table_pointer; |
| 132 | |
Patrick Georgi | bab4f92 | 2009-05-26 19:39:14 +0000 | [diff] [blame] | 133 | rom_table_end = ALIGN(rom_table_end, 16); |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 134 | new_high_table_pointer = write_acpi_tables(high_table_pointer); |
| 135 | if (new_high_table_pointer > ( high_table_pointer + MAX_ACPI_SIZE)) { |
| 136 | printk(BIOS_ERR, "ERROR: Increase ACPI size\n"); |
| 137 | } |
| 138 | printk(BIOS_DEBUG, "ACPI tables: %ld bytes.\n", |
| 139 | new_high_table_pointer - high_table_pointer); |
| 140 | |
| 141 | /* Now we need to create a low table copy of the RSDP. */ |
| 142 | |
| 143 | /* First we look for the high table RSDP */ |
| 144 | while (acpi_start < new_high_table_pointer) { |
Patrick Georgi | bab4f92 | 2009-05-26 19:39:14 +0000 | [diff] [blame] | 145 | if (memcmp(((acpi_rsdp_t *)acpi_start)->signature, RSDP_SIG, 8) == 0) { |
| 146 | break; |
| 147 | } |
| 148 | acpi_start++; |
| 149 | } |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 150 | |
| 151 | /* Now, if we found the RSDP, we take the RSDT and XSDT pointer |
| 152 | * from it in order to write the low RSDP |
| 153 | */ |
| 154 | if (acpi_start < new_high_table_pointer) { |
Stefan Reinauer | b657a3c | 2009-07-21 21:38:33 +0000 | [diff] [blame] | 155 | acpi_rsdp_t *low_rsdp = (acpi_rsdp_t *)rom_table_end, |
| 156 | *high_rsdp = (acpi_rsdp_t *)acpi_start; |
| 157 | |
| 158 | acpi_write_rsdp(low_rsdp, |
| 159 | (acpi_rsdt_t *)(high_rsdp->rsdt_address), |
Stefan Reinauer | de3206a | 2010-02-22 06:09:43 +0000 | [diff] [blame] | 160 | (acpi_xsdt_t *)((unsigned long)high_rsdp->xsdt_address)); |
Patrick Georgi | bab4f92 | 2009-05-26 19:39:14 +0000 | [diff] [blame] | 161 | } else { |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 162 | printk(BIOS_ERR, "ERROR: Didn't find RSDP in high table.\n"); |
Patrick Georgi | bab4f92 | 2009-05-26 19:39:14 +0000 | [diff] [blame] | 163 | } |
Patrick Georgi | bab4f92 | 2009-05-26 19:39:14 +0000 | [diff] [blame] | 164 | rom_table_end = ALIGN(rom_table_end + sizeof(acpi_rsdp_t), 16); |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 165 | } else { |
| 166 | rom_table_end = write_acpi_tables(rom_table_end); |
| 167 | rom_table_end = ALIGN(rom_table_end, 1024); |
Myles Watson | fa12b67 | 2009-04-30 22:45:41 +0000 | [diff] [blame] | 168 | } |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 169 | |
Patrick Georgi | 8f047de | 2009-05-13 14:39:59 +0000 | [diff] [blame] | 170 | #endif |
Sven Schnelle | 164bcfd | 2011-08-14 20:56:34 +0200 | [diff] [blame] | 171 | #define MAX_SMBIOS_SIZE 2048 |
| 172 | #if CONFIG_GENERATE_SMBIOS_TABLES |
| 173 | high_table_pointer = (unsigned long)cbmem_add(CBMEM_ID_SMBIOS, MAX_SMBIOS_SIZE); |
| 174 | if (high_table_pointer) { |
| 175 | unsigned long new_high_table_pointer; |
Steven J. Magnani | d94e1d6 | 2005-09-12 18:41:30 +0000 | [diff] [blame] | 176 | |
Sven Schnelle | 164bcfd | 2011-08-14 20:56:34 +0200 | [diff] [blame] | 177 | new_high_table_pointer = smbios_write_tables(high_table_pointer); |
| 178 | rom_table_end = ALIGN(rom_table_end, 16); |
| 179 | memcpy((void *)rom_table_end, (void *)high_table_pointer, sizeof(struct smbios_entry)); |
| 180 | rom_table_end += sizeof(struct smbios_entry); |
| 181 | |
| 182 | if (new_high_table_pointer > ( high_table_pointer + MAX_SMBIOS_SIZE)) { |
| 183 | printk(BIOS_ERR, "ERROR: Increase SMBIOS size\n"); |
| 184 | } |
| 185 | printk(BIOS_DEBUG, "SMBIOS tables: %ld bytes.\n", |
| 186 | new_high_table_pointer - high_table_pointer); |
| 187 | } else { |
| 188 | unsigned long new_rom_table_end = smbios_write_tables(rom_table_end); |
| 189 | printk(BIOS_DEBUG, "SMBIOS size %ld bytes\n", new_rom_table_end - rom_table_end); |
| 190 | rom_table_end = ALIGN(new_rom_table_end, 16); |
| 191 | } |
| 192 | #endif |
Robert Millan | 81af3d4 | 2008-11-11 20:20:54 +0000 | [diff] [blame] | 193 | |
Aaron Durbin | 5a767fd | 2013-03-23 00:12:19 -0500 | [diff] [blame] | 194 | post_code(0x9e); |
| 195 | |
| 196 | #if CONFIG_HAVE_ACPI_RESUME |
| 197 | /* Only add CBMEM_ID_RESUME when the ramstage isn't relocatable. */ |
| 198 | #if !CONFIG_RELOCATABLE_RAMSTAGE |
| 199 | /* Let's prepare the ACPI S3 Resume area now already, so we can rely on |
| 200 | * it begin there during reboot time. We don't need the pointer, nor |
| 201 | * the result right now. If it fails, ACPI resume will be disabled. |
| 202 | */ |
| 203 | cbmem_add(CBMEM_ID_RESUME, HIGH_MEMORY_SAVE); |
| 204 | #endif |
Siyuan Wang | 3e32cc0 | 2013-07-09 17:16:20 +0800 | [diff] [blame] | 205 | #if CONFIG_NORTHBRIDGE_AMD_AGESA_FAMILY14 || CONFIG_NORTHBRIDGE_AMD_AGESA_FAMILY15_TN || CONFIG_NORTHBRIDGE_AMD_AGESA_FAMILY16_KB |
Aaron Durbin | 5a767fd | 2013-03-23 00:12:19 -0500 | [diff] [blame] | 206 | cbmem_add(CBMEM_ID_RESUME_SCRATCH, CONFIG_HIGH_SCRATCH_MEMORY_SIZE); |
| 207 | #endif |
| 208 | #endif |
| 209 | |
Stefan Reinauer | 294edb2 | 2011-08-14 13:52:03 -0700 | [diff] [blame] | 210 | #define MAX_COREBOOT_TABLE_SIZE (32 * 1024) |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 211 | post_code(0x9d); |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 212 | |
Uwe Hermann | 312673c | 2009-10-27 21:49:33 +0000 | [diff] [blame] | 213 | high_table_pointer = (unsigned long)cbmem_add(CBMEM_ID_CBTABLE, MAX_COREBOOT_TABLE_SIZE); |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 214 | |
| 215 | if (high_table_pointer) { |
| 216 | unsigned long new_high_table_pointer; |
| 217 | |
Kyösti Mälkki | 1ae305e | 2013-09-04 13:05:01 +0300 | [diff] [blame] | 218 | /* FIXME: The high_table_base parameter is not reference when tables are high, |
| 219 | * or high_table_pointer >1 MB. |
| 220 | */ |
| 221 | #if CONFIG_DYNAMIC_CBMEM |
| 222 | u64 fixme_high_tables_base = 0; |
| 223 | #else |
Kyösti Mälkki | e1ea802 | 2013-09-04 14:11:08 +0300 | [diff] [blame] | 224 | u64 fixme_high_tables_base = (u32)get_cbmem_toc(); |
Kyösti Mälkki | 1ae305e | 2013-09-04 13:05:01 +0300 | [diff] [blame] | 225 | #endif |
| 226 | |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 227 | /* Also put a forwarder entry into 0-4K */ |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 228 | new_high_table_pointer = write_coreboot_table(low_table_start, low_table_end, |
Kyösti Mälkki | 1ae305e | 2013-09-04 13:05:01 +0300 | [diff] [blame] | 229 | fixme_high_tables_base, high_table_pointer); |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 230 | |
| 231 | if (new_high_table_pointer > (high_table_pointer + |
| 232 | MAX_COREBOOT_TABLE_SIZE)) |
Stefan Reinauer | c02b4fc | 2010-03-22 11:42:32 +0000 | [diff] [blame] | 233 | printk(BIOS_ERR, "%s: coreboot table didn't fit (%lx)\n", |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 234 | __func__, new_high_table_pointer - |
| 235 | high_table_pointer); |
| 236 | |
| 237 | printk(BIOS_DEBUG, "coreboot table: %ld bytes.\n", |
| 238 | new_high_table_pointer - high_table_pointer); |
Stefan Reinauer | efab4ba | 2009-03-17 14:38:48 +0000 | [diff] [blame] | 239 | } else { |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 240 | /* The coreboot table must be in 0-4K or 960K-1M */ |
Juhana Helovuo | 6b56e43 | 2010-09-13 14:47:22 +0000 | [diff] [blame] | 241 | rom_table_end = write_coreboot_table( |
| 242 | low_table_start, low_table_end, |
| 243 | rom_table_start, rom_table_end); |
Stefan Reinauer | efab4ba | 2009-03-17 14:38:48 +0000 | [diff] [blame] | 244 | } |
Stefan Reinauer | 14e2277 | 2010-04-27 06:56:47 +0000 | [diff] [blame] | 245 | |
Juhana Helovuo | 6b56e43 | 2010-09-13 14:47:22 +0000 | [diff] [blame] | 246 | #if CONFIG_MULTIBOOT |
| 247 | post_code(0x9d); |
| 248 | |
| 249 | /* The Multiboot information structure */ |
| 250 | write_multiboot_info(rom_table_end); |
| 251 | #endif |
| 252 | |
Stefan Reinauer | 3e4e303 | 2013-03-20 14:08:04 -0700 | [diff] [blame] | 253 | /* Print CBMEM sections */ |
Stefan Reinauer | 3b31402 | 2009-10-26 17:04:28 +0000 | [diff] [blame] | 254 | cbmem_list(); |
Stefan Reinauer | f2152ec | 2009-05-26 14:07:44 +0000 | [diff] [blame] | 255 | |
Eric Biederman | 8ca8d76 | 2003-04-22 19:02:15 +0000 | [diff] [blame] | 256 | return get_lb_mem(); |
| 257 | } |