bl_common.c 17.6 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
357
358
359
/*******************************************************************************
 * 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)
{
	return load_auth_image_internal(image_id, image_data, 0);
}

360
361
#else /* LOAD_IMAGE_V2 */

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

	assert(mem_layout != NULL);
393
	assert(image_data != NULL);
394
	assert(image_data->h.version == VERSION_1);
395
396

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

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

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

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

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

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

441
442
443
	image_data->image_base = image_base;
	image_data->image_size = image_size;

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

460
#if !TRUSTED_BOARD_BOOT
461
	/*
462
	 * File has been successfully loaded.
463
464
465
466
	 * 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.
467
	 */
468
	flush_dcache_range(image_base, image_size);
469
#endif /* TRUSTED_BOARD_BOOT */
470

471
472
	INFO("Image id=%u loaded at address %p, size = 0x%zx\n", image_id,
		(void *) image_base, image_size);
473
474

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

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

482
	return io_result;
483
}
484

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

	/* Load the image */
	rc = load_image(mem_layout, image_id, image_base, image_data,
			entry_point_info);
511
512
	if (rc != 0) {
		return rc;
513
514
515
516
517
518
519
520
	}

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

540
	return 0;
541
}
542

543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
/*******************************************************************************
 * 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)
{
	return load_auth_image_internal(mem_layout, image_id, image_base,
					image_data, entry_point_info, 0);
}

560
561
#endif /* LOAD_IMAGE_V2 */

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

#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);
578
#ifndef AARCH32
579
580
581
582
	PRINT_IMAGE_ARG(4);
	PRINT_IMAGE_ARG(5);
	PRINT_IMAGE_ARG(6);
	PRINT_IMAGE_ARG(7);
583
#endif
584
585
#undef PRINT_IMAGE_ARG
}