STM32通用定时器使用

发布者:Chunjie2022最新更新时间:2017-11-07 来源: eefocus关键字:STM32  通用定时器 手机看文章 扫描二维码
随时随地手机看文章

STM32中一共有11个定时器,其中2个高级控制定时器,4个普通定时器和2个基本定时器,以及2个看门狗定时器和1个系统嘀嗒定时器。(TIM1和TIM8是能够产生3对PWM互补输出的高级登时其,常用于三相电机的驱动,时钟由APB2的输出产生;TIM2-TIM5是普通定时器;TIM6和TIM7是基本定时器,其时钟由APB1输出产生)

本实验要实现的功能是:用普通定时器TIM2每一秒发生一次更新事件,进入中断服务程序翻转LED1的状态。

预备知识:

① STM32通用定时器TIM2是16位自动重装载计数器。

② 向上计数模式:从0开始计数,计到自动装载寄存器(TIMx_ARR)中的数值时,清0,依次循环。

需要弄清楚的两个问题:

1. 计数器的计数频率是什么?

这个问题涉及到RCC时钟部分,如下图所示:

 

STM32通用定时器使用

       定时器的时钟不是直接来自APB1或APB2,而是来自于输入为APB1或APB2的一个倍频器。

       下面以定时器2~7的时钟说明这个倍频器的作用:当APB1的预分频系数为1时,这个倍频器不起作用,定时器的时钟频率等于APB1的频率;当APB1的预分频系数为其它数值(即预分频系数为2、4、8或16)时,这个倍频器起作用,定时器的时钟频率等于APB1的频率两倍。 

       假定AHB=36MHz,因为APB1允许的最大频率为36MHz,所以APB1的预分频系数可以取任意数值;当预分频系数=1时,APB1=36MHz,TIM2~7的时钟频率=36MHz(倍频器不起作用);当预分频系数=2时,APB1=18MHz,在倍频器的作用下,TIM2~7的时钟频率=36MHz。

有人会问,既然需要TIM2~7的时钟频率=36MHz,为什么不直接取APB1的预分频系数=1?答案是:APB1不但要为TIM2~7提供时钟,而且还要为其它外设提供时钟;设置这个倍频器可以在保证其它外设使用较低时钟频率时,TIM2~7仍能得到较高的时钟频率。

       再举个例子:当AHB=72MHz时,APB1的预分频系数必须大于2,因为APB1的最大频率只能为36MHz。如果APB1的预分频系数=2,则因为这个倍频器,TIM2~7仍然能够得到72MHz的时钟频率。能够使用更高的时钟频率,无疑提高了定时器的分辨率,这也正是设计这个倍频器的初衷。

注意:APB1和APB2上挂的外设如图所示:

 STM32通用定时器使用

       定时器的计数频率有个公式:

TIMx_CLK = CK_INT / (TIM_Prescaler + 1)

其中:TIMx_CLK       定时器的计数频率

      CK_INT         内部时钟源频率(APB1的倍频器送出时钟)

      TIM_Prescaler  用户设定的预分频系数,取值范围0~65535。

例如:RCC中AHB=72MHZ、APB1=36MHZ、APB2=72MHZ,则CK_INT=72MKZ。

2. 如何计算定时时间?

上述公式中TIM_Prescaler涉及到寄存器TIMx_PSC

 

STM32通用定时器使用

STM32通用定时器使用

如果TIM_Prescaler设为36000,由上面公式可知:

定时器的计数频率 TIMx_CLK = 72MKZ / 36000 = 2000HZ,则定时器的计数周期=1/2000HZ=0.5ms.

如果要定时1秒,则需要计数2000次,这也是自动重装载的值。又涉及到TIMx_ARR

STM32通用定时器使用

STM32通用定时器使用

只要上述两个问题搞清楚了,剩下的就是设置相应寄存器的对应位了。

 

LED硬件连接如下图所示:高电平点亮LED。

 STM32通用定时器使用

 

第一步:配置系统时钟。见STM32F103x RCC寄存器配置

除此之外,还需将GPIO和TIM2外设时钟打开。

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);

注意:TIM2是挂在APB1上的,打开时钟时别写错了,调用RCC_APB1PeriphClockCmd函数,而不是RCC_APB2PeriphClockCmd。

第二步:配置中断向量表。见stm32_exti(含NVIC)配置及库函数讲解

void NVIC_Configuration(void)

