微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > 关于contiki系统到STM32的移植

关于contiki系统到STM32的移植

时间:11-13 来源:互联网 点击:
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初始

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top