合宙AIR105(三): 定時器, 定時器中斷和PWM輸出

来源:https://www.cnblogs.com/milton/archive/2022/06/18/16389098.html
-Advertisement-
Play Games

Air105 有 1 個 Timer 單元,包含 8 個獨立定時器: Timer0 到 Time7, 8 個定時器中斷源獨立,每個定時器單獨占 1 個中斷源, 使用 PCLK 時鐘頻率作為定時器計時鐘源, 定時器採用向下計數方式. 每個 Timer 單元定時器都支持 PWM 模式, PWM 模式最高... ...


目錄

Air105 的 Timer

定時器

  • 1 個 Timer 單元,包含 8 個獨立定時器: Timer0 - Time7
  • 8 個定時器中斷源獨立,每個定時器單獨占 1 個中斷源
  • 使用 PCLK 時鐘頻率作為定時器計時鐘源
  • 定時器採用向下計數方式

定時器的兩種運行模式

  • user-defined: 定時器計數值載入TimerNLoadCount寄存器設定值, 使用用戶模式可以產生固定時間的定時器中斷
  • free-runing: 定時器計數值會載入其允許的最大值, 即0xFFFFFFFF. 在定時器產生中斷(計數到0)前, 用戶可以再編程或禁止定時器中斷. 使用這個模式, 定時器只產生1次中斷, 中斷產生後計數重置為 0xFFFFFFFF 並向下計數, 但不會再產生中斷.

PWM

  • 每個 Timer 單元定時器都支持 PWM 模式
  • PWM 模式最高頻率 PCLK/2
  • PWM 單次觸發(one shot)功能

定時器相關代碼

以下代碼基於 air105_project https://gitee.com/iosetting/air105_project 的庫函數

定時器模塊結構

在Air105中, 全局只有一個定時器模塊, TIMM0

typedef struct
{
    TIM_TypeDef TIM[TIM_NUM];
    __I  uint32_t TIM_IntStatus;
    __I  uint32_t TIM_EOI;
    __I  uint32_t TIM_RawIntStatus;
    __I  uint32_t TIM_Comp;
    __IO uint32_t TIM_ReloadCount[TIM_NUM];
} TIM_Module_TypeDef;

這個 TIMM0 的地址定義在 air105.h 中

#define TIMM0                                   ((TIM_Module_TypeDef *)TIMM0_BASE)

#define AIR105_PERIPH_BASE                      (0x40000000UL)   /*!< (Peripheral) Base Address */
#define AIR105_APB0_BASE                        (AIR105_PERIPH_BASE + 0x10000)
#define TIMM0_BASE                              (AIR105_APB0_BASE + 0x3000)
  • 地址 = 0x40000000UL + 0x10000 + 0x3000 = 0x4001 3000
  • 範圍 [0x4001_3000, 0x4001_3FFF]

定時器初始化

定時器的初始化只需要兩個參數: TIMx, 周期(時鐘數), 為配合定時器使用, 還需要定義中斷

void Timer_Init(void)
{
    TIM_InitTypeDef TIM_InitStruct;
    NVIC_InitTypeDef NVIC_InitStructure;

    // 開啟定時器的外設時鐘
    SYSCTRL_APBPeriphClockCmd(SYSCTRL_APBPeriph_TIMM0, ENABLE);
    SYSCTRL_APBPeriphResetCmd(SYSCTRL_APBPeriph_TIMM0, ENABLE);

    // 定時器的時鐘是 PCLK, 計數間隔為 1ms 對應的時鐘數
    TIM_InitStruct.TIM_Period = SYSCTRL->PCLK_1MS_VAL;
    // 使用 定時器0
    TIM_InitStruct.TIMx = TIM_0;
    // 初始化
    TIM_Init(TIMM0, &TIM_InitStruct);
    // 開啟定時器0的中斷
    TIM_ITConfig(TIMM0, TIM_InitStruct.TIMx, ENABLE);

    //NVIC
    NVIC_SetPriorityGrouping(NVIC_PriorityGroup_0);

    NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
    NVIC_InitStructure.NVIC_IRQChannel = TIM0_0_IRQn;
    NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
    NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;

    NVIC_Init(&NVIC_InitStructure);

    // 啟動定時器0
    TIM_Cmd(TIMM0, (TIM_NumTypeDef)TIM_0, ENABLE);
}

在庫函數中, 會將模式設為 user-defined, 即自動迴圈, 重覆載入周期並產生中斷.

/**
  * @brief  Initializes the TIMx Unit peripheral according to the specified parameters.
  * @param  TIMMx: x can be 0 to select the TIM peripheral
  * @param  TIM_InitStruct: pointer to a TIM_InitTypeDef structor that contains the configuration information
  * @retval None
  */
