aboutsummaryrefslogtreecommitdiff
path: root/npcm8xx/image.c
diff options
context:
space:
mode:
Diffstat (limited to 'npcm8xx/image.c')
-rw-r--r--npcm8xx/image.c86
1 files changed, 86 insertions, 0 deletions
diff --git a/npcm8xx/image.c b/npcm8xx/image.c
new file mode 100644
index 0000000..f799774
--- /dev/null
+++ b/npcm8xx/image.c
@@ -0,0 +1,86 @@
+/*
+ * Boot image parsing and loading.
+ *
+ * Copyright 2022 Google LLC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#include <stdint.h>
+
+#define SPI0CS0 0x80000000
+#define CLK 0xf0801000
+#define FIU0 0xfb000000
+#define CLK_CLKSEL 0x04
+#define CLK_CLKSEL_DEFAULT 0x1f18fc9
+#define FIU_DRD_CFG 0x00
+
+/*
+ * This structure must reside at offset 0x100 in SRAM.
+ *
+ * See the Check_ROMCode_Status function in the Nuvoton bootblock:
+ * https://github.com/Nuvoton-Israel/bootblock/blob/master/Src/bootblock_main.c#L795
+ */
+struct rom_status {
+ uint8_t reserved[12];
+ uint8_t start_tag[8];
+ uint32_t status;
+} rom_status __attribute__((section(".data.rom_status"))) = {
+ .status = 0x21, /* SPI0 CS0 offset 0 */
+};
+
+extern void panic(const char *);
+
+static void reg_write(uintptr_t base, uintptr_t offset, uint32_t value)
+{
+ asm volatile("str %w0, [%1, %2]"
+ :
+ : "r"(value), "r"(base), "i"(offset)
+ : "memory");
+}
+
+static uint32_t image_read_u8(uintptr_t base, uintptr_t offset)
+{
+ return *(uint8_t *)(base + offset);
+}
+
+static uint32_t image_read_u32(uintptr_t base, uintptr_t offset)
+{
+ return *(uint32_t *)(base + offset);
+}
+
+void copy_boot_image(uintptr_t dest_addr, uintptr_t src_addr, int32_t len)
+{
+ uint32_t *dst = (uint32_t *)dest_addr;
+ uint32_t *src = (uint32_t *)src_addr;
+
+ while (len > 0) {
+ *dst++ = *src++;
+ len -= sizeof(*dst);
+ }
+}
+
+uintptr_t load_boot_image(void)
+{
+ uintptr_t dest_addr = 0x8000;
+
+ /* Set CLKSEL to similar values as NPCM7XX */
+ reg_write(CLK, CLK_CLKSEL, CLK_CLKSEL_DEFAULT);
+
+ /* Load the U-BOOT image to DRAM */
+ copy_boot_image(dest_addr, SPI0CS0 + 0x20200, 0xa6e80);
+ /* Set FIU to use 4 byte mode, similar to what TIP does in reality. */
+ reg_write(FIU0, FIU_DRD_CFG, 0x0301100b);
+
+ return dest_addr;
+}