blob: be4a235bf16d0f8f38c9636a524be8244758b679 [file] [log] [blame]
Patrick Georgi11f00792020-03-04 15:10:45 +01001/* SPDX-License-Identifier: GPL-2.0-only */
Aaron Durbin7f8afe02016-03-18 12:21:23 -05002
Kyösti Mälkkia963acd2019-08-16 20:34:25 +03003#include <arch/romstage.h>
Aaron Durbin7f8afe02016-03-18 12:21:23 -05004#include <cbmem.h>
5#include <console/console.h>
6#include <cpu/x86/msr.h>
7#include <cpu/x86/mtrr.h>
Subrata Banik8edc6dc2019-08-22 11:30:52 +05308#include <cpu/x86/smm.h>
Aaron Durbin7f8afe02016-03-18 12:21:23 -05009#include <program_loading.h>
Kyösti Mälkki4f14cd82019-12-18 19:40:48 +020010#include <reset.h>
Aaron Durbin7f8afe02016-03-18 12:21:23 -050011#include <rmodule.h>
Elyes Haouasae1ca822022-10-07 10:02:38 +020012#include <security/vboot/vboot_common.h>
Aaron Durbind0084132016-11-29 15:52:08 -060013#include <stage_cache.h>
Subrata Banik2847e1e2019-02-25 20:31:22 +053014#include <timestamp.h>
Elyes Haouasae1ca822022-10-07 10:02:38 +020015#include <types.h>
Aaron Durbin7f8afe02016-03-18 12:21:23 -050016
Arthur Heymans46b409d2021-05-14 13:19:43 +020017static size_t var_mtrr_ctx_size(void)
Aaron Durbin7f8afe02016-03-18 12:21:23 -050018{
Arthur Heymans46b409d2021-05-14 13:19:43 +020019 int mtrr_count = get_var_mtrr_count();
20 return sizeof(struct var_mtrr_context) + mtrr_count * 2 * sizeof(msr_t);
Aaron Durbin7f8afe02016-03-18 12:21:23 -050021}
22
Arthur Heymansba00d102022-02-15 11:04:15 +010023static int postcar_frame_init(struct postcar_frame *pcf)
Kyösti Mälkki029cebc2016-12-02 19:47:07 +020024{
Arthur Heymans876a1b42022-02-15 11:06:10 +010025 memset(pcf, 0, sizeof(*pcf));
26
Arthur Heymans46b409d2021-05-14 13:19:43 +020027 struct var_mtrr_context *ctx;
Kyösti Mälkki029cebc2016-12-02 19:47:07 +020028
Arthur Heymans46b409d2021-05-14 13:19:43 +020029 ctx = cbmem_add(CBMEM_ID_ROMSTAGE_RAM_STACK, var_mtrr_ctx_size());
30 if (ctx == NULL) {
31 printk(BIOS_ERR, "Couldn't add var_mtrr_ctx setup in cbmem.\n");
Aaron Durbin7f8afe02016-03-18 12:21:23 -050032 return -1;
33 }
34
Arthur Heymans46b409d2021-05-14 13:19:43 +020035 pcf->mtrr = ctx;
36 var_mtrr_context_init(pcf->mtrr);
37
Aaron Durbin7f8afe02016-03-18 12:21:23 -050038 return 0;
39}
40
41void postcar_frame_add_mtrr(struct postcar_frame *pcf,
42 uintptr_t addr, size_t size, int type)
43{
Arthur Heymans46b409d2021-05-14 13:19:43 +020044 var_mtrr_set(pcf->mtrr, addr, size, type);
Aaron Durbin7f8afe02016-03-18 12:21:23 -050045}
46
Nico Huber36ec3e92018-05-27 14:32:27 +020047void postcar_frame_add_romcache(struct postcar_frame *pcf, int type)
48{
Julius Wernercd49cce2019-03-05 16:53:33 -080049 if (!CONFIG(BOOT_DEVICE_MEMORY_MAPPED))
Nico Huber36ec3e92018-05-27 14:32:27 +020050 return;
51 postcar_frame_add_mtrr(pcf, CACHE_ROM_BASE, CACHE_ROM_SIZE, type);
52}
53
Aaron Durbine8936742020-04-29 14:20:05 -060054static void postcar_frame_common_mtrrs(struct postcar_frame *pcf)
Kyösti Mälkki544878b2019-08-09 11:41:15 +030055{
56 if (pcf->skip_common_mtrr)
57 return;
58
Kyösti Mälkki544878b2019-08-09 11:41:15 +030059 /* Cache the ROM as WP just below 4GiB. */
60 postcar_frame_add_romcache(pcf, MTRR_TYPE_WRPROT);
61}
62
Arthur Heymansba00d102022-02-15 11:04:15 +010063static void run_postcar_phase(struct postcar_frame *pcf);
64
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030065/* prepare_and_run_postcar() determines the stack to use after
66 * cache-as-ram is torn down as well as the MTRR settings to use. */
Arthur Heymansefd27202022-07-13 07:20:47 +020067void __noreturn prepare_and_run_postcar(void)
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030068{
Arthur Heymans876a1b42022-02-15 11:06:10 +010069 struct postcar_frame pcf;
70
71 if (postcar_frame_init(&pcf))
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030072 die("Unable to initialize postcar frame.\n");
73
Arthur Heymans876a1b42022-02-15 11:06:10 +010074 fill_postcar_frame(&pcf);
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030075
Arthur Heymans876a1b42022-02-15 11:06:10 +010076 postcar_frame_common_mtrrs(&pcf);
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030077
Arthur Heymans876a1b42022-02-15 11:06:10 +010078 run_postcar_phase(&pcf);
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030079 /* We do not return here. */
Arthur Heymansefd27202022-07-13 07:20:47 +020080 die("Failed to load postcar\n!");
Kyösti Mälkkicd7a70f2019-08-17 20:51:08 +030081}
82
Arthur Heymans46b409d2021-05-14 13:19:43 +020083static void finalize_load(uintptr_t *reloc_params, uintptr_t mtrr_frame_ptr)
Rizwan Qureshib1b44d32016-08-26 21:08:50 +053084{
Arthur Heymans46b409d2021-05-14 13:19:43 +020085 *reloc_params = mtrr_frame_ptr;
Kyösti Mälkkia7421fb2017-09-05 22:43:05 +030086 /*
87 * Signal to rest of system that another update was made to the
88 * postcar program prior to running it.
89 */
Arthur Heymans46b409d2021-05-14 13:19:43 +020090 prog_segment_loaded((uintptr_t)reloc_params, sizeof(uintptr_t), SEG_FINAL);
91 prog_segment_loaded((uintptr_t)mtrr_frame_ptr, var_mtrr_ctx_size(), SEG_FINAL);
Kyösti Mälkkia7421fb2017-09-05 22:43:05 +030092}
93
Aaron Durbind0084132016-11-29 15:52:08 -060094static void load_postcar_cbfs(struct prog *prog, struct postcar_frame *pcf)
Aaron Durbin7f8afe02016-03-18 12:21:23 -050095{
Aaron Durbin7f8afe02016-03-18 12:21:23 -050096 struct rmod_stage_load rsl = {
97 .cbmem_id = CBMEM_ID_AFTER_CAR,
Aaron Durbind0084132016-11-29 15:52:08 -060098 .prog = prog,
Aaron Durbin7f8afe02016-03-18 12:21:23 -050099 };
100
Wim Vervoorn1058dd82019-11-01 10:22:22 +0100101 vboot_run_logic();
102
Aaron Durbin7f8afe02016-03-18 12:21:23 -0500103 if (rmodule_stage_load(&rsl))
Keith Short70064582019-05-06 16:12:57 -0600104 die_with_post_code(POST_INVALID_ROM,
105 "Failed to load after CAR program.\n");
Aaron Durbin7f8afe02016-03-18 12:21:23 -0500106
107 /* Set the stack pointer within parameters of the program loaded. */
108 if (rsl.params == NULL)
Keith Short70064582019-05-06 16:12:57 -0600109 die_with_post_code(POST_INVALID_ROM,
110 "No parameters found in after CAR program.\n");
Aaron Durbin7f8afe02016-03-18 12:21:23 -0500111
Arthur Heymans46b409d2021-05-14 13:19:43 +0200112 finalize_load(rsl.params, (uintptr_t)pcf->mtrr);
Aaron Durbindd95e002016-03-31 13:36:33 -0500113
Subrata Banik90f750b2019-06-11 17:52:06 +0530114 stage_cache_add(STAGE_POSTCAR, prog);
Aaron Durbind0084132016-11-29 15:52:08 -0600115}
116
Subrata Banik8edc6dc2019-08-22 11:30:52 +0530117/*
118 * Cache the TSEG region at the top of ram. This region is
119 * not restricted to SMM mode until SMM has been relocated.
120 * By setting the region to cacheable it provides faster access
121 * when relocating the SMM handler as well as using the TSEG
122 * region for other purposes.
123 */
124void postcar_enable_tseg_cache(struct postcar_frame *pcf)
125{
126 uintptr_t smm_base;
127 size_t smm_size;
128
129 smm_region(&smm_base, &smm_size);
130 postcar_frame_add_mtrr(pcf, smm_base, smm_size,
131 MTRR_TYPE_WRBACK);
132}
133
Kyösti Mälkki4f14cd82019-12-18 19:40:48 +0200134static void postcar_cache_invalid(void)
135{
136 printk(BIOS_ERR, "postcar cache invalid.\n");
137 board_reset();
138}
139
Arthur Heymansba00d102022-02-15 11:04:15 +0100140static void run_postcar_phase(struct postcar_frame *pcf)
Aaron Durbind0084132016-11-29 15:52:08 -0600141{
142 struct prog prog =
Philipp Deppenwiese01797b12018-11-08 10:39:39 +0100143 PROG_INIT(PROG_POSTCAR, CONFIG_CBFS_PREFIX "/postcar");
Aaron Durbind0084132016-11-29 15:52:08 -0600144
Kyösti Mälkkie0165fb2021-01-09 13:30:57 +0200145 if (resume_from_stage_cache()) {
Aaron Durbind0084132016-11-29 15:52:08 -0600146 stage_cache_load_stage(STAGE_POSTCAR, &prog);
Aaron Durbinc546c762018-04-23 14:55:09 -0600147 /* This is here to allow platforms to pass different stack
148 parameters between S3 resume and normal boot. On the
149 platforms where the values are the same it's a nop. */
Arthur Heymans46b409d2021-05-14 13:19:43 +0200150 finalize_load(prog.arg, (uintptr_t)pcf->mtrr);
Kyösti Mälkki4f14cd82019-12-18 19:40:48 +0200151
152 if (prog_entry(&prog) == NULL)
153 postcar_cache_invalid();
Kyösti Mälkkia7421fb2017-09-05 22:43:05 +0300154 } else
Aaron Durbind0084132016-11-29 15:52:08 -0600155 load_postcar_cbfs(&prog, pcf);
156
Subrata Banik2847e1e2019-02-25 20:31:22 +0530157 /* As postcar exist, it's end of romstage here */
Jakub Czapigaad6157e2022-02-15 11:50:31 +0100158 timestamp_add_now(TS_ROMSTAGE_END);
Subrata Banik2847e1e2019-02-25 20:31:22 +0530159
Kyösti Mälkki45ddb432019-11-02 14:12:18 +0200160 console_time_report();
161
Arthur Heymans5331a7c2019-10-23 17:07:15 +0200162 prog_set_arg(&prog, cbmem_top());
163
Aaron Durbin7f8afe02016-03-18 12:21:23 -0500164 prog_run(&prog);
165}