***配套代码工程***
LLC数字控制TMS320F28034,3-DSP的timer定时器配置介绍
- LLC数字控制TMS320F28034,3-DSP的timer定时器配置介绍
 - 1 TMS320F28034
 - 1.1 系统时钟大小
 - 1.2 TMS320F28034 芯片系统时钟源介绍
 
- 2 时钟库函数说明示例
 - 3 TMS320F28034手写定时器功能
 - 4 定时器运用
 - 5 工程代码
 - 6 定时器中断总结
 
***配套代码工程***
LLC数字控制TMS320F28034,3-DSP的timer定时器配置介绍
***配套代码工程***
1 TMS320F28034
1.1 系统时钟大小
TMS320F28034 的系统时钟即为其主频。根据第一节手册的介绍,该系统时钟的最大值为 60MHz。在实际工程应用中,为了防止系统在满频时出现不稳定的运行状况,通常会将时钟频率降低一些,例如配置为 56MHz 或 40MHz。当然,具体的配置需要根据各功能模块的需求,合理地设定主频大小。
 在某些数字高速控制 PWM 的情况下,主频使用会比较高;
 在某些高速通讯的情况下,主频使用也会比较高;
 在需要系统快速响应的功能设计时,主频使用同样会比较高。
1.2 TMS320F28034 芯片系统时钟源介绍

TMS320F28034 是德州仪器(TI)生产的一款高性能数字信号处理器(DSP)。它具有多种时钟源,可以灵活地配置系统时钟,以满足不同应用场景的需求。
 系统时钟源
 TMS320F28034 提供了多个时钟源供用户选择,包括:
- 内部振荡器(Internal Oscillator)
内部低频振荡器 (INTOSC1 和 INTOSC2):
芯片内置两个低频振荡器,通常工作在 10MHz 左右。它们可以作为系统时钟的备用或低功耗模式下的时钟源。 - 外部晶振(External Crystal Oscillator)
外部晶振:
支持外部晶振输入,可以通过外部晶体或振荡器产生高精度的时钟信号。常用频率有 20MHz、25MHz 等。 - 外部时钟输入(External Clock Input)
外部时钟输入 (XCLKIN):
可以直接从外部输入时钟信号,灵活性较高,适用于对时钟源有特殊要求的应用场景。 - 零等待状态片上振荡器(Zero Wait-State On-Chip Oscillator)
零等待状态片上振荡器:
提供高性能的时钟源,适用于需要快速响应和高性能的应用。
时钟配置
为了满足不同应用的需求,TMS320F28034 提供了灵活的时钟配置选项:
锁相环 (PLL):
TMS320F28034 集成了一个锁相环 (PLL),可以将输入时钟频率倍增,以生成高频系统时钟。PLL 的倍频系数可以编程配置,以满足不同的频率需求。
时钟分频器 (Clock Dividers):
时钟分频器可以将高频系统时钟分频,生成适合不同模块使用的时钟信号。这使得系统能够在不同的功能模块之间合理分配时钟资源。
应用场景
高性能控制应用:
在需要高性能和快速响应的控制应用中,可以使用外部晶振加锁相环的组合,生成高频系统时钟,以满足性能需求。
低功耗应用:
在低功耗应用中,可以使用内部振荡器或外部低频时钟源,减少功耗。
灵活配置:
根据具体应用需求,灵活配置系统时钟源和时钟频率,以实现最佳的系统性能和功耗平衡。
以上是 TMS320F28034 芯片系统时钟源的简要介绍。通过合理配置时钟源和时钟频率,可以充分发挥该芯片的性能,满足各种应用场景的需求。 
2 时钟库函数说明示例
TMS320F28034 芯片官方时钟库函数说明
 TMS320F28034 是德州仪器(TI)生产的一款高性能数字信号处理器(DSP),用于嵌入式控制应用。TI 提供了丰富的库函数,以简化开发过程。以下是一些常用的与时钟配置相关的官方库函数说明。
 库函数
- InitSysCtrl
功能:初始化系统控制寄存器,包括启用时钟、配置PLL、设置系统时钟频率等。
使用示例: 
InitSysCtrl();
 