{

  NVIC_InitTypeDef NVIC_InitStructure;

  

#ifdef  VECT_TAB_RAM  

   

  NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); 

#else  

   

  NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);   

#endif

  

 

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

  NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 4;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

}

该函数完成两个功能

1. 决定将程序下载到RAM中还是FLASH中

2. 配置中断分组。(NVIC中断分组只能设置一次)

3. 选择中断通道号,抢占式优先级和响应优先级,使能中断

第三步:配置GPIO的模式。输入模式还是输出模式。点亮LED已讲过,见STM32_GPIO配置及库函数讲解——LED跑马灯

void GPIO_Configuration(void)

{

  GPIO_InitTypeDef GPIO_InitStructure;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 

  GPIO_Init(GPIOC, &GPIO_InitStructure);

}

第四步:定时器配置,本章重点!

void TIM2_Configuration(void)

{

  TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

  //重新将Timer设置为缺省值

  TIM_DeInit(TIM2);

  //采用内部时钟给TIM2提供时钟源

  TIM_InternalClockConfig(TIM2);

  //预分频系数为36000-1,这样计数器时钟为72MHz/36000 = 2kHz

  TIM_TimeBaseStructure.TIM_Prescaler = 36000 - 1;

  //设置时钟分割

  TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;

  //设置计数器模式为向上计数模式

  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;

  //设置计数溢出大小,每计2000个数就产生一个更新事件

  TIM_TimeBaseStructure.TIM_Period = 2000;

  //将配置应用到TIM2中

  TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
  //清除溢出中断标志

  TIM_ClearFlag(TIM2, TIM_FLAG_Update);

  //禁止ARR预装载缓冲器

  TIM_ARRPreloadConfig(TIM2, DISABLE);  //预装载寄存器的内容被立即传送到影子寄存器 

  //开启TIM2的中断

  TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);

}

该函数完成两个功能

1. 设定预分频系数TIM_Prescaler = 36000 - 1

2. 设定自动重装载值TIM_Period = 2000

注意:上述只是配置好了TIM2,但还没有开启TIM2。

下面给出timer2.c的完整代码

#include "stm32f10x_lib.h"

void RCC_Configuration(void);

void NVIC_Configuration(void);

void GPIO_Configuration(void);

void TIM2_Configuration(void);

void Delay(vu32 nCount);

int main(void)

{

#ifdef DEBUG

  debug();

#endif

  

  RCC_Configuration();

  NVIC_Configuration();

  GPIO_Configuration();

  TIM2_Configuration();

  TIM_Cmd(TIM2, ENABLE); //开启定时器2

  

  while (1)

  {    

  }

}

void RCC_Configuration(void)

{

  ErrorStatus HSEStartUpStatus; 

  RCC_DeInit();

  RCC_HSEConfig(RCC_HSE_ON);

  HSEStartUpStatus = RCC_WaitForHSEStartUp()

  if (HSEStartUpStatus == SUCCESS)

  {

    FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

    FLASH_SetLatency(FLASH_Latency_2);

    RCC_HCLKConfig(RCC_SYSCLK_Div1); 

    RCC_PCLK2Config(RCC_HCLK_Div1); 

    RCC_PCLK1Config(RCC_HCLK_Div2);

    RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);

    RCC_PLLCmd(ENABLE);

    while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET) {}

    RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

    while(RCC_GetSYSCLKSource() != 0x08) {}

  }

  

  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);

  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);

}

void NVIC_Configuration(void)

{

  NVIC_InitTypeDef NVIC_InitStructure;

  

#ifdef  VECT_TAB_RAM  

  NVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); 

#else  

  NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0);   

#endif

  

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

  NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 4;

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

}

void GPIO_Configuration(void)

{

  GPIO_InitTypeDef GPIO_InitStructure;

  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;

  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; 

  GPIO_Init(GPIOC, &GPIO_InitStructure);

}

void TIM2_Configuration(void)

{

  TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

  //重新将Timer设置为缺省值

  TIM_DeInit(TIM2);

  //采用内部时钟给TIM2提供时钟源

  TIM_InternalClockConfig(TIM2);

  //预分频系数为36000-1,这样计数器时钟为72MHz/36000 = 2kHz

  TIM_TimeBaseStructure.TIM_Prescaler = 36000 - 1;

  //设置时钟分割

  TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;

  //设置计数器模式为向上计数模式

  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;

  //设置计数溢出大小,每计2000个数就产生一个更新事件

  TIM_TimeBaseStructure.TIM_Period = 2000;

  //将配置应用到TIM2中

  TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);

  //清除溢出中断标志

  TIM_ClearFlag(TIM2, TIM_FLAG_Update);

  //禁止ARR预装载缓冲器

  TIM_ARRPreloadConfig(TIM2, DISABLE);  //预装载寄存器的内容被立即传送到影子寄存器 

  //开启TIM2的中断

  TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);

}

