bl_common.c 18.5 KB
Newer Older
1
/*
2
 * Copyright (c) 2013-2018, 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
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
#if TRUSTED_BOARD_BOOT
# ifdef DYN_DISABLE_AUTH
static int disable_auth;

/******************************************************************************
 * API to dynamically disable authentication. Only meant for development
 * systems. This is only invoked if DYN_DISABLE_AUTH is defined. This
 * capability is restricted to LOAD_IMAGE_V2.
 *****************************************************************************/
void dyn_disable_auth(void)
{
	INFO("Disabling authentication of images dynamically\n");
	disable_auth = 1;
}
# endif /* DYN_DISABLE_AUTH */

/******************************************************************************
 * Function to determine whether the authentication is disabled dynamically.
 *****************************************************************************/
static int dyn_is_auth_disabled(void)
{
# ifdef DYN_DISABLE_AUTH
	return disable_auth;
# else
	return 0;
# endif
}
#endif /* TRUSTED_BOARD_BOOT */

49
uintptr_t page_align(uintptr_t value, unsigned dir)
50
51
{
	/* Round up the limit to the next page boundary */
52
53
	if (value & (PAGE_SIZE - 1)) {
		value &= ~(PAGE_SIZE - 1);
54
		if (dir == UP)
55
			value += PAGE_SIZE;
56
57
58
59
60
	}

	return value;
}

61
62
63
/******************************************************************************
 * Determine whether the memory region delimited by 'addr' and 'size' is free,
 * given the extents of free memory.
64
65
 * Return 1 if it is free, 0 if it is not free or if the input values are
 * invalid.
66
 *****************************************************************************/
67
int is_mem_free(uintptr_t free_base, size_t free_size,
68
		uintptr_t addr, size_t size)
69
{
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
	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);
102
}
103

104
#if !LOAD_IMAGE_V2
105
106
107
108
109
110
/******************************************************************************
 * 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'.
 *****************************************************************************/
111
112
113
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)
114
{
115
	size_t top_chunk_size, bottom_chunk_size;
116

117
118
119
120
	assert(mem_start <= submem_start);
	assert(submem_start <= submem_end);
	assert(submem_end <= mem_end);
	assert(small_chunk_size != NULL);
121

122
123
124
125
126
127
	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;
128
	} else {
129
130
		*small_chunk_size = bottom_chunk_size;
		return BOTTOM;
131
	}
132
}
133

134
135
136
137
/******************************************************************************
 * 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.
138
139
 * The caller must ensure the memory to reserve is free and that the addresses
 * and sizes passed in arguments are sane.
140
 *****************************************************************************/
141
142
void reserve_mem(uintptr_t *free_base, size_t *free_size,
		 uintptr_t addr, size_t size)
143
144
145
146
{
	size_t discard_size;
	size_t reserved_size;
	unsigned int pos;
147

148
149
150
151
	assert(free_base != NULL);
	assert(free_size != NULL);
	assert(is_mem_free(*free_base, *free_size, addr, size));

152
153
154
155
156
157
158
	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),
159
160
161
162
163
164
165
166
			     &discard_size);

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

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

167
	VERBOSE("Reserved 0x%zx bytes (discarded 0x%zx bytes %s)\n",
168
169
	     reserved_size, discard_size,
	     pos == TOP ? "above" : "below");
170
171
}

172
173
static void dump_load_info(uintptr_t image_load_addr,
			   size_t image_size,
174
			   const meminfo_t *mem_layout)
175
{
176
177
	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
178
	INFO("Current memory layout:\n");
179
180
181
182
	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);
183
}
184
#endif /* LOAD_IMAGE_V2 */
185

