blob: 562829801ed92fefce8510944a3e6df476c54718 [file] [log] [blame]
Stefan Reinauer3b314022009-10-26 17:04:28 +00001/*
2 * This file is part of the coreboot project.
3 *
4 * Copyright (C) 2009 coresystems GmbH
Aaron Durbindf3a1092013-03-13 12:41:44 -05005 * Copyright (C) 2013 Google, Inc.
Stefan Reinauer3b314022009-10-26 17:04:28 +00006 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; version 2 of the License.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Stefan Reinauer3b314022009-10-26 17:04:28 +000015 */
16
17#ifndef _CBMEM_H_
18#define _CBMEM_H_
19
Aaron Durbindc9f5cd2015-09-08 13:34:43 -050020#include <commonlib/cbmem_id.h>
Kyösti Mälkki4fbac462015-01-07 04:48:43 +020021#include <rules.h>
Aaron Durbin0f333072014-01-30 17:19:46 -060022#include <stddef.h>
Aaron Durbindf3a1092013-03-13 12:41:44 -050023#include <stdint.h>
Aaron Durbin1ca2d862015-09-30 12:26:54 -050024#include <boot/coreboot_tables.h>
Aaron Durbindf3a1092013-03-13 12:41:44 -050025
Kyösti Mälkkid72cc412016-06-15 06:15:46 +030026#define CBMEM_FSP_HOB_PTR 0x614
27
Aaron Durbindf3a1092013-03-13 12:41:44 -050028struct cbmem_entry;
29
Aaron Durbindf3a1092013-03-13 12:41:44 -050030/*
31 * The dynamic cbmem infrastructure allows for growing cbmem dynamically as
32 * things are added. It requires an external function, cbmem_top(), to be
33 * implemented by the board or chipset to define the upper address where
34 * cbmem lives. This address is required to be a 32-bit address. Additionally,
35 * the address needs to be consistent in both romstage and ramstage. The
Martin Roth0cb07e32013-07-09 21:46:01 -060036 * dynamic cbmem infrastructure allocates new regions below the last allocated
Aaron Durbindf3a1092013-03-13 12:41:44 -050037 * region. Regions are defined by a cbmem_entry struct that is opaque. Regions
38 * may be removed, but the last one added is the only that can be removed.
Aaron Durbindf3a1092013-03-13 12:41:44 -050039 */
40
41#define DYN_CBMEM_ALIGN_SIZE (4096)
Lee Leahy522149c2015-05-08 11:33:55 -070042#define CBMEM_ROOT_SIZE DYN_CBMEM_ALIGN_SIZE
43
44/* The root region is at least DYN_CBMEM_ALIGN_SIZE . */
45#define CBMEM_ROOT_MIN_SIZE DYN_CBMEM_ALIGN_SIZE
46#define CBMEM_LG_ALIGN CBMEM_ROOT_MIN_SIZE
47
48/* Small allocation parameters. */
49#define CBMEM_SM_ROOT_SIZE 1024
50#define CBMEM_SM_ALIGN 32
51
52/* Determine the size for CBMEM root and the small allocations */
53static inline size_t cbmem_overhead_size(void)
54{
Lee Leahy708fc272017-03-07 12:18:53 -080055 return 2 * CBMEM_ROOT_MIN_SIZE;
Lee Leahy522149c2015-05-08 11:33:55 -070056}
Aaron Durbindf3a1092013-03-13 12:41:44 -050057
Kyösti Mälkki2d8520b2014-01-06 17:20:31 +020058/* By default cbmem is attempted to be recovered. Returns 0 if cbmem was
59 * recovered or 1 if cbmem had to be reinitialized. */
60int cbmem_initialize(void);
Lee Leahy522149c2015-05-08 11:33:55 -070061int cbmem_initialize_id_size(u32 id, u64 size);
62
Martin Roth0cb07e32013-07-09 21:46:01 -060063/* Initialize cbmem to be empty. */
Aaron Durbindf3a1092013-03-13 12:41:44 -050064void cbmem_initialize_empty(void);
Lee Leahy522149c2015-05-08 11:33:55 -070065void cbmem_initialize_empty_id_size(u32 id, u64 size);
Aaron Durbindf3a1092013-03-13 12:41:44 -050066
Aaron Durbindfdea2a2017-08-01 10:27:10 -060067/* Optional hook for platforms to initialize cbmem_top() value. When employed
68 * it's called a single time during boot at cbmem initialization/recovery
69 * time. */
70void cbmem_top_init(void);
71
Aaron Durbindf3a1092013-03-13 12:41:44 -050072/* Return the top address for dynamic cbmem. The address returned needs to
73 * be consistent across romstage and ramstage, and it is required to be
Furquan Shaikh6f009dc2016-03-28 13:23:13 -070074 * below 4GiB.
Julius Werner3c814b22016-08-19 16:20:40 -070075 * x86 boards or chipsets must return NULL before the cbmem backing store has
76 * been initialized. */
Aaron Durbindf3a1092013-03-13 12:41:44 -050077void *cbmem_top(void);
78
79/* Add a cbmem entry of a given size and id. These return NULL on failure. The
80 * add function performs a find first and do not check against the original
81 * size. */
82const struct cbmem_entry *cbmem_entry_add(u32 id, u64 size);
83
84/* Find a cbmem entry of a given id. These return NULL on failure. */
85const struct cbmem_entry *cbmem_entry_find(u32 id);
86
87/* Remove a region defined by a cbmem_entry. Returns 0 on success, < 0 on
88 * error. Note: A cbmem_entry cannot be removed unless it was the last one
89 * added. */
90int cbmem_entry_remove(const struct cbmem_entry *entry);
91
92/* cbmem_entry accessors to get pointer and size of a cbmem_entry. */
93void *cbmem_entry_start(const struct cbmem_entry *entry);
94u64 cbmem_entry_size(const struct cbmem_entry *entry);
95
Kyösti Mälkki2d8520b2014-01-06 17:20:31 +020096/* Returns 0 if old cbmem was recovered. Recovery is only attempted if
97 * s3resume is non-zero. */
98int cbmem_recovery(int s3resume);
Aaron Durbindf3a1092013-03-13 12:41:44 -050099/* Add a cbmem entry of a given size and id. These return NULL on failure. The
100 * add function performs a find first and do not check against the original
101 * size. */
102void *cbmem_add(u32 id, u64 size);
103/* Find a cbmem entry of a given id. These return NULL on failure. */
104void *cbmem_find(u32 id);
105
Aaron Durbin41607a42015-06-09 13:54:10 -0500106/* Indicate to each hook if cbmem is being recovered or not. */
107typedef void (* const cbmem_init_hook_t)(int is_recovery);
108void cbmem_run_init_hooks(int is_recovery);
Kyösti Mälkki823edda2014-12-18 18:30:29 +0200109void cbmem_fail_resume(void);
110
Aaron Durbindf3a1092013-03-13 12:41:44 -0500111/* Ramstage only functions. */
Aaron Durbin49048022014-02-18 21:55:02 -0600112/* Add the cbmem memory used to the memory map at boot. */
113void cbmem_add_bootmem(void);
Philipp Deppenwiese84258db2018-08-16 00:31:26 +0200114/* Return the cbmem memory used */
115void cbmem_get_region(void **baseptr, size_t *size);
Aaron Durbindf3a1092013-03-13 12:41:44 -0500116void cbmem_list(void);
Aaron Durbin1ca2d862015-09-30 12:26:54 -0500117void cbmem_add_records_to_cbtable(struct lb_header *header);
Aaron Durbindf3a1092013-03-13 12:41:44 -0500118
Kyösti Mälkki4fbac462015-01-07 04:48:43 +0200119#if ENV_RAMSTAGE
Lee Leahy22c28e02017-03-07 15:47:44 -0800120#define ROMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
121 static cbmem_init_hook_t init_fn_ ## _unused_ = init_fn_;
Kyösti Mälkki4fbac462015-01-07 04:48:43 +0200122#define RAMSTAGE_CBMEM_INIT_HOOK(init_fn_) \
123 static cbmem_init_hook_t init_fn_ ## _ptr_ __attribute__((used, \
124 section(".rodata.cbmem_init_hooks"))) = init_fn_;
Lee Leahy22c28e02017-03-07 15:47:44 -0800125#define POSTCAR_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
126 static cbmem_init_hook_t init_fn_ ## _unused2_ = init_fn_;
Kyösti Mälkki4fbac462015-01-07 04:48:43 +0200127#elif ENV_ROMSTAGE
128#define ROMSTAGE_CBMEM_INIT_HOOK(init_fn_) \
129 static cbmem_init_hook_t init_fn_ ## _ptr_ __attribute__((used, \
130 section(".rodata.cbmem_init_hooks"))) = init_fn_;
Lee Leahy22c28e02017-03-07 15:47:44 -0800131#define RAMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
132 static cbmem_init_hook_t init_fn_ ## _unused_ = init_fn_;
133#define POSTCAR_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
134 static cbmem_init_hook_t init_fn_ ## _unused2_ = init_fn_;
Aaron Durbin1e9a9142016-09-16 16:23:21 -0500135#elif ENV_POSTCAR
Lee Leahy22c28e02017-03-07 15:47:44 -0800136#define ROMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
137 static cbmem_init_hook_t init_fn_ ## _unused2_ = init_fn_;
138#define RAMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
139 static cbmem_init_hook_t init_fn_ ## _unused_ = init_fn_;
Aaron Durbin1e9a9142016-09-16 16:23:21 -0500140#define POSTCAR_CBMEM_INIT_HOOK(init_fn_) \
141 static cbmem_init_hook_t init_fn_ ## _ptr_ __attribute__((used, \
142 section(".rodata.cbmem_init_hooks"))) = init_fn_;
Kyösti Mälkki4fbac462015-01-07 04:48:43 +0200143#else
Lee Leahy22c28e02017-03-07 15:47:44 -0800144#define ROMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
145 static cbmem_init_hook_t init_fn_ ## _unused_ = init_fn_;
146#define RAMSTAGE_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
147 static cbmem_init_hook_t init_fn_ ## _unused2_ = init_fn_;
148#define POSTCAR_CBMEM_INIT_HOOK(init_fn_) __attribute__((unused)) \
149 static cbmem_init_hook_t init_fn_ ## _unused3_ = init_fn_;
Kyösti Mälkki4fbac462015-01-07 04:48:43 +0200150#endif /* ENV_RAMSTAGE */
151
152
Kyösti Mälkkia7dd6452017-04-19 07:37:38 +0300153/* Any new chipset and board must implement cbmem_top() for both
154 * romstage and ramstage to support early features like COLLECT_TIMESTAMPS
155 * and CBMEM_CONSOLE. Sometimes it is necessary to have cbmem_top()
156 * value stored in nvram to enable early recovery on S3 path.
Kyösti Mälkki2fb6b402014-12-19 08:20:45 +0200157 */
Kyösti Mälkkia7dd6452017-04-19 07:37:38 +0300158#if IS_ENABLED(CONFIG_ARCH_X86)
Kyösti Mälkki70d92b92017-04-19 19:57:01 +0300159/* Note that with LATE_CBMEM_INIT, restore_top_of_low_cacheable()
160 * may conditionally return 0 when the sleep type is non S3,
161 * i.e. cold and warm boots would return NULL also for cbmem_top. */
162void backup_top_of_low_cacheable(uintptr_t ramtop);
163uintptr_t restore_top_of_low_cacheable(void);
164uintptr_t restore_cbmem_top(void);
165
166/* Deprecated, only use with LATE_CBMEM_INIT. */
167void set_late_cbmem_top(uintptr_t ramtop);
Kyösti Mälkki2fb6b402014-12-19 08:20:45 +0200168#endif
169
Aaron Durbinb2a5f482016-12-14 14:40:43 -0600170/*
171 * Returns 0 for the stages where we know that cbmem does not come online.
172 * Even if this function returns 1 for romstage, depending upon the point in
173 * bootup, cbmem might not actually be online.
174 */
175static inline int cbmem_possibly_online(void)
176{
177 if (ENV_BOOTBLOCK)
178 return 0;
179
180 if (ENV_VERSTAGE && IS_ENABLED(CONFIG_VBOOT_STARTS_IN_BOOTBLOCK))
181 return 0;
182
183 return 1;
184}
185
Aaron Durbindf3a1092013-03-13 12:41:44 -0500186#endif /* _CBMEM_H_ */