void Delay(vu32 nCount)

{

  for(; nCount != 0; nCount--);

}

#ifdef  DEBUG

void assert_failed(u8* file, u32 line)

  while (1)

  {

  }

}

#endif

stm32f10x_it.c有关TIM2_IRQHandler代码如下

void TIM2_IRQHandler(void)

{

  if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)

  {

    TIM_ClearITPendingBit(TIM2,TIM_IT_Update);

    GPIO_WriteBit(GPIOC, GPIO_Pin_6, (BitAction)((1 - GPIO_ReadOutputDataBit(GPIOC, GPIO_Pin_6))));

  }

}


 

STM32通用定时器库函数设置

 

STM32的通用定时器为:TIM2、TIM3、TIM4和TIM5

在使用通用定时器时利用库函数直接设置定时器如下:

1.使能定时器TIM_X的时钟:(X=2、3、4、5)

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIMX,ENABLE);

2.计算要定时的时间,根据定时时间来设定分频数和最大计数值(以向上计数为例子),其中计算关系如下:

系统时钟(一般为72MHZ) =定时器分频数 * 计数值

假如分频数为7200,则定时器时钟为:72MHZ/7200=10KHZ,定时器每次计数时间间隔为1/10000秒,假如定时1秒,则要计数10000次,因此计数器的最大计数值为9999,因为计数器从零开始计数。

3.将计算好的分频数和计数值分别赋值以上面定时为例,如下:

/自动重装的计数值 

TIM_TimeBaseStructure.TIM_Period = (10000 - 1);

// 这个就是预分频系数

 TIM_TimeBaseStructure.TIM_Prescaler =7200 ;

//数字滤波器,定时的时候不涉及此功能,为零即

TIM_TimeBaseStructure.TIM_ClockDivision = 0;

//计数模式选择,此处设置为向上模式

TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounteMode_Up;

//定时基本设置((X=2、3、4、5))

TIM_TimeBaseInit(TIMX, &TIM_TimeBaseStructure);

//清除定时器X的中断溢出标识

TIM_ClearITPendingBit(TIMX,TIM_IT_Update);

//开定时器X溢出中断

TIM_ITConfig(TIM5,TIM_IT_Update, ENABLE);

//计数器使能,开始工作

   TIM_Cmd(TIM5, ENABLE); 

到此通用定时器的定时功能配置完成,以上配置代码可写入

void TIMX_Init(void)函数中,函数名自己可变。别忘了在函数中的第一句写入:TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;

4.中断配置,这个按照下面的函数复制即可

void NVIC_Configuration(void)//定时器中断配置

{

NVIC_InitTypeDefNVIC_InitStructure;

NVIC_SetVectorTable(NVIC_VectTab_FLASH,0x0000);

NVIC_InitStructure.NVIC_IRQChannel= TIM5_IRQn;

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority= 0;

NVIC_InitStructure.NVIC_IRQChannelSubPriority= 1;

NVIC_InitStructure.NVIC_IRQChannelCmd= ENABLE;

NVIC_Init(&NVIC_InitStructure);

}

5. 在stm32f10x_it.c添加定时器TIMX的中断函数:

voidTIM5_IRQHandler(void)

{

//如果定时器产生了中断(X=2、3、4、5)

if (TIM_GetITStatus(TIMX,TIM_IT_Update) != RESET)

{

  //这是你要完成的事情和相关判断,自己去写

}

//完成事情和判断后,清除中断 

TIM_ClearITPendingBit(TIM5,TIM_IT_Update);

  

  }

}

到此,定时器定时中断功能实现了

 

 

 

 

STM32 的定时器功能十分强大,有 TIME1 和 TIME8 等高级定时器,也有 TIME2~TIME5 等通用定时器,还有 TIME6 和TIME7 等基本定时器。

