aboutsummaryrefslogtreecommitdiffstats
path: root/roms/openbios/libopenbios/fcode_load.c
diff options
context:
space:
mode:
authorfishsoupisgood <github@madingley.org>2019-04-29 01:17:54 +0100
committerfishsoupisgood <github@madingley.org>2019-05-27 03:43:43 +0100
commit3f2546b2ef55b661fd8dd69682b38992225e86f6 (patch)
tree65ca85f13617aee1dce474596800950f266a456c /roms/openbios/libopenbios/fcode_load.c
downloadqemu-master.tar.gz
qemu-master.tar.bz2
qemu-master.zip
Initial import of qemu-2.4.1HEADmaster
Diffstat (limited to 'roms/openbios/libopenbios/fcode_load.c')
-rw-r--r--roms/openbios/libopenbios/fcode_load.c109
1 files changed, 109 insertions, 0 deletions
diff --git a/roms/openbios/libopenbios/fcode_load.c b/roms/openbios/libopenbios/fcode_load.c
new file mode 100644
index 00000000..f4eb7bf3
--- /dev/null
+++ b/roms/openbios/libopenbios/fcode_load.c
@@ -0,0 +1,109 @@
+/*
+ * FCode boot loader
+ */
+
+#include "config.h"
+#include "kernel/kernel.h"
+#include "libopenbios/bindings.h"
+#include "libopenbios/fcode_load.h"
+#include "libopenbios/sys_info.h"
+#include "libc/diskio.h"
+#define printf printk
+#define debug printk
+
+static int fd;
+
+int
+is_fcode(unsigned char *fcode)
+{
+ return (fcode[0] == 0xf0 // start0
+ || fcode[0] == 0xf1 // start1
+ || fcode[0] == 0xf2 // start2
+ || fcode[0] == 0xf3 // start4
+ || fcode[0] == 0xfd); // version1
+}
+
+int
+fcode_load(ihandle_t dev)
+{
+ int retval = -1;
+ uint8_t fcode_header[8];
+ unsigned long start, size;
+ unsigned int offset;
+
+ /* Mark the saved-program-state as invalid */
+ feval("0 state-valid !");
+
+ fd = open_ih(dev);
+ if (fd == -1) {
+ goto out;
+ }
+
+ for (offset = 0; offset < 16 * 512; offset += 512) {
+ seek_io(fd, offset);
+ if (read_io(fd, &fcode_header, sizeof(fcode_header))
+ != sizeof(fcode_header)) {
+ debug("Can't read FCode header from ihandle " FMT_ucellx "\n", dev);
+ retval = LOADER_NOT_SUPPORT;
+ goto out;
+ }
+
+ if (is_fcode(fcode_header))
+ goto found;
+ }
+
+ debug("Not a bootable FCode image\n");
+ retval = LOADER_NOT_SUPPORT;
+ goto out;
+
+ found:
+ size = (fcode_header[4] << 24) | (fcode_header[5] << 16) |
+ (fcode_header[6] << 8) | fcode_header[7];
+
+ fword("load-base");
+ start = POP();
+
+ printf("\nLoading FCode image...\n");
+
+ seek_io(fd, offset);
+
+ if ((size_t)read_io(fd, (void *)start, size) != size) {
+ printf("Can't read file (size 0x%lx)\n", size);
+ goto out;
+ }
+
+ debug("Loaded %lu bytes\n", size);
+ debug("entry point is %#lx\n", start);
+
+ // Initialise saved-program-state
+ PUSH(start);
+ feval("saved-program-state >sps.entry !");
+ PUSH(size);
+ feval("saved-program-state >sps.file-size !");
+ feval("fcode saved-program-state >sps.file-type !");
+
+ feval("-1 state-valid !");
+
+out:
+ close_io(fd);
+ return retval;
+}
+
+void
+fcode_init_program(void)
+{
+ /* If the payload is Fcode then we execute it immediately */
+ ucell address;
+
+ fword("load-base");
+ address = POP();
+
+ if (!is_fcode((unsigned char *)address)) {
+ debug("Not a valid Fcode memory image\n");
+ return;
+ }
+
+ PUSH(address);
+ PUSH(1);
+ fword("byte-load");
+}