新聞中心

STM32 RTC時(shí)鐘源LSE

作者: 時(shí)間:2016-11-17 來源:網(wǎng)絡(luò) 收藏
一開始,所有實(shí)驗(yàn)都是在神舟板上去完成,根本就沒有發(fā)現(xiàn)RTC的問題。直到我們自己畫板來后調(diào)試時(shí),才發(fā)現(xiàn)STM32 RTC的外部時(shí)鐘源存在問題。

這也算是STM32的一個(gè)雞肋,對(duì)于LSE外部晶振太過于苛刻,手冊(cè)上要求使用6pf,這個(gè)規(guī)格的晶振市場(chǎng)上太少,魚龍混雜,中招的高手菜鳥不在少數(shù)。我們自己的板也是如此,幾經(jīng)波折,反反復(fù)復(fù)嘗試使用不同的規(guī)格的晶振,替換外部的電容,電阻都沒有能讓這個(gè)32.768K的LSE起振。但是又需要有RTC來提供時(shí)間,考慮的方法主要有2種,第一采用外部RTC時(shí)鐘芯片,如DS1302。第二是使用內(nèi)部其它的時(shí)鐘源來提供RTC時(shí)鐘。毫無疑問,目前板已經(jīng)制好,添加時(shí)鐘芯片肯定造成板上布局更改,還得重新打板,這里采用了第二種方法。

本文引用地址:http://m.butianyuan.cn/article/201611/315515.htm

查看STM32的手冊(cè)上時(shí)鐘樹,如下:

除去不能起振的外部低速LSE外,可供使用的只有LSI和HSE的128分頻,LSI這個(gè)是內(nèi)部的40KHz RC振蕩器,頻率在30~60KHz浮動(dòng),自然這個(gè)不能用于RTC計(jì)時(shí),誤差太大。

我們的板上配的是STM32F107這款芯片,外部高速晶振是25MHz的。128分頻后頻率為 25000000 / 128 = 195312.5 Hz,很顯然這里也不能做到很精確,有小許誤差。

然后設(shè)置RTC_PRL寄存器,寫入195312這個(gè)分頻值,便可以得到1Hz的頻率。使用HSE作為RTC時(shí)鐘,缺點(diǎn)就是無法在斷開電源后使用后備電池進(jìn)行供電,維持RTC的正常。下次需要上位機(jī)重新去設(shè)置時(shí)間。

