bl_common.c 17.5 KB
Newer Older
1
/*
2
 * Copyright (c) 2013-2017, ARM Limited and Contributors. All rights reserved.
3
 *
dp-arm's avatar
dp-arm committed
4
 * SPDX-License-Identifier: BSD-3-Clause
5
6
 */

7
#include <arch.h>
8
#include <arch_helpers.h>
9
#include <assert.h>
10
#include <auth_mod.h>
11
#include <bl_common.h>
12
#include <debug.h>
13
#include <errno.h>
14
15
#include <io_storage.h>
#include <platform.h>
16
#include <string.h>
17
#include <utils.h>
18
#include <xlat_tables_defs.h>
19

20
uintptr_t page_align(uintptr_t value, unsigned dir)
21
22
{
	/* Round up the limit to the next page boundary */
23
24
	if (value & (PAGE_SIZE - 1)) {
		value &= ~(PAGE_SIZE - 1);
25
		if (dir == UP)
26
			value += PAGE_SIZE;
27
28
29
30
31
	}

	return value;
}

32
33
34
/******************************************************************************
 * Determine whether the memory region delimited by 'addr' and 'size' is free,
 * given the extents of free memory.
35
36
 * Return 1 if it is free, 0 if it is not free or if the input values are
 * invalid.
37
 *****************************************************************************/
38
int is_mem_free(uintptr_t free_base, size_t free_size,
39
		uintptr_t addr, size_t size)
40
{
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
	uintptr_t free_end, requested_end;

	/*
	 * Handle corner cases first.
	 *
	 * The order of the 2 tests is important, because if there's no space
	 * left (i.e. free_size == 0) but we don't ask for any memory
	 * (i.e. size == 0) then we should report that the memory is free.
	 */
	if (size == 0)
		return 1;	/* A zero-byte region is always free */
	if (free_size == 0)
		return 0;

	/*
	 * Check that the end addresses don't overflow.
	 * If they do, consider that this memory region is not free, as this
	 * is an invalid scenario.
	 */
	if (check_uptr_overflow(free_base, free_size - 1))
		return 0;
	free_end = free_base + (free_size - 1);

	if (check_uptr_overflow(addr, size - 1))
		return 0;
	requested_end = addr + (size - 1);

	/*
	 * Finally, check that the requested memory region lies within the free
	 * region.
	 */
	return (addr >= free_base) && (requested_end <= free_end);
73
}
74

75
#if !LOAD_IMAGE_V2
76
77
78
79
80
81
/******************************************************************************
 * Inside a given memory region, determine whether a sub-region of memory is
 * closer from the top or the bottom of the encompassing region. Return the
 * size of the smallest chunk of free memory surrounding the sub-region in
 * 'small_chunk_size'.
 *****************************************************************************/
82
83
84
static unsigned int choose_mem_pos(uintptr_t mem_start, uintptr_t mem_end,
				  uintptr_t submem_start, uintptr_t submem_end,
				  size_t *small_chunk_size)
85
{
86
	size_t top_chunk_size, bottom_chunk_size;
87

88
89
90
91
	assert(mem_start <= submem_start);
	assert(submem_start <= submem_end);
	assert(submem_end <= mem_end);
	assert(small_chunk_size != NULL);
92

93
94
95
96
97
98
	top_chunk_size = mem_end - submem_end;
	bottom_chunk_size = submem_start - mem_start;

	if (top_chunk_size < bottom_chunk_size) {
		*small_chunk_size = top_chunk_size;
		return TOP;
99
	} else {
100
101
		*small_chunk_size = bottom_chunk_size;
		return BOTTOM;
102
	}
103
}
104

105
106
107
108
/******************************************************************************
 * Reserve the memory region delimited by 'addr' and 'size'. The extents of free
 * memory are passed in 'free_base' and 'free_size' and they will be updated to
 * reflect the memory usage.
109
110
 * The caller must ensure the memory to reserve is free and that the addresses
 * and sizes passed in arguments are sane.
111
 *****************************************************************************/
112
113
void reserve_mem(uintptr_t *free_base, size_t *free_size,
		 uintptr_t addr, size_t size)