void TIM_Init(TIM_Module_TypeDef *TIMMx, TIM_InitTypeDef *TIM_InitStruct)
{
    TIM_Cmd(TIMMx, TIM_InitStruct->TIMx, DISABLE);
    
    TIMMx->TIM[TIM_InitStruct->TIMx].ControlReg = 0;
    TIMMx->TIM[TIM_InitStruct->TIMx].ControlReg |= TIMER_CONTROL_REG_TIMER_MODE;
    TIMMx->TIM[TIM_InitStruct->TIMx].ControlReg &= ~TIMER_CONTROL_REG_TIMER_PWM;

    TIMMx->TIM[TIM_InitStruct->TIMx].LoadCount = TIM_InitStruct->TIM_Period;
}

定時器中斷處理

Air105對應每個定時器, 各有一個中斷處理函數, 可以查看 startup.air105.s 中的中斷向量定義

TIM0_0_IRQHandler
TIM0_1_IRQHandler
TIM0_2_IRQHandler
TIM0_3_IRQHandler
TIM0_4_IRQHandler
TIM0_5_IRQHandler
TIM0_6_IRQHandler
TIM0_7_IRQHandler

對應 Timer0 的中斷處理, 寫在 air105_it.c. TIM_ClearITPendingBit 和 NVIC_ClearPendingIRQ 是必須調用的, 用於清除中斷

void TIM0_0_IRQHandler(void)
{
    TIM_ClearITPendingBit(TIMM0, TIM_0);
    NVIC_ClearPendingIRQ(TIM0_0_IRQn);
}

下麵加入處理邏輯的例子, 每秒調用一次 timer_handler(), 註意不要在中斷處理中使用耗時的工作

extern uint32_t timer_count;
extern void timer_handler(void);

void TIM0_0_IRQHandler(void)
{
    timer_count++;
    if (timer_count >= 1000)
    {
        timer_count = 0;
        timer_handler();
    }

    TIM_ClearITPendingBit(TIMM0, TIM_0);
    NVIC_ClearPendingIRQ(TIM0_0_IRQn);
}

定時器示例代碼

使用Timer0控制板載LED每隔一秒閃爍

https://gitee.com/iosetting/air105_project/tree/master/Demos/Timer/Timer_Blink

Air105 的 PWM

Air105 的8個獨立定時器均可編程產生PWM信號. 當用戶設定TimerNControlReg中PWM比特位為1後,定時器進入PWM工作模式. 此時 PWM 由 TimerNLoadCount2 和 TimerNLoadCount 寄存器分別控制高電平及低電平周期翻轉輸出.

頻率和占空比設置

  • 高電平周期 = (TimerNLoadCount2 + 1) * PCLK_Period
  • 低電平周期 = (TimerNLoadCount + 1) * PCLK_Period

PWM 相關代碼

PWM初始化也只需要三個參數 TIMx 和高低電平兩個周期, 兩者之和就是一個PWM周期

typedef struct 
{
	TIM_NumTypeDef TIMx;
	uint32_t TIM_LowLevelPeriod;
	uint32_t TIM_HighLevelPeriod;
}TIM_PWMInitTypeDef;

用Timer5初始化

void TimerPWM_Init(void)
{
    TIM_PWMInitTypeDef TIM_PWMInitStruct;

    SYSCTRL_APBPeriphClockCmd(SYSCTRL_APBPeriph_TIMM0, ENABLE);
    SYSCTRL_APBPeriphResetCmd(SYSCTRL_APBPeriph_TIMM0, ENABLE);

    //Timer5 -> PWM5
    TIM_PWMInitStruct.TIM_HighLevelPeriod = SYSCTRL->PCLK_1MS_VAL;
    TIM_PWMInitStruct.TIM_HighLevelPeriod = 0;
    TIM_PWMInitStruct.TIMx = TIM_5;
    TIM_PWMInit(TIMM0, &TIM_PWMInitStruct);
    TIM_Cmd(TIMM0, TIM_5, ENABLE);
}

在初始化PWM的庫函數中, 預設將模式設為 user-defined, 自動迴圈載入周期, 並屏蔽中斷

/**
  * @brief  Initializes the TIMx PWM Unit peripheral according to the specified parameters.
  * @param  TIMMx: x can be 0 to select the TIM peripheral
  * @param  TIM_PWMInitStruct: pointer to a TIM_PWMInitTypeDef structor that contains the configuration information
  * @retval None
  */
