Angel Pons | 8a3453f | 2020-04-02 23:48:19 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 2 | |
Duncan Laurie | 59be624 | 2016-03-07 13:21:56 -0800 | [diff] [blame] | 3 | #include <bootmode.h> |
Furquan Shaikh | 76cedd2 | 2020-05-02 10:24:23 -0700 | [diff] [blame] | 4 | #include <acpi/acpi.h> |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 5 | #include <console/console.h> |
Lee Leahy | 94b856e | 2015-10-15 12:07:03 -0700 | [diff] [blame] | 6 | #include <fsp/ramstage.h> |
Aaron Durbin | 789f2b6 | 2015-09-09 17:05:06 -0500 | [diff] [blame] | 7 | #include <fsp/util.h> |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 8 | #include <framebuffer_info.h> |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 9 | #include <lib.h> |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 10 | #include <stage_cache.h> |
Aaron Durbin | 39bdb0b | 2015-08-04 23:59:43 -0500 | [diff] [blame] | 11 | #include <string.h> |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 12 | #include <timestamp.h> |
Frans Hendriks | 50b999f | 2019-11-08 13:55:45 +0100 | [diff] [blame] | 13 | #include <cbmem.h> |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 14 | |
| 15 | /* SOC initialization after FSP silicon init */ |
Aaron Durbin | 6403167 | 2018-04-21 14:45:32 -0600 | [diff] [blame] | 16 | __weak void soc_after_silicon_init(void) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 17 | { |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 18 | } |
| 19 | |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 20 | static void display_hob_info(FSP_INFO_HEADER *fsp_info_header) |
| 21 | { |
| 22 | const EFI_GUID graphics_info_guid = EFI_PEI_GRAPHICS_INFO_HOB_GUID; |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 23 | void *hob_list_ptr = get_hob_list(); |
| 24 | |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 25 | /* Verify the HOBs */ |
| 26 | if (hob_list_ptr == NULL) { |
Frans Hendriks | 509f469 | 2019-06-28 14:11:41 +0200 | [diff] [blame] | 27 | printk(BIOS_ERR, "ERROR - HOB pointer is NULL!\n"); |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 28 | return; |
| 29 | } |
| 30 | |
Frans Hendriks | 509f469 | 2019-06-28 14:11:41 +0200 | [diff] [blame] | 31 | if (CONFIG(DISPLAY_HOBS)) |
| 32 | print_hob_type_structure(0, hob_list_ptr); |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 33 | |
| 34 | /* |
| 35 | * Verify that FSP is generating the required HOBs: |
| 36 | * 7.1: FSP_BOOTLOADER_TEMP_MEMORY_HOB only produced for FSP 1.0 |
| 37 | * 7.2: FSP_RESERVED_MEMORY_RESOURCE_HOB verified by raminit |
| 38 | * 7.3: FSP_NON_VOLATILE_STORAGE_HOB verified by raminit |
| 39 | * 7.4: FSP_BOOTLOADER_TOLUM_HOB verified by raminit |
| 40 | * 7.5: EFI_PEI_GRAPHICS_INFO_HOB verified below, |
| 41 | * if the ImageAttribute bit is set |
| 42 | * FSP_SMBIOS_MEMORY_INFO HOB verified by raminit |
| 43 | */ |
| 44 | if ((fsp_info_header->ImageAttribute & GRAPHICS_SUPPORT_BIT) && |
Frans Hendriks | 509f469 | 2019-06-28 14:11:41 +0200 | [diff] [blame] | 45 | !get_next_guid_hob(&graphics_info_guid, hob_list_ptr) && |
| 46 | CONFIG(DISPLAY_HOBS)) { |
| 47 | printk(BIOS_ERR, "7.5: EFI_PEI_GRAPHICS_INFO_HOB missing!\n"); |
| 48 | printk(BIOS_ERR, |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 49 | "ERROR - Missing one or more required FSP HOBs!\n"); |
Frans Hendriks | 509f469 | 2019-06-28 14:11:41 +0200 | [diff] [blame] | 50 | } |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 51 | } |
| 52 | |
Lee Leahy | cff5f09 | 2016-02-08 08:37:53 -0800 | [diff] [blame] | 53 | void fsp_run_silicon_init(FSP_INFO_HEADER *fsp_info_header, int is_s3_wakeup) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 54 | { |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 55 | FSP_SILICON_INIT fsp_silicon_init; |
| 56 | SILICON_INIT_UPD *original_params; |
| 57 | SILICON_INIT_UPD silicon_init_params; |
| 58 | EFI_STATUS status; |
| 59 | UPD_DATA_REGION *upd_ptr; |
| 60 | VPD_DATA_REGION *vpd_ptr; |
Wim Vervoorn | 67117c3 | 2019-12-16 14:21:09 +0100 | [diff] [blame] | 61 | const struct cbmem_entry *logo_entry = NULL; |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 62 | |
Lee Leahy | cff5f09 | 2016-02-08 08:37:53 -0800 | [diff] [blame] | 63 | /* Display the FSP header */ |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 64 | if (fsp_info_header == NULL) { |
| 65 | printk(BIOS_ERR, "FSP_INFO_HEADER not set!\n"); |
| 66 | return; |
| 67 | } |
| 68 | print_fsp_info(fsp_info_header); |
| 69 | |
| 70 | /* Initialize the UPD values */ |
| 71 | vpd_ptr = (VPD_DATA_REGION *)(fsp_info_header->CfgRegionOffset + |
| 72 | fsp_info_header->ImageBase); |
Julius Werner | 540a980 | 2019-12-09 13:03:29 -0800 | [diff] [blame] | 73 | printk(BIOS_DEBUG, "%p: VPD Data\n", vpd_ptr); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 74 | upd_ptr = (UPD_DATA_REGION *)(vpd_ptr->PcdUpdRegionOffset + |
| 75 | fsp_info_header->ImageBase); |
Julius Werner | 540a980 | 2019-12-09 13:03:29 -0800 | [diff] [blame] | 76 | printk(BIOS_DEBUG, "%p: UPD Data\n", upd_ptr); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 77 | original_params = (void *)((u8 *)upd_ptr + |
| 78 | upd_ptr->SiliconInitUpdOffset); |
| 79 | memcpy(&silicon_init_params, original_params, |
| 80 | sizeof(silicon_init_params)); |
| 81 | soc_silicon_init_params(&silicon_init_params); |
| 82 | |
| 83 | /* Locate VBT and pass to FSP GOP */ |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 84 | if (CONFIG(RUN_FSP_GOP)) |
Kyösti Mälkki | 0a54685 | 2021-01-24 18:06:26 +0200 | [diff] [blame] | 85 | load_vbt(&silicon_init_params); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 86 | mainboard_silicon_init_params(&silicon_init_params); |
| 87 | |
Wim Vervoorn | 67117c3 | 2019-12-16 14:21:09 +0100 | [diff] [blame] | 88 | if (CONFIG(FSP1_1_DISPLAY_LOGO) && !is_s3_wakeup) |
| 89 | logo_entry = soc_load_logo(&silicon_init_params); |
Frans Hendriks | 50b999f | 2019-11-08 13:55:45 +0100 | [diff] [blame] | 90 | |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 91 | /* Display the UPD data */ |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 92 | if (CONFIG(DISPLAY_UPD_DATA)) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 93 | soc_display_silicon_init_params(original_params, |
| 94 | &silicon_init_params); |
| 95 | |
| 96 | /* Perform silicon initialization after RAM is configured */ |
| 97 | printk(BIOS_DEBUG, "Calling FspSiliconInit\n"); |
| 98 | fsp_silicon_init = (FSP_SILICON_INIT)(fsp_info_header->ImageBase |
| 99 | + fsp_info_header->FspSiliconInitEntryOffset); |
| 100 | timestamp_add_now(TS_FSP_SILICON_INIT_START); |
Julius Werner | 540a980 | 2019-12-09 13:03:29 -0800 | [diff] [blame] | 101 | printk(BIOS_DEBUG, "Calling FspSiliconInit(%p) at %p\n", |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 102 | &silicon_init_params, fsp_silicon_init); |
Duncan Laurie | fb50983 | 2015-11-22 14:53:57 -0800 | [diff] [blame] | 103 | post_code(POST_FSP_SILICON_INIT); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 104 | status = fsp_silicon_init(&silicon_init_params); |
| 105 | timestamp_add_now(TS_FSP_SILICON_INIT_END); |
| 106 | printk(BIOS_DEBUG, "FspSiliconInit returned 0x%08x\n", status); |
| 107 | |
Frans Hendriks | 50b999f | 2019-11-08 13:55:45 +0100 | [diff] [blame] | 108 | /* The logo_entry can be freed up now as it is not required any longer */ |
Wim Vervoorn | 67117c3 | 2019-12-16 14:21:09 +0100 | [diff] [blame] | 109 | if (logo_entry && !is_s3_wakeup) |
Frans Hendriks | 50b999f | 2019-11-08 13:55:45 +0100 | [diff] [blame] | 110 | cbmem_entry_remove(logo_entry); |
| 111 | |
Duncan Laurie | 59be624 | 2016-03-07 13:21:56 -0800 | [diff] [blame] | 112 | /* Mark graphics init done after SiliconInit if VBT was provided */ |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 113 | #if CONFIG(RUN_FSP_GOP) |
Duncan Laurie | 59be624 | 2016-03-07 13:21:56 -0800 | [diff] [blame] | 114 | /* GraphicsConfigPtr doesn't exist in Quark X1000's FSP, so this needs |
Elyes HAOUAS | 2e4d806 | 2016-08-25 20:50:50 +0200 | [diff] [blame] | 115 | * to be #if'd out instead of using if (). */ |
Duncan Laurie | 59be624 | 2016-03-07 13:21:56 -0800 | [diff] [blame] | 116 | if (silicon_init_params.GraphicsConfigPtr) |
| 117 | gfx_set_init_done(1); |
| 118 | #endif |
| 119 | |
Patrick Rudolph | 92106b1 | 2020-02-19 12:54:06 +0100 | [diff] [blame] | 120 | if (CONFIG(RUN_FSP_GOP)) { |
| 121 | const EFI_GUID vbt_guid = EFI_PEI_GRAPHICS_INFO_HOB_GUID; |
| 122 | u32 *vbt_hob; |
| 123 | |
| 124 | void *hob_list_ptr = get_hob_list(); |
| 125 | vbt_hob = get_next_guid_hob(&vbt_guid, hob_list_ptr); |
| 126 | if (vbt_hob == NULL) { |
| 127 | printk(BIOS_ERR, "FSP_ERR: Graphics Data HOB is not present\n"); |
| 128 | } else { |
| 129 | EFI_PEI_GRAPHICS_INFO_HOB *gop; |
| 130 | |
| 131 | printk(BIOS_DEBUG, "FSP_DEBUG: Graphics Data HOB present\n"); |
| 132 | gop = GET_GUID_HOB_DATA(vbt_hob); |
| 133 | |
| 134 | fb_add_framebuffer_info(gop->FrameBufferBase, |
| 135 | gop->GraphicsMode.HorizontalResolution, |
| 136 | gop->GraphicsMode.VerticalResolution, |
| 137 | gop->GraphicsMode.PixelsPerScanLine * 4, |
| 138 | 32); |
| 139 | } |
| 140 | } |
| 141 | |
Alexandru Gagniuc | 41c003c | 2015-08-28 19:07:35 -0400 | [diff] [blame] | 142 | display_hob_info(fsp_info_header); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 143 | soc_after_silicon_init(); |
| 144 | } |
| 145 | |
Aaron Durbin | abf87a2 | 2015-08-05 12:26:56 -0500 | [diff] [blame] | 146 | static int fsp_find_and_relocate(struct prog *fsp) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 147 | { |
Aaron Durbin | 5d6f0f9 | 2015-10-08 15:06:28 -0500 | [diff] [blame] | 148 | if (prog_locate(fsp)) { |
| 149 | printk(BIOS_ERR, "ERROR: Couldn't find %s\n", prog_name(fsp)); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 150 | return -1; |
| 151 | } |
| 152 | |
Aaron Durbin | 5d6f0f9 | 2015-10-08 15:06:28 -0500 | [diff] [blame] | 153 | if (fsp_relocate(fsp, prog_rdev(fsp))) { |
Aaron Durbin | 22ea007 | 2015-08-05 10:17:33 -0500 | [diff] [blame] | 154 | printk(BIOS_ERR, "ERROR: FSP relocation failed.\n"); |
| 155 | return -1; |
| 156 | } |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 157 | |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 158 | return 0; |
| 159 | } |
| 160 | |
Kyösti Mälkki | d0bc92d | 2021-01-10 00:23:58 +0200 | [diff] [blame] | 161 | static void fsp_load(void) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 162 | { |
Aaron Durbin | 7e7a4df | 2015-12-08 14:34:35 -0600 | [diff] [blame] | 163 | struct prog fsp = PROG_INIT(PROG_REFCODE, "fsp.bin"); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 164 | |
Kyösti Mälkki | e0165fb | 2021-01-09 13:30:57 +0200 | [diff] [blame] | 165 | if (resume_from_stage_cache()) { |
Aaron Durbin | abf87a2 | 2015-08-05 12:26:56 -0500 | [diff] [blame] | 166 | stage_cache_load_stage(STAGE_REFCODE, &fsp); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 167 | } else { |
Aaron Durbin | abf87a2 | 2015-08-05 12:26:56 -0500 | [diff] [blame] | 168 | fsp_find_and_relocate(&fsp); |
Kyösti Mälkki | e0165fb | 2021-01-09 13:30:57 +0200 | [diff] [blame] | 169 | |
| 170 | if (prog_entry(&fsp)) |
| 171 | stage_cache_add(STAGE_REFCODE, &fsp); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 172 | } |
| 173 | |
Aaron Durbin | abf87a2 | 2015-08-05 12:26:56 -0500 | [diff] [blame] | 174 | /* FSP_INFO_HEADER is set as the program entry. */ |
| 175 | fsp_update_fih(prog_entry(&fsp)); |
Furquan Shaikh | f4b20af | 2017-02-20 13:33:32 -0800 | [diff] [blame] | 176 | } |
| 177 | |
| 178 | void intel_silicon_init(void) |
| 179 | { |
| 180 | fsp_load(); |
| 181 | fsp_run_silicon_init(fsp_get_fih(), acpi_is_wakeup_s3()); |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 182 | } |
| 183 | |
| 184 | /* Initialize the UPD parameters for SiliconInit */ |
Aaron Durbin | 6403167 | 2018-04-21 14:45:32 -0600 | [diff] [blame] | 185 | __weak void mainboard_silicon_init_params( |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 186 | SILICON_INIT_UPD *params) |
| 187 | { |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 188 | }; |
| 189 | |
| 190 | /* Display the UPD parameters for SiliconInit */ |
Aaron Durbin | 6403167 | 2018-04-21 14:45:32 -0600 | [diff] [blame] | 191 | __weak void soc_display_silicon_init_params( |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 192 | const SILICON_INIT_UPD *old, SILICON_INIT_UPD *new) |
| 193 | { |
| 194 | printk(BIOS_SPEW, "UPD values for SiliconInit:\n"); |
| 195 | hexdump32(BIOS_SPEW, new, sizeof(*new)); |
| 196 | } |
| 197 | |
| 198 | /* Initialize the UPD parameters for SiliconInit */ |
Aaron Durbin | 6403167 | 2018-04-21 14:45:32 -0600 | [diff] [blame] | 199 | __weak void soc_silicon_init_params(SILICON_INIT_UPD *params) |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 200 | { |
Lee Leahy | 0946ec3 | 2015-04-20 15:24:54 -0700 | [diff] [blame] | 201 | } |