STM32F103 USART2仅输入,使用PA2输出PWM

开启USART2之后PA2无论是否使用都会被USART2影响,导致TIM2输出PWM时PA2的状态不会改变,但是TIM的中断事件可以正常触发,PWM的本质就是IO口的电平变换,所以这里完全可以在中断里面通过软件设置的方式实现电平的转换。

GPIO初始化

GPIO_InitTypeDef GPIO_InitStruct;

GPIO_InitStruct.GPIO_Pin = GPIO_Pin_2;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStruct);

GPIO_InitStruct.GPIO_Pin = GPIO_Pin_3;
GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStruct);

USART初始化

USART_InitTypeDef USART_InitStruct;

USART_InitStruct.USART_BaudRate = 9600;
USART_InitStruct.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStruct.USART_Mode = USART_Mode_Rx; // 仅初始化RX
USART_InitStruct.USART_Parity = USART_Parity_No;
USART_InitStruct.USART_StopBits = USART_StopBits_1;
USART_InitStruct.USART_WordLength = USART_WordLength_8b;
USART_Init(USART2, &USART_InitStruct);

TIM定时器初始化

TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
TIM_OCInitTypeDef TIM_OCInitStruct;
NVIC_InitTypeDef NVIC_InitStruct;

TIM_TimeBaseInitStruct.TIM_Period = 0xFFFF;
TIM_TimeBaseInitStruct.TIM_Prescaler = 71;
TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseInitStruct);

TIM_OCInitStruct.TIM_Pulse = 0;
TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_Toggle; // 使用输出比较模式
TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;
TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;
TIM_OCInitStruct.TIM_OCIdleState = TIM_OCIdleState_Set;
TIM_OC3Init(TIM2, &TIM_OCInitStruct);
TIM_OC3PreloadConfig(TIM2, TIM_OCPreload_Enable);

NVIC_InitStruct.NVIC_IRQChannel = TIM2_IRQn;
NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;
NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0;
NVIC_Init(&NVIC_InitStruct);

TIM_ITConfig(TIM2, TIM_IT_CC3, ENABLE);
TIM_Cmd(TIM2, ENABLE);

定时器中断

typedef struct {
    uint32_t frequency;
    uint32_t dutyCycle;
} TIMOC_TypeDef;

// 设置frequency可以修改频率
// 设置dutyCycle可以修改占空比
__IO TIMOC_TypeDef TIM2OC3 = { 100, 0 };

void TIM2_IRQHandler(){
    uint16_t cap, ccr = 1;
    if(TIM_GetITStatus(TIM2, TIM_IT_CC3)){
        if(TIM2OC3.dutyCycle >= 100){ // 占空比100%,持续输出高电平
            GPIOA->BSRR = GPIO_Pin_2;
        }else if(TIM2OC3.dutyCycle <= 0){ // 占空比0%,持续输出低电平
            GPIOA->BRR = GPIO_Pin_2;
        }else{
            ccr = 1000000 / TIM2OC3.frequency;
            cap = TIM_GetCapture3(TIM2);
            if(GPIOA->ODR & GPIO_Pin_2){
                ccr -= ccr * TIM2OC3.dutyCycle / 100;
                GPIOA->BRR = GPIO_Pin_2;
            }else{
                ccr = ccr * TIM2OC3.dutyCycle / 100;
                GPIOA->BSRR = GPIO_Pin_2;
            }
        }
        TIM_SetCompare3(TIM2, cap + ccr);
    }
    TIM_ClearITPendingBit(TIM2, TIM_IT_CC3);
}

不过这个方法有个缺点,在PWM输出期间串口会一直输出数据

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值