作者共发了1篇帖子。 内容转换:不转换▼
 
点击 回复
648 0
【应用】使用STM32定时器的Encoder模式驱动数字旋转编码开关
一派护法 十九级
1楼 发表于:2017-3-15 20:42
/* 本程序使用的是锐志电子六合一扩展板上的数字旋转编码开关 */
/* 注意:一定要把板子下面的JPBMA和JPBMB跳线插上 */
#include <stm32f10x.h>

int main(void)
{
    /* 打开外设时钟 */
    RCC->APB1ENR = RCC_APB1ENR_TIM2EN | RCC_APB1ENR_TIM3EN;
    RCC->APB2ENR = RCC_APB2ENR_IOPAEN | RCC_APB2ENR_IOPBEN | RCC_APB2ENR_IOPCEN;
    
    /* I/O端口配置 */
    GPIOA->CRL = 0x88000000; // PA6为通道1, PA7为通道2, 设为带电阻输入
    GPIOA->BSRR = GPIO_BSRR_BS6 | GPIO_BSRR_BS7; // PA6~7带上拉电阻输入
    GPIOA->CRH = 0x00080000; // PA12设为带电阻输入
    GPIOA->BSRR = GPIO_BSRR_BS12;
    GPIOB->CRH = 0x33333333; // 数码管段选
    GPIOC->ODR = 0x3000; // 熄灭数码管
    GPIOC->CRH = 0x00333333; // 数码管位选
    
    /* 数码管扫描中断配置 */
    // 每隔一定时间点亮一个数码管
    // 从低位向高位扫描
    TIM2->ARR = 29; // 0~29有30个数字, 定时时间30*0.1ms=3ms, 向上计数, 每当TIM2->CNT跳变到0时触发一次中断
    TIM2->PSC = 7199; // 对72MHz进行7200分频, 得到的频率是10kHz, 也就是0.1ms
    TIM2->EGR = TIM_EGR_UG; // 立即触发TIM2中断, 点亮数码管, 同时刷新寄存器
    TIM2->DIER = TIM_DIER_UIE; // 允许触发定时器溢出中断
    NVIC_EnableIRQ(TIM2_IRQn); // 允许执行中断服务函数, 该函数位于core_cm3.h, 是MDK自带的函数, 不是库函数
    TIM2->CR1 = TIM_CR1_CEN; // 开定时器
    
    /* 数字编码开关计数器配置 */
    // 将数字编码开关的1、3脚分别接到TIM3定时器的通道1~2输入端口PA6~7上
    TIM3->ARR = TIM_ARR_ARR; // 计数的最大值为65535
    TIM3->CCMR1 = TIM_CCMR1_CC1S_0; // CC1S=01(IC1映射到TI1)
    TIM3->CCMR1 |= TIM_CCMR1_CC2S_0; // CC2S=01(IC2映射到TI2)
    
    // 以下三种模式任选一种即可
    TIM3->SMCR = TIM_SMCR_SMS_0; // SMS=001, 对通道1计数
    //TIM3->SMCR = TIM_SMCR_SMS_1; // SMS=010, 对通道2计数
    //TIM3->SMCR = TIM_SMCR_SMS_1 | TIM_SMCR_SMS_0; // SMS=011, 同时对通道1,2计数(计数速度更快)
    
    TIM3->CCER = TIM_CCER_CC1P; // 反转TI1的电平, 顺时针旋转时使计数值增加而不是减少(默认是减少)
    // 也可以改为反转TI2的电平, 这个和上面SMS的选择无关
    
    TIM3->EGR = TIM_EGR_UG; // 刷新寄存器
    TIM3->CR1 = TIM_CR1_CEN; // 开始计数
    
    /* 开PA12外中断 */
    EXTI->IMR = EXTI_IMR_MR12;
    EXTI->FTSR = EXTI_FTSR_TR12; // 下降沿触发
    NVIC_EnableIRQ(EXTI15_10_IRQn);
    NVIC_SetPriority(EXTI15_10_IRQn, 1); // 该中断优先级比数码管扫描中断的优先级低, 不会影响数码管扫描
    
    while (1);
}

// 数码管动态扫描
void TIM2_IRQHandler(void)
{
    static uint8_t pos = 5;
    static uint32_t numbuf;
    const uint8_t seg8[] = {0xc0, 0xf9, 0xa4, 0xb0, 0x99, 0x92, 0x82, 0xf8, 0x80, 0x90};
    if (TIM2->SR & TIM_SR_UIF) // 定时器2溢出中断
    {
        TIM2->SR &= ~TIM_SR_UIF; // 清中断标志
        if (pos == 5)
            numbuf = TIM3->CNT; // 取定时器3的计数值
        else
            numbuf /= 10;
        
        GPIOC->ODR = (GPIOC->ODR & 0xf0ff) | 0x3000; // 关闭位选
        GPIOB->ODR = (GPIOB->ODR & 0xff) | (seg8[numbuf % 10] << 8); // 设置显示字符
        if (pos <= 3)
            GPIOC->ODR |= 1 << (pos + 8); // 数码管1~4的位选直连I/O端
        else
            GPIOC->ODR &= ~(1 << (pos + 8)); // 数码管5~6的位选是由PNP三极管驱动的
        if (pos == 0)
            pos = 5;
        else
            pos--;
    }
}

// 编码开关按键按下中断
void EXTI15_10_IRQHandler(void)
{
    EXTI->PR |= EXTI_PR_PR12; // 清中断标志
    TIM3->EGR = TIM_EGR_UG; // 重置定时器3, 使计数值归0
}

回复帖子

内容:
用户名: 您目前是匿名发表
验证码:
(快捷键:Ctrl+Enter)
 

本帖信息

点击数:648 回复数:0
评论数: ?
作者:巨大八爪鱼
最后回复:巨大八爪鱼
最后回复时间:2017-3-15 20:42
 
©2010-2024 Arslanbar Ver2.0
除非另有声明,本站采用知识共享署名-相同方式共享 3.0 Unported许可协议进行许可。