void TIM_PWMInit(TIM_Module_TypeDef *TIMMx, TIM_PWMInitTypeDef *TIM_PWMInitStruct)
{
    TIM_Cmd(TIMMx, TIM_PWMInitStruct->TIMx, DISABLE);

    TIMMx->TIM[TIM_PWMInitStruct->TIMx].ControlReg = 0;
    TIMMx->TIM[TIM_PWMInitStruct->TIMx].ControlReg |= TIMER_CONTROL_REG_TIMER_MODE;
    TIMMx->TIM[TIM_PWMInitStruct->TIMx].ControlReg |= TIMER_CONTROL_REG_TIMER_PWM;
    TIMMx->TIM[TIM_PWMInitStruct->TIMx].ControlReg |= TIMER_CONTROL_REG_TIMER_INTERRUPT;
    TIMMx->TIM[TIM_PWMInitStruct->TIMx].LoadCount = TIM_PWMInitStruct->TIM_LowLevelPeriod;
    TIMMx->TIM_ReloadCount[TIM_PWMInitStruct->TIMx] = TIM_PWMInitStruct->TIM_HighLevelPeriod;
}

將 PB5 功能復用為 PWM5

GPIO_InitTypeDef gpio;
gpio.GPIO_Pin = GPIO_Pin_5;
gpio.GPIO_Mode = GPIO_Mode_Out_PP;
gpio.GPIO_Remap = GPIO_Remap_2;
GPIO_Init(GPIOB, &gpio);
printf("GPIO Init\r\n");

實時調節占空比, 後兩個參數代表PCLK時鐘周期個數

TIM_SetPWMPeriod(TIMM0, TIM_5, period - high_period, high_period);

PWM示例代碼

使用PWM5(Timer5)控制LED產生呼吸燈效果

https://gitee.com/iosetting/air105_project/tree/master/Demos/PWM/PWM_FadeLED

示例接線:

