您的位置 首页 软件

关于contiki体系到STM32的移植

1.contiki简介Contiki是一个小型的,开源的,极易移植的多任务操作系统。它专门设计以适用于一系列的内存优先的网络系统,包括从8位电脑到…

1.contiki简介

“Contiki是一个小型的,开源的,极易移植的多使命操作体系。它专门规划以适用于一系列的内存优先的网络体系,包括从8位电脑到微型控制器的嵌入体系。它的姓名来自于托尔·海尔达尔的康提基号。Contiki只需几kilobyte的代码和几百字节的内存就能供给多使命环境和内建TCP/IP支撑。

2.移植前的预备

首要树立一个最简略工程。一个最简略的使命莫过于LED闪耀了,从学习51单片机开端,到AVR,到ARM,从移植uCOS到移植contiki。LED闪耀无疑是最棒的使命。假定这个使命便是LED点亮1秒,然后LED平息1秒。Contiki的选用工作驱动机制,那么怎么才能够发生“工作“呢。答案只要两个:榜首,经过时钟守时,守时工作到就发生一个工作;第二,经过某种中止,某个中止发生,就发生某个工作例如外部中止。那么移植contiki究竟要做哪些作业呢。先来回忆一下uCOS在STM32移植,uCOS的移植也便是做了两件工作,榜首,在PendSV这个异常中止中,保存上下文;第二,运用systick供给体系时钟。因为contiki对错抢占的操作体系,所以移植时并不需要PendSV中保存上下文。那么时钟一定是必要的,移植contiki的移植要点就应该在systick上。

先上悉数的代码,给我们一个全体的形象。

  1. #include”stm32f10x.h”
  2. #include
  3. #include
  4. #includeuart.h>
  5. #include
  6. #include
  7. #include
  8. #include
  9. #include
  10. unsignedintidle_count=0;
  11. voidled_init();
  12. PROCESS(blink_process,”Blink”);
  13. AUTOSTART_PROCESSES(&blink_process);
  14. PROCESS_THREAD(blink_process,ev,data)
  15. {
  16. PROCESS_BEGIN();
  17. while(1)
  18. {
  19. staticstructetimeret;
  20. etimer_set(&et,CLOCK_SECOND);
  21. PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
  22. //翻开LED
  23. GPIO_ResetBits(GPIOC,GPIO_Pin_6);
  24. printf(“LEDON\r\n”);
  25. etimer_set(&et,CLOCK_SECOND);
  26. PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
  27. //封闭LED
  28. GPIO_SetBits(GPIOC,GPIO_Pin_6);
  29. printf(“LEDOFF\r\n”);
  30. }
  31. PROCESS_END();
  32. }
  33. intmain()
  34. {
  35. dbg_setup_uart();
  36. led_init();
  37. printf(“Initialising\r\n”);
  38. clock_init();
  39. process_init();
  40. process_start(&etimer_process,NULL);
  41. autostart_start(autostart_processes);
  42. //process_start(&blink_process,NULL);
  43. printf(“Processesrunning\r\n”);
  44. while(1){
  45. do
  46. {
  47. }
  48. while(process_run()>0);
  49. idle_count++;
  50. /*Idle!*/
  51. /*Stopprocessorclock*/
  52. /*asm(“wfi”::);*/
  53. }
  54. return0;
  55. }
  56. voidled_init()
  57. {
  58. GPIO_InitTypeDefGPIO_InitStructure;
  59. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);
  60. //PC6推挽输出
  61. GPIO_InitStructure.GPIO_Pin=GPIO_Pin_6;
  62. GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  63. GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
  64. GPIO_Init(GPIOC,&GPIO_InitStructure);
  65. }

3.寻觅一些头绪

阅览contiki-2.5 源码中,stm32移植的相关内容涣散在两个文件夹中,榜首, cpu\arm\stm32f103,这个文件夹寄存的stm32移植的相关文件;第二,platform\stm32test,这个文件夹中有一个不是那么完好的比如。详细的源码如下:

  1. #include
  2. #include
  3. #include
  4. #include
  5. #include
  6. #include
  7. #include
  8. #include
  9. #include
  10. #include
  11. #include
  12. #include
  13. unsignedintidle_count=0;
  14. int
  15. main()
  16. {
  17. dbg_setup_uart();
  18. printf(“Initialising\n”);
  19. clock_init();
  20. process_init();
  21. process_start(&etimer_process,NULL);
  22. autostart_start(autostart_processes);
  23. printf(“Processesrunning\n”);
  24. while(1){
  25. do{
  26. }while(process_run()>0);
  27. idle_count++;
  28. /*Idle!*/
  29. /*Stopprocessorclock*/
  30. /*asm(“wfi”::);*/
  31. }
  32. return0;
  33. }

简略剖析一下,首要文件中包括了一些头文件。看着有点了解,应该是V2.0库的头文件,后边的移植作业会悉数替换掉,运用V3.4的库文件。在main函数中,榜首步初始化串口并经过串口发送某些信息。接着,初始化时钟,经过盯梢源代码,发现clock_init函数坐落cpu\arm\stm32f103文件夹中的clock文件夹中。详细的函数如下:

  1. void
  2. clock_init()
  3. {
  4. NVIC_SET_SYSTICK_PRI(8);
  5. SysTick->LOAD=MCK/8/CLOCK_SECOND;
  6. SysTick->CTRL=SysTick_CTRL_ENABLE|SysTick_CTRL_TICKINT;
  7. }

这段代码的原理也十分的简略,初始化systick守时器。其功用是每秒发生CLOCK_SECOND次溢出。装备了systick也少不了systick中止了,systick的中止的源码如下: 在systick中止中不断更新了etimer,有了时钟contiki就能够运行了。

