Tom Rini | 10e4779 | 2018-05-06 17:58:06 -0400 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0+ */ |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 2 | /* |
| 3 | * Copyright (c) 2011-2012 The Chromium OS Authors. |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 4 | */ |
| 5 | |
| 6 | #ifndef __SANDBOX_STATE_H |
| 7 | #define __SANDBOX_STATE_H |
| 8 | |
Simon Glass | 8a3e035 | 2012-02-15 15:51:16 -0800 | [diff] [blame] | 9 | #include <config.h> |
Stephen Warren | 859f256 | 2016-05-12 12:03:35 -0600 | [diff] [blame] | 10 | #include <sysreset.h> |
Simon Glass | f498e43 | 2013-11-10 10:27:02 -0700 | [diff] [blame] | 11 | #include <stdbool.h> |
Simon Glass | fc1ebd3 | 2018-09-15 00:50:56 -0600 | [diff] [blame] | 12 | #include <linux/list.h> |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 13 | #include <linux/stringify.h> |
Simon Glass | 8a3e035 | 2012-02-15 15:51:16 -0800 | [diff] [blame] | 14 | |
Simon Glass | 678ef47 | 2014-02-27 13:26:22 -0700 | [diff] [blame] | 15 | /** |
| 16 | * Selects the behavior of the serial terminal. |
| 17 | * |
| 18 | * If Ctrl-C is processed by U-Boot, then the only way to quit sandbox is with |
| 19 | * the 'reset' command, or equivalent. |
| 20 | * |
| 21 | * If the terminal is cooked, then Ctrl-C will terminate U-Boot, and the |
| 22 | * command line will not be quite such a faithful emulation. |
| 23 | * |
| 24 | * Options are: |
| 25 | * |
| 26 | * raw-with-sigs - Raw, but allow signals (Ctrl-C will quit) |
| 27 | * raw - Terminal is always raw |
| 28 | * cooked - Terminal is always cooked |
| 29 | */ |
| 30 | enum state_terminal_raw { |
| 31 | STATE_TERM_RAW_WITH_SIGS, /* Default */ |
| 32 | STATE_TERM_RAW, |
| 33 | STATE_TERM_COOKED, |
| 34 | |
| 35 | STATE_TERM_COUNT, |
| 36 | }; |
| 37 | |
Mike Frysinger | 1d8e57c | 2013-12-03 16:43:26 -0700 | [diff] [blame] | 38 | struct sandbox_spi_info { |
Simon Glass | 95429fe | 2014-10-13 23:41:57 -0600 | [diff] [blame] | 39 | struct udevice *emul; |
Mike Frysinger | 1d8e57c | 2013-12-03 16:43:26 -0700 | [diff] [blame] | 40 | }; |
| 41 | |
maxims@google.com | daea6d4 | 2017-04-17 12:00:21 -0700 | [diff] [blame] | 42 | struct sandbox_wdt_info { |
| 43 | unsigned long long counter; |
| 44 | uint reset_count; |
| 45 | bool running; |
| 46 | }; |
| 47 | |
Simon Glass | fc1ebd3 | 2018-09-15 00:50:56 -0600 | [diff] [blame] | 48 | /** |
| 49 | * struct sandbox_mapmem_entry - maps pointers to/from U-Boot addresses |
| 50 | * |
| 51 | * When map_to_sysmem() is called with an address outside sandbox's emulated |
| 52 | * RAM, a record is created with a tag that can be used to reference that |
| 53 | * pointer. When map_sysmem() is called later with that tag, the pointer will |
| 54 | * be returned, just as it would for a normal sandbox address. |
| 55 | * |
| 56 | * @tag: Address tag (a value which U-Boot uses to refer to the address) |
| 57 | * @ptr: Associated pointer for that tag |
| 58 | */ |
| 59 | struct sandbox_mapmem_entry { |
| 60 | ulong tag; |
| 61 | void *ptr; |
| 62 | struct list_head sibling_node; |
| 63 | }; |
| 64 | |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 65 | /* The complete state of the test system */ |
| 66 | struct sandbox_state { |
| 67 | const char *cmd; /* Command to execute */ |
Simon Glass | f498e43 | 2013-11-10 10:27:02 -0700 | [diff] [blame] | 68 | bool interactive; /* Enable cmdline after execute */ |
Sjoerd Simons | 335f470 | 2015-04-30 22:16:09 +0200 | [diff] [blame] | 69 | bool run_distro_boot; /* Automatically run distro bootcommands */ |
Simon Glass | 1539343 | 2013-04-20 08:42:41 +0000 | [diff] [blame] | 70 | const char *fdt_fname; /* Filename of FDT binary */ |
Simon Glass | 8a3e035 | 2012-02-15 15:51:16 -0800 | [diff] [blame] | 71 | const char *parse_err; /* Error to report from parsing */ |
| 72 | int argc; /* Program arguments */ |
Simon Glass | e990653 | 2014-02-27 13:26:16 -0700 | [diff] [blame] | 73 | char **argv; /* Command line arguments */ |
Simon Glass | f0b534f | 2022-10-20 18:23:02 -0600 | [diff] [blame^] | 74 | const char *jumped_fname; /* Jumped from previous U-Boot */ |
| 75 | const char *prog_fname; /* U-Boot executable filename */ |
Simon Glass | 9dd10bf | 2013-11-10 10:27:03 -0700 | [diff] [blame] | 76 | uint8_t *ram_buf; /* Emulated RAM buffer */ |
Heinrich Schuchardt | fff251e | 2020-06-07 18:47:35 +0200 | [diff] [blame] | 77 | unsigned long ram_size; /* Size of RAM buffer */ |
Simon Glass | 9dd10bf | 2013-11-10 10:27:03 -0700 | [diff] [blame] | 78 | const char *ram_buf_fname; /* Filename to use for RAM buffer */ |
Simon Glass | 47acfc6 | 2014-02-27 13:26:23 -0700 | [diff] [blame] | 79 | bool ram_buf_rm; /* Remove RAM buffer file after read */ |
Simon Glass | 9dd10bf | 2013-11-10 10:27:03 -0700 | [diff] [blame] | 80 | bool write_ram_buf; /* Write RAM buffer on exit */ |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 81 | const char *state_fname; /* File containing sandbox state */ |
| 82 | void *state_fdt; /* Holds saved state for sandbox */ |
| 83 | bool read_state; /* Read sandbox state on startup */ |
| 84 | bool write_state; /* Write sandbox state on exit */ |
| 85 | bool ignore_missing_state_on_read; /* No error if state missing */ |
Simon Glass | b9ddbf4 | 2014-02-27 13:26:19 -0700 | [diff] [blame] | 86 | bool show_lcd; /* Show LCD on start-up */ |
Simon Glass | f91de0b | 2020-02-03 07:36:13 -0700 | [diff] [blame] | 87 | bool double_lcd; /* Double display size for high-DPI */ |
Stephen Warren | 859f256 | 2016-05-12 12:03:35 -0600 | [diff] [blame] | 88 | enum sysreset_t last_sysreset; /* Last system reset type */ |
| 89 | bool sysreset_allowed[SYSRESET_COUNT]; /* Allowed system reset types */ |
Simon Glass | 678ef47 | 2014-02-27 13:26:22 -0700 | [diff] [blame] | 90 | enum state_terminal_raw term_raw; /* Terminal raw/cooked */ |
Simon Glass | b25fa31 | 2015-11-08 23:47:43 -0700 | [diff] [blame] | 91 | bool skip_delays; /* Ignore any time delays (for test) */ |
Simon Glass | fe6d12a | 2015-11-08 23:47:50 -0700 | [diff] [blame] | 92 | bool show_test_output; /* Don't suppress stdout in tests */ |
Simon Glass | 4e9a64d | 2018-10-01 11:55:11 -0600 | [diff] [blame] | 93 | int default_log_level; /* Default log level for sandbox */ |
Simon Glass | 24a284a | 2018-11-23 21:29:29 -0700 | [diff] [blame] | 94 | bool ram_buf_read; /* true if we read the RAM buffer */ |
Simon Glass | a4e289b | 2020-10-25 20:38:28 -0600 | [diff] [blame] | 95 | bool run_unittests; /* Run unit tests */ |
Simon Glass | eff9658 | 2020-10-25 20:38:33 -0600 | [diff] [blame] | 96 | const char *select_unittests; /* Unit test to run */ |
Simon Glass | b78cc9b | 2021-03-22 18:21:01 +1300 | [diff] [blame] | 97 | bool handle_signals; /* Handle signals within sandbox */ |
Simon Glass | d8c6017 | 2021-07-24 15:14:39 -0600 | [diff] [blame] | 98 | bool autoboot_keyed; /* Use keyed-autoboot feature */ |
Mike Frysinger | 1d8e57c | 2013-12-03 16:43:26 -0700 | [diff] [blame] | 99 | |
| 100 | /* Pointer to information for each SPI bus/cs */ |
| 101 | struct sandbox_spi_info spi[CONFIG_SANDBOX_SPI_MAX_BUS] |
| 102 | [CONFIG_SANDBOX_SPI_MAX_CS]; |
maxims@google.com | daea6d4 | 2017-04-17 12:00:21 -0700 | [diff] [blame] | 103 | |
| 104 | /* Information about Watchdog */ |
| 105 | struct sandbox_wdt_info wdt; |
Simon Glass | fc1ebd3 | 2018-09-15 00:50:56 -0600 | [diff] [blame] | 106 | |
| 107 | ulong next_tag; /* Next address tag to allocate */ |
| 108 | struct list_head mapmem_head; /* struct sandbox_mapmem_entry */ |
Benjamin Gaignard | a550b54 | 2018-11-27 13:49:50 +0100 | [diff] [blame] | 109 | bool hwspinlock; /* Hardware Spinlock status */ |
Simon Glass | 0a341a7 | 2019-09-25 08:56:09 -0600 | [diff] [blame] | 110 | bool allow_memio; /* Allow readl() etc. to work */ |
Simon Glass | 17d133b | 2019-02-16 20:24:45 -0700 | [diff] [blame] | 111 | |
Simon Glass | f17b967 | 2022-09-06 20:27:09 -0600 | [diff] [blame] | 112 | void *other_fdt_buf; /* 'other' FDT blob used by tests */ |
| 113 | int other_size; /* size of other FDT blob */ |
| 114 | |
Simon Glass | 17d133b | 2019-02-16 20:24:45 -0700 | [diff] [blame] | 115 | /* |
| 116 | * This struct is getting large. |
| 117 | * |
| 118 | * Consider putting test data in driver-private structs, like |
| 119 | * sandbox_pch.c. |
| 120 | * |
| 121 | * If you add new members, please put them above this comment. |
| 122 | */ |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 123 | }; |
| 124 | |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 125 | /* Minimum space we guarantee in the state FDT when calling read/write*/ |
| 126 | #define SANDBOX_STATE_MIN_SPACE 0x1000 |
| 127 | |
| 128 | /** |
| 129 | * struct sandbox_state_io - methods to saved/restore sandbox state |
| 130 | * @name: Name of of the device tree node, also the name of the variable |
| 131 | * holding this data so it should be an identifier (use underscore |
| 132 | * instead of minus) |
| 133 | * @compat: Compatible string for the node containing this state |
| 134 | * |
| 135 | * @read: Function to read state from FDT |
| 136 | * If data is available, then blob and node will provide access to it. If |
| 137 | * not (blob == NULL and node == -1) this function should set up an empty |
| 138 | * data set for start-of-day. |
| 139 | * @param blob: Pointer to device tree blob, or NULL if no data to read |
| 140 | * @param node: Node offset to read from |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 141 | * Return: 0 if OK, -ve on error |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 142 | * |
| 143 | * @write: Function to write state to FDT |
| 144 | * The caller will ensure that there is a node ready for the state. The |
| 145 | * node may already contain the old state, in which case it should be |
| 146 | * overridden. There is guaranteed to be SANDBOX_STATE_MIN_SPACE bytes |
| 147 | * of free space, so error checking is not required for fdt_setprop...() |
| 148 | * calls which add up to less than this much space. |
| 149 | * |
| 150 | * For adding larger properties, use state_setprop(). |
| 151 | * |
| 152 | * @param blob: Device tree blob holding state |
| 153 | * @param node: Node to write our state into |
| 154 | * |
| 155 | * Note that it is possible to save data as large blobs or as individual |
| 156 | * hierarchical properties. However, unless you intend to keep state files |
| 157 | * around for a long time and be able to run an old state file on a new |
| 158 | * sandbox, it might not be worth using individual properties for everything. |
| 159 | * This is certainly supported, it is just a matter of the effort you wish |
| 160 | * to put into the state read/write feature. |
| 161 | */ |
| 162 | struct sandbox_state_io { |
| 163 | const char *name; |
| 164 | const char *compat; |
| 165 | int (*write)(void *blob, int node); |
| 166 | int (*read)(const void *blob, int node); |
| 167 | }; |
| 168 | |
| 169 | /** |
| 170 | * SANDBOX_STATE_IO - Declare sandbox state to read/write |
| 171 | * |
| 172 | * Sandbox permits saving state from one run and restoring it in another. This |
| 173 | * allows the test system to retain state between runs and thus better |
| 174 | * emulate a real system. Examples of state that might be useful to save are |
| 175 | * the emulated GPIOs pin settings, flash memory contents and TPM private |
| 176 | * data. U-Boot memory contents is dealth with separately since it is large |
| 177 | * and it is not normally useful to save it (since a normal system does not |
| 178 | * preserve DRAM between runs). See the '-m' option for this. |
| 179 | * |
| 180 | * See struct sandbox_state_io above for member documentation. |
| 181 | */ |
| 182 | #define SANDBOX_STATE_IO(_name, _compat, _read, _write) \ |
| 183 | ll_entry_declare(struct sandbox_state_io, _name, state_io) = { \ |
| 184 | .name = __stringify(_name), \ |
| 185 | .read = _read, \ |
| 186 | .write = _write, \ |
| 187 | .compat = _compat, \ |
| 188 | } |
| 189 | |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 190 | /** |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 191 | * Gets a pointer to the current state. |
| 192 | * |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 193 | * Return: pointer to state |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 194 | */ |
| 195 | struct sandbox_state *state_get_current(void); |
| 196 | |
| 197 | /** |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 198 | * Read the sandbox state from the supplied device tree file |
| 199 | * |
| 200 | * This calls all registered state handlers to read in the sandbox state |
| 201 | * from a previous test run. |
| 202 | * |
| 203 | * @param state Sandbox state to update |
| 204 | * @param fname Filename of device tree file to read from |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 205 | * Return: 0 if OK, -ve on error |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 206 | */ |
| 207 | int sandbox_read_state(struct sandbox_state *state, const char *fname); |
| 208 | |
| 209 | /** |
| 210 | * Write the sandbox state to the supplied device tree file |
| 211 | * |
| 212 | * This calls all registered state handlers to write out the sandbox state |
| 213 | * so that it can be preserved for a future test run. |
| 214 | * |
| 215 | * If the file exists it is overwritten. |
| 216 | * |
| 217 | * @param state Sandbox state to update |
| 218 | * @param fname Filename of device tree file to write to |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 219 | * Return: 0 if OK, -ve on error |
Simon Glass | d7c8d8d | 2013-11-10 10:27:04 -0700 | [diff] [blame] | 220 | */ |
| 221 | int sandbox_write_state(struct sandbox_state *state, const char *fname); |
| 222 | |
| 223 | /** |
| 224 | * Add a property to a sandbox state node |
| 225 | * |
| 226 | * This is equivalent to fdt_setprop except that it automatically enlarges |
| 227 | * the device tree if necessary. That means it is safe to write any amount |
| 228 | * of data here. |
| 229 | * |
| 230 | * This function can only be called from within struct sandbox_state_io's |
| 231 | * ->write method, i.e. within state I/O drivers. |
| 232 | * |
| 233 | * @param node Device tree node to write to |
| 234 | * @param prop_name Property to write |
| 235 | * @param data Data to write into property |
| 236 | * @param size Size of data to write into property |
| 237 | */ |
| 238 | int state_setprop(int node, const char *prop_name, const void *data, int size); |
| 239 | |
| 240 | /** |
Simon Glass | b25fa31 | 2015-11-08 23:47:43 -0700 | [diff] [blame] | 241 | * Control skipping of time delays |
| 242 | * |
| 243 | * Some tests have unnecessay time delays (e.g. USB). Allow these to be |
| 244 | * skipped to speed up testing |
| 245 | * |
| 246 | * @param skip_delays true to skip delays from now on, false to honour delay |
| 247 | * requests |
| 248 | */ |
| 249 | void state_set_skip_delays(bool skip_delays); |
| 250 | |
| 251 | /** |
| 252 | * See if delays should be skipped |
| 253 | * |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 254 | * Return: true if delays should be skipped, false if they should be honoured |
Simon Glass | b25fa31 | 2015-11-08 23:47:43 -0700 | [diff] [blame] | 255 | */ |
| 256 | bool state_get_skip_delays(void); |
| 257 | |
| 258 | /** |
Simon Glass | 36d08d82 | 2017-05-18 20:09:13 -0600 | [diff] [blame] | 259 | * state_reset_for_test() - Reset ready to re-run tests |
| 260 | * |
| 261 | * This clears out any test state ready for another test run. |
| 262 | */ |
| 263 | void state_reset_for_test(struct sandbox_state *state); |
| 264 | |
| 265 | /** |
Simon Glass | 46508c9 | 2018-11-15 18:44:03 -0700 | [diff] [blame] | 266 | * state_show() - Show information about the sandbox state |
| 267 | * |
| 268 | * @param state Sandbox state to show |
| 269 | */ |
| 270 | void state_show(struct sandbox_state *state); |
| 271 | |
| 272 | /** |
Simon Glass | d74c461 | 2022-09-06 20:27:08 -0600 | [diff] [blame] | 273 | * state_get_rel_filename() - Get a filename relative to the executable |
| 274 | * |
| 275 | * This uses argv[0] to obtain a filename path |
| 276 | * |
| 277 | * @rel_path: Relative path to build, e.g. "arch/sandbox/dts/test.dtb". Must not |
| 278 | * have a trailing / |
| 279 | * @buf: Buffer to use to return the filename |
| 280 | * @size: Size of buffer |
| 281 | * @return length of filename (including terminator), -ENOSPC if @size is too |
| 282 | * small |
| 283 | */ |
| 284 | int state_get_rel_filename(const char *rel_path, char *buf, int size); |
| 285 | |
| 286 | /** |
Simon Glass | f17b967 | 2022-09-06 20:27:09 -0600 | [diff] [blame] | 287 | * state_load_other_fdt() - load the 'other' FDT into a buffer |
| 288 | * |
| 289 | * This loads the other.dtb file into a buffer. This is typically used in tests. |
| 290 | * |
| 291 | * @bufp: Place to put allocated buffer pointer. The buffer is read using |
| 292 | * os_read_file() which calls os_malloc(), so does affect U-Boot's own malloc() |
| 293 | * space |
| 294 | * @sizep: Returns the size of the buffer |
| 295 | * @return 0 if OK, -ve on error |
| 296 | */ |
| 297 | int state_load_other_fdt(const char **bufp, int *sizep); |
| 298 | |
| 299 | /** |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 300 | * Initialize the test system state |
| 301 | */ |
| 302 | int state_init(void); |
| 303 | |
Simon Glass | 9dd10bf | 2013-11-10 10:27:03 -0700 | [diff] [blame] | 304 | /** |
| 305 | * Uninitialize the test system state, writing out state if configured to |
| 306 | * do so. |
| 307 | * |
Heinrich Schuchardt | 47b4c02 | 2022-01-19 18:05:50 +0100 | [diff] [blame] | 308 | * Return: 0 if OK, -ve on error |
Simon Glass | 9dd10bf | 2013-11-10 10:27:03 -0700 | [diff] [blame] | 309 | */ |
| 310 | int state_uninit(void); |
| 311 | |
Simon Glass | 20bf89a | 2012-02-15 15:51:15 -0800 | [diff] [blame] | 312 | #endif |