根據 開發板的BOM PCB 查看 https://wiki.luatos.com/_static/bom/Air105.html
示例中使用Timer4, Timer5對應的PWM4和PWM5輸出, 使用的是PB4和PB5, 對應開發板的SP2_MOSP2_MI, 開發板上的PWM5對應的是PC7, 要註意, 別接錯了.
運行示例, 將兩個LED各自串接一個1-5K的電阻, 分別接GND後接在SP2_MOSP2_MI上, 就能看到呼吸燈的效果了


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • pycharm 常用快捷鍵 更多教程請點擊查看gale博客🚀 最重要的快捷鍵 ctrl+shift+A:萬能命令行 shift兩次:查看資源文件 新建工程第一步操作 module設置把空包分層去掉,compact empty middle package 設置當前的工程是utf-8,設置的Edit ...
  • 背景 我在的學校校園網登錄是web式的,即隨便打開一個網頁就會自動跳轉到登錄頁面,然後輸入用戶名密碼,點登錄,便可以上網了。 但這種登錄方式有個缺點:登錄狀態不會一直保持下去。即過一段時間就會掉線,然後你需要重新登陸才行。這個時間大概是一天。 這就蛋疼了,想讓實驗室的電腦隨時保持聯網狀態怎麼辦呢?( ...
  • 今天分享一篇文章,是關於如何使用 Manim 這個工具 Python 工具庫來製作視頻的。 據我所知,目前應該是沒有專門的書籍和教程來介紹這個工具的。至於教程,不同版本的Manim有一部分文檔,其中 Manim社區 版的文檔相對而言要完善些。 本次僅介紹 Manim 中 文本 的使用,使用的版本為 ...
  • 程式員都知道寫代碼是一件低調又枯燥的事情,一天到晚盯著電腦屏幕看。怎麼能讓寫代碼變成一件酷炫的事情,那就從裝扮編輯器開始。 安裝了這些插件,保證同事看到後,都會問你。 兄弟,你安裝了什麼插件,讓我也裝一下。 1. 先換個漂亮主題 Vuesion Theme Idea編輯器的界麵灰矇矇的,有點性冷淡的 ...
  • 一、題目 描述 給定一個僅包含0和1的n*n二維矩陣,請計算二維矩陣的最大值。 計算規則如下 1、每行元素按下標順序組成一個二進位數(下標越大約排在低位),二進位數的值就是該行的值,矩陣各行之和為矩陣的值 2、允許通過向左或向右整體迴圈移動每個元素來改變元素在行中的位置 比如:[1,0,1,1,1] ...
  • 做下記錄, 首先插入一個dataGridView控制項,兩個button按鈕(導入數據,導出數據),一個ComboBox(獲取列標題使用),一個textbox(輸入關鍵字),一個定位按鈕(定位使用) 1,導入數據(NPOI) 1 2 private void daoRuShuJu_cmd_Click( ...
  • 最近在看 C++ 的方法和類模板,我就在想 C# 中也是有這個概念的,不過叫法不一樣,人家叫模板,我們叫泛型,哈哈,有點意思,這一篇我們來聊聊它們底層是怎麼玩的? 一:C++ 中的模板玩法 畢竟 C++ 是相容 C 語言,而 C 是過程式的玩法,所以 C++ 就出現了兩種模板類型,分別為:函數模板 ...
  • 一、CDN是什麼? CDN的全稱是Content Delivery Network,即內容分髮網絡。其目的是通過在現有的Internet中增加一層新的CACHE(緩存)層,將網站的內容發佈到最接近用戶的網路”邊緣“的節點,使用戶可以就近取得所需的內容(就近原則),提高用戶訪問網站的響應速度。從技術上 ...
一周排行
    -Advertisement-
    Play Games
  • 概述:在C#中,++i和i++都是自增運算符,其中++i先增加值再返回,而i++先返回值再增加。應用場景根據需求選擇,首碼適合先增後用,尾碼適合先用後增。詳細示例提供清晰的代碼演示這兩者的操作時機和實際應用。 在C#中,++i 和 i++ 都是自增運算符,但它們在操作上有細微的差異,主要體現在操作的 ...
  • 上次發佈了:Taurus.MVC 性能壓力測試(ap 壓測 和 linux 下wrk 壓測):.NET Core 版本,今天計劃準備壓測一下 .NET 版本,來測試並記錄一下 Taurus.MVC 框架在 .NET 版本的性能,以便後續持續優化改進。 為了方便對比,本文章的電腦環境和測試思路,儘量和... ...
  • .NET WebAPI作為一種構建RESTful服務的強大工具,為開發者提供了便捷的方式來定義、處理HTTP請求並返迴響應。在設計API介面時,正確地接收和解析客戶端發送的數據至關重要。.NET WebAPI提供了一系列特性,如[FromRoute]、[FromQuery]和[FromBody],用 ...
  • 原因:我之所以想做這個項目,是因為在之前查找關於C#/WPF相關資料時,我發現講解圖像濾鏡的資源非常稀缺。此外,我註意到許多現有的開源庫主要基於CPU進行圖像渲染。這種方式在處理大量圖像時,會導致CPU的渲染負擔過重。因此,我將在下文中介紹如何通過GPU渲染來有效實現圖像的各種濾鏡效果。 生成的效果 ...
  • 引言 上一章我們介紹了在xUnit單元測試中用xUnit.DependencyInject來使用依賴註入,上一章我們的Sample.Repository倉儲層有一個批量註入的介面沒有做單元測試,今天用這個示例來演示一下如何用Bogus創建模擬數據 ,和 EFCore 的種子數據生成 Bogus 的優 ...
  • 一、前言 在自己的項目中,涉及到實時心率曲線的繪製,項目上的曲線繪製,一般很難找到能直接用的第三方庫,而且有些還是定製化的功能,所以還是自己繪製比較方便。很多人一聽到自己畫就害怕,感覺很難,今天就分享一個完整的實時心率數據繪製心率曲線圖的例子;之前的博客也分享給DrawingVisual繪製曲線的方 ...
  • 如果你在自定義的 Main 方法中直接使用 App 類並啟動應用程式,但發現 App.xaml 中定義的資源沒有被正確載入,那麼問題可能在於如何正確配置 App.xaml 與你的 App 類的交互。 確保 App.xaml 文件中的 x:Class 屬性正確指向你的 App 類。這樣,當你創建 Ap ...
  • 一:背景 1. 講故事 上個月有個朋友在微信上找到我,說他們的軟體在客戶那邊隔幾天就要崩潰一次,一直都沒有找到原因,讓我幫忙看下怎麼回事,確實工控類的軟體環境複雜難搞,朋友手上有一個崩潰的dump,剛好丟給我來分析一下。 二:WinDbg分析 1. 程式為什麼會崩潰 windbg 有一個厲害之處在於 ...
  • 前言 .NET生態中有許多依賴註入容器。在大多數情況下,微軟提供的內置容器在易用性和性能方面都非常優秀。外加ASP.NET Core預設使用內置容器,使用很方便。 但是筆者在使用中一直有一個頭疼的問題:服務工廠無法提供請求的服務類型相關的信息。這在一般情況下並沒有影響,但是內置容器支持註冊開放泛型服 ...
  • 一、前言 在項目開發過程中,DataGrid是經常使用到的一個數據展示控制項,而通常表格的最後一列是作為操作列存在,比如會有編輯、刪除等功能按鈕。但WPF的原始DataGrid中,預設只支持固定左側列,這跟大家習慣性操作列放最後不符,今天就來介紹一種簡單的方式實現固定右側列。(這裡的實現方式參考的大佬 ...