| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 嵌入式 -> stm32单片机实现旋钮功能 -> 正文阅读 |
|
[嵌入式]stm32单片机实现旋钮功能 |
旋钮初始化设置对应的管脚为输入模式,然后再设置初始电平。 代码如下 void?Task_knob_init() { u32 pin_value;; bxd_gpio_set_mode(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_MODE_INPUT); ?//设置引脚23为输入模式 bxd_gpio_set_pull(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_PULLDOWN); ??//将引脚23电平拉低 bxd_gpio_set_mode(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_MODE_INPUT); ?//设置引脚15为输入模式 bxd_gpio_set_pull(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_PULLUP); ??//将引脚15电平拉高 bxd_gpio_set_mode(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_MODE_INPUT); ?//设置引脚17为输入模式 bxd_gpio_set_pull(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_PULLUP); ??//将引脚17电平拉高 }? 根据EncoderScan的返回值res判断左旋右旋,并打印log该函数主要通过“res = EncoderScan(gpio_A, gpio_B);”这条语句来执行左旋右旋的判断。EncoderScan函数应在编码器的AB两路信号中任何一个发生变化时,两者进行异或运算。如果AB两路信号都能产生双边沿中断,则两个中断程序里都调用此程序 代码如下 void?left_right_deal() { char?data[2] = { 0 }; unsigned?char?res = 0; if?(2 == Flag) { data[0] = 0X03; bx_logln("Main \n"); Flag = 0; } else { res = EncoderScan(gpio_A, gpio_B); if?(res & (ENC_1_R | ENC_1_L)) { if?(res & ENC_1_R) { data[0] = 0X01; bx_logln("Right \n"); Flag = 0; } else?if?(res & ENC_1_L) { data[0] = 0X02; bx_logln("Left \n"); Flag = 0; } } } } 设置中断模式并中断使能在Task_knob函数里面,读取初始话后的电平状态并打印出来。用if语句进行判断,如果高电平则设置为下降沿触发,如果低电平则设置为上降沿触发。设置使能中断模式。 代码如下 void?Task_knob() { u32 pin_value;; bxd_gpio_read(KEY_M_GPIO_PORT, &pin_value); ?//读取所有引脚的电平状态 old_gpio_M = (pin_value >> KEY_M_GPIO_PIN) & 0x1; bxd_gpio_read(KEY_A_GPIO_PORT, &pin_value); ?//读取所有引脚的电平状态 old_gpio_A = (pin_value >> KEY_A_GPIO_PIN) & 0x1; bxd_gpio_read(KEY_B_GPIO_PORT, &pin_value); ?//读取所有引脚的电平状态 old_gpio_B = (pin_value >> KEY_B_GPIO_PIN) & 0x1; bx_logln("old_gpio_A=%d, old_gpio_B=%d", old_gpio_A, old_gpio_B); if?(old_gpio_A == 1) { gpio_set_it_mode(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_MODE_IT_FALLING);//下降沿触发中断 } else { gpio_set_it_mode(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_MODE_IT_RISING);//上升沿触发中断 } if?(old_gpio_B == 1) { gpio_set_it_mode(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_MODE_IT_FALLING);//下降沿触发中断 } else { gpio_set_it_mode(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_MODE_IT_RISING);//上升沿触发中断 } if?(old_gpio_M == 1) { gpio_set_it_mode(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_MODE_IT_LOW);//低电平触发中断 } else { gpio_set_it_mode(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_MODE_IT_HIGH);//高电平触发中断 } bxd_gpio_enable_intr(KEY_M_GPIO_PORT); //使能中断 bxd_gpio_enable_intr(KEY_A_GPIO_PORT); //使能中断 bxd_gpio_enable_intr(KEY_B_GPIO_PORT); //使能中断 } while?(1) { if?(Flag) { left_right_deal(); } BX_DELAY_US(10 * 1000); } } 中断处理中断处理过程中需要中断标志,这是因为单片机要靠查询中断标志来判断是否要进入中断,如果不清除中断标志,本次中断退出,单片机又会检测到中断标志,因此重复进入中断。 void?GPIO_IRQHandler(void) { uint32_t int_stat = BX_GPIOA->IS; //设置中断标志 u32 pin_value;//电平状态 static?unsigned?int?gpio_eint_A = 0; static?unsigned?int?gpio_eint_B = 0; static?unsigned?int?gpio_eint_M = 0; gpio_eint_M = (int_stat >> KEY_M_GPIO_PIN) & 0x1; gpio_eint_A = (int_stat >> KEY_A_GPIO_PIN) & 0x1; gpio_eint_B = (int_stat >> KEY_B_GPIO_PIN) & 0x1; if?(gpio_eint_A || gpio_eint_B) { bxd_gpio_read(KEY_A_GPIO_PORT, &pin_value); ?//读取15号引脚的电平状态 gpio_A = (pin_value >> KEY_A_GPIO_PIN) & 0x1; bxd_gpio_read(KEY_B_GPIO_PORT, &pin_value); ?//读取17号引脚的电平状态 gpio_B = (pin_value >> KEY_B_GPIO_PIN) & 0x1; //printf("A B gpio_state is %d, %d, %d, %d\n", gpio_A, gpio_B, old_gpio_A, old_gpio_B); if?(gpio_eint_A) { BX_GPIOA->EOI |= (int_stat >> KEY_A_GPIO_PIN) & 0x1; //清除中断状态 if?(gpio_A != old_gpio_A) { if?(gpio_A == 1) { gpio_set_it_mode(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_MODE_IT_FALLING);//下降沿触发中断 // bx_logln( "A" ); } else?{ gpio_set_it_mode(KEY_A_GPIO_PORT, KEY_A_GPIO_PIN, BX_GPIO_MODE_IT_RISING);//上升沿触发中断 // bx_logln( "a" ); } old_gpio_A = gpio_A; Flag = 1; } bxd_gpio_enable_intr(KEY_A_GPIO_PORT); //15号引脚中断打开 } if?(gpio_eint_B) { BX_GPIOA->EOI |= (int_stat >> KEY_B_GPIO_PIN) & 0x1; //清除中断状态 if?(gpio_B != old_gpio_B) { if?(gpio_B == 1) { gpio_set_it_mode(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_MODE_IT_FALLING);//下降沿触发中断 // bx_logln( "B" ); } else?{ gpio_set_it_mode(KEY_B_GPIO_PORT, KEY_B_GPIO_PIN, BX_GPIO_MODE_IT_RISING);//上升沿触发中断 // bx_logln( "b" ); } old_gpio_B = gpio_B; Flag = 1; } bxd_gpio_enable_intr(KEY_B_GPIO_PORT); //17号引脚中断打开 } } if?(gpio_eint_M) { bxd_gpio_read(KEY_M_GPIO_PORT, &pin_value); ?//读取23号引脚的电平状态 gpio_M = (pin_value >> KEY_M_GPIO_PIN) & 0x1; if?(gpio_eint_M) { BX_GPIOA->EOI |= (int_stat >> KEY_M_GPIO_PIN) & 0x1; //清除中断状态 // bx_logln( "MMMM" ); if?(gpio_M != old_gpio_M) { if?(gpio_M == 1) { gpio_set_it_mode(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_MODE_IT_LOW);//下降沿触发中断 } else?{ gpio_set_it_mode(KEY_M_GPIO_PORT, KEY_M_GPIO_PIN, BX_GPIO_MODE_IT_HIGH);//上升沿触发中断 Flag = 2; } old_gpio_M = gpio_M; } bxd_gpio_enable_intr(KEY_M_GPIO_PORT); //23号引脚外部中断打开 } } else?{ BX_GPIOA->EOI |= (int_stat); //清除中断状态 } } 主函数实现代码如下 int?main(void) { ????ble_init(); ????bx_kernel_init(); ????app_init(); ????Task_knob(); ??while?(1) ????{ ????????ble_schedule(); ????????bx_kernel_schedule(); ????} } ? |
|
嵌入式 最新文章 |
基于高精度单片机开发红外测温仪方案 |
89C51单片机与DAC0832 |
基于51单片机宠物自动投料喂食器控制系统仿 |
《痞子衡嵌入式半月刊》 第 68 期 |
多思计组实验实验七 简单模型机实验 |
CSC7720 |
启明智显分享| ESP32学习笔记参考--PWM(脉冲 |
STM32初探 |
STM32 总结 |
【STM32】CubeMX例程四---定时器中断(附工 |
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 | -2025/1/9 1:36:01- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |