blob: 6953d3010a65d203b90834b24640b46aeee846a8 [file] [log] [blame]
Simon Glass919e7a82018-11-15 18:43:53 -07001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * Copyright (c) 2018, Google Inc. All rights reserved.
4 */
5
6#include <common.h>
7#include <bloblist.h>
8#include <log.h>
9#include <mapmem.h>
Simon Glass401d1c42020-10-30 21:38:53 -060010#include <asm/global_data.h>
Simon Glass919e7a82018-11-15 18:43:53 -070011#include <test/suites.h>
12#include <test/test.h>
13#include <test/ut.h>
14
15DECLARE_GLOBAL_DATA_PTR;
16
17/* Declare a new compression test */
18#define BLOBLIST_TEST(_name, _flags) \
19 UNIT_TEST(_name, _flags, bloblist_test)
20
21enum {
22 TEST_TAG = 1,
23 TEST_TAG2 = 2,
24 TEST_TAG_MISSING = 3,
25
26 TEST_SIZE = 10,
27 TEST_SIZE2 = 20,
Simon Glass4c1497e2020-09-19 18:49:29 -060028 TEST_SIZE_LARGE = 0x3e0,
Simon Glass919e7a82018-11-15 18:43:53 -070029
30 TEST_ADDR = CONFIG_BLOBLIST_ADDR,
Simon Glass4c1497e2020-09-19 18:49:29 -060031 TEST_BLOBLIST_SIZE = 0x400,
Simon Glass751b7c72020-09-19 18:49:28 -060032
33 ERASE_BYTE = '\xff',
Simon Glass919e7a82018-11-15 18:43:53 -070034};
35
36static struct bloblist_hdr *clear_bloblist(void)
37{
38 struct bloblist_hdr *hdr;
39
Simon Glassb83994d2020-01-27 08:49:52 -070040 /*
41 * Clear out any existing bloblist so we have a clean slate. Zero the
42 * header so that existing records are removed, but set everything else
43 * to 0xff for testing purposes.
44 */
Simon Glass919e7a82018-11-15 18:43:53 -070045 hdr = map_sysmem(CONFIG_BLOBLIST_ADDR, TEST_BLOBLIST_SIZE);
Simon Glass751b7c72020-09-19 18:49:28 -060046 memset(hdr, ERASE_BYTE, TEST_BLOBLIST_SIZE);
Simon Glassb83994d2020-01-27 08:49:52 -070047 memset(hdr, '\0', sizeof(*hdr));
Simon Glass919e7a82018-11-15 18:43:53 -070048
49 return hdr;
50}
51
Simon Glassb83994d2020-01-27 08:49:52 -070052static int check_zero(void *data, int size)
53{
54 u8 *ptr;
55 int i;
56
57 for (ptr = data, i = 0; i < size; i++, ptr++) {
58 if (*ptr)
59 return -EINVAL;
60 }
61
62 return 0;
63}
64
Simon Glass919e7a82018-11-15 18:43:53 -070065static int bloblist_test_init(struct unit_test_state *uts)
66{
67 struct bloblist_hdr *hdr;
68
69 hdr = clear_bloblist();
70 ut_asserteq(-ENOENT, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
71 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
72 hdr->version++;
73 ut_asserteq(-EPROTONOSUPPORT, bloblist_check(TEST_ADDR,
74 TEST_BLOBLIST_SIZE));
75
76 ut_asserteq(-ENOSPC, bloblist_new(TEST_ADDR, 0x10, 0));
77 ut_asserteq(-EFAULT, bloblist_new(1, TEST_BLOBLIST_SIZE, 0));
78 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
79
80 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
81 ut_assertok(bloblist_finish());
82 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
83 hdr->flags++;
84 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
85
86 return 1;
87}
88BLOBLIST_TEST(bloblist_test_init, 0);
89
90static int bloblist_test_blob(struct unit_test_state *uts)
91{
92 struct bloblist_hdr *hdr;
93 struct bloblist_rec *rec, *rec2;
94 char *data;
95
96 /* At the start there should be no records */
97 hdr = clear_bloblist();
98 ut_assertnull(bloblist_find(TEST_TAG, TEST_BLOBLIST_SIZE));
99 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
Simon Glasscdd4e302020-09-19 18:49:27 -0600100 ut_asserteq(map_to_sysmem(hdr), TEST_ADDR);
Simon Glass919e7a82018-11-15 18:43:53 -0700101
102 /* Add a record and check that we can find it */
Simon Glass4c1497e2020-09-19 18:49:29 -0600103 data = bloblist_add(TEST_TAG, TEST_SIZE, 0);
Simon Glass919e7a82018-11-15 18:43:53 -0700104 rec = (void *)(hdr + 1);
Simon Glasscdd4e302020-09-19 18:49:27 -0600105 ut_asserteq_addr(rec + 1, data);
Simon Glass919e7a82018-11-15 18:43:53 -0700106 data = bloblist_find(TEST_TAG, TEST_SIZE);
Simon Glasscdd4e302020-09-19 18:49:27 -0600107 ut_asserteq_addr(rec + 1, data);
Simon Glass919e7a82018-11-15 18:43:53 -0700108
Simon Glassb83994d2020-01-27 08:49:52 -0700109 /* Check the data is zeroed */
110 ut_assertok(check_zero(data, TEST_SIZE));
111
Simon Glass919e7a82018-11-15 18:43:53 -0700112 /* Check the 'ensure' method */
Simon Glasscdd4e302020-09-19 18:49:27 -0600113 ut_asserteq_addr(data, bloblist_ensure(TEST_TAG, TEST_SIZE));
Simon Glass919e7a82018-11-15 18:43:53 -0700114 ut_assertnull(bloblist_ensure(TEST_TAG, TEST_SIZE2));
115 rec2 = (struct bloblist_rec *)(data + ALIGN(TEST_SIZE, BLOBLIST_ALIGN));
Simon Glassb83994d2020-01-27 08:49:52 -0700116 ut_assertok(check_zero(data, TEST_SIZE));
Simon Glass919e7a82018-11-15 18:43:53 -0700117
118 /* Check for a non-existent record */
Simon Glasscdd4e302020-09-19 18:49:27 -0600119 ut_asserteq_addr(data, bloblist_ensure(TEST_TAG, TEST_SIZE));
120 ut_asserteq_addr(rec2 + 1, bloblist_ensure(TEST_TAG2, TEST_SIZE2));
Simon Glass919e7a82018-11-15 18:43:53 -0700121 ut_assertnull(bloblist_find(TEST_TAG_MISSING, 0));
122
123 return 0;
124}
125BLOBLIST_TEST(bloblist_test_blob, 0);
126
Simon Glass5b044542020-01-27 08:49:50 -0700127/* Check bloblist_ensure_size_ret() */
128static int bloblist_test_blob_ensure(struct unit_test_state *uts)
129{
130 void *data, *data2;
131 int size;
132
133 /* At the start there should be no records */
134 clear_bloblist();
135 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
136
137 /* Test with an empty bloblist */
138 size = TEST_SIZE;
139 ut_assertok(bloblist_ensure_size_ret(TEST_TAG, &size, &data));
140 ut_asserteq(TEST_SIZE, size);
Simon Glassb83994d2020-01-27 08:49:52 -0700141 ut_assertok(check_zero(data, TEST_SIZE));
Simon Glass5b044542020-01-27 08:49:50 -0700142
143 /* Check that we get the same thing again */
144 ut_assertok(bloblist_ensure_size_ret(TEST_TAG, &size, &data2));
145 ut_asserteq(TEST_SIZE, size);
Simon Glasscdd4e302020-09-19 18:49:27 -0600146 ut_asserteq_addr(data, data2);
Simon Glass5b044542020-01-27 08:49:50 -0700147
148 /* Check that the size remains the same */
149 size = TEST_SIZE2;
150 ut_assertok(bloblist_ensure_size_ret(TEST_TAG, &size, &data));
151 ut_asserteq(TEST_SIZE, size);
152
153 /* Check running out of space */
154 size = TEST_SIZE_LARGE;
155 ut_asserteq(-ENOSPC, bloblist_ensure_size_ret(TEST_TAG2, &size, &data));
156
157 return 0;
158}
159BLOBLIST_TEST(bloblist_test_blob_ensure, 0);
160
Simon Glass919e7a82018-11-15 18:43:53 -0700161static int bloblist_test_bad_blob(struct unit_test_state *uts)
162{
163 struct bloblist_hdr *hdr;
164 void *data;
165
166 hdr = clear_bloblist();
167 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
168 data = hdr + 1;
169 data += sizeof(struct bloblist_rec);
Simon Glasscdd4e302020-09-19 18:49:27 -0600170 ut_asserteq_addr(data, bloblist_ensure(TEST_TAG, TEST_SIZE));
171 ut_asserteq_addr(data, bloblist_ensure(TEST_TAG, TEST_SIZE));
Simon Glass919e7a82018-11-15 18:43:53 -0700172
173 return 0;
174}
175BLOBLIST_TEST(bloblist_test_bad_blob, 0);
176
177static int bloblist_test_checksum(struct unit_test_state *uts)
178{
179 struct bloblist_hdr *hdr;
180 char *data, *data2;
181
182 hdr = clear_bloblist();
183 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
184 ut_assertok(bloblist_finish());
185 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
186
187 /*
188 * Now change things amd make sure that the checksum notices. We cannot
189 * change the size or alloced fields, since that will crash the code.
190 * It has to rely on these being correct.
191 */
192 hdr->flags--;
193 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
194 hdr->flags++;
195
196 hdr->size--;
197 ut_asserteq(-EFBIG, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
198 hdr->size++;
199
200 hdr->spare++;
201 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
202 hdr->spare--;
203
204 hdr->chksum++;
205 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
206 hdr->chksum--;
207
208 /* Make sure the checksum changes when we add blobs */
Simon Glass4c1497e2020-09-19 18:49:29 -0600209 data = bloblist_add(TEST_TAG, TEST_SIZE, 0);
Simon Glass919e7a82018-11-15 18:43:53 -0700210 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
211
Simon Glass4c1497e2020-09-19 18:49:29 -0600212 data2 = bloblist_add(TEST_TAG2, TEST_SIZE2, 0);
Simon Glass919e7a82018-11-15 18:43:53 -0700213 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
214 ut_assertok(bloblist_finish());
215
216 /* It should also change if we change the data */
217 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
218 *data += 1;
219 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
220 *data -= 1;
221
222 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
223 *data2 += 1;
224 ut_asserteq(-EIO, bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
225 *data2 -= 1;
226
227 /*
228 * Changing data outside the range of valid data should not affect
229 * the checksum.
230 */
231 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
232 data[TEST_SIZE]++;
233 data2[TEST_SIZE2]++;
234 ut_assertok(bloblist_check(TEST_ADDR, TEST_BLOBLIST_SIZE));
235
236 return 0;
237}
Simon Glass919e7a82018-11-15 18:43:53 -0700238BLOBLIST_TEST(bloblist_test_checksum, 0);
239
Simon Glass4aed2272020-09-19 18:49:26 -0600240/* Test the 'bloblist info' command */
241static int bloblist_test_cmd_info(struct unit_test_state *uts)
242{
Simon Glass4aed2272020-09-19 18:49:26 -0600243 struct bloblist_hdr *hdr;
244 char *data, *data2;
245
246 hdr = clear_bloblist();
247 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
248 data = bloblist_ensure(TEST_TAG, TEST_SIZE);
249 data2 = bloblist_ensure(TEST_TAG2, TEST_SIZE2);
250
251 console_record_reset_enable();
Simon Glassef7e2642020-11-08 21:08:43 -0700252 ut_silence_console(uts);
Simon Glass4aed2272020-09-19 18:49:26 -0600253 console_record_reset();
254 run_command("bloblist info", 0);
Simon Glasscdd4e302020-09-19 18:49:27 -0600255 ut_assert_nextline("base: %lx", (ulong)map_to_sysmem(hdr));
Simon Glass4c1497e2020-09-19 18:49:29 -0600256 ut_assert_nextline("size: 400 1 KiB");
Simon Glass4aed2272020-09-19 18:49:26 -0600257 ut_assert_nextline("alloced: 70 112 Bytes");
Simon Glass4c1497e2020-09-19 18:49:29 -0600258 ut_assert_nextline("free: 390 912 Bytes");
Simon Glass4aed2272020-09-19 18:49:26 -0600259 ut_assert_console_end();
Simon Glassef7e2642020-11-08 21:08:43 -0700260 ut_unsilence_console(uts);
Simon Glass4aed2272020-09-19 18:49:26 -0600261
262 return 0;
263}
264BLOBLIST_TEST(bloblist_test_cmd_info, 0);
265
266/* Test the 'bloblist list' command */
267static int bloblist_test_cmd_list(struct unit_test_state *uts)
268{
Simon Glass4aed2272020-09-19 18:49:26 -0600269 struct bloblist_hdr *hdr;
270 char *data, *data2;
271
272 hdr = clear_bloblist();
273 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
274 data = bloblist_ensure(TEST_TAG, TEST_SIZE);
275 data2 = bloblist_ensure(TEST_TAG2, TEST_SIZE2);
276
277 console_record_reset_enable();
Simon Glassef7e2642020-11-08 21:08:43 -0700278 ut_silence_console(uts);
Simon Glass4aed2272020-09-19 18:49:26 -0600279 console_record_reset();
280 run_command("bloblist list", 0);
281 ut_assert_nextline("Address Size Tag Name");
Simon Glasscdd4e302020-09-19 18:49:27 -0600282 ut_assert_nextline("%08lx %8x 1 EC host event",
283 (ulong)map_to_sysmem(data), TEST_SIZE);
284 ut_assert_nextline("%08lx %8x 2 SPL hand-off",
285 (ulong)map_to_sysmem(data2), TEST_SIZE2);
Simon Glass4aed2272020-09-19 18:49:26 -0600286 ut_assert_console_end();
Simon Glassef7e2642020-11-08 21:08:43 -0700287 ut_unsilence_console(uts);
Simon Glass4aed2272020-09-19 18:49:26 -0600288
289 return 0;
290}
291BLOBLIST_TEST(bloblist_test_cmd_list, 0);
292
Simon Glass751b7c72020-09-19 18:49:28 -0600293/* Test alignment of bloblist blobs */
294static int bloblist_test_align(struct unit_test_state *uts)
295{
296 struct bloblist_hdr *hdr;
Simon Glass4c1497e2020-09-19 18:49:29 -0600297 ulong addr;
298 char *data;
Simon Glass751b7c72020-09-19 18:49:28 -0600299 int i;
300
301 /* At the start there should be no records */
302 hdr = clear_bloblist();
303 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
304 ut_assertnull(bloblist_find(TEST_TAG, TEST_BLOBLIST_SIZE));
305
Simon Glass4c1497e2020-09-19 18:49:29 -0600306 /* Check the default alignment */
Simon Glass751b7c72020-09-19 18:49:28 -0600307 for (i = 0; i < 3; i++) {
308 int size = i * 3;
309 ulong addr;
310 char *data;
311 int j;
312
Simon Glass4c1497e2020-09-19 18:49:29 -0600313 data = bloblist_add(i, size, 0);
Simon Glass751b7c72020-09-19 18:49:28 -0600314 ut_assertnonnull(data);
315 addr = map_to_sysmem(data);
316 ut_asserteq(0, addr & (BLOBLIST_ALIGN - 1));
317
318 /* Only the bytes in the blob data should be zeroed */
319 for (j = 0; j < size; j++)
320 ut_asserteq(0, data[j]);
321 for (; j < BLOBLIST_ALIGN; j++)
322 ut_asserteq(ERASE_BYTE, data[j]);
323 }
324
Simon Glass4c1497e2020-09-19 18:49:29 -0600325 /* Check larger alignment */
326 for (i = 0; i < 3; i++) {
327 int align = 32 << i;
328
329 data = bloblist_add(3 + i, i * 4, align);
330 ut_assertnonnull(data);
331 addr = map_to_sysmem(data);
332 ut_asserteq(0, addr & (align - 1));
333 }
334
335 /* Check alignment with an bloblist starting on a smaller alignment */
336 hdr = map_sysmem(TEST_ADDR + BLOBLIST_ALIGN, TEST_BLOBLIST_SIZE);
337 memset(hdr, ERASE_BYTE, TEST_BLOBLIST_SIZE);
338 memset(hdr, '\0', sizeof(*hdr));
339 ut_assertok(bloblist_new(TEST_ADDR + BLOBLIST_ALIGN, TEST_BLOBLIST_SIZE,
340 0));
341
342 data = bloblist_add(1, 5, BLOBLIST_ALIGN * 2);
343 ut_assertnonnull(data);
344 addr = map_to_sysmem(data);
345 ut_asserteq(0, addr & (BLOBLIST_ALIGN * 2 - 1));
346
Simon Glass751b7c72020-09-19 18:49:28 -0600347 return 0;
348}
349BLOBLIST_TEST(bloblist_test_align, 0);
350
Simon Glass9fe06462021-01-13 20:29:43 -0700351/* Test relocation of a bloblist */
352static int bloblist_test_reloc(struct unit_test_state *uts)
353{
354 const uint large_size = TEST_BLOBLIST_SIZE;
355 const uint small_size = 0x20;
356 void *old_ptr, *new_ptr;
357 void *blob1, *blob2;
358 ulong new_addr;
359 ulong new_size;
360
361 ut_assertok(bloblist_new(TEST_ADDR, TEST_BLOBLIST_SIZE, 0));
362 old_ptr = map_sysmem(TEST_ADDR, TEST_BLOBLIST_SIZE);
363
364 /* Add one blob and then one that won't fit */
365 blob1 = bloblist_add(TEST_TAG, small_size, 0);
366 ut_assertnonnull(blob1);
367 blob2 = bloblist_add(TEST_TAG2, large_size, 0);
368 ut_assertnull(blob2);
369
370 /* Relocate the bloblist somewhere else, a bit larger */
371 new_addr = TEST_ADDR + TEST_BLOBLIST_SIZE;
372 new_size = TEST_BLOBLIST_SIZE + 0x100;
373 new_ptr = map_sysmem(new_addr, TEST_BLOBLIST_SIZE);
374 bloblist_reloc(new_ptr, new_size, old_ptr, TEST_BLOBLIST_SIZE);
375 gd->bloblist = new_ptr;
376
377 /* Check the old blob is there and that we can now add the bigger one */
378 ut_assertnonnull(bloblist_find(TEST_TAG, small_size));
379 ut_assertnull(bloblist_find(TEST_TAG2, small_size));
380 blob2 = bloblist_add(TEST_TAG2, large_size, 0);
381 ut_assertnonnull(blob2);
382
383 return 0;
384}
385BLOBLIST_TEST(bloblist_test_reloc, 0);
386
Simon Glass09140112020-05-10 11:40:03 -0600387int do_ut_bloblist(struct cmd_tbl *cmdtp, int flag, int argc,
388 char *const argv[])
Simon Glass919e7a82018-11-15 18:43:53 -0700389{
390 struct unit_test *tests = ll_entry_start(struct unit_test,
391 bloblist_test);
392 const int n_ents = ll_entry_count(struct unit_test, bloblist_test);
393
Philippe Reynes4ad4edf2019-12-17 19:07:04 +0100394 return cmd_ut_category("bloblist", "bloblist_test_",
395 tests, n_ents, argc, argv);
Simon Glass919e7a82018-11-15 18:43:53 -0700396}