Patrick Georgi | ac95903 | 2020-05-05 22:49:26 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-or-later */ |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 2 | |
Arthur Heymans | fdf6d12 | 2022-05-17 13:07:30 +0200 | [diff] [blame] | 3 | #include <arch/null_breakpoint.h> |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 4 | #include <bootstate.h> |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 5 | #include <console/console.h> |
Nico Huber | d67edca | 2018-11-13 19:28:07 +0100 | [diff] [blame] | 6 | #include <cpu/x86/mtrr.h> |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 7 | #include <fsp/util.h> |
Patrick Rudolph | 40beb36 | 2020-12-01 10:08:38 +0100 | [diff] [blame] | 8 | #include <mode_switch.h> |
Angel Pons | 654930e | 2022-01-01 17:08:00 +0100 | [diff] [blame] | 9 | #include <timestamp.h> |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 10 | #include <types.h> |
| 11 | |
| 12 | struct fsp_notify_phase_data { |
| 13 | enum fsp_notify_phase notify_phase; |
Subrata Banik | e8feab0 | 2021-12-27 10:25:55 +0000 | [diff] [blame] | 14 | bool skip; |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 15 | uint8_t post_code_before; |
| 16 | uint8_t post_code_after; |
| 17 | enum timestamp_id timestamp_before; |
| 18 | enum timestamp_id timestamp_after; |
| 19 | }; |
| 20 | |
| 21 | static const struct fsp_notify_phase_data notify_data[] = { |
| 22 | { |
| 23 | .notify_phase = AFTER_PCI_ENUM, |
Subrata Banik | 34f26b2 | 2022-02-10 12:38:02 +0530 | [diff] [blame] | 24 | .skip = !CONFIG(USE_FSP_NOTIFY_PHASE_POST_PCI_ENUM), |
lilacious | 40cb3fe | 2023-06-21 23:24:14 +0200 | [diff] [blame^] | 25 | .post_code_before = POSTCODE_FSP_NOTIFY_BEFORE_ENUMERATE, |
| 26 | .post_code_after = POSTCODE_FSP_NOTIFY_AFTER_ENUMERATE, |
Jakub Czapiga | ad6157e | 2022-02-15 11:50:31 +0100 | [diff] [blame] | 27 | .timestamp_before = TS_FSP_ENUMERATE_START, |
| 28 | .timestamp_after = TS_FSP_ENUMERATE_END, |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 29 | }, |
| 30 | { |
| 31 | .notify_phase = READY_TO_BOOT, |
Subrata Banik | 34f26b2 | 2022-02-10 12:38:02 +0530 | [diff] [blame] | 32 | .skip = !CONFIG(USE_FSP_NOTIFY_PHASE_READY_TO_BOOT), |
lilacious | 40cb3fe | 2023-06-21 23:24:14 +0200 | [diff] [blame^] | 33 | .post_code_before = POSTCODE_FSP_NOTIFY_BEFORE_FINALIZE, |
| 34 | .post_code_after = POSTCODE_FSP_NOTIFY_AFTER_FINALIZE, |
Jakub Czapiga | ad6157e | 2022-02-15 11:50:31 +0100 | [diff] [blame] | 35 | .timestamp_before = TS_FSP_FINALIZE_START, |
| 36 | .timestamp_after = TS_FSP_FINALIZE_END, |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 37 | }, |
| 38 | { |
| 39 | .notify_phase = END_OF_FIRMWARE, |
Subrata Banik | 34f26b2 | 2022-02-10 12:38:02 +0530 | [diff] [blame] | 40 | .skip = !CONFIG(USE_FSP_NOTIFY_PHASE_END_OF_FIRMWARE), |
lilacious | 40cb3fe | 2023-06-21 23:24:14 +0200 | [diff] [blame^] | 41 | .post_code_before = POSTCODE_FSP_NOTIFY_BEFORE_END_OF_FIRMWARE, |
| 42 | .post_code_after = POSTCODE_FSP_NOTIFY_AFTER_END_OF_FIRMWARE, |
Jakub Czapiga | ad6157e | 2022-02-15 11:50:31 +0100 | [diff] [blame] | 43 | .timestamp_before = TS_FSP_END_OF_FIRMWARE_START, |
| 44 | .timestamp_after = TS_FSP_END_OF_FIRMWARE_END, |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 45 | }, |
| 46 | }; |
| 47 | |
| 48 | static const struct fsp_notify_phase_data *get_notify_phase_data(enum fsp_notify_phase phase) |
| 49 | { |
| 50 | for (size_t i = 0; i < ARRAY_SIZE(notify_data); i++) { |
| 51 | if (notify_data[i].notify_phase == phase) |
| 52 | return ¬ify_data[i]; |
| 53 | } |
| 54 | die("Unknown FSP notify phase %u\n", phase); |
| 55 | } |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 56 | |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 57 | static void fsp_notify(enum fsp_notify_phase phase) |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 58 | { |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 59 | const struct fsp_notify_phase_data *data = get_notify_phase_data(phase); |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 60 | struct fsp_notify_params notify_params = { .phase = phase }; |
Angel Pons | 654930e | 2022-01-01 17:08:00 +0100 | [diff] [blame] | 61 | fsp_notify_fn fspnotify; |
| 62 | uint32_t ret; |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 63 | |
Subrata Banik | e8feab0 | 2021-12-27 10:25:55 +0000 | [diff] [blame] | 64 | if (data->skip) { |
| 65 | printk(BIOS_INFO, "coreboot skipped calling FSP notify phase: %08x.\n", phase); |
| 66 | return; |
| 67 | } |
| 68 | |
Lee Leahy | 9671faa | 2016-07-24 18:18:52 -0700 | [diff] [blame] | 69 | if (!fsps_hdr.notify_phase_entry_offset) |
| 70 | die("Notify_phase_entry_offset is zero!\n"); |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 71 | |
Angel Pons | 654930e | 2022-01-01 17:08:00 +0100 | [diff] [blame] | 72 | fspnotify = (void *)(uintptr_t)(fsps_hdr.image_base + |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 73 | fsps_hdr.notify_phase_entry_offset); |
Lee Leahy | 672df16 | 2016-07-24 18:21:13 -0700 | [diff] [blame] | 74 | fsp_before_debug_notify(fspnotify, ¬ify_params); |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 75 | |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 76 | timestamp_add_now(data->timestamp_before); |
| 77 | post_code(data->post_code_before); |
Alexandru Gagniuc | 010225c | 2016-05-06 08:22:45 -0700 | [diff] [blame] | 78 | |
Arthur Heymans | fdf6d12 | 2022-05-17 13:07:30 +0200 | [diff] [blame] | 79 | /* FSP disables the interrupt handler so remove debug exceptions temporarily */ |
| 80 | null_breakpoint_disable(); |
Patrick Rudolph | 31218a4 | 2020-11-30 15:50:06 +0100 | [diff] [blame] | 81 | if (ENV_X86_64 && CONFIG(PLATFORM_USES_FSP2_X86_32)) |
Patrick Rudolph | 40beb36 | 2020-12-01 10:08:38 +0100 | [diff] [blame] | 82 | ret = protected_mode_call_1arg(fspnotify, (uintptr_t)¬ify_params); |
| 83 | else |
| 84 | ret = fspnotify(¬ify_params); |
Arthur Heymans | fdf6d12 | 2022-05-17 13:07:30 +0200 | [diff] [blame] | 85 | null_breakpoint_init(); |
Alexandru Gagniuc | 010225c | 2016-05-06 08:22:45 -0700 | [diff] [blame] | 86 | |
Angel Pons | 355d844 | 2022-01-01 17:03:10 +0100 | [diff] [blame] | 87 | timestamp_add_now(data->timestamp_after); |
| 88 | post_code(data->post_code_after); |
| 89 | |
Lee Leahy | 672df16 | 2016-07-24 18:21:13 -0700 | [diff] [blame] | 90 | fsp_debug_after_notify(ret); |
Alexandru Gagniuc | 010225c | 2016-05-06 08:22:45 -0700 | [diff] [blame] | 91 | |
Lee Leahy | 9671faa | 2016-07-24 18:18:52 -0700 | [diff] [blame] | 92 | /* Handle any errors returned by FspNotify */ |
| 93 | fsp_handle_reset(ret); |
Angel Pons | 2b1f8d4 | 2022-01-01 17:20:00 +0100 | [diff] [blame] | 94 | if (ret != FSP_SUCCESS) |
| 95 | die("FspNotify returned with error 0x%08x!\n", ret); |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 96 | |
| 97 | /* Allow the platform to run something after FspNotify */ |
| 98 | platform_fsp_notify_status(phase); |
| 99 | } |
| 100 | |
| 101 | static void fsp_notify_dummy(void *arg) |
| 102 | { |
Patrick Rudolph | 90fda02 | 2020-11-30 13:44:17 +0100 | [diff] [blame] | 103 | enum fsp_notify_phase phase = (uint32_t)(uintptr_t)arg; |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 104 | |
Nico Huber | d67edca | 2018-11-13 19:28:07 +0100 | [diff] [blame] | 105 | display_mtrrs(); |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 106 | |
| 107 | fsp_notify(phase); |
| 108 | if (phase == READY_TO_BOOT) |
| 109 | fsp_notify(END_OF_FIRMWARE); |
| 110 | } |
| 111 | |
Angel Pons | 654930e | 2022-01-01 17:08:00 +0100 | [diff] [blame] | 112 | BOOT_STATE_INIT_ENTRY(BS_DEV_ENABLE, BS_ON_ENTRY, fsp_notify_dummy, (void *)AFTER_PCI_ENUM); |
| 113 | BOOT_STATE_INIT_ENTRY(BS_PAYLOAD_LOAD, BS_ON_EXIT, fsp_notify_dummy, (void *)READY_TO_BOOT); |
| 114 | BOOT_STATE_INIT_ENTRY(BS_OS_RESUME, BS_ON_ENTRY, fsp_notify_dummy, (void *)READY_TO_BOOT); |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 115 | |
Angel Pons | 654930e | 2022-01-01 17:08:00 +0100 | [diff] [blame] | 116 | __weak void platform_fsp_notify_status(enum fsp_notify_phase phase) |
Lee Leahy | 806fa24 | 2016-08-01 13:55:02 -0700 | [diff] [blame] | 117 | { |
Andrey Petrov | 9de55cc | 2016-02-25 14:19:07 -0800 | [diff] [blame] | 118 | } |