- InitPll
功能:配置并启用锁相环 (PLL)。
参数: 
pllMult: PLL 的倍频系数。
clkDiv: 系统时钟分频系数。
 
使用示例:
InitPll(10, 2);  // 设置 PLL 倍频系数为 10,系统时钟分频系数为 2
 
- SetVCLK
功能:设置外设时钟 (VCLK) 的分频系数。
参数: 
div: VCLK 的分频系数。
 
使用示例:
SetVCLK(2);  // 设置 VCLK 分频系数为 2
 
- InitPeripheralClocks
功能:初始化外设时钟,启用所需的外设时钟。
使用示例: 
InitPeripheralClocks();
 
- DisablePeripheralClocks
功能:禁用特定外设的时钟,以节省功耗。
参数: 
peripheral: 需要禁用时钟的外设。
 
使用示例:
DisablePeripheralClocks(SYSCTL_PERIPH_CLK_SPIA);  // 禁用 SPI A 的时钟
 
- EnablePeripheralClocks
功能:启用特定外设的时钟。
参数: 
peripheral: 需要启用时钟的外设。
 
使用示例:
EnablePeripheralClocks(SYSCTL_PERIPH_CLK_SPIA);  // 启用 SPI A 的时钟
 
- SysClkOutConfig
功能:配置系统时钟输出到外部引脚。
参数: 
clkOutDiv: 系统时钟输出的分频系数。
 
使用示例:
SysClkOutConfig(2);  // 设置系统时钟输出的分频系数为 2
 
示例代码
 以下是一个完整的示例代码,展示如何使用上述库函数来配置系统时钟:
#include "F2803x_Device.h"
#include "F2803x_Examples.h"
void main(void)
{
    // 初始化系统控制
    InitSysCtrl();
    // 配置并启用 PLL
    InitPll(10, 2);  // 设置 PLL 倍频系数为 10,系统时钟分频系数为 2
    // 初始化外设时钟
    InitPeripheralClocks();
    // 启用特定外设时钟
    EnablePeripheralClocks(SYSCTL_PERIPH_CLK_SPIA);
    // 配置系统时钟输出到外部引脚
    SysClkOutConfig(2);  // 设置系统时钟输出的分频系数为 2
    while (1)
    {
        // 主循环
    }
}
 
通过使用这些库函数,开发者可以方便地配置 TMS320F28034 的时钟系统,以满足各种应用需求。详细的函数说明和更多使用示例可以参考 TI 官方提供的文档和示例代码。
3 TMS320F28034手写定时器功能
我们通过定时器配置,实现在主频60M情况下的两个定时器溢出中断功能,定时市场分别为20us和1ms定时器。
 (1)首先我们选择内部时钟振荡器
 (2)配置分频倍频系数
// Defines
//
//#define DSP28_DIVSEL   0      // Enable /4 for SYSCLKOUT
//#define DSP28_DIVSEL   1      // Disable /4 for SYSCKOUT
#define DSP28_DIVSEL   2      // Enable /2 for SYSCLKOUT
//#define DSP28_DIVSEL   3      // Enable /1 for SYSCLKOUT
#define DSP28_PLLCR   12    //Uncomment for 60 MHz devs [60 MHz=(10MHz * 12)/2]
//#define DSP28_PLLCR   11
//#define DSP28_PLLCR   10
//#define DSP28_PLLCR    9
//#define DSP28_PLLCR    8  //Uncomment for 40 MHz devs [40 MHz=(10MHz * 8)/2]
//#define DSP28_PLLCR    7
//#define DSP28_PLLCR    6
//#define DSP28_PLLCR    5
//#define DSP28_PLLCR    4
//#define DSP28_PLLCR    3
//#define DSP28_PLLCR    2
//#define DSP28_PLLCR    1
//#define DSP28_PLLCR    0  // PLL is bypassed in this mode
//
 
