summaryrefslogtreecommitdiff
path: root/src/lib/selfboot.c
diff options
context:
space:
mode:
authorRonald G. Minnich <rminnich@gmail.com>2018-10-24 15:46:51 -0700
committerRonald G. Minnich <rminnich@gmail.com>2018-10-25 16:57:51 +0000
commitc308554c10a33c977657b627c880cde60a590eb4 (patch)
treea030259c0bcd8557f792e693dc3dbf1eac655a33 /src/lib/selfboot.c
parentde332f35da72814722c053ae591b8dace0d54169 (diff)
downloadcoreboot-c308554c10a33c977657b627c880cde60a590eb4.tar.xz
selfboot: create selfboot_check function, remove check param
The selfboot function was changed at some point to take a parameter which meant "check the allocated descriptors to see if they target regions of real memory." The region check had to be buried deep in the last step of loading since that is where those descriptors were created and used. An issue with the use of the parameter was that it was not possible for compilers to easily divine whether the check code was used, and it was hence possible for the code, and its dependencies, to be compiled in even if never used (which caused problems for the rampayload code). Now that bounce buffers are gone, we can hoist the check code to the outermost level. Further, by creating a selfload_check and selfload function, we can make it easy for compilers to discard unused code: if selfload_check is never called, all the code it uses can be discarded too. Change-Id: Id5b3f450fd18480d54ffb6e395429fba71edcd77 Signed-off-by: Ronald G. Minnich <rminnich@gmail.com> Reviewed-on: https://review.coreboot.org/29259 Tested-by: build bot (Jenkins) <no-reply@coreboot.org> Reviewed-by: Aaron Durbin <adurbin@chromium.org> Reviewed-by: Paul Menzel <paulepanter@users.sourceforge.net>
Diffstat (limited to 'src/lib/selfboot.c')
-rw-r--r--src/lib/selfboot.c57
1 files changed, 47 insertions, 10 deletions
diff --git a/src/lib/selfboot.c b/src/lib/selfboot.c
index c9c8beefd8..efd2d3d5a9 100644
--- a/src/lib/selfboot.c
+++ b/src/lib/selfboot.c
@@ -30,6 +30,9 @@
#include <timestamp.h>
#include <cbmem.h>
+/* The type syntax for C is essentially unparsable. -- Rob Pike */
+typedef int (*checker_t)(struct cbfs_payload_segment *cbfssegs);
+
/* Decode a serialized cbfs payload segment
* from memory into native endianness.
*/
@@ -138,10 +141,26 @@ static int last_loadable_segment(struct cbfs_payload_segment *seg)
return read_be32(&(seg + 1)->type) == PAYLOAD_SEGMENT_ENTRY;
}
-static int load_payload_segments(
- struct cbfs_payload_segment *cbfssegs,
- int check_regions,
- uintptr_t *entry)
+static int check_payload_segments(struct cbfs_payload_segment *cbfssegs)
+{
+ uint8_t *dest;
+ size_t memsz;
+ struct cbfs_payload_segment *first_segment, *seg, segment;
+
+ for (first_segment = seg = cbfssegs;; ++seg) {
+ printk(BIOS_DEBUG, "Checking segment from ROM address 0x%p\n", seg);
+ cbfs_decode_payload_segment(&segment, seg);
+ dest = (uint8_t *)(uintptr_t)segment.load_addr;
+ memsz = segment.mem_len;
+ if (segment.type == PAYLOAD_SEGMENT_ENTRY)
+ break;
+ if (!segment_targets_usable_ram(dest, memsz))
+ return -1;
+ }
+ return 0;
+}
+
+static int load_payload_segments(struct cbfs_payload_segment *cbfssegs, uintptr_t *entry)
{
uint8_t *dest, *src;
size_t filesz, memsz;
@@ -202,8 +221,6 @@ static int load_payload_segments(
printk(BIOS_EMERG, "Bad segment type %x\n", segment.type);
return -1;
}
- if (check_regions && !segment_targets_usable_ram(dest, memsz))
- return -1;
/* Note that the 'seg + 1' is safe as we only call this
* function on "not the last" * items, since entry
* is always last. */
@@ -221,19 +238,29 @@ __weak int payload_arch_usable_ram_quirk(uint64_t start, uint64_t size)
return 0;
}
-bool selfload(struct prog *payload, bool check_regions)
+static void *selfprepare(struct prog *payload)
+{
+ void *data;
+ data = rdev_mmap_full(prog_rdev(payload));
+ return data;
+}
+
+static bool _selfload(struct prog *payload, checker_t f)
{
uintptr_t entry = 0;
struct cbfs_payload_segment *cbfssegs;
void *data;
- data = rdev_mmap_full(prog_rdev(payload));
-
+ data = selfprepare(payload);
if (data == NULL)
return false;
cbfssegs = &((struct cbfs_payload *)data)->segments;
- if (load_payload_segments(cbfssegs, check_regions, &entry))
+
+ if (f && f(cbfssegs))
+ goto out;
+
+ if (load_payload_segments(cbfssegs, &entry))
goto out;
printk(BIOS_SPEW, "Loaded segments\n");
@@ -248,3 +275,13 @@ out:
rdev_munmap(prog_rdev(payload), data);
return false;
}
+
+bool selfload_check(struct prog *payload)
+{
+ return _selfload(payload, check_payload_segments);
+}
+
+bool selfload(struct prog *payload)
+{
+ return _selfload(payload, NULL);
+}