<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          STM32 RTC時鐘源LSE

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

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

          本文引用地址:http://www.ex-cimer.com/article/201611/315515.htm

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

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

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

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

          代碼大致如下:

          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. /*將外設(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級別
          19. for(i=0;i<10;i++)//10次檢測,如果LSE仍然沒有起振,證明這玩意有問題,跳出循環(huán)
          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;//開啟內(nèi)部低速晶振
          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晶振預(yù)分頻值是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,則這里應(yīng)該填為62499
          60. /*WaituntillastwriteoperationonRTCregistershasfinished*/
          61. /*等待上一次對RTC寄存器的寫操作完成*/
          62. RTC_WaitForLastTask();
          63. }
          64. voidInit_RTC(void)
          65. {
          66. /*以下if...else....if判斷系統(tǒng)時間是否已經(jīng)設(shè)置,判斷RTC后備寄存器1的值
          67. 是否為事先寫入的0XA5A5,如果不是,則說明RTC是第一次上電,需要配置RTC,
          68. 提示用戶通過串口更改系統(tǒng)時間,把實際時間轉(zhuǎn)化為RTC計數(shù)值寫入RTC寄存器,
          69. 并修改后備寄存器1的值為0XA5A5。
          70. else表示已經(jīng)設(shè)置了系統(tǒng)時間,打印上次系統(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時鐘
          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晶振,這東西簡直不能忍受......



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

          評論


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

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();