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();
}
}
stm32单片机实现旋钮功能相关推荐
- 基于STM32单片机实现多功能MP3播放器系统设计
百度网盘下载地址(942):点击下载 本项目是基于STM32F103来制作一个多功能MP3,除了可以实现MP3播放的基本功能之外,同时拥有丰富并实用的外扩功能,整个系统的功能包括:MP3播放功能.收音 ...
- STM32单片机使用ADC功能驱动手指检测心跳模块
一.模块简介 某宝或某多,两三块钱一个,如下图. 该模块采用超亮红外LED和光敏晶体管来探测手指的脉搏,将手指放在发射和接收端之间,血压随着脉搏变化,接收端收到的光会发生相应的变化,因此可用来检测心跳 ...
- 基于STM32单片机的多功能智能时钟【DHT11温湿度传感器蜂鸣器报警12864液晶显示】
系统功能 系统可以显示日期.时间.星期.温度.湿度,可以对日期时间进行设置,可以设置闹钟,可以打开和关闭背光显示,有两种计时模式:正计时和倒计时. 液晶显示界面
- STM32单片机智能全自动多功能洗碗机加热放水烘干紫外消毒
实践制作DIY- GC0090-全自动多功能洗碗机 一.功能说明: 基于STM32单片机设计-全自动多功能洗碗机 功能介绍: 硬件组成: STM32F103C系列最小系统板 +LCD1602显示器+水 ...
- STM32单片机实现DMA+ADC+UART功能
突然想测试一下STM32单片机ADC采样速率问题,按照常规方法,可以通过ADC采样,然后将采样值打印出来.但是这种方法在处理和打印数据的时候会占用很多时间,导致处理数据的时间超过了ADC的采样时间.于 ...
- 【单片机】Android手机USB外接STM32单片机通过ADB实现投屏反向控制的功能
Android手机USB外接STM32单片机通过ADB实现投屏反向控制的功能 前言 一.功能演示 二.实现步骤 1.开发环境 2.代码分析 (1)USBHost处理函数 (2)ADB事务处理函数 (3 ...
- CubeMX配置STM32实现httpd服务器CGI功能并使用网页控制STM32单片机(四)
CubeMX配置STM32实现httpd服务器CGI功能并使用网页控制STM32单片机 引言 CubeMX配置HTTPD的CGI功能 实验过程 发现的问题 总结 引言 在前三篇文章中自己介绍了如何配置 ...
- 基于stm32单片机多功能可调时钟闹钟万年历计时器闹铃提醒Proteus仿真(源码+仿真+原理图+PCB)
资料编号:148 视频讲解: 148-基于stm32单片机多功能可调时钟闹钟万年历计时器闹铃提醒Proteus仿真(源码+仿真+原理图+PCB) 本设计采用stm32单片机作为主控,LCD1602显 ...
- 基于STM32单片机的智能停车场车位管理系统设计
摘 要 通过调查发现,现有的许多公共场所的停车位管理落后,智能化程度不高.为顺应现代自动化狂潮的发展趋势,本项目以STM32单片机为主控芯片,基于RFID智能识别技术,设计了一个具有IC识别的智能停 ...
最新文章
- 如何查看方法在哪里被调用
- hdu4932 Miaomiao#39;s Geometry (BestCoder Round #4 枚举)
- 在windows中手动安装第三方模块
- 《众妙之门——网页排版设计制胜秘诀》——3.4 展现品牌视觉的同时保持网页的可读性...
- 递归和迭代路由_静态路由在以太网接口中的不同书写会导致路由器怎样的操作结果?...
- WCF入门学习3-配置文件与部署iis
- 课后作业-阅读任务-阅读笔记3
- ubuntu环境下android开发环境安装
- JavaScript 函数参数默认值
- 翻译软件Bob安装教程
- 简单的程序诠释C++ STL算法系列之八:mismatch
- React SSR 服务器端渲染
- 用request模块爬取拉钩招聘信息
- 潜在解决方法-系统映像还原失败,找不到可用于恢复系统盘的磁盘
- 【SQL学习笔记】之数据定义语言(DDL)
- 2022年全国职业院校技能大赛:网络系统管理项目 B模块-Windows部署(10套样题)
- 数学中随机性和不确定性有何不同?
- RhcsaLinux初次使用进行简单的操作
- 洛谷:P5707 【深基2.例12】上学迟到
- NOIP2018提高组省一冲奖班模测训练2 T3 XYK的音游
热门文章
- android读写速度,安卓第一机皇堆“狠”料:UFS 3.0读取速度超1500Mb/s
- LM详解 GPT3,GPT2, GPT1 论文译读
- 商淘软件单店版功能特点
- 全站最全Win快捷键!
- oppoA83怎么升级android版本,OPPO A83 A1刷机教程_OPPO A83 A1卡刷升级更新官方系统包...
- English--初识音标--phonetic symbol_Introduction
- NFT正在 + 什么?
- 【阿里云ACE 北京】[AA 羽毛球休闲] 3月11日 大兴体育局羽毛球馆
- android手势密码csdn,Android手势密码LockPatternView、LockPasswordUtils、LockPatternUtils等分析...
- 达索系统SIMULIA助力油气行业