(3)定时器配置和中断配置
void FunTimerInit(void)
{
    EALLOW;  // This is needed to write to EALLOW protected registers
    //打开对应定时器时钟
    SysCtrlRegs.PCLKCR3.bit.CPUTIMER0ENCLK = 1;
    SysCtrlRegs.PCLKCR3.bit.CPUTIMER1ENCLK = 1;
    // CPU 定时器 0
    // 寄存器地址指针和各自计时器初始化:
    CpuTimer0.RegsAddr = &CpuTimer0Regs;
    // 定时器周期寄存器最大值初始化:
    CpuTimer0Regs.PRD.all  = 0xFFFFFFFF;
    // 预分频寄存器初始化为1 (SYSCLKOUT):
    CpuTimer0Regs.TPR.all  = 0;
    CpuTimer0Regs.TPRH.all = 0;
    // 确保定时器0停止:
    CpuTimer0Regs.TCR.bit.TSS = 1;
    // 重加载计数器周期值:
    CpuTimer0Regs.TCR.bit.TRB = 1;
    // 复位中断计时器:
    CpuTimer0.InterruptCount = 0;
    // CPU 定时器 1
    // 寄存器地址指针和各自计时器初始化:
    CpuTimer1.RegsAddr = &CpuTimer1Regs;
    // 定时器周期寄存器最大值初始化:
    CpuTimer1Regs.PRD.all  = 0xFFFFFFFF;
    // 预分频寄存器初始化为1 (SYSCLKOUT):
    CpuTimer1Regs.TPR.all  = 0;
    CpuTimer1Regs.TPRH.all = 0;
    // 确保定时器1停止:
    CpuTimer1Regs.TCR.bit.TSS = 1;
    // 重加载计数器周期值:
    CpuTimer1Regs.TCR.bit.TRB = 1;
    // 复位中断计时器:
    CpuTimer1.InterruptCount = 0;
    //定时器时长配置
    //60:代表主频60M,
    //20:代表20us,
    //1000:代表1ms
    ConfigCpuTimer(&CpuTimer0,60,20);
    ConfigCpuTimer(&CpuTimer1,60,1000);
    StartCpuTimer0();
    StartCpuTimer1();
    PieVectTable.TINT0 = &CpuTimer0Isr;//往中断矢量表中填写定时器0一个指针
    PieVectTable.TINT1 = &CpuTimer1Isr;//往中断矢量表中填写定时器1一个指针
    //打开PIE组对应中断
    PieCtrlRegs.PIEIER1.bit.INTx7 = 1; //使能第一组7通道定时器0
    IER |= M_INT1;  // Enable CPU INT
    IER |= M_INT13; //根据中断向量表使能定时器1
    EDIS;   // This is needed to disable write to EALLOW protected registers
}
//定时器中断0的中断服务函数
interrupt void CpuTimer0Isr(void)
{
    //============================================================
    CpuTimer0Regs.TCR.bit.TIF  = 1; //清除外设级中断标志位
    PieCtrlRegs.PIEACK.bit.ACK1= 1; //清除PIE级中断应答
}
//定时器中断0的中断服务函数
interrupt void CpuTimer1Isr(void)
{
    //============================================================
    CpuTimer1Regs.TCR.bit.TIF  = 1; //清除外设级中断标志位
    PieCtrlRegs.PIEACK.bit.ACK1= 1; //清除PIE级中断应答
}
 
4 定时器运用
(1)使用定时器0实现板子一个灯的1S闪烁功能
//定时器中断0的中断服务函数==20us
static unsigned int  timercnt= 0;
interrupt void CpuTimer0Isr(void)
{
    timercnt++;
    //1S时长
    if(timercnt>50000)
    {
        timercnt = 0;
        //指示灯的引脚状态进行反转
        LEDRTOGGLE();
        LEDYTOGGLE();
    }
    //============================================================
    CpuTimer0Regs.TCR.bit.TIF  = 1; //清除外设级中断标志位
    PieCtrlRegs.PIEACK.bit.ACK1= 1; //清除PIE级中断应答
}
 
(2)使用定时器1实现精准的毫秒级计时函数
//delay_ms时间函数
static unsigned char delay_flag = 0;
static unsigned int delay_cnt = 0;
void delay_ms(unsigned int timer)
{
    delay_flag = 1;
    delay_cnt = 0;
    while(delay_cnt<timer)
    {
        ;
    }
    delay_flag = 0;
    delay_cnt = 0;
}
 
