新闻中心

EEPW首页 > 嵌入式系统 > 设计应用 > Cortex-M3 (NXP LPC1788)之SysTick系统节拍定时器

Cortex-M3 (NXP LPC1788)之SysTick系统节拍定时器

作者:时间:2016-11-19来源:网络收藏
在GPIO控制篇中的延时闪烁LED只用了简单的for循环,为了精确的计时本篇介绍使用SysTick定时器每1ms产生中断,从而实现精确定时的目的。要使用系统节拍定时器主要进行两个部分的配置。1:系统时钟控制。2系统节拍定时器的控制。

一,系统时钟控制

LPC1788有3个独立的振荡器。他们是主振荡器,内部RC振荡器,RTC振荡器。复位后,LPC1788将用内部的RC振荡器运行,直到被软件切换。这样就能在没有任何外部晶振的情况下运行。LPC1788的时钟控制如图1所示

本文引用地址:http://www.eepw.com.cn/article/201611/318446.htm

在开发板上使用12M的晶振作为主振荡器,它通过锁相环PLL0来提高频率提供CPU。由于芯片总是从内部的RC振荡器开始工作,因此主振荡器只会应软件的请求而启动。实现方法是设定SCS寄存器中的OSCEN位使能。主振荡器提供一个状态标志SCS寄存器的OSCSTAT位,这样软件就可以确定何时主振荡器在运行稳定。此时,软件可以控制切换到主振荡器,使其作为时钟源。在启动以前,必须通过SCS的OSCRANGE位,选择一个频率范围。在确定了主振荡器之后,需要进行锁相环的配置。1,配置CLKSRCSEL选择正确的时钟源。2,将正确的PLL设置值写入PLLCFG寄存器并且在PLLCON中使能PLL。3,向PLLFEED寄存器中写入馈送序列0xAA,0x55。4,设置所需的时钟分配器如CCLKSEL,PCLSEL,EMCCLKSEL,以及USBCLKSEL寄存器。5,查询PLLSTAT寄存器等待PLL锁定。

二,系统节拍定时器的控制

LPC1788的系统节拍定时器是一个24位的定时器,当数值达到0时产生中断。系统节拍定时器的时钟信号可以由CPU时钟提供(即图1中的cclk)。想要在规定的时间间隔循环的产生中断,必须将指定的正确时间间隔值装入STRELOAD寄存器进行初始化。假如我们选择cclk作为系统节拍的时钟信号,并且根据开发板将系统时钟设置成12MHZ,为了循环1ms产生一次中断,我们写入STRELOAD的值为cclk/1000 - 1 。

程序的代码如下,使LED灯每500ms闪烁。SystemInit函数在启动文件中被调用。

  1. #definerFIO1DIR(*(volatileunsigned*)0x20098020)
  2. #definerFIO1MASK(*(volatileunsigned*)0x20098030)
  3. #definerFIO1PIN(*(volatileunsigned*)0x20098034)
  4. #definerFIO1SET(*(volatileunsigned*)0x20098038)
  5. #definerFIO1CLR(*(volatileunsigned*)0x2009803c)
  6. #definerCLKSRCSEL(*(unsigned*)(0x400FC10C))//时钟源选择寄存器
  7. #definerPLL0CON(*(unsigned*)(0x400FC080))//PLL0控制寄存器
  8. #definerPLL0CFG(*(unsigned*)(0x400FC084))//PLL0配置寄存器
  9. #definerPLL0STAT(*(unsigned*)(0x400FC088))//PLL0状态寄存器
  10. #definerPLL0FEED(*(unsigned*)(0x400FC08C))//PLL0馈送寄存器
  11. #definerPLL1CON(*(unsigned*)(0x400FC0A0))
  12. #definerPLL1CFG(*(unsigned*)(0x400FC0A4))
  13. #definerPLL1STAT(*(unsigned*)(0x400FC0A8))
  14. #definerPLL1FEED(*(unsigned*)(0x400FC0AC))
  15. #definerCCLKSEL(*(unsigned*)(0x400FC104))//CPU时钟选择寄存器
  16. #definerUSBCLKSEL(*(unsigned*)(0x400FC108))//USB时钟选择寄存器
  17. #definerPCLKSEL(*(unsigned*)(0x400FC1A8))//外设时钟寄存器
  18. #definerPCON(*(unsigned*)(0x400FC0C0))
  19. #definerPXCONP(*(unsigned*)(0x400FC0C4))
  20. #definerSCS(*(unsigned*)(0x400FC1A0))//系统控制和状态寄存器
  21. #definerCLKOUTCFG(*(unsigned*)(0x400FC1C8))
  22. #definerSTCTRL(*(unsigned*)(0xE000E010))
  23. #definerSTRELOAD(*(unsigned*)(0xE000E014))
  24. #definerSTCURR(*(unsigned*)(0xE000E018))
  25. #definerSTALIB(*(unsigned*)(0xE000E01C))
  26. #defineCCLK120000000
  27. volatileunsignedlongSysTickCnt;
  28. /*
  29. 系统时钟初始化
  30. */
  31. voidSystemInit()
  32. {
  33. rSCS&=~(0x1<<4);//频率12M
  34. rSCS|=(0x1<<5);//使能主振荡器
  35. while(0==(rSCS&(0x1<<6)));//等待主振荡器稳定
  36. rCLKSRCSEL=0x1;
  37. rPLL0CFG=0x9;//配置CCLK=120M
  38. rPLL0CON=0x01;
  39. rPLL0FEED=0xAA;
  40. rPLL0FEED=0x55;
  41. while(0==(rPLL0STAT&(0x1<<10)));
  42. rCCLKSEL=(0x1|(0x1<<8));
  43. rPCLKSEL=0x2;//配置PCLK=60M
  44. rCLKOUTCFG=0x0|(0xb<<4)|(0x1<<8);
  45. }
  46. /*
  47. 系统节拍定时器初始化
  48. */
  49. unsignedcharSysTick_Config(unsignedintticks)
  50. {
  51. if(ticks>0xFFFFFFUL)
  52. return0;
  53. rSTRELOAD=ticks;
  54. rSTCURR=0x0;
  55. rSTCTRL=(0x1)|(0x1<<1)|(0x1<<2);
  56. return1;
  57. }
  58. /*
  59. 系统节拍定时器中断处理函数
  60. */
  61. voidSysTick_Handler(void)
  62. {
  63. SysTickCnt++;
  64. }
  65. intmain()
  66. {
  67. unsignedcharvalue=1;
  68. SysTick_Config(CCLK/1000-1);//每1ms产生一次SysTick系统异常
  69. rFIO1DIR|=(1<<18);//GPIO1.18->OUTPUT
  70. while(1)
  71. {
  72. if(SysTickCnt>=500)
  73. {
  74. SysTickCnt=0;
  75. value=!value;
  76. }
  77. if(0==value)
  78. {
  79. rFIO1PIN&=~(1<<18);
  80. }
  81. elseif(1==value)
  82. {
  83. rFIO1PIN|=(1<<18);
  84. }
  85. }
  86. }

关于SysTick的异常的优先级可以在SHPR3中进行设置,优先级等级可以从0~31。初始化默认为0,只低于固定的负数优先级的复位,硬件故障和NMI。


评论


相关推荐

技术专区

关闭