代碼大致如下:

  1. voidRTC_Configuration(void)
  2. {
  3. u8i=0;
  4. /*EnablePWRandBKPclocks*/
  5. /*PWR時(shí)鐘(電源控制)與BKP時(shí)鐘(RTC后備寄存器)使能*/
  6. RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP,ENABLE);
  7. /*AllowaccesstoBKPDomain*/
  8. /*使能RTC和后備寄存器訪問*/
  9. PWR_BackupAccessCmd(ENABLE);
  10. /*ResetBackupDomain*/
  11. /*將外設(shè)BKP的全部寄存器重設(shè)為缺省值*/
  12. BKP_DeInit();
  13. /*EnableLSE*/
  14. /*使能LSE(外部32.768KHz低速晶振)*/
  15. RCC_LSEConfig(RCC_LSE_ON);
  16. /*WaittillLSEisready*/
  17. /*等待外部晶振震蕩穩(wěn)定輸出*/
  18. TIM5_Init_Query(CALC_TYPE_MS);//ms級(jí)別
  19. for(i=0;i<10;i++)//10次檢測(cè),如果LSE仍然沒有起振,證明這玩意有問題,跳出循環(huán)
  20. {
  21. if(RCC_GetFlagStatus(RCC_FLAG_LSERDY)!=RESET)
  22. break;
  23. TIM5_MS_CALC(1);//1ms延時(shí)
  24. }
  25. //while(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET){}
  26. if(i==10)
  27. {
  28. //RCC->CSR|=0x1;//開啟內(nèi)部低速晶振
  29. //while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY)==RESET);
  30. //RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);//使用LSI提供RTC時(shí)鐘
  31. //使用外部高速晶振128分頻
  32. RCC_RTCCLKConfig(RCC_RTCCLKSource_HSE_Div128);
  33. }else
  34. {
  35. /*SelectLSEasRTCClockSource*/
  36. /*使用外部32.768KHz晶振作為RTC時(shí)鐘*/
  37. RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);
  38. }
  39. /*EnableRTCClock*/
  40. /*使能RTC的時(shí)鐘供給*/
  41. RCC_RTCCLKCmd(ENABLE);
  42. /*WaitforRTCregisterssynchronization*/
  43. /*等待RTC寄存器同步*/
  44. RTC_WaitForSynchro();
  45. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  46. /*等待上一次對(duì)RTC寄存器的寫操作完成*/
  47. RTC_WaitForLastTask();
  48. /*EnabletheRTCSecond*/
  49. /*使能RTC的秒中斷*/
  50. RTC_ITConfig(RTC_IT_SEC,ENABLE);
  51. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  52. /*等待上一次對(duì)RTC寄存器的寫操作完成*/
  53. RTC_WaitForLastTask();
  54. /*SetRTCprescaler:setRTCperiodto1sec*/
  55. /*32.768KHz晶振預(yù)分頻值是32767,如果對(duì)精度要求很高可以修改此分頻值來校準(zhǔn)晶振*/
  56. if(i!=10)//LSE不能正常
  57. RTC_SetPrescaler(32767);/*RTCperiod=RTCCLK/RTC_PR=(32.768KHz)/(32767+1)*/
  58. else
  59. RTC_SetPrescaler(195312);//25000000/128=195312.5,如果是8M/128=62500,則這里應(yīng)該填為62499
  60. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  61. /*等待上一次對(duì)RTC寄存器的寫操作完成*/
  62. RTC_WaitForLastTask();
  63. }
  64. voidInit_RTC(void)
  65. {
  66. /*以下if...else....if判斷系統(tǒng)時(shí)間是否已經(jīng)設(shè)置,判斷RTC后備寄存器1的值
  67. 是否為事先寫入的0XA5A5,如果不是,則說明RTC是第一次上電,需要配置RTC,
  68. 提示用戶通過串口更改系統(tǒng)時(shí)間,把實(shí)際時(shí)間轉(zhuǎn)化為RTC計(jì)數(shù)值寫入RTC寄存器,
  69. 并修改后備寄存器1的值為0XA5A5。
  70. else表示已經(jīng)設(shè)置了系統(tǒng)時(shí)間,打印上次系統(tǒng)復(fù)位的原因,并使能RTC秒中斷
  71. */
  72. if(BKP_ReadBackupRegister(BKP_DR1)!=RTC_SEQ_ID)
  73. {
  74. /*Backupdataregistervalueisnotcorrectornotyetprogrammed(when
  75. thefirsttimetheprogramisexecuted)*/
  76. /*RTCConfiguration*/
  77. RTC_Configuration();
  78. /*Adjusttimebyvaluesentredbytheuseronthehyperterminal*/
  79. RTC_SetCounter(Time_Regulate(YEAR_BASE,01,01,0,0,0));//2008-1-10:0:0
  80. /*修改后備寄存器1的值為0XA5A5*/
  81. BKP_WriteBackupRegister(BKP_DR1,RTC_SEQ_ID);
  82. }else
  83. {
  84. /*CheckifthePowerOnResetflagisset*/
  85. //RCC_GetFlagStatus(RCC_FLAG_PORRST)!=RESET
  86. //printf("rnnPowerOnResetoccurred....");
  87. /*CheckifthePinResetflagisset*/
  88. //elseif(RCC_GetFlagStatus(RCC_FLAG_PINRST)!=RESET)
  89. //printf("rnnExternalResetoccurred....");
  90. if(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET)
  91. {
  92. //RCC->CSR|=0x1;//開啟內(nèi)部低速晶振
  93. //while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY)==RESET);
  94. //RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);//使用LSI提供RTC時(shí)鐘
  95. //RCC_RTCCLKConfig(RCC_RTCCLKSource_HSE_Div128);
  96. RTC_Configuration();
  97. }
  98. //printf("rnNoneedtoconfigureRTC....");
  99. /*WaitforRTCregisterssynchronization*/
  100. RTC_WaitForSynchro();
  101. /*EnabletheRTCSecond*/
  102. RTC_ITConfig(RTC_IT_SEC,ENABLE);
  103. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  104. RTC_WaitForLastTask();
  105. }
  106. #ifdefRTCClockOutput_Enable
  107. /*EnablePWRandBKPclocks*/
  108. RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP,ENABLE);
  109. /*AllowaccesstoBKPDomain*/
  110. PWR_BackupAccessCmd(ENABLE);
  111. /*DisabletheTamperPin*/
  112. BKP_TamperPinCmd(DISABLE);/*TooutputRTCCLK/64onTamperpin,thetamper
  113. functionalitymustbedisabled*/
  114. /*EnableRTCClockOutputonTamperPin*/
  115. BKP_RTCOutputConfig(BKP_RTCOutputSource_CalibClock);
  116. #endif
  117. /*Clearresetflags*/
  118. RCC_ClearFlag();
  119. }


實(shí)際測(cè)試,RTC效果還行,然后配合上位機(jī)隔一定的時(shí)間后同步時(shí)間基本上能夠滿足要求。

萬惡的LSE晶振,這東西簡直不能忍受......



關(guān)鍵詞: STM32RTC時(shí)鐘源LS

評(píng)論


技術(shù)專區(qū)

關(guān)閉