114
115
116
117
{
	size_t discard_size;
	size_t reserved_size;
	unsigned int pos;
118

119
120
121
122
	assert(free_base != NULL);
	assert(free_size != NULL);
	assert(is_mem_free(*free_base, *free_size, addr, size));

123
124
125
126
127
128
129
	if (size == 0) {
		WARN("Nothing to allocate, requested size is zero\n");
		return;
	}

	pos = choose_mem_pos(*free_base, *free_base + (*free_size - 1),
			     addr, addr + (size - 1),
130
131
132
133
134
135
136
137
			     &discard_size);

	reserved_size = size + discard_size;
	*free_size -= reserved_size;

	if (pos == BOTTOM)
		*free_base = addr + size;

138
	VERBOSE("Reserved 0x%zx bytes (discarded 0x%zx bytes %s)\n",
139
140
	     reserved_size, discard_size,
	     pos == TOP ? "above" : "below");
141
142
}

143
144
static void dump_load_info(uintptr_t image_load_addr,
			   size_t image_size,
145
			   const meminfo_t *mem_layout)
146
{
147
148
	INFO("Trying to load image at address %p, size = 0x%zx\n",
		(void *)image_load_addr, image_size);
Dan Handley's avatar
Dan Handley committed
149
	INFO("Current memory layout:\n");
150
151
152
153
	INFO("  total region = [base = %p, size = 0x%zx]\n",
		(void *) mem_layout->total_base, mem_layout->total_size);
	INFO("  free region = [base = %p, size = 0x%zx]\n",
		(void *) mem_layout->free_base, mem_layout->free_size);
154
}
155
#endif /* LOAD_IMAGE_V2 */
156