STM32 的通用定时器是一个通过可编程预分频器(PSC)驱动的 16 位自动装载计数器(CNT)构成。STM32 的通用定时器可以被用于:测量输入信号的脉冲长度(输入捕获)或者产生输出波形(输出比较和 PWM)等。   使用定时器预分频器和 RCC 时钟控制器预分频器,脉冲长度和波形周期可以在几个微秒到几个毫秒间调整。STM32 的每个通用定时器都是完全独立的,没有互相共享的任何资源。

STM32 的通用 TIMx (TIM2、TIM3、TIM4 和 TIM5)定时器功能包括:

1)16 位向上、向下、向上/向下自动装载计数器(TIMx_CNT)。

2)16 位可编程(可以实时修改)预分频器(TIMx_PSC),计数器时钟频率的分频系数为 1~65535 之间的任意数值。

3)4 个独立通道(TIMx_CH1~4),这些通道可以用来作为: 

A.输入捕获 

B.输出比较 

C.PWM 生成(边缘或中间对齐模式) 

D.单脉冲模式输出 

4)可使用外部信号(TIMx_ETR)控制定时器和定时器互连(可以用 1 个定时器控制另外一个定时器)的同步电路。

5)如下事件发生时产生中断/DMA: 

A.更新:计数器向上溢出/向下溢出,计数器初始化(通过软件或者内部/外部触发) 

B.触发事件(计数器启动、停止、初始化或者由内部/外部触发计数) 

C.输入捕获 

D.输出比较 

E.支持针对定位的增量(正交)编码器和霍尔传感器电路 

F.触发输入作为外部时钟或者按周期的电流管理

定时器相关的库函数主要集中在固件库文件 stm32f10x_tim.h 和 stm32f10x_tim.c 文件中。

1)TIM3 时钟使能。

TIM3 是挂载在 APB1 之下,所以我们通过 APB1 总线下的使能使能函数来使能 TIM3。调用的函数是:

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //时钟使能

2)初始化定时器参数,设置自动重装值,分频系数,计数方式等。

在库函数中,定时器的初始化参数是通过初始化函数 TIM_TimeBaseInit 实现的:

voidTIM_TimeBaseInit(TIM_TypeDef*TIMx,TIM_TimeBaseInitTypeDef* TIM_TimeBaseInitStruct);

第一个参数是确定是哪个定时器,这个比较容易理解。第二个参数是定时器初始化参数结构体指针,结构体类型为 TIM_TimeBaseInitTypeDef。针对 TIM3 初始化范例代码格式:

TIM_TimeBaseInitTypeDef   TIM_TimeBaseStructure;

TIM_TimeBaseStructure.TIM_Period = 5000; //设置自动重载计数周期值

TIM_TimeBaseStructure.TIM_Prescaler =7199;  //设置分频系数

TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分频因子

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //设置计数方式

TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); 

3)设置 TIM3_DIER 允许更新中断。

在库函数里面定时器中断使能是通过 TIM_ITConfig 函数来实现的:

void TIM_ITConfig(TIM_TypeDef* TIMx, uint16_t TIM_IT, FunctionalState NewState);

第一个参数是选择定时器号,这个容易理解,取值为 TIM1~TIM17。

第二个参数非常关键,是用来指明我们使能的定时器中断的类型,定时器中断的类型有很多种,包括更新中断 TIM_IT_Update,触发中断 TIM_IT_Trigger,以及输入捕获中断等等。

第三个参数就很简单了,就是失能还是使能。

例如我们要使能 TIM3 的更新中断,格式为:

TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); 

4)TIM3 中断优先级设置。

在定时器中断使能之后,因为要产生中断,必不可少的要设置 NVIC 相关寄存器,设置中断优先级。

5)允许 TIM3 工作,也就是使能 TIM3。

在配置完后要开启定时器,通过 TIM3_CR1 的 CEN 位来设置。 在固件库里面使能定时器的函数是通过 TIM_Cmd 函数来实现的:

void TIM_Cmd(TIM_TypeDef* TIMx, FunctionalState NewState)

这个函数非常简单,比如我们要使能定时器 3,方法为:

TIM_Cmd(TIM3, ENABLE);   //使能 TIMx 外设

6)编写中断服务函数。

在最后,还是要编写定时器中断服务函数,通过该函数来处理定时器产生的相关中断。在中断产生后,通过状态寄存器的值来判断此次产生的中断属于什么类型。然后执行相关的操作,我们这里使用的是更新(溢出)中断,所以在状态寄存器 SR 的最低位。在处理完中断之后应该向 TIM3_SR 的最低位写 0,来清除该中断标志。

