chakokuのブログ(rev4)

テック・コミック・DTM・・・ごくまれにチャリ

Cortex-MのstartupをCで書く例

startupは一般的にはアセンブラで書かれるのですが、Cortex-MではCで書くこともできるようで、startup.cが公開されています。以下(そのまま引用)(自分の覚書用)
GitHubの公開場所は以下
cortex-uni-startup/startup.c at master · pfalcon/cortex-uni-startup · GitHub
linker scriptもかなりシンプルなので、これからはこのstartupを使うつもりです。
このstartup.cと先ほどのLチカを組み合わせてビルドして、BluePillで動作することを確認した。

/*
 * Unified Cortex Startup - C startup file
 *
 * This file is in public domain
 *
 * Put together by Paul Sokolovsky based on article by Vanya Sergeev
 * http://dev.frozeneskimo.com/notes/cortex_cmsis/ , GNU ld documentation
 * and numerous other public resources.
 *
 */

#include <stdint.h>

/* Declare linker-defined symbols. The only thing of interest
   regarding these symbols is their *address*. uint32_t hints
   of alignment. */

extern uint32_t _end_text;
extern uint32_t _start_data;
extern uint32_t _end_data;
extern uint32_t _start_bss;
extern uint32_t _end_bss;
extern uint32_t _start_stack;
extern uint32_t _end_stack;

/* C main function */
extern int main(void);

/* Device-specific initialization function. Optional, any Cortex-M
   should be able to start up in its default mode on its own, though
   some may have errata for some peripherals (including PLL) which
   this function may "patch". */
extern void SystemInit(void);

void Dummy_Handler(void);

/* Cortex-M core interrupt handlers */
#define ALIAS(sym) __attribute__((weak, alias (sym)))

void Reset_Handler(void);
void NMI_Handler(void) ALIAS("Dummy_Handler");
void HardFault_Handler(void) ALIAS("Dummy_Handler");
void MemManage_Handler(void) ALIAS("Dummy_Handler");
void BusFault_Handler(void) ALIAS("Dummy_Handler");
void UsageFault_Handler(void) ALIAS("Dummy_Handler");
void SVC_Handler(void) ALIAS("Dummy_Handler");
void DebugMon_Handler(void) ALIAS("Dummy_Handler");
void PendSV_Handler(void) ALIAS("Dummy_Handler");
void SysTick_Handler(void) ALIAS("Dummy_Handler");


/* 16 standard Cortex-M vectors - these are present in every MCU */
void *core_vector_table[16] __attribute__ ((section(".cortex_vectors"))) = {
    // See http://sourceware.org/binutils/docs/ld/Source-Code-Reference.html
    // why the address is used here (if not intuitive)
    &_end_stack,
    Reset_Handler,
    NMI_Handler,
    HardFault_Handler,
    MemManage_Handler,
    BusFault_Handler,
    UsageFault_Handler,
    0,
    0,
    0,
    0,
    SVC_Handler,
    DebugMon_Handler,
    0,
    PendSV_Handler,
    SysTick_Handler,
};

/* Based on http://sourceware.org/binutils/docs/ld/Output-Section-LMA.html */
void Reset_Handler(void) {
    register uint32_t *src, *dst;

    /* Copy data section from flash to RAM */
    src = &_end_text;
    dst = &_start_data;
    while (dst < &_end_data)
        *dst++ = *src++;

    /* Clear the bss section, assumes .bss goes directly after .data */
    dst = &_start_bss;
    while (dst < &_end_bss)
        *dst++ = 0;

#ifndef NO_SYSTEMINIT
    SystemInit();
#endif
    main();
    while (1);
}

void Dummy_Handler(void) {
    /* Receiving an unexpected interrupt is considered a bug
       in the program. Let's make it very visible by just
       hanging the processing. Ignoring it by just returning
       may result in very complicated debugging sessions. */
    while (1);
}

以下がBluePill用のLチカメイン

/*
 * LED Blink Test for the STM32F103C8
 * 
 * original source :  Author : Chuck McManis
 * http://robotics.mcmanis.com/src/arm-blink/blink-example.html
 *
 */
#include <stdint.h>

#define GPIOC       0x40011000ul  
#define GPIOC_CRL  (*(uint32_t *)(GPIOC+0x00ul))
#define GPIOC_CRH  (*(uint32_t *)(GPIOC+0x04ul))
#define GPIOC_IDR  (*(uint32_t *)(GPIOC+0x08ul))
#define GPIOC_ODR  (*(uint32_t *)(GPIOC+0x0cul))
#define GPIOC_BSRR (*(uint32_t *)(GPIOC+0x10ul))
#define GPIOC_BRR  (*(uint32_t *)(GPIOC+0x14ul))
#define GPIOC_LCKR (*(uint32_t *)(GPIOC+0x14ul))

#define RCC_BASE      0x40021000ul
#define RCC_APB2ENR  (*(uint32_t *)(RCC_BASE + 0x18ul))
#define GPIOC_ENA     0x01 << 4   // C:bit4

#define GPIOC_13 13
#define GPIOC_8   8
#define OUT10MH 0x01 
#define CNF13_GENPP    0x00000000ul
#define MODE13_OUT10MH OUT10MH << ((GPIOC_13 - GPIOC_8) *4)
#define DELAY_TIME 300000

volatile int var;

void wait(int delay_time);
void blink(void);

void main()
{
  blink();

  // infinit loop
  while(1)
    ;
}


void blink(void){

  // setup GPIOC PORT
  RCC_APB2ENR = GPIOC_ENA; // Enable clocks to GPIOC
  GPIOC_CRH = CNF13_GENPP | MODE13_OUT10MH;  // set GPIOC_13 as OUTPUT
  GPIOC_BSRR = 1 << GPIOC_13 ; //LED off

  while (1) {
    GPIOC_BRR = 1 << GPIOC_13; // LED on
    wait(DELAY_TIME);

    GPIOC_BSRR = 1 << GPIOC_13 ; //LED off
    wait(DELAY_TIME);

  }
}

void wait(int delay_time){
  unsigned int i = delay_time;
  while(i--){
    asm("nop");
  }
}