Ryan Harkin's avatar
Ryan Harkin committed
157
/* Generic function to return the size of an image */
158
size_t image_size(unsigned int image_id)
Ryan Harkin's avatar
Ryan Harkin committed
159
{
160
161
162
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
Ryan Harkin's avatar
Ryan Harkin committed
163
	size_t image_size = 0;
164
	int io_result;
Ryan Harkin's avatar
Ryan Harkin committed
165
166

	/* Obtain a reference to the image by querying the platform layer */
167
	io_result = plat_get_image_source(image_id, &dev_handle, &image_spec);
168
	if (io_result != 0) {
169
170
		WARN("Failed to obtain reference to image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
171
172
173
174
175
		return 0;
	}

	/* Attempt to access the image */
	io_result = io_open(dev_handle, image_spec, &image_handle);
176
	if (io_result != 0) {
177
178
		WARN("Failed to access image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
179
180
181
182
183
		return 0;
	}

	/* Find the size of the image */
	io_result = io_size(image_handle, &image_size);
184
	if ((io_result != 0) || (image_size == 0)) {
185
186
		WARN("Failed to determine the size of the image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
187
188
189
190
191
192
193
194
195
196
197
198
	}
	io_result = io_close(image_handle);
	/* Ignore improbable/unrecoverable error in 'close' */

	/* TODO: Consider maintaining open device connection from this
	 * bootloader stage
	 */
	io_result = io_dev_close(dev_handle);
	/* Ignore improbable/unrecoverable error in 'dev_close' */

	return image_size;
}
199

200
201
202
#if LOAD_IMAGE_V2

/*******************************************************************************
203
 * Internal function to load an image at a specific address given
204
205
206
207
208
209
 * an image ID and extents of free memory.
 *
 * If the load is successful then the image information is updated.
 *
 * Returns 0 on success, a negative error code otherwise.
 ******************************************************************************/
210
static int load_image(unsigned int image_id, image_info_t *image_data)
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
{
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
	uintptr_t image_base;
	size_t image_size;
	size_t bytes_read;
	int io_result;

	assert(image_data != NULL);
	assert(image_data->h.version >= VERSION_2);

	image_base = image_data->image_base;

	/* Obtain a reference to the image by querying the platform layer */
	io_result = plat_get_image_source(image_id, &dev_handle, &image_spec);
	if (io_result != 0) {
		WARN("Failed to obtain reference to image id=%u (%i)\n",
			image_id, io_result);
		return io_result;
	}

	/* Attempt to access the image */
	io_result = io_open(dev_handle, image_spec, &image_handle);
	if (io_result != 0) {
		WARN("Failed to access image id=%u (%i)\n",
			image_id, io_result);
		return io_result;
	}

	INFO("Loading image id=%u at address %p\n", image_id,
		(void *) image_base);

	/* Find the size of the image */
	io_result = io_size(image_handle, &image_size);
	if ((io_result != 0) || (image_size == 0)) {
		WARN("Failed to determine the size of the image id=%u (%i)\n",
			image_id, io_result);
		goto exit;
	}

	/* Check that the image size to load is within limit */
	if (image_size > image_data->image_max_size) {
		WARN("Image id=%u size out of bounds\n", image_id);
		io_result = -EFBIG;
		goto exit;
	}

	image_data->image_size = image_size;

	/* We have enough space so load the image now */
	/* TODO: Consider whether to try to recover/retry a partially successful read */
	io_result = io_read(image_handle, image_base, image_size, &bytes_read);
	if ((io_result != 0) || (bytes_read < image_size)) {
		WARN("Failed to load image id=%u (%i)\n", image_id, io_result);
		goto exit;
	}

	INFO("Image id=%u loaded: %p - %p\n", image_id, (void *) image_base,
	     (void *) (image_base + image_size));

exit:
	io_close(image_handle);
	/* Ignore improbable/unrecoverable error in 'close' */

	/* TODO: Consider maintaining open device connection from this bootloader stage */
	io_dev_close(dev_handle);
	/* Ignore improbable/unrecoverable error in 'dev_close' */

	return io_result;
}

283
284
285
static int load_auth_image_internal(unsigned int image_id,
				    image_info_t *image_data,
				    int is_parent_image)
286
287
288
289
290
291
292
293
294
{
	int rc;

#if TRUSTED_BOARD_BOOT
	unsigned int parent_id;

	/* Use recursion to authenticate parent images */
	rc = auth_mod_get_parent_id(image_id, &parent_id);
	if (rc == 0) {
295
		rc = load_auth_image_internal(parent_id, image_data, 1);
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
		if (rc != 0) {
			return rc;
		}
	}
#endif /* TRUSTED_BOARD_BOOT */

	/* Load the image */
	rc = load_image(image_id, image_data);
	if (rc != 0) {
		return rc;
	}

#if TRUSTED_BOARD_BOOT
	/* Authenticate it */
	rc = auth_mod_verify_img(image_id,
				 (void *)image_data->image_base,
				 image_data->image_size);
	if (rc != 0) {
314
		/* Authentication error, zero memory and flush it right away. */
315
		zero_normalmem((void *)image_data->image_base,
316
317
318
319
320
		       image_data->image_size);
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
		return -EAUTH;
	}
321
#endif /* TRUSTED_BOARD_BOOT */
322
323
324

	/*
	 * Flush the image to main memory so that it can be executed later by
325
326
327
	 * any CPU, regardless of cache and MMU state. If TBB is enabled, then
	 * the file has been successfully loaded and authenticated and flush
	 * only for child images, not for the parents (certificates).
328
	 */
329
330
331
332
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
333

334
335
336
337

	return 0;
}

338
339
340
341
342
343
344
345
346
/*******************************************************************************
 * Generic function to load and authenticate an image. The image is actually
 * loaded by calling the 'load_image()' function. Therefore, it returns the
 * same error codes if the loading operation failed, or -EAUTH if the
 * authentication failed. In addition, this function uses recursion to
 * authenticate the parent images up to the root of trust.
 ******************************************************************************/
int load_auth_image(unsigned int image_id, image_info_t *image_data)
{
347
348
349
350
351
352
353
	int err;

	do {
		err = load_auth_image_internal(image_id, image_data, 0);
	} while (err != 0 && plat_try_next_boot_source());

	return err;
354
355
}

356
357
#else /* LOAD_IMAGE_V2 */

358
/*******************************************************************************
359
360
361
362
363
364
365
366
367
368
369
370
371
372
 * Generic function to load an image at a specific address given an image ID and
 * extents of free memory.
 *
 * If the load is successful then the image information is updated.
 *
 * If the entry_point_info argument is not NULL then this function also updates:
 * - the memory layout to mark the memory as reserved;
 * - the entry point information.
 *
 * The caller might pass a NULL pointer for the entry point if they are not
 * interested in this information. This is typically the case for non-executable
 * images (e.g. certificates) and executable images that won't ever be executed
 * on the application processor (e.g. additional microcontroller firmware).
 *
373
 * Returns 0 on success, a negative error code otherwise.
374
 ******************************************************************************/
375
int load_image(meminfo_t *mem_layout,
376
	       unsigned int image_id,
377
	       uintptr_t image_base,
378
379
	       image_info_t *image_data,
	       entry_point_info_t *entry_point_info)
380
{
381
382
383
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
384
385
	size_t image_size;
	size_t bytes_read;
386
	int io_result;
387
388

	assert(mem_layout != NULL);
389
	assert(image_data != NULL);
390
	assert(image_data->h.version == VERSION_1);
391
392

	/* Obtain a reference to the image by querying the platform layer */
393
	io_result = plat_get_image_source(image_id, &dev_handle, &image_spec);
394
	if (io_result != 0) {
395
396
		WARN("Failed to obtain reference to image id=%u (%i)\n",
			image_id, io_result);
397
		return io_result;
398
	}
399

400
401
	/* Attempt to access the image */
	io_result = io_open(dev_handle, image_spec, &image_handle);
402
	if (io_result != 0) {
403
404
		WARN("Failed to access image id=%u (%i)\n",
			image_id, io_result);
405
		return io_result;
406
407
	}

408
409
	INFO("Loading image id=%u at address %p\n", image_id,
		(void *) image_base);
410

411
412
	/* Find the size of the image */
	io_result = io_size(image_handle, &image_size);
413
	if ((io_result != 0) || (image_size == 0)) {
414
415
		WARN("Failed to determine the size of the image id=%u (%i)\n",
			image_id, io_result);
416
		goto exit;
417
418
	}

419
420
421
	/* Check that the memory where the image will be loaded is free */
	if (!is_mem_free(mem_layout->free_base, mem_layout->free_size,
			 image_base, image_size)) {
422
423
		WARN("Failed to reserve region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
424
425
426
		dump_load_info(image_base, image_size, mem_layout);
		io_result = -ENOMEM;
		goto exit;
427
428
429
	}

	/* We have enough space so load the image now */
430
	/* TODO: Consider whether to try to recover/retry a partially successful read */
431
	io_result = io_read(image_handle, image_base, image_size, &bytes_read);
432
	if ((io_result != 0) || (bytes_read < image_size)) {
433
		WARN("Failed to load image id=%u (%i)\n", image_id, io_result);
434
		goto exit;
435
436
	}

437
438
439
	image_data->image_base = image_base;
	image_data->image_size = image_size;

440
441
442
443
	/*
	 * Update the memory usage info.
	 * This is done after the actual loading so that it is not updated when
	 * the load is unsuccessful.
444
445
	 * If the caller does not provide an entry point, bypass the memory
	 * reservation.
446
	 */
447
448
449
	if (entry_point_info != NULL) {
		reserve_mem(&mem_layout->free_base, &mem_layout->free_size,
				image_base, image_size);
450
		entry_point_info->pc = image_base;
451
	} else {
452
453
		INFO("Skip reserving region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
454
	}
455

456
#if !TRUSTED_BOARD_BOOT
457
	/*
458
	 * File has been successfully loaded.
459
460
461
462
	 * Flush the image to main memory so that it can be executed later by
	 * any CPU, regardless of cache and MMU state.
	 * When TBB is enabled the image is flushed later, after image
	 * authentication.
463
	 */
464
	flush_dcache_range(image_base, image_size);
465
#endif /* TRUSTED_BOARD_BOOT */
466

467
468
	INFO("Image id=%u loaded at address %p, size = 0x%zx\n", image_id,
		(void *) image_base, image_size);
469
470

exit:
471
	io_close(image_handle);
472
473
474
	/* Ignore improbable/unrecoverable error in 'close' */

	/* TODO: Consider maintaining open device connection from this bootloader stage */
475
	io_dev_close(dev_handle);
476
	/* Ignore improbable/unrecoverable error in 'dev_close' */
477

478
	return io_result;
479
}
480

481
482
483
484
485
486
static int load_auth_image_internal(meminfo_t *mem_layout,
				    unsigned int image_id,
				    uintptr_t image_base,
				    image_info_t *image_data,
				    entry_point_info_t *entry_point_info,
				    int is_parent_image)
487
488
489
490
491
492
493
494
495
{
	int rc;

#if TRUSTED_BOARD_BOOT
	unsigned int parent_id;

	/* Use recursion to authenticate parent images */
	rc = auth_mod_get_parent_id(image_id, &parent_id);
	if (rc == 0) {
496
497
		rc = load_auth_image_internal(mem_layout, parent_id, image_base,
				     image_data, NULL, 1);
498
		if (rc != 0) {
499
500
501
502
503
504
505
506
			return rc;
		}
	}
#endif /* TRUSTED_BOARD_BOOT */

	/* Load the image */
	rc = load_image(mem_layout, image_id, image_base, image_data,
			entry_point_info);
507
508
	if (rc != 0) {
		return rc;
509
510
511
512
513
514
515
516
	}

#if TRUSTED_BOARD_BOOT
	/* Authenticate it */
	rc = auth_mod_verify_img(image_id,
				 (void *)image_data->image_base,
				 image_data->image_size);
	if (rc != 0) {
517
		/* Authentication error, zero memory and flush it right away. */
518
		zero_normalmem((void *)image_data->image_base,
519
520
521
		       image_data->image_size);
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
522
		return -EAUTH;
523
	}
524
525
526
527
	/*
	 * File has been successfully loaded and authenticated.
	 * Flush the image to main memory so that it can be executed later by
	 * any CPU, regardless of cache and MMU state.
528
	 * Do it only for child images, not for the parents (certificates).
529
	 */
530
531
532
533
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
534
535
#endif /* TRUSTED_BOARD_BOOT */

536
	return 0;
537
}
538

539
540
541
542
543
544
545
546
547
548
549
550
551
/*******************************************************************************
 * Generic function to load and authenticate an image. The image is actually
 * loaded by calling the 'load_image()' function. Therefore, it returns the
 * same error codes if the loading operation failed, or -EAUTH if the
 * authentication failed. In addition, this function uses recursion to
 * authenticate the parent images up to the root of trust.
 ******************************************************************************/
int load_auth_image(meminfo_t *mem_layout,
		    unsigned int image_id,
		    uintptr_t image_base,
		    image_info_t *image_data,
		    entry_point_info_t *entry_point_info)
{
552
553
554
555
556
557
558
559
	int err;

	do {
		err = load_auth_image_internal(mem_layout, image_id, image_base,
					       image_data, entry_point_info, 0);
	} while (err != 0 && plat_try_next_boot_source());

	return err;
560
561
}

562
563
#endif /* LOAD_IMAGE_V2 */

564
565
566
567
568
/*******************************************************************************
 * Print the content of an entry_point_info_t structure.
 ******************************************************************************/
void print_entry_point_info(const entry_point_info_t *ep_info)
{
569
570
	INFO("Entry point address = %p\n", (void *)ep_info->pc);
	INFO("SPSR = 0x%x\n", ep_info->spsr);
571
572
573
574
575
576
577
578
579

#define PRINT_IMAGE_ARG(n)					\
	VERBOSE("Argument #" #n " = 0x%llx\n",			\
		(unsigned long long) ep_info->args.arg##n)

	PRINT_IMAGE_ARG(0);
	PRINT_IMAGE_ARG(1);
	PRINT_IMAGE_ARG(2);
	PRINT_IMAGE_ARG(3);
580
#ifndef AARCH32
581
582
583
584
	PRINT_IMAGE_ARG(4);
	PRINT_IMAGE_ARG(5);
	PRINT_IMAGE_ARG(6);
	PRINT_IMAGE_ARG(7);
585
#endif
586
587
#undef PRINT_IMAGE_ARG
}