在固件库函数里面,用来读取中断状态寄存器的值判断中断类型的函数是:

ITStatus TIM_GetITStatus(TIM_TypeDef* TIMx, uint16_t)

该函数的作用是,判断定时器 TIMx 的中断类型 TIM_IT 是否发生中断。比如,我们要判断定时器 3 是否发生更新(溢出)中断,方法为:

if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET){}

固件库中清除中断标志位的函数是:

void TIM_ClearITPendingBit(TIM_TypeDef* TIMx, uint16_t TIM_IT) 

该函数的作用是,清除定时器 TIMx 的中断 TIM_IT 标志位。使用起来非常简单,比如我们在TIM3 的溢出中断发生后,我们要清除中断标志位,方法是:

TIM_ClearITPendingBit(TIM3, TIM_IT_Update   );

 

参考代码如下:


  1. void TIM3_Int_Init(u16 arr,u16 psc)

  2. {

  3.     TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

  4.     RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); 


  5.     

  6.     TIM_TimeBaseStructure.TIM_Period = arr; 

  7.     TIM_TimeBaseStructure.TIM_Prescaler = psc; 

  8.     TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; 

  9.     TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; 

  10.     TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); 


  11.     TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE); 

  12.     TIM3_NVIC_Init();


  13.     TIM_Cmd(TIM3,ENABLE);

  14. }



  15. void TIM3_NVIC_Init(void)

  16. {

  17.     NVIC_InitTypeDef NVIC_InitStructure;

  18.     NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; 

  19.     NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; 

  20.     NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; 

  21.     NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; 

  22.     NVIC_Init(&NVIC_InitStructure);

  23. }



  24. void TIM3_IRQHandler(void)

  25. {

  26.     if(TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET ) 

  27.     {

  28.         TIM_ClearITPendingBit(TIM3, TIM_IT_Update); 

  29.         LED0 = !LED0;

  30.     }

  31. }


关键字:STM32  通用定时器 引用地址:STM32通用定时器使用

上一篇:stm32f4xx标准外设固件库
下一篇:STM32与无源蜂鸣器

推荐阅读最新更新时间:2024-03-16 15:43

