stm32输出指定数量PWM脉冲终于搞定啦!!

2019-07-21 05:53发布

本帖最后由 lmywudi 于 2017-11-6 16:45 编辑

摸索了一天,终于解决了这个问题!很有成就感。
[mw_shl_code=c,true]GPIO_InitTypeDef GPIO_InitStructure;
        TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
        USART_InitTypeDef  USART_InitStructure;
        TIM_OCInitTypeDef  TIM_OCInitStructure;
         NVIC_InitTypeDef   NVIC_InitStructure;

        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_TIM1|RCC_APB2Periph_USART1,ENABLE);
        //RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
        TIM_DeInit(TIM3);
        GPIO_DeInit(GPIOA);
        USART_DeInit(USART1);

        GPIO_InitStructure.GPIO_Pin=GPIO_Pin_8|GPIO_Pin_9;
        GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
        GPIO_InitStructure.GPIO_Mode=GPIO_Mode_AF_PP;
        GPIO_Init(GPIOA,&GPIO_InitStructure);
        
        GPIO_InitStructure.GPIO_Pin=GPIO_Pin_7;
        GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;
        GPIO_Init(GPIOA,&GPIO_InitStructure);
        GPIO_SetBits(GPIOA,GPIO_Pin_7);

        GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10; //USART1_RX PA.10
        GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //¸¡¿Õêäèë
        GPIO_Init(GPIOA, &GPIO_InitStructure);

        USART_InitStructure.USART_BaudRate = 9600;
        USART_InitStructure.USART_WordLength = USART_WordLength_8b;
        USART_InitStructure.USART_StopBits = USART_StopBits_1;
        USART_InitStructure.USART_Parity = USART_Parity_No;
        USART_InitStructure.USART_HardwareFlowControl =USART_HardwareFlowControl_None;
        USART_InitStructure.USART_Mode = USART_Mode_Tx | USART_Mode_Rx;
        USART_Init(USART1, &USART_InitStructure);
        USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);
        USART_Cmd(USART1, ENABLE);

        TIM_TimeBaseStructure.TIM_Period=x;
        TIM_TimeBaseStructure.TIM_Prescaler=71;
        TIM_TimeBaseStructure.TIM_ClockDivision=TIM_CKD_DIV1;
        TIM_TimeBaseStructure.TIM_CounterMode=TIM_CounterMode_Up;
        TIM_TimeBaseInit(TIM1,&TIM_TimeBaseStructure);
        TIM_ITConfig(TIM1,TIM_IT_Update,ENABLE );

        TIM_OCInitStructure.TIM_OCMode=TIM_OCMode_PWM2;
        //TIM_OCInitStructure.TIM_Channel=TIM_Channel_1;
        TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable;
        TIM_OCInitStructure.TIM_Pulse=0;
        TIM_OCInitStructure.TIM_OCPolarity=TIM_OCPolarity_Low;
        TIM_OC1Init(TIM1, &TIM_OCInitStructure);
        TIM_OC1PreloadConfig(TIM1, TIM_OCPreload_Enable);
        TIM_CtrlPWMOutputs(TIM1,ENABLE);
        TIM_ARRPreloadConfig(TIM1, ENABLE);
        TIM_CtrlPWMOutputs(TIM1,ENABLE);

        NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ; //ÇàÕ¼óÅÏ輶 3
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //×óóÅÏ輶 3
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ í¨μàê1Äü
        NVIC_Init(&NVIC_InitStructure);
   
        NVIC_InitStructure.NVIC_IRQChannel = TIM1_UP_IRQn; //TIM3 ÖD¶Ï
        NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //ÏèÕ¼óÅÏ輶 0 ¼¶
        NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //′óóÅÏ輶 3 ¼¶
        NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ í¨μà±»ê1Äü
        NVIC_Init(&NVIC_InitStructure);

        TIM_Cmd(TIM1,ENABLE);
        TIM_SetCompare1(TIM1, x/2);
        
        while(1)
        {
                 
        }
}
void TIM1_UP_IRQHandler(void) //TIM3 ÖD¶Ï
{
if (TIM_GetITStatus(TIM1, TIM_IT_Update) != RESET) //¼ì2é TIM3 ¸üDÂÖD¶Ï·¢éúóë·ñ
  {
//          x=~x;
    TIM_ClearITPendingBit(TIM1, TIM_IT_Update ); //Çå3y TIM3 ¸üDÂÖD¶Ï±êÖ¾
        y++;
        if(y>1600)//1600为指定脉冲数,1600个脉冲刚好使步进电机转一圈。
        {
                y=0;
                TIM_Cmd(TIM1,DISABLE);
        }
        USART_SendData(USART1, 0x00);
//     if(x)GPIO_SetBits(GPIOA,GPIO_Pin_8);
//          else GPIO_ResetBits(GPIOA,GPIO_Pin_8);
  }[/mw_shl_code]

用这个方法可以很精确的控制步进电机的位置

不过还有个问题不明白,NVIC_InitStructure.NVIC_IRQChannel = TIM1_UP_IRQn;

在官方库中,中断通道是这样写的:TIM1_UP_IRQChannel
但没法这样用,提示未定义,但在库中没有发现TIM1_UP_IRQn这种写法,现在已经搞不清楚了
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
15条回答
aiyeba
1楼-- · 2019-07-22 01:13
lmywudi 发表于 2017-11-10 11:00
使用TIM1的中断啊,每发一个脉冲就中断一次

妙哉!!

其实用的就是定时器的溢出中断吧。
1115074159
2楼-- · 2019-07-22 04:19
 精彩回答 2  元偷偷看……
lmywudi
3楼-- · 2019-07-22 09:35
1115074159 发表于 2017-11-22 10:02
你好 可以私发我一个您做的整个文件吗 我想仔细研究一下 如果方便的话 发到 不胜感激!!!

这就是main函数里所有代码了,其它文件都是函数库
三生有幸
4楼-- · 2019-07-22 12:06
中断计数只适合低频状态,我测试过一路脉冲时,6M仿真就看不到脉冲,3M的时候可以看到。

两路定时器的时候,3M时,仿真都没有脉冲
jinfeihan57
5楼-- · 2019-07-22 13:16
查看STM32F10x_StdPeriph_Driver_3.5.0这个文件,可以找到TIM1_UP_IRQn的位置和用法。
风呼呼
6楼-- · 2019-07-22 14:59
翱翔云端的鸟 发表于 2018-3-15 09:30
定时器内部互连可以解决

你好,有相应的代码供参考吗?分享分享吧多谢

一周热门 更多>