在定时器1中进行计时
//时间计数的代码程序
    if(delay_flag)
    {
        delay_cnt++;
        if(delay_cnt>65500)
        {
            delay_cnt = 65500;
        }
        else
        {
            ;
        }
    }
    else
    {
        delay_cnt = 0;
    }
 
5 工程代码

/*
- includes.h
 - Created on: 2024年7月29日
 -  
Author: 
*/
#ifndef APP_INCLUDES_H_
#define APP_INCLUDES_H_
#include "Flash2803x_API_Library.h"
#include "Flash2803x_API_Config.h"
#include "DSP28x_Project.h"
#include "DSP2803x_DefaultISR.h"
#include "DSP2803x_Adc.h"
#include "Flash2803x_API_Config.h"
#include "string.h"
#include "sys_gpio.h"
#include "sys_timer.h"
//------------------------------------------
void MemCopy(Uint16 *SourceAddr, Uint16* SourceEndAddr, Uint16* DestAddr);
extern void Init_system(void);
extern void Init_system_end(void);
#endif /* APP_INCLUDES_H_ */
 
//Main.c文件中
#include "includes.h"
/********************************************************
* 函数名称:Init_system
* 功 能:   系统初始化
* 入口参数:
* 出口参数:
* 修 改:
********************************************************/
void Init_system(void)
{
    InitSysCtrl();
    DINT;
    InitPieCtrl();
    IER = 0x0000;
    IFR = 0x0000;
    InitPieVectTable();
    EALLOW;
    Flash_CPUScaleFactor = SCALE_FACTOR;
    Flash_CallbackPtr=NULL;
    EDIS;
    MemCopy(&RamfuncsLoadStart,&RamfuncsLoadEnd, &RamfuncsRunStart);
    InitFlash();
}
/********************************************************
* 函数名称:void Init_system_end(void)
* 功 能:   系统初始化结束
* 入口参数:
* 出口参数:
* 修 改:
********************************************************/
void Init_system_end(void)
{
    EALLOW;  // This is needed to write to EALLOW protected registers
    PieCtrlRegs.PIECTRL.bit.ENPIE = 1;   // Enable the PIE block
    EDIS;    // This is needed to disable write to EALLOW protected registers
    //================================================================================
    EINT;
    ERTM;
}
/********************************************************
函数名称:main
函数功能:主程序
入口参数:
出口参数:
修    改:
********************************************************/
void main(void)
{
    Init_system();
    FunTimerInit();
    FunGpioInit();
    Init_system_end();
//
    while(1)
    {
        //计时200ms
        delay_ms(200);
        //指示灯的引脚状态进行反转
        LEDGTOGGLE();
    }
}
 
/*
- sys_gpio.c
 - Created on: 2024年7月30日
 -  
Author: 
*/
#include "includes.h"
void FunGpioInit(void)
{
    GpioDataRegs.GPADAT.all = 0ul;
    GpioDataRegs.GPBDAT.all = 0ul;
    //GPIO初始化
    //开始寄存器配置前执行指令
    EALLOW;
    // GPIOA DATA数据全清0
    GpioDataRegs.GPADAT.all = 0ul;
    // GPIOB DATA数据全清0
    GpioDataRegs.GPBDAT.all = 0ul;
    //LED G
    GpioCtrlRegs.GPAMUX2.bit.GPIO24=0;
    GpioCtrlRegs.GPADIR.bit.GPIO24=1;
    //LED Y
    GpioCtrlRegs.GPAMUX2.bit.GPIO21=0;
    GpioCtrlRegs.GPADIR.bit.GPIO21=1;
    //LED R
    GpioCtrlRegs.GPAMUX2.bit.GPIO20=0;
    GpioCtrlRegs.GPADIR.bit.GPIO20=1;
    // GPIO0 <-> EPWM1A == 0:GPIO功能 1:PWM功能
    GpioCtrlRegs.GPAMUX1.bit.GPIO0=0;
    // GPIO1 <-> EPWM1B == 0:GPIO功能 1:PWM功能
    GpioCtrlRegs.GPAMUX1.bit.GPIO1=0;
    // GPIO0 <-> EPWM1A == 0:输出功能 1:输入功能
    GpioCtrlRegs.GPADIR.bit.GPIO0=1;
    // GPIO1 <-> EPWM1B == 0:输出功能 1:输入功能
    GpioCtrlRegs.GPADIR.bit.GPIO1=1;
    // GPIO0 <-> EPWM1A == 1:输出0
    GpioDataRegs.GPACLEAR.bit.GPIO0=1;
    // GPIO1 <-> EPWM1B == 1:输出0
    GpioDataRegs.GPACLEAR.bit.GPIO1=1;
    // COMP_OUT GPIO42 <-> OPP_COMP1_OUT
    GpioCtrlRegs.GPBMUX1.bit.GPIO42 = 3;
    // AI02<-> COPM1+ ==2:比较器功能
    GpioCtrlRegs.AIOMUX1.bit.AIO2 = 2;
    // GPIO15 <-> TZ1
    GpioCtrlRegs.GPAMUX1.bit.GPIO15 = 1;
    //寄存器配置结束后执行指令
    EDIS;
}
 