stm32专题二十:扩展SRAM
STM32 控制器芯片内部有一定大小的 SRAM 及 FLASH 作为内存和程序存储空间,但当程序较大,内存和程序空间不足时,就需要在 STM32芯片的外部扩展存储器了。注意,STM32F103VET6是不支持外扩SRAM的,因为引脚数量不够,而ZET6可以。 这里以IS62WV51216BLL为例: 芯片引脚图如下(40引脚来驱动): 结构框图如下: 引脚说明(要注意UB和LB,也就是高字节和低字节): 读操作时序要求: 可以看到,当给出地址后,最少要经过tAA的时间,才能去读取数据,以及输出时能后,要经过tODE的时间。只有满足这些时间要求,读到的数据才是正确的。 具体的时间参数: 同样的,写操作时
[单片机]
<font color='red'>stm32</font>专题二十:扩展SRAM
基于STM32单片机的DLP驱动电路研究
DLP投影技术是应用美国德州仪器公司开发的数字微镜元件--DMD(Digital Micromirror Device)作为主要关键处理元件以实现数字光学处理过程的技术。DLP显示的色彩清晰度高、艳丽、细腻、逼真,且为全数字显示即可靠性极高,能在各类产品(如大屏幕数字电视、公司/家庭/专业会议投影机和数码相机(DLP Cinema))中提供最佳图像效果。目前,大部分的家用或商用DLP投影机都采用了单片结构,使得其便于移动携带,因而得到越来越广泛的应用。在目前应用发展的基础上,又对其结构的精简性、携带的方便性提出了更高的要求。传统的DLP投影仪是通过DVI接口接收外部信号,并且经过信号转换传送给DLP控制器来控制DLP的显示,占用的
[单片机]
基于<font color='red'>STM32</font>单片机的DLP驱动电路研究
STM32按键控制流水灯 转换|加速|减慢
单片机源程序如下: #include stdio.h #include stm32f4xx.h #include led.h #include key.h int main(void) { int pwm_cmd; SysTick_CLKSourceConfig(SysTick_CLKSource_HCLK); NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); LED_Init(); KEY_Init(); EXTI_Config(); while(1) { }
[单片机]
<font color='red'>STM32</font>按键控制流水灯 转换|加速|减慢
STM32+Keil 中使用printf函数
Keil不支持Host-semi机制,即不支持直接在IDE打印字符串。 那么只能通过程序向硬件串口发数据了,这样调用的时候用自定义的函数即可,也很方便,例如: void send_char_to_usart(unsigned char c){} 但是可否直接使用printf函数呢?毕竟人家都做好了,我们给他定一个打印输出的接口就可以了,答案是肯定的,看ST的官方源码: /** ****************************************************************************** * @file Lib_DEBUG/Lib_DEBUG_Example/main.c *
[单片机]
STM32单片机按键电路设计
以前GPIO上接的电路都是低电平触发的那种,新画了个板子有一个按键设计的是高电平触发,结果IO口设置成上拉输入后,怎么读都不正确,按键电路如图1。无奈只能去调试一下,发现当设置为上拉输入后,其ODR(GPIO输出数据寄存器)相应的也置为1,百思不得其解。 图1 按键电路 于是去看一下GPIO的结构图: I/O引脚结构图 由图可以发现其实输出寄存器与输入寄存器之间在I/O引脚处是线与状态,在GPIO设置成上位输入时,其上拉电阻闭合,这时如果输出寄存器设置为0那么在IO口内部就会自己损耗电流,而这些电流的损耗是会增加功耗的。因此当GPIO设置成上拉输入时,相应的输出寄存器也设置为1. 将按键电路与I/O
[单片机]
<font color='red'>STM32</font>单片机按键电路设计
基于STM32的伺服压力机控制系统设计
1 引言 随着制造业朝着生产规模化、产品个性化的方向发展,产品型号变化加快,生产批量相对变小,多种型号共线生产、覆盖件大型化,一体化的趋势日益明显,要求压力机不仅能够高速度、高精度、大负载的运转,而且应具有更大的柔性,能迅速、方便地改变输出运动规律。以上这些不断提高的加工工艺和生产条件的多样化都促进了锻压设备行业的重大变革。在需求的推动下,目前,世界上出现伺服电机直接驱动的伺服压力机,能够满足塑性加工、难成形材料成形、复杂形状零件成形、复合成形以及高精度成形等成形工艺的要求。伺服压力机采用交流伺服电机代替普通交流电机,去除传统曲柄压力机上的飞轮和离合器,使得压力机机械结构得以简化,可靠性增强。所以研制适合不同材料加工要求的压力机数
[单片机]
基于<font color='red'>STM32</font>的伺服压力机控制系统设计
2023 STM32全国巡展,米尔限量发STM32MP135开发板优惠券
2023年9月12日至10月27日,以“STM32,不止于芯”为主题的第十六届STM32全国巡回研讨会将走进11个城市,本届研讨会为全天会议,我们将围绕STM32最新产品开展技术演讲和方案演示。 本次STM32全国研讨会,米尔电子将现场展出STM32相关的最新产品技术和应用演示,为工程师们提供共话交流共同探索开放式平台,届时,米尔电子的技术工程师将与各位嘉宾面对面深入交流。感兴趣的客户和朋友欢迎莅临参观了解! 米尔电子的明星产品将精彩亮相 米尔MYC-YF13X 核心板及开发板 基于STM32MP1系列MPU,STM32MP135高性价比入门级MPU设计平台,完整的安全生态系统让MCU开发者友好地过渡到MPU平
[单片机]
2023 <font color='red'>STM32</font>全国巡展,米尔限量发STM32MP135开发板优惠券
STM32】HAL库 STM32CubeMX教程十二---IIC(读取AT24C02 )
前言: 本系列教程将HAL库与STM32CubeMX结合在一起讲解,使您可以更快速的学会各个模块的使用 在之前的标准库中,STM32的硬件IIC非常复杂,更重要的是它并不稳定,所以都不推荐使用。 但是在我们的HAL库中,对硬件IIC做了全新的优化,使得之前软件IIC几百行代码,在HAL库中,只需要寥寥几行就可以完成 那么这篇文章将带你去感受下它的优异之处 这可能是目前关于STM32CubeMX的硬件iic 讲的最全面和详细的一篇文章之一了 所用工具: 1、芯片: STM32F103ZET6 2、STM32CubeMx软件 3、IDE: MDK-Keil软件 4、STM32F1xx/STM32F4xxHAL库 5、II
[单片机]
【<font color='red'>STM32</font>】HAL库 STM32CubeMX教程十二---IIC(读取AT24C02 )
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved