久久ER99热精品一区二区-久久精品99国产精品日本-久久精品免费一区二区三区-久久综合九色综合欧美狠狠

新聞中心

EEPW首頁 > 嵌入式系統 > 設計應用 > STM32 RTC時鐘源LSE

STM32 RTC時鐘源LSE

作者: 時間:2016-11-17 來源:網絡 收藏
一開始,所有實驗都是在神舟板上去完成,根本就沒有發現RTC的問題。直到我們自己畫板來后調試時,才發現STM32 RTC的外部時鐘源存在問題。

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

本文引用地址:http://cqxgywz.com/article/201611/315515.htm

查看STM32的手冊上時鐘樹,如下:

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

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

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

代碼大致如下:

  1. voidRTC_Configuration(void)
  2. {
  3. u8i=0;
  4. /*EnablePWRandBKPclocks*/
  5. /*PWR時鐘(電源控制)與BKP時鐘(RTC后備寄存器)使能*/
  6. RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP,ENABLE);
  7. /*AllowaccesstoBKPDomain*/
  8. /*使能RTC和后備寄存器訪問*/
  9. PWR_BackupAccessCmd(ENABLE);
  10. /*ResetBackupDomain*/
  11. /*將外設BKP的全部寄存器重設為缺省值*/
  12. BKP_DeInit();
  13. /*EnableLSE*/
  14. /*使能LSE(外部32.768KHz低速晶振)*/
  15. RCC_LSEConfig(RCC_LSE_ON);
  16. /*WaittillLSEisready*/
  17. /*等待外部晶振震蕩穩定輸出*/
  18. TIM5_Init_Query(CALC_TYPE_MS);//ms級別
  19. for(i=0;i<10;i++)//10次檢測,如果LSE仍然沒有起振,證明這玩意有問題,跳出循環
  20. {
  21. if(RCC_GetFlagStatus(RCC_FLAG_LSERDY)!=RESET)
  22. break;
  23. TIM5_MS_CALC(1);//1ms延時
  24. }
  25. //while(RCC_GetFlagStatus(RCC_FLAG_LSERDY)==RESET){}
  26. if(i==10)
  27. {
  28. //RCC->CSR|=0x1;//開啟內部低速晶振
  29. //while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY)==RESET);
  30. //RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);//使用LSI提供RTC時鐘
  31. //使用外部高速晶振128分頻
  32. RCC_RTCCLKConfig(RCC_RTCCLKSource_HSE_Div128);
  33. }else
  34. {
  35. /*SelectLSEasRTCClockSource*/
  36. /*使用外部32.768KHz晶振作為RTC時鐘*/
  37. RCC_RTCCLKConfig(RCC_RTCCLKSource_LSE);
  38. }
  39. /*EnableRTCClock*/
  40. /*使能RTC的時鐘供給*/
  41. RCC_RTCCLKCmd(ENABLE);
  42. /*WaitforRTCregisterssynchronization*/
  43. /*等待RTC寄存器同步*/
  44. RTC_WaitForSynchro();
  45. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  46. /*等待上一次對RTC寄存器的寫操作完成*/
  47. RTC_WaitForLastTask();
  48. /*EnabletheRTCSecond*/
  49. /*使能RTC的秒中斷*/
  50. RTC_ITConfig(RTC_IT_SEC,ENABLE);
  51. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  52. /*等待上一次對RTC寄存器的寫操作完成*/
  53. RTC_WaitForLastTask();
  54. /*SetRTCprescaler:setRTCperiodto1sec*/
  55. /*32.768KHz晶振預分頻值是32767,如果對精度要求很高可以修改此分頻值來校準晶振*/
  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,則這里應該填為62499
  60. /*WaituntillastwriteoperationonRTCregistershasfinished*/
  61. /*等待上一次對RTC寄存器的寫操作完成*/
  62. RTC_WaitForLastTask();
  63. }
  64. voidInit_RTC(void)
  65. {
  66. /*以下if...else....if判斷系統時間是否已經設置,判斷RTC后備寄存器1的值
  67. 是否為事先寫入的0XA5A5,如果不是,則說明RTC是第一次上電,需要配置RTC,
  68. 提示用戶通過串口更改系統時間,把實際時間轉化為RTC計數值寫入RTC寄存器,
  69. 并修改后備寄存器1的值為0XA5A5。
  70. else表示已經設置了系統時間,打印上次系統復位的原因,并使能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;//開啟內部低速晶振
  93. //while(RCC_GetFlagStatus(RCC_FLAG_LSIRDY)==RESET);
  94. //RCC_RTCCLKConfig(RCC_RTCCLKSource_LSI);//使用LSI提供RTC時鐘
  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. }


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

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



關鍵詞: STM32RTC時鐘源LS

評論


技術專區

關閉