Ryan Harkin's avatar
Ryan Harkin committed
186
/* Generic function to return the size of an image */
Daniel Boulby's avatar
Daniel Boulby committed
187
size_t get_image_size(unsigned int image_id)
Ryan Harkin's avatar
Ryan Harkin committed
188
{
189
190
191
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
Ryan Harkin's avatar
Ryan Harkin committed
192
	size_t image_size = 0;
193
	int io_result;
Ryan Harkin's avatar
Ryan Harkin committed
194
195

	/* Obtain a reference to the image by querying the platform layer */
196
	io_result = plat_get_image_source(image_id, &dev_handle, &image_spec);
197
	if (io_result != 0) {
198
199
		WARN("Failed to obtain reference to image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
200
201
202
203
204
		return 0;
	}

	/* Attempt to access the image */
	io_result = io_open(dev_handle, image_spec, &image_handle);
205
	if (io_result != 0) {
206
207
		WARN("Failed to access image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
208
209
210
211
212
		return 0;
	}

	/* Find the size of the image */
	io_result = io_size(image_handle, &image_size);
213
	if ((io_result != 0) || (image_size == 0)) {
214
215
		WARN("Failed to determine the size of the image id=%u (%i)\n",
			image_id, io_result);
Ryan Harkin's avatar
Ryan Harkin committed
216
217
218
219
220
221
222
223
224
225
226
227
	}
	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;
}
228

229
230
231
#if LOAD_IMAGE_V2

/*******************************************************************************
232
 * Internal function to load an image at a specific address given
233
234
235
236
237
238
 * 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.
 ******************************************************************************/
239
static int load_image(unsigned int image_id, image_info_t *image_data)
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
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
{
	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;
}

312
313
314
static int load_auth_image_internal(unsigned int image_id,
				    image_info_t *image_data,
				    int is_parent_image)
315
316
317
318
{
	int rc;

#if TRUSTED_BOARD_BOOT
319
320
321
322
323
324
325
326
327
328
	if (dyn_is_auth_disabled() == 0) {
		unsigned int parent_id;

		/* Use recursion to authenticate parent images */
		rc = auth_mod_get_parent_id(image_id, &parent_id);
		if (rc == 0) {
			rc = load_auth_image_internal(parent_id, image_data, 1);
			if (rc != 0) {
				return rc;
			}
329
330
331
332
333
334
335
336
337
338
339
		}
	}
#endif /* TRUSTED_BOARD_BOOT */

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

#if TRUSTED_BOARD_BOOT
340
341
342
343
344
345
346
347
348
349
350
351
352
	if (dyn_is_auth_disabled() == 0) {
		/* Authenticate it */
		rc = auth_mod_verify_img(image_id,
					 (void *)image_data->image_base,
					 image_data->image_size);
		if (rc != 0) {
			/* Authentication error, zero memory and flush it right away. */
			zero_normalmem((void *)image_data->image_base,
			       image_data->image_size);
			flush_dcache_range(image_data->image_base,
					   image_data->image_size);
			return -EAUTH;
		}
353
	}
354
#endif /* TRUSTED_BOARD_BOOT */
355
356
357

	/*
	 * Flush the image to main memory so that it can be executed later by
358
359
360
	 * 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).
361
	 */
362
363
364
365
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
366

367
368
369
370

	return 0;
}

371
372
373
374
375
376
377
378
379
/*******************************************************************************
 * 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)
{
380
381
382
383
384
385
386
	int err;

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

	return err;
387
388
}

389
390
#else /* LOAD_IMAGE_V2 */

391
/*******************************************************************************
392
393
394
395
396
397
398
399
400
401
402
403
404
405
 * 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).
 *
406
 * Returns 0 on success, a negative error code otherwise.
407
 ******************************************************************************/
408
int load_image(meminfo_t *mem_layout,
409
	       unsigned int image_id,
410
	       uintptr_t image_base,
411
412
	       image_info_t *image_data,
	       entry_point_info_t *entry_point_info)
413
{
414
415
416
	uintptr_t dev_handle;
	uintptr_t image_handle;
	uintptr_t image_spec;
417
418
	size_t image_size;
	size_t bytes_read;
419
	int io_result;
420
421

	assert(mem_layout != NULL);
422
	assert(image_data != NULL);
423
	assert(image_data->h.version == VERSION_1);
424
425

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

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

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

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

452
453
454
	/* 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)) {
455
456
		WARN("Failed to reserve region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
457
458
459
		dump_load_info(image_base, image_size, mem_layout);
		io_result = -ENOMEM;
		goto exit;
460
461
462
	}

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

470
471
472
	image_data->image_base = image_base;
	image_data->image_size = image_size;

473
474
475
476
	/*
	 * Update the memory usage info.
	 * This is done after the actual loading so that it is not updated when
	 * the load is unsuccessful.
477
478
	 * If the caller does not provide an entry point, bypass the memory
	 * reservation.
479
	 */
480
481
482
	if (entry_point_info != NULL) {
		reserve_mem(&mem_layout->free_base, &mem_layout->free_size,
				image_base, image_size);
483
		entry_point_info->pc = image_base;
484
	} else {
485
486
		INFO("Skip reserving region [base = %p, size = 0x%zx]\n",
		     (void *) image_base, image_size);
487
	}
488

489
#if !TRUSTED_BOARD_BOOT
490
	/*
491
	 * File has been successfully loaded.
492
493
494
495
	 * 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.
496
	 */
497
	flush_dcache_range(image_base, image_size);
498
#endif /* TRUSTED_BOARD_BOOT */
499

500
501
	INFO("Image id=%u loaded at address %p, size = 0x%zx\n", image_id,
		(void *) image_base, image_size);
502
503

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

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

511
	return io_result;
512
}
513

514
515
516
517
518
519
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)
520
521
522
523
524
525
526
527
528
{
	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) {
529
530
		rc = load_auth_image_internal(mem_layout, parent_id, image_base,
				     image_data, NULL, 1);
531
		if (rc != 0) {
532
533
534
535
536
537
538
539
			return rc;
		}
	}
#endif /* TRUSTED_BOARD_BOOT */

	/* Load the image */
	rc = load_image(mem_layout, image_id, image_base, image_data,
			entry_point_info);
540
541
	if (rc != 0) {
		return rc;
542
543
544
545
546
547
548
549
	}

#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) {
550
		/* Authentication error, zero memory and flush it right away. */
551
		zero_normalmem((void *)image_data->image_base,
552
553
554
		       image_data->image_size);
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
555
		return -EAUTH;
556
	}
557
558
559
560
	/*
	 * 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.
561
	 * Do it only for child images, not for the parents (certificates).
562
	 */
563
564
565
566
	if (!is_parent_image) {
		flush_dcache_range(image_data->image_base,
				   image_data->image_size);
	}
567
568
#endif /* TRUSTED_BOARD_BOOT */

569
	return 0;
570
}
571

572
573
574
575
576
577
578
579
580
581
582
583
584
/*******************************************************************************
 * 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)
{
585
586
587
588
589
590
591
592
	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;
593
594
}

595
596
#endif /* LOAD_IMAGE_V2 */

597
598
599
600
601
/*******************************************************************************
 * Print the content of an entry_point_info_t structure.
 ******************************************************************************/
void print_entry_point_info(const entry_point_info_t *ep_info)
{
602
603
	INFO("Entry point address = %p\n", (void *)ep_info->pc);
	INFO("SPSR = 0x%x\n", ep_info->spsr);
604
605
606
607
608
609
610
611
612

#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);
613
#ifndef AARCH32
614
615
616
617
	PRINT_IMAGE_ARG(4);
	PRINT_IMAGE_ARG(5);
	PRINT_IMAGE_ARG(6);
	PRINT_IMAGE_ARG(7);
618
#endif
619
620
#undef PRINT_IMAGE_ARG
}