這篇文章主要介紹“如何解決SysTick定時器錯誤問題”,在日常操作中,相信很多人在如何解決SysTick定時器錯誤問題問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”如何解決SysTick定時器錯誤問題”的疑惑有所幫助!接下來,請跟著小編一起來學習吧!
創(chuàng)新互聯(lián)公司服務(wù)項目包括饒平網(wǎng)站建設(shè)、饒平網(wǎng)站制作、饒平網(wǎng)頁制作以及饒平網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,饒平網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到饒平省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
使用的就是systick定時器,具體代碼如下
void delay_us(uint32_t nus) { uint32_t temp; SysTick->LOAD = RCC_Clocks.HCLK_Frequency/1000000/8*nus; SysTick->VAL=0X00;//清空計數(shù)器 SysTick->CTRL=0X01;//使能,減到零是無動作,采用外部時鐘源 do { temp=SysTick->CTRL;//讀取當前倒計數(shù)值 }while((temp&0x01)&&(!(temp&(1<<16))));//等待時間到達 SysTick->CTRL=0x00; //關(guān)閉計數(shù)器 SysTick->VAL =0X00; //清空計數(shù)器 } void delay_ms(uint16_t nms) { uint32_t temp; SysTick->LOAD = RCC_Clocks.HCLK_Frequency/1000/8*nms; SysTick->VAL=0X00;//清空計數(shù)器 SysTick->CTRL=0X01;//使能,減到零是無動作,采用外部時鐘源 do { temp=SysTick->CTRL;//讀取當前倒計數(shù)值 }while((temp&0x01)&&(!(temp&(1<<16))));//等待時間到達 SysTick->CTRL=0x00; //關(guān)閉計數(shù)器 SysTick->VAL =0X00; //清空計數(shù)器 }
對于《STM32延時的四種方法》文中所說的內(nèi)容如下
也就是下面代碼中/8的原因。
SysTick->LOAD = RCC_Clocks.HCLK_Frequency/1000/8*nms;
我對此深信不疑,并在STM32F207參考手冊(RM0033)上找到“證據(jù)”。
上圖①處直接是8分頻,而不像②出的1/2/4/8分頻。所以我確信是SYSTICK的時鐘固定為HCLK時鐘的1/8。
我在學習RTThread的時候,看到配置SysTick定制器代碼如下
我心里一堆問號,STM32官方手冊,明明寫了SYSTICK的時鐘固定為HCLK時鐘的1/8。我使用示波器測量,RTThread的配置是沒有問題,可以正常延時的。
位2置1,表示時鐘頻率為AHB,也就是默認的120000000Hz。
位2清0,表示時鐘頻率為AHB/8,也就是120000000/8Hz。
RTThread配置為內(nèi)部時鐘
之前的文章配置為外部時鐘源
這個細節(jié)我沒有留意,導致我看RTThread代碼時有點懵逼。
準確的描述是:
SYSTICK的時鐘可以為HCLK時鐘的1分頻或8分頻,在這里我們選用外部時鐘源120M,所以SYSTICK的時鐘為(120/8)M。
特此更正。
關(guān)于這點,STM32的標準外設(shè)庫提供的SysTick_Config函數(shù),也是使用內(nèi)部時鐘的
/** \brief System Tick Configuration This function initialises the system tick timer and its interrupt and start the system tick timer. Counter is in free running mode to generate periodical interrupts. \param [in] ticks Number of ticks between two interrupts \return 0 Function succeeded \return 1 Function failed */ static __INLINE uint32_t SysTick_Config(uint32_t ticks) { if (ticks > SysTick_LOAD_RELOAD_Msk) return (1); /* Reload value impossible */ SysTick->LOAD = (ticks & SysTick_LOAD_RELOAD_Msk) - 1; /* set reload register */ NVIC_SetPriority (SysTick_IRQn, (1<<__NVIC_PRIO_BITS) - 1); /* set Priority for Cortex-M0 System Interrupts */ SysTick->VAL = 0; /* Load the SysTick Counter Value */ SysTick->CTRL = SysTick_CTRL_CLKSOURCE_Msk | SysTick_CTRL_TICKINT_Msk | SysTick_CTRL_ENABLE_Msk; /* Enable SysTick IRQ and SysTick Timer */ return (0); /* Function successful */ }
調(diào)用方法,產(chǎn)生1ms中斷調(diào)用方法
SysTick_Config(SystemCoreClock / 1000);
關(guān)于時鐘源的選擇,除了操作寄存器外,還有庫函數(shù)可以選擇。
/** * @brief Configures the SysTick clock source. * @param SysTick_CLKSource: specifies the SysTick clock source. * This parameter can be one of the following values: * @arg SysTick_CLKSource_HCLK_Div8: AHB clock divided by 8 selected as SysTick clock source. * @arg SysTick_CLKSource_HCLK: AHB clock selected as SysTick clock source. * @retval None */ void SysTick_CLKSourceConfig(uint32_t SysTick_CLKSource) { /* Check the parameters */ assert_param(IS_SYSTICK_CLK_SOURCE(SysTick_CLKSource)); if (SysTick_CLKSource == SysTick_CLKSource_HCLK) { SysTick->CTRL |= SysTick_CLKSource_HCLK; } else { SysTick->CTRL &= SysTick_CLKSource_HCLK_Div8; } }
除上外,我找到了其他證據(jù)來說明,SYSTICK的時鐘可以為HCLK時鐘的1分頻或8分頻。
在STM32CubeMx配置軟件中,可以選擇1分頻或8分頻。
把涉及的代碼修改成1分頻的。
void delay_ms(uint16_t nms) { uint32_t temp; SysTick->LOAD = RCC_Clocks.HCLK_Frequency/1000*nms-1; SysTick->VAL=0X00;//清空計數(shù)器 SysTick->CTRL=0X01; SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK); do { temp=SysTick->CTRL;//讀取當前倒計數(shù)值 }while((temp&0x01)&&(!(temp&(1<<16))));//等待時間到達 SysTick->CTRL=0x00; //關(guān)閉計數(shù)器 SysTick->VAL =0X00; //清空計數(shù)器 }
然后調(diào)用
GPIO_SetBits(GPIOE,GPIO_Pin_4); //熄滅LED燈 delay_ms(500);//延時500ms GPIO_ResetBits(GPIOE,GPIO_Pin_4);//點亮LED燈 delay_ms(500);//延時500ms
就踩到另一個坑,延時不準。
原因是:此時SYSTICK時鐘頻率是120MHz的24位的倒計數(shù)定時器,也就是說一個周期,最多定時139.810125ms。不能延時500ms。
這里再更正之前的一個錯誤,如下圖
這個計數(shù)器的值,我們減去了1,這樣才更準確。
到此,關(guān)于“如何解決SysTick定時器錯誤問題”的學習就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學習,快去試試吧!若想繼續(xù)學習更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
新聞名稱:如何解決SysTick定時器錯誤問題
轉(zhuǎn)載來于:http://weahome.cn/article/gcpjdg.html