Skip to content
bootloader.c 5.16 KiB
Newer Older
/*******************************************************************************
f4grx's avatar
f4grx committed
 * configs/hn70ap/src/bootloader.c
f4grx's avatar
f4grx committed
 *   Copyright (C) 2016-2018 Sebastien Lorquet. All rights reserved.
 *   Author: Sebastien Lorquet <sebastien@lorquet.fr>
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 3. Neither the name NuttX nor the names of its contributors may be
 *    used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 *
 ******************************************************************************/
/*******************************************************************************
 * Pre-processor Definitions
 ******************************************************************************/
f4grx's avatar
f4grx committed
#include "bootloader.h"

#define BOOT_STACKSIZE  (16 * 1024)
#define BOOT_PERIPHERAL_INTERRUPTS   16

/*******************************************************************************
 * Public Functions
 ******************************************************************************/

/* Chip-specific entrypoint */

extern void __boot_start(void);

/* Common exception entrypoint */

extern void __boot_exception_common(void);

/*******************************************************************************
 * Public data
 ******************************************************************************/
/* Provided by the linker script to indicate the end of the BSS
 * and start of text */
f4grx's avatar
f4grx committed
extern const uint32_t _ebss;
extern const uint32_t _bootstack;

/* The v7m vector table consists of an array of function pointers, with the first
 * slot (vector zero) used to hold the initial stack pointer.
 *
 * As all exceptions (interrupts) are routed via exception_common, we just need to
 * fill this array with pointers to it.
 *
 * Note that the [ ... ] designated initialiser is a GCC extension.
 */

unsigned __boot_vectors[] __attribute__((section(".boot.vectors"))) =
{
f4grx's avatar
f4grx committed
  (unsigned)&_bootstack + BOOT_STACKSIZE, /* Initial stack */
  (unsigned)&__boot_start,                /* Reset exception handler */

  /* Vectors 2 - n point directly at the generic handler */

  [2 ... (15 + BOOT_PERIPHERAL_INTERRUPTS)] = (unsigned)&__boot_exception_common
};

/* -------------------------------------------------------------------------- */
f4grx's avatar
f4grx committed
BOOTCODE __attribute__((naked, noreturn)) void __boot_app(void)
  {
    __asm__ __volatile__ ("\t"
        /* load SP (from 08004000) */

        "movw r0, #0x4000              \n\t"
        "movt r0, #0x0800              \n\t"
        "ldr  sp, [r0]                 \n\t"

        /* load LR (from 08004004) */

        "movw r0, #0x4004              \n\t"
        "movt r0, #0x0800              \n\t"
        "ldr  r0, [r0]                 \n\t"
        "mov  lr, r0                   \n\t"

        /* setup VTOR (at E000ED08) to remap vectors*/

        "movw r0, #0xED08        /*adr lo*/ \n\t"
        "movt r0, #0xE000        /*adr hi*/ \n\t"
        "movw r1, #0x4000        /*val lo*/ \n\t"
        "movt r1, #0x0800        /*val hi*/ \n\t"
        "str  r1, [r0]           /*store value at address*/ \n\t"

        "bx   lr                 /*take the jump*/ \n"
    );
  }

/* -------------------------------------------------------------------------- */
f4grx's avatar
f4grx committed
BOOTCODE __attribute__((noreturn)) void __boot_start(void)
f4grx's avatar
f4grx committed
    bootloader_inithardware();

    if(bootloader_buttonpressed())
f4grx's avatar
f4grx committed
      {
        /* Boot button is pressed -> dont boot app, do a download */
f4grx's avatar
f4grx committed
        bootloader_download();
        while(1); /* We have to reboot manually */
f4grx's avatar
f4grx committed
    /* Button not pressed: normal process */
f4grx's avatar
f4grx committed
    if(bootloader_check())
f4grx's avatar
f4grx committed
        bootloader_apply(); /* If app was updated, only boot if update was successful */
        bootloader_cleanup();
f4grx's avatar
f4grx committed
    /* Update was not interrupted. Do the boot. */
    bootloader_stophardware();
    __boot_app();
/* -------------------------------------------------------------------------- */
BOOTCODE void __boot_exception_common(void)
  {
    while(1);
  }