summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/stm/ao_boot.h3
-rw-r--r--src/stm/ao_boot_chain.c7
-rw-r--r--src/stm/ao_interrupt.c7
3 files changed, 11 insertions, 6 deletions
diff --git a/src/stm/ao_boot.h b/src/stm/ao_boot.h
index 862e8755..e0ed4de7 100644
--- a/src/stm/ao_boot.h
+++ b/src/stm/ao_boot.h
@@ -24,7 +24,8 @@ ao_boot_chain(uint32_t *base);
void
ao_boot_check_pin(void);
-void
+/* Return true to switch to application (if present) */
+int
ao_boot_check_chain(void);
void
diff --git a/src/stm/ao_boot_chain.c b/src/stm/ao_boot_chain.c
index 668f6e6d..6a3864a7 100644
--- a/src/stm/ao_boot_chain.c
+++ b/src/stm/ao_boot_chain.c
@@ -26,7 +26,7 @@ ao_boot_chain(uint32_t *base)
sp = base[0];
pc = base[1];
- if (0x08000100 <= pc && pc <= 0x08200000) {
+ if (0x08000100 <= pc && pc <= 0x08200000 && (pc & 1) == 1) {
asm ("mov sp, %0" : : "r" (sp));
asm ("mov lr, %0" : : "r" (pc));
asm ("bx lr");
@@ -44,14 +44,17 @@ struct ao_boot {
static struct ao_boot __attribute__ ((section(".boot"))) ao_boot;
-void
+int
ao_boot_check_chain(void)
{
if (ao_boot.signal == AO_BOOT_SIGNAL && ao_boot.check == AO_BOOT_CHECK) {
ao_boot.signal = 0;
ao_boot.check = 0;
+ if (ao_boot.base == 0)
+ return 0;
ao_boot_chain(ao_boot.base);
}
+ return 1;
}
void
diff --git a/src/stm/ao_interrupt.c b/src/stm/ao_interrupt.c
index 49156285..969e6a0f 100644
--- a/src/stm/ao_interrupt.c
+++ b/src/stm/ao_interrupt.c
@@ -42,10 +42,11 @@ const void *stm_interrupt_vector[];
void start(void)
{
#ifdef AO_BOOT_CHAIN
- ao_boot_check_chain();
-#endif
+ if (ao_boot_check_chain()) {
#ifdef AO_BOOT_PIN
- ao_boot_check_pin();
+ ao_boot_check_pin();
+#endif
+ }
#endif
/* Set interrupt vector table offset */
stm_nvic.vto = (uint32_t) &stm_interrupt_vector;