關(guān)于contiki系統(tǒng)到STM32的移植
“Contiki是一個(gè)小型的,開源的,極易移植的多任務(wù)操作系統(tǒng)。它專門設(shè)計(jì)以適用于一系列的內(nèi)存優(yōu)先的網(wǎng)絡(luò)系統(tǒng),包括從8位電腦到微型控制器的嵌入系統(tǒng)。它的名字來自于托爾·海爾達(dá)爾的康提基號。Contiki只需幾kilobyte的代碼和幾百字節(jié)的內(nèi)存就能提供多任務(wù)環(huán)境和內(nèi)建TCP/IP支持。
本文引用地址:http://www.ex-cimer.com/article/201611/316108.htm2.移植前的準(zhǔn)備
首先建立一個(gè)最簡單工程。一個(gè)最簡單的任務(wù)莫過于LED閃爍了,從學(xué)習(xí)51單片機(jī)開始,到AVR,到ARM,從移植uCOS到移植contiki。LED閃爍無疑是最棒的任務(wù)。假設(shè)這個(gè)任務(wù)就是LED點(diǎn)亮1秒,然后LED熄滅1秒。Contiki的采用事件驅(qū)動機(jī)制,那么如何才能夠產(chǎn)生“事件“呢。答案只有兩個(gè):第一,通過時(shí)鐘定時(shí),定時(shí)事件到就產(chǎn)生一個(gè)事件;第二,通過某種中斷,某個(gè)中斷發(fā)生,就產(chǎn)生某個(gè)事件例如外部中斷。那么移植contiki到底要做哪些工作呢。先來回顧一下uCOS在STM32移植,uCOS的移植也就是做了兩件事情,第一,在PendSV這個(gè)異常中斷中,保存上下文;第二,使用systick提供系統(tǒng)時(shí)鐘。由于contiki是非搶占的操作系統(tǒng),所以移植時(shí)并不需要PendSV中保存上下文。那么時(shí)鐘一定是必要的,移植contiki的移植重點(diǎn)就應(yīng)該在systick上。
先上全部的代碼,給大家一個(gè)整體的印象。
- #include"stm32f10x.h"
- #include
- #include
- #include
uart.h> - #include
- #include
- #include
- #include
- #include
- unsignedintidle_count=0;
- voidled_init();
- PROCESS(blink_process,"Blink");
- AUTOSTART_PROCESSES(&blink_process);
- PROCESS_THREAD(blink_process,ev,data)
- {
- PROCESS_BEGIN();
- while(1)
- {
- staticstructetimeret;
- etimer_set(&et,CLOCK_SECOND);
- PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
- //打開LED
- GPIO_ResetBits(GPIOC,GPIO_Pin_6);
- printf("LEDONrn");
- etimer_set(&et,CLOCK_SECOND);
- PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
- //關(guān)閉LED
- GPIO_SetBits(GPIOC,GPIO_Pin_6);
- printf("LEDOFFrn");
- }
- PROCESS_END();
- }
- intmain()
- {
- dbg_setup_uart();
- led_init();
- printf("Initialisingrn");
- clock_init();
- process_init();
- process_start(&etimer_process,NULL);
- autostart_start(autostart_processes);
- //process_start(&blink_process,NULL);
- printf("Processesrunningrn");
- while(1){
- do
- {
- }
- while(process_run()>0);
- idle_count++;
- /*Idle!*/
- /*Stopprocessorclock*/
- /*asm("wfi"::);*/
- }
- return0;
- }
- voidled_init()
- {
- GPIO_InitTypeDefGPIO_InitStructure;
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
- //PC6推挽輸出
- GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6;
- GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
- GPIO_Init(GPIOC,&GPIO_InitStructure);
- }
3.尋找一些線索
閱讀contiki-2.5 源碼中,stm32移植的相關(guān)內(nèi)容分散在兩個(gè)文件夾中,第一, cpuarmstm32f103,這個(gè)文件夾存放的stm32移植的相關(guān)文件;第二,platformstm32test,這個(gè)文件夾中有一個(gè)不是那么完整的例子。具體的源碼如下:
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- #include
- unsignedintidle_count=0;
- int
- main()
- {
- dbg_setup_uart();
- printf("Initialisingn");
- clock_init();
- process_init();
- process_start(&etimer_process,NULL);
- autostart_start(autostart_processes);
- printf("Processesrunningn");
- while(1){
- do{
- }while(process_run()>0);
- idle_count++;
- /*Idle!*/
- /*Stopprocessorclock*/
- /*asm("wfi"::);*/
- }
- return0;
- }
簡單分析一下,首先文件中包含了一些頭文件。看著有點(diǎn)熟悉,應(yīng)該是V2.0庫的頭文件,后面的移植工作會全部替換掉,使用V3.4的庫文件。在main函數(shù)中,第一步初始化串口并通過串口發(fā)送某些信息。接著,初始化時(shí)鐘,通過跟蹤源代碼,發(fā)現(xiàn)clock_init函數(shù)位于cpuarmstm32f103文件夾中的clock文件夾中。具體的函數(shù)如下:
- void
- clock_init()
- {
- NVIC_SET_SYSTICK_PRI(8);
- SysTick->LOAD=MCK/8/CLOCK_SECOND;
- SysTick->CTRL=SysTick_CTRL_ENABLE|SysTick_CTRL_TICKINT;
- }
這段代碼的原理也非常的簡單,初始化systick定時(shí)器。其功能是每秒發(fā)生CLOCK_SECOND次溢出。配置了systick也少不了systick中斷了,systick的中斷的源碼如下: 在systick中斷中不斷更新了etimer,有了時(shí)鐘contiki就可以運(yùn)行了。
4.開始移植 先在clock源文件中添加頭文件
#include "stm32f10x.h"
#include "stm32f10x_it.h"
刪除原來的
#include
#include
把systick初始化改成
- void
- clock_init()
- {
- if(SysTick_Config(SystemCoreClock/CLOCK_SECOND))
- {
- while(1);
- }
- }
把systick中斷改為
- voidSysTick_Handler(void)
- {
- current_clock++;
- if(etimer_pending()&&etimer_next_expiration_time()<=current_clock){
- etimer_request_poll();
- //printf("%d,%dn",clock_time(),etimer_next_expiration_time());
- }
- if(--second_countdown==0){
- current_seconds++;
- second_countdown=CLOCK_SECOND;
- }
- }
最后,把stm32f10x_it.c的void SysTick_Handler(void){}刪除。。 再來配置一下debug接口。配置串口位于debug_uart文件中,我把原代碼中的DMA相關(guān)代碼刪除,只剩串口初始化和fputc函數(shù)。具體的代碼如下:
- void
- dbg_setup_uart_default()
- {
- USART_InitTypeDefUSART_InitStructure;
- GPIO_InitTypeDefGPIO_InitStructure;
- //使能GPIOA時(shí)鐘
- RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA
- |RCC_APB2Periph_USART1,ENABLE);
- //PA9TX1復(fù)用推挽輸出
- GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;
- GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
- GPIO_Init(GPIOA,&GPIO_InitStructure);
- //PA10RX1浮動輸入
- GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;
- GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
- GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;
- GPIO_Init(GPIOA,&GPIO_InitStructure);
- USART_InitStructure.USART_BaudRate=9600;
- USART_InitStructure.USART_WordLength=USART_WordLength_8b;
- USART_InitStructure.USART_StopBits=USART_StopBits_1;
- USART_InitStructure.USART_Parity=USART_Parity_No;
- USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
- USART_InitStructure.USART_Mode=USART_Mode_Rx|USART_Mode_Tx;
- USART_Init(USART1,&USART_InitStructure);
- //使能USART1
- USART_Cmd(USART1,ENABLE);
- }
- intfputc(intch,FILE*f)
- {
- USART_SendData(USART1,(uint8_t)ch);
- while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
- returnch;
- }
5.新建一個(gè)任務(wù)
通過上網(wǎng)搜索和閱讀書籍,我寫了以下任務(wù)。
- PROCESS(blink_process,"Blink");
- AUTOSTART_PROCESSES(&blink_process);
- PROCESS_THREAD(blink_process,ev,data)
- {
- PROCESS_BEGIN();
- while(1)
- {
- staticstructetimeret;
- etimer_set(&et,CLOCK_SECOND);
- PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
- //打開LED
- GPIO_ResetBits(GPIOC,GPIO_Pin_6);
- printf("LEDONrn");
- etimer_set(&et,CLOCK_SECOND);
- PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
- //關(guān)閉LED
- GPIO_SetBits(GPIOC,GPIO_Pin_6);
- printf("LEDOFFrn");
- }
- PROCESS_END();
- }
該任務(wù)是從contiki-2.5中例子修改而來的。任務(wù)非常的簡單,打開LED,通過串口發(fā)送提示信息,然后關(guān)閉LED,通過串口發(fā)送提示信息。
【1】PROCESS(blink_process,"Blink");相關(guān)于函數(shù)的聲明
【2】AUTOSTART_PROCESSES(&blink_process);是指該任務(wù)自動啟動,也可以調(diào)用process_start函數(shù)啟動任務(wù)。AUTOSTART_PROCESSES其實(shí)也是一個(gè)宏東定義:
- #if!CC_NO_VA_ARGS
- #ifAUTOSTART_ENABLE
- #defineAUTOSTART_PROCESSES(...)
- structprocess*constautostart_processes[]={__VA_ARGS__,NULL}
- #else//AUTOSTART_ENABLE
- #defineAUTOSTART_PROCESSES(...)
- externint_dummy
- #endif//AUTOSTART_ENABLE
- #else
- #error"Ccompilermustsupport__VA_ARGS__macro"
- #endif
要想使用它的話,還需要添加AUTOSTART_ENABLE定義。
#define AUTOSTART_ENABLE 1
最后請大家不要忘記LED相關(guān)IO口的初始化操作。請查看前文代碼。
6.實(shí)驗(yàn)結(jié)果
先給出contiki的IAR 工程目錄和文件目錄
再來一個(gè)頭文件包含路徑:
$PROJ_DIR$CMSIS
$PROJ_DIR$StdPeriph_Driverinc
$PROJ_DIR$User
$PROJ_DIR$contiki-2.5core
$PROJ_DIR$contiki-2.5coresys
$PROJ_DIR$contiki-2.5corelib
$PROJ_DIR$contiki-2.5cpu
【小技巧】在編譯文件的時(shí)候會發(fā)生一些莫名奇妙的警告,這個(gè)警告產(chǎn)生的原因是 linux的文件換行和window文件換行不同! 采用以下方法可以屏蔽這個(gè)警告,如下圖所示:
如果移植順利的話,就可以看到以下實(shí)驗(yàn)結(jié)果。
寫到這里你會發(fā)現(xiàn),contiki的移植還是非常簡單的。
評論