bl_common.c 17.8 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
203
204
205
206
207
208
209
210
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
283
284
285
286
287
288
289
290
291
292
293
#if LOAD_IMAGE_V2

/*******************************************************************************
 * 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.
 *
 * Returns 0 on success, a negative error code otherwise.
 ******************************************************************************/
int load_image(unsigned int image_id, image_info_t *image_data)
{
	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;
	}

#if !TRUSTED_BOARD_BOOT
	/*
	 * File has been successfully loaded.
	 * 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.
	 */
	flush_dcache_range(image_base, image_size);
#endif /* TRUSTED_BOARD_BOOT */

	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;
}

294
295
296
static int load_auth_image_internal(unsigned int image_id,
				    image_info_t *image_data,
				    int is_parent_image)
297
298
299
300
301
302
303
304
305
{
	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) {
306
		rc = load_auth_image_internal(parent_id, image_data, 1);
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
		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) {
325
		/* Authentication error, zero memory and flush it right away. */
326
		zero_normalmem((void *)image_data->image_base,
327
328
329
330
331
332
333
334
335
336
		       image_data->image_size);
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
		return -EAUTH;
	}

	/*
	 * 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.
337
	 * Do it only for child images, not for the parents (certificates).
338
	 */
339
340
341
342
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
343
344
345
346
347
#endif /* TRUSTED_BOARD_BOOT */

	return 0;
}

348
349
350
351
352
353
354
355
356
/*******************************************************************************
 * 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)
{
357
358
359
360
361
362
363
	int err;

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

	return err;
364
365
}

366
367
#else /* LOAD_IMAGE_V2 */

368
/*******************************************************************************
369
370
371
372
373
374
375
376
377
378
379
380
381
382
 * 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).
 *
383
 * Returns 0 on success, a negative error code otherwise.
384
 ******************************************************************************/
385
int load_image(meminfo_t *mem_layout,
386
	       unsigned int image_id,
387
	       uintptr_t image_base,
388
389
	       image_info_t *image_data,
	       entry_point_info_t *entry_point_info)
390
{
391
392
393
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
394
395
	size_t image_size;
	size_t bytes_read;
396
	int io_result;
397
398

	assert(mem_layout != NULL);
399
	assert(image_data != NULL);
400
	assert(image_data->h.version == VERSION_1);
401
402

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

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

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

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

429
430
431
	/* 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)) {
432
433
		WARN("Failed to reserve region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
434
435
436
		dump_load_info(image_base, image_size, mem_layout);
		io_result = -ENOMEM;
		goto exit;
437
438
439
	}

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

447
448
449
	image_data->image_base = image_base;
	image_data->image_size = image_size;

450
451
452
453
	/*
	 * Update the memory usage info.
	 * This is done after the actual loading so that it is not updated when
	 * the load is unsuccessful.
454
455
	 * If the caller does not provide an entry point, bypass the memory
	 * reservation.
456
	 */
457
458
459
	if (entry_point_info != NULL) {
		reserve_mem(&mem_layout->free_base, &mem_layout->free_size,
				image_base, image_size);
460
		entry_point_info->pc = image_base;
461
	} else {
462
463
		INFO("Skip reserving region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
464
	}
465

466
#if !TRUSTED_BOARD_BOOT
467
	/*
468
	 * File has been successfully loaded.
469
470
471
472
	 * 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.
473
	 */
474
	flush_dcache_range(image_base, image_size);
475
#endif /* TRUSTED_BOARD_BOOT */
476

477
478
	INFO("Image id=%u loaded at address %p, size = 0x%zx\n", image_id,
		(void *) image_base, image_size);
479
480

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

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

488
	return io_result;
489
}
490

491
492
493
494
495
496
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)
497
498
499
500
501
502
503
504
505
{
	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) {
506
507
		rc = load_auth_image_internal(mem_layout, parent_id, image_base,
				     image_data, NULL, 1);
508
		if (rc != 0) {
509
510
511
512
513
514
515
516
			return rc;
		}
	}
#endif /* TRUSTED_BOARD_BOOT */

	/* Load the image */
	rc = load_image(mem_layout, image_id, image_base, image_data,
			entry_point_info);
517
518
	if (rc != 0) {
		return rc;
519
520
521
522
523
524
525
526
	}

#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) {
527
		/* Authentication error, zero memory and flush it right away. */
528
		zero_normalmem((void *)image_data->image_base,
529
530
531
		       image_data->image_size);
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
532
		return -EAUTH;
533
	}
534
535
536
537
	/*
	 * 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.
538
	 * Do it only for child images, not for the parents (certificates).
539
	 */
540
541
542
543
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
544
545
#endif /* TRUSTED_BOARD_BOOT */

546
	return 0;
547
}
548

549
550
551
552
553
554
555
556
557
558
559
560
561
/*******************************************************************************
 * 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)
{
562
563
564
565
566
567
568
569
	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;
570
571
}

572
573
#endif /* LOAD_IMAGE_V2 */

574
575
576
577
578
/*******************************************************************************
 * Print the content of an entry_point_info_t structure.
 ******************************************************************************/
void print_entry_point_info(const entry_point_info_t *ep_info)
{
579
580
	INFO("Entry point address = %p\n", (void *)ep_info->pc);
	INFO("SPSR = 0x%x\n", ep_info->spsr);
581
582
583
584
585
586
587
588
589

#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);
590
#ifndef AARCH32
591
592
593
594
	PRINT_IMAGE_ARG(4);
	PRINT_IMAGE_ARG(5);
	PRINT_IMAGE_ARG(6);
	PRINT_IMAGE_ARG(7);
595
#endif
596
597
#undef PRINT_IMAGE_ARG
}