/*
- sys_gpio.h
 - Created on: 2024年7月30日
 -  
Author: 
*/
#ifndef APP_SYS_GPIO_H_
#define APP_SYS_GPIO_H_
//===========================================================================
#define LEDRGPIO        GPIO20
#define LEDRMUX         GpioCtrlRegs.GPAMUX2.bit.LEDRGPIO
#define LEDRDIR         GpioCtrlRegs.GPADIR.bit.LEDRGPIO
#define LEDRTOGGLE()    GpioDataRegs.GPATOGGLE.bit.LEDRGPIO = 1
#define LEDRON()        GpioDataRegs.GPASET.bit.LEDRGPIO =1
#define LEDRLOW()       GpioDataRegs.GPACLEAR.bit.LEDRGPIO =1
#define LEDYGPIO        GPIO21
#define LEDYMUX         GpioCtrlRegs.GPAMUX2.bit.LEDYGPIO
#define LEDYDIR         GpioCtrlRegs.GPADIR.bit.LEDYGPIO
#define LEDYTOGGLE()    GpioDataRegs.GPATOGGLE.bit.LEDYGPIO = 1
#define LEDYON()        GpioDataRegs.GPASET.bit.LEDYGPIO =1
#define LEDYLOW()       GpioDataRegs.GPACLEAR.bit.LEDYGPIO =1
#define LEDGGPIO        GPIO24
#define LEDGMUX         GpioCtrlRegs.GPAMUX2.bit.LEDGGPIO
#define LEDGDIR         GpioCtrlRegs.GPADIR.bit.LEDGGPIO
#define LEDGTOGGLE()    GpioDataRegs.GPATOGGLE.bit.LEDGGPIO = 1
#define LEDGON()        GpioDataRegs.GPASET.bit.LEDGGPIO =1
#define LEDGLOW()       GpioDataRegs.GPACLEAR.bit.LEDGGPIO =1
extern void FunGpioInit(void);
#endif /* APP_SYS_GPIO_H_ */
 