4.开端移植 先在clock源文件中增加头文件

#include “stm32f10x.h”

#include “stm32f10x_it.h”

删去本来的

#include

#include

把systick初始化改成

  1. void
  2. clock_init()
  3. {
  4. if(SysTick_Config(SystemCoreClock/CLOCK_SECOND))
  5. {
  6. while(1);
  7. }
  8. }

把systick中止改为

  1. voidSysTick_Handler(void)
  2. {
  3. current_clock++;
  4. if(etimer_pending()&&etimer_next_expiration_time()<=current_clock){
  5. etimer_request_poll();
  6. //printf(“%d,%d\n”,clock_time(),etimer_next_expiration_time());
  7. }
  8. if(–second_countdown==0){
  9. current_seconds++;
  10. second_countdown=CLOCK_SECOND;
  11. }
  12. }

最终,把stm32f10x_it.c的void SysTick_Handler(void){}删去。。 再来装备一下debug接口。装备串口坐落debug_uart文件中,我把原代码中的DMA相关代码删去,只剩串口初始化和fputc函数。详细的代码如下:

  1. void
  2. dbg_setup_uart_default()
  3. {
  4. USART_InitTypeDefUSART_InitStructure;
  5. GPIO_InitTypeDefGPIO_InitStructure;
  6. //使能GPIOA时钟
  7. RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA\
  8. |RCC_APB2Periph_USART1,ENABLE);
  9. //PA9TX1复用推挽输出
  10. GPIO_InitStructure.GPIO_Pin=GPIO_Pin_9;
  11. GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  12. GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
  13. GPIO_Init(GPIOA,&GPIO_InitStructure);
  14. //PA10RX1起浮输入
  15. GPIO_InitStructure.GPIO_Pin=GPIO_Pin_10;
  16. GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  17. GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IN_FLOATING;
  18. GPIO_Init(GPIOA,&GPIO_InitStructure);
  19. USART_InitStructure.USART_BaudRate=9600;
  20. USART_InitStructure.USART_WordLength=USART_WordLength_8b;
  21. USART_InitStructure.USART_StopBits=USART_StopBits_1;
  22. USART_InitStructure.USART_Parity=USART_Parity_No;
  23. USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None;
  24. USART_InitStructure.USART_Mode=USART_Mode_Rx|USART_Mode_Tx;
  25. USART_Init(USART1,&USART_InitStructure);
  26. //使能USART1
  27. USART_Cmd(USART1,ENABLE);
  28. }
  29. intfputc(intch,FILE*f)
  30. {
  31. USART_SendData(USART1,(uint8_t)ch);
  32. while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);
  33. returnch;
  34. }

5.新建一个使命

经过上网查找和阅览书本,我写了以下使命。

  1. PROCESS(blink_process,”Blink”);
  2. AUTOSTART_PROCESSES(&blink_process);
  3. PROCESS_THREAD(blink_process,ev,data)
  4. {
  5. PROCESS_BEGIN();
  6. while(1)
  7. {
  8. staticstructetimeret;
  9. etimer_set(&et,CLOCK_SECOND);
  10. PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
  11. //翻开LED
  12. GPIO_ResetBits(GPIOC,GPIO_Pin_6);
  13. printf(“LEDON\r\n”);
  14. etimer_set(&et,CLOCK_SECOND);
  15. PROCESS_WAIT_EVENT_UNTIL(etimer_expired(&et));
  16. //封闭LED
  17. GPIO_SetBits(GPIOC,GPIO_Pin_6);
  18. printf(“LEDOFF\r\n”);
  19. }
  20. PROCESS_END();
  21. }

该使命是从contiki-2.5中比如修正而来的。使命十分的简略,翻开LED,经过串口发送提示信息,然后封闭LED,经过串口发送提示信息。

【1】PROCESS(blink_process,”Blink”);相关于函数的声明

【2】AUTOSTART_PROCESSES(&blink_process);是指该使命主动发动,也能够调用process_start函数发动使命。AUTOSTART_PROCESSES其实也是一个宏东界说:

  1. #if!CC_NO_VA_ARGS
  2. #ifAUTOSTART_ENABLE
  3. #defineAUTOSTART_PROCESSES(…)\
  4. structprocess*constautostart_processes[]={__VA_ARGS__,NULL}
  5. #else//AUTOSTART_ENABLE
  6. #defineAUTOSTART_PROCESSES(…)\
  7. externint_dummy
  8. #endif//AUTOSTART_ENABLE
  9. #else
  10. #error”Ccompilermustsupport__VA_ARGS__macro”
  11. #endif

要想运用它的话,还需要增加AUTOSTART_ENABLE界说。

#define AUTOSTART_ENABLE 1

最终请我们不要忘掉LED相关IO口的初始化操作。请检查前文代码。

6.试验成果

先给出contiki的IAR 工程目录和文件目录

再来一个头文件包括途径:

$PROJ_DIR$\CMSIS

$PROJ_DIR$\StdPeriph_Driver\inc

$PROJ_DIR$\User

$PROJ_DIR$\contiki-2.5\core

$PROJ_DIR$\contiki-2.5\core\sys

$PROJ_DIR$\contiki-2.5\core\lib

$PROJ_DIR$\contiki-2.5\cpu

【小技巧】在编译文件的时分会发生一些莫名美妙的正告,这个正告发生的原因是 linux的文件换行和window文件换行不同! 选用以下办法能够屏蔽这个正告,如下图所示:

假如移植顺畅的话,就能够看到以下试验成果。

写到这儿你会发现,contiki的移植还对错常简略的。

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/qianrushi/ruanjian/272204.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部