/*
- sys_timer.c
 - Created on: 2024年8月1日
 -  
Author: 
*/
#include "includes.h"
void FunTimerInit(void)
{
    EALLOW;  // This is needed to write to EALLOW protected registers
    //打开对应定时器时钟
    SysCtrlRegs.PCLKCR3.bit.CPUTIMER0ENCLK = 1;
    SysCtrlRegs.PCLKCR3.bit.CPUTIMER1ENCLK = 1;
    // CPU 定时器 0
    // 寄存器地址指针和各自计时器初始化:
    CpuTimer0.RegsAddr = &CpuTimer0Regs;
    // 定时器周期寄存器最大值初始化:
    CpuTimer0Regs.PRD.all  = 0xFFFFFFFF;
    // 预分频寄存器初始化为1 (SYSCLKOUT):
    CpuTimer0Regs.TPR.all  = 0;
    CpuTimer0Regs.TPRH.all = 0;
    // 确保定时器0停止:
    CpuTimer0Regs.TCR.bit.TSS = 1;
    // 重加载计数器周期值:
    CpuTimer0Regs.TCR.bit.TRB = 1;
    // 复位中断计时器:
    CpuTimer0.InterruptCount = 0;
    // CPU 定时器 1
    // 寄存器地址指针和各自计时器初始化:
    CpuTimer1.RegsAddr = &CpuTimer1Regs;
    // 定时器周期寄存器最大值初始化:
    CpuTimer1Regs.PRD.all  = 0xFFFFFFFF;
    // 预分频寄存器初始化为1 (SYSCLKOUT):
    CpuTimer1Regs.TPR.all  = 0;
    CpuTimer1Regs.TPRH.all = 0;
    // 确保定时器1停止:
    CpuTimer1Regs.TCR.bit.TSS = 1;
    // 重加载计数器周期值:
    CpuTimer1Regs.TCR.bit.TRB = 1;
    // 复位中断计时器:
    CpuTimer1.InterruptCount = 0;
    //定时器时长配置
    //60:代表主频60M,
    //20:代表20us,
    //1000:代表1ms
    ConfigCpuTimer(&CpuTimer0,60,20);
    ConfigCpuTimer(&CpuTimer1,60,1000);
    StartCpuTimer0();
    StartCpuTimer1();
    PieVectTable.TINT0 = &CpuTimer0Isr;//往中断矢量表中填写定时器0一个指针
    PieVectTable.TINT1 = &CpuTimer1Isr;//往中断矢量表中填写定时器1一个指针
    //打开PIE组对应中断
    PieCtrlRegs.PIEIER1.bit.INTx7 = 1; //使能第一组7通道定时器0
    IER |= M_INT1;  // Enable CPU INT
    IER |= M_INT13; //根据中断向量表使能定时器1
    EDIS;   // This is needed to disable write to EALLOW protected registers
}
//delay_ms时间函数
static unsigned char delay_flag = 0;
static unsigned int delay_cnt = 0;
void delay_ms(unsigned int timer)
{
    delay_flag = 1;
    delay_cnt = 0;
    while(delay_cnt<timer)
    {
        ;
    }
    delay_flag = 0;
    delay_cnt = 0;
}
//定时器中断0的中断服务函数==20us
static unsigned int  timercnt= 0;
interrupt void CpuTimer0Isr(void)
{
    timercnt++;
    //1S时长
    if(timercnt>50000)
    {
        timercnt = 0;
        //指示灯的引脚状态进行反转
        LEDRTOGGLE();
        LEDYTOGGLE();
    }
    //============================================================
    CpuTimer0Regs.TCR.bit.TIF  = 1; //清除外设级中断标志位
    PieCtrlRegs.PIEACK.bit.ACK1= 1; //清除PIE级中断应答
}
//定时器中断1的中断服务函数==1000us
interrupt void CpuTimer1Isr(void)
{
    //时间计数的代码程序
    if(delay_flag)
    {
        delay_cnt++;
        if(delay_cnt>65500)
        {
            delay_cnt = 65500;
        }
        else
        {
            ;
        }
    }
    else
    {
        delay_cnt = 0;
    }
    //============================================================
    CpuTimer1Regs.TCR.bit.TIF  = 1; //清除外设级中断标志位
    PieCtrlRegs.PIEACK.bit.ACK1= 1; //清除PIE级中断应答
}
 
/*
- sys_timer.h
 - Created on: 2024年8月1日
 -  
Author: 
*/
#ifndef APP_SYS_TIMER_H_
#define APP_SYS_TIMER_H_
extern void FunTimerInit(void);
extern void delay_ms(unsigned int timer);
extern interrupt void CpuTimer0Isr(void);
extern interrupt void CpuTimer1Isr(void);
#endif /* APP_SYS_TIMER_H_ */
 
***配套代码工程***
6 定时器中断总结
TMS320F28034 提供了多种定时器模块,包括 CPU 定时器、eCAP、ePWM 和 eQEP 等。通过合理配置和使用这些定时器模块,可以实现各种定时任务,例如周期性任务、输入捕获、PWM 信号生成等。掌握这些定时器的使用方法,对于开发高性能嵌入式控制系统至关重要。
***配套代码工程***


















