零基础RT-thread第五节:电容按键(2)

上一章的电容按键完全使用的HAL库的代码,并没有使用线程。这里尝试使用线程来控制电容按键。
依旧是 F767

本来以为会很容易实现,没想到尝试了很久,电容按键一直没有反应。

static rt_uint32_t measure_charge_time(void)
{// 步骤1: 放电 (PA5输出低电平)rt_pin_mode(CAP_KEY_PIN, PIN_MODE_OUTPUT);rt_pin_write(CAP_KEY_PIN, PIN_LOW);rt_hw_us_delay(200);  // 充分放电// 步骤2: 切换到输入捕获模式rt_pin_mode(CAP_KEY_PIN, PIN_MODE_INPUT);// 步骤3: 重置计数器并启动捕获__HAL_TIM_SET_COUNTER(&htim2, 0);HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL);// 步骤4: 开始充电rt_pin_write(CAP_KEY_PIN, PIN_HIGH);rt_hw_us_delay(1);  // 确保充电开始// 等待捕获完成 (等待10ms)rt_uint32_t start_tick = rt_tick_get();while (__HAL_TIM_GET_FLAG(&htim2, TIM_FLAG_CC1) == RESET) {if ((rt_tick_get() - start_tick) > rt_tick_from_millisecond(10)) {break;}}// 获取捕获值rt_uint32_t capture_value = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL);HAL_TIM_IC_Stop_IT(&htim2, TIM_CHANNEL);// 清除捕获标志__HAL_TIM_CLEAR_FLAG(&htim2, TIM_FLAG_CC1);return capture_value;
}

上面这一段是原来的,按键捕获函数,也就是靠 这段代码来捕获PA5的上升沿。
然后,函数卡在了
在这里插入图片描述
这部分。这是为什么呢?逻辑上没有什么问题。其实很简单,充电错了:
在这里插入图片描述
这里是有3.3V电源的。所以不应该 通过PA5充电。
在这里插入图片描述
那我们改成PIN_LOW.对吗?
还是不对,要通过使用TIM得到电容按键的充电时间,就需要捕捉上升沿,我们下拉引脚是得不到上升沿的,所以,我们这里需要NOPULL。
可是。。。RTthread的PIN模式里面竟然没有NOPULL
在这里插入图片描述
原来是RT直接将 INPUT拆分了。PIN_MODE_INPUT,就是NOPULL。看来我的基础 还需要巩固,引脚的电平模式都还没有区分清楚。

按照上一章配置RTthread后,接下来就是完整的代码:
Tpad_tim2.c:

#include <rtthread.h>
#include <rthw.h>
#include <board.h>
#include <rtdevice.h>
#include <drv_common.h>
//#include <tim.h>  // 包含STM32Cube HAL的TIM头文件// 硬件配置
#define CAP_KEY_PIN     GET_PIN(A, 5)  // PA5引脚
#define TIM_CHANNEL     TIM_CHANNEL_1   // TIM2_CH1// 软件配置
#define SAMPLE_INTERVAL 20             // 采样间隔(ms)
#define TOUCH_THRESHOLD 1.2            // 触发阈值倍数
#define CALIBRATION_COUNT 50           // 校准采样次数
#define MAX_CHARGE_TIME 5000           // 最大充电时间(us)static TIM_HandleTypeDef htim2;// 全局状态变量
static rt_uint32_t baseline = 0;        // 基准电容值(无触摸)
static struct rt_timer touch_timer;     // 触摸定时器
static rt_bool_t touched = RT_FALSE;    // 当前触摸状态// 回调函数指针
static void (*key_press_callback)(void) = RT_NULL;
static void (*key_release_callback)(void) = RT_NULL;/*** @brief 设置按键回调函数* @param press_cb 按键按下回调函数* @param release_cb 按键释放回调函数*/
void cap_key_set_callback(void (*press_cb)(void), void (*release_cb)(void))
{key_press_callback = press_cb;key_release_callback = release_cb;
}/*** @brief 初始化TIM2输入捕获*/
static void tim2_capture_init(void)
{// 1. 使能TIM2时钟__HAL_RCC_TIM2_CLK_ENABLE();// 2. 配置基础定时器设置htim2.Instance = TIM2;htim2.Init.Prescaler = 84 - 1;  // 84MHz/84 = 1MHz (1us精度)htim2.Init.CounterMode = TIM_COUNTERMODE_UP;htim2.Init.Period = 0xFFFF;htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE;if (HAL_TIM_IC_Init(&htim2) != HAL_OK) {rt_kprintf("TIM2 init failed!\n");return;}// 3. 配置输入捕获通道TIM_IC_InitTypeDef sConfigIC = {0};sConfigIC.ICPolarity = TIM_INPUTCHANNELPOLARITY_RISING;sConfigIC.ICSelection = TIM_ICSELECTION_DIRECTTI;sConfigIC.ICPrescaler = TIM_ICPSC_DIV1;sConfigIC.ICFilter = 0;if (HAL_TIM_IC_ConfigChannel(&htim2, &sConfigIC, TIM_CHANNEL) != HAL_OK) {rt_kprintf("TIM2 channel config failed!\n");return;}// 4. 配置GPIO引脚复用GPIO_InitTypeDef GPIO_InitStruct = {0};__HAL_RCC_GPIOA_CLK_ENABLE();GPIO_InitStruct.Pin = GPIO_PIN_5;GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;GPIO_InitStruct.Alternate = GPIO_AF1_TIM2;HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);// 5. 配置中断HAL_NVIC_SetPriority(TIM2_IRQn, 2, 0);HAL_NVIC_EnableIRQ(TIM2_IRQn);// 6. 启动定时器基础计数HAL_TIM_Base_Start(&htim2);rt_kprintf("TIM2 capture initialized\n");
}/*** @brief 测量充电时间(单位:us)** 使用TIM2的输入捕获功能测量PA5的充电时间*/
static rt_uint32_t measure_charge_time(void)
{GPIO_InitTypeDef GPIO_InitStruct = {0};// 步骤1: 放电 (PA5输出低电平)rt_pin_mode(CAP_KEY_PIN, PIN_MODE_OUTPUT);rt_pin_write(CAP_KEY_PIN, PIN_LOW);rt_hw_us_delay(200);  // 充分放电// 步骤2: 切换到输入捕获模式rt_pin_mode(CAP_KEY_PIN, PIN_MODE_INPUT);// 步骤3: 重置计数器并启动捕获__HAL_TIM_SET_COUNTER(&htim2, 0);HAL_TIM_IC_Start_IT(&htim2, TIM_CHANNEL_1);// 步骤4: 开始充电//rt_pin_mode(CAP_KEY_PIN, PIN_MODE_INPUT);GPIO_InitStruct.Pin = GPIO_PIN_5;GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;GPIO_InitStruct.Alternate = GPIO_AF1_TIM2;GPIO_InitStruct.Pull = GPIO_NOPULL;GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);rt_hw_us_delay(1);  // 确保充电开始// 等待捕获完成 (等待10ms)rt_uint32_t start_tick = rt_tick_get();while (__HAL_TIM_GET_FLAG(&htim2, TIM_FLAG_CC1) == RESET) {// 超时保护if ((rt_tick_get() - start_tick) > rt_tick_from_millisecond(10)) {rt_kprintf("Capture timeout! Counter: %d\n", __HAL_TIM_GET_COUNTER(&htim2));break;}}// 获取捕获值rt_uint32_t capture_value = HAL_TIM_ReadCapturedValue(&htim2, TIM_CHANNEL);HAL_TIM_IC_Stop_IT(&htim2, TIM_CHANNEL);// 清除捕获标志__HAL_TIM_CLEAR_FLAG(&htim2, TIM_FLAG_CC1);return capture_value;
}/*** @brief 高级校准算法*/
static void advanced_calibration(void)
{rt_uint32_t samples[CALIBRATION_COUNT];rt_uint32_t min = 0xFFFFFFFF, max = 0;rt_uint32_t sum = 0;// 第一阶段:采集原始样本for (int i = 0; i < CALIBRATION_COUNT; i++) {samples[i] = measure_charge_time();if (samples[i] < min) min = samples[i];if (samples[i] > max) max = samples[i];sum += samples[i];rt_thread_mdelay(10);}// 第二阶段:剔除异常值(20%边界)rt_uint32_t range = max - min;rt_uint32_t low_bound = min + range / 5;rt_uint32_t high_bound = max - range / 5;rt_uint32_t valid_sum = 0;rt_uint16_t valid_count = 0;for (int i = 0; i < CALIBRATION_COUNT; i++) {if (samples[i] >= low_bound && samples[i] <= high_bound) {valid_sum += samples[i];valid_count++;}}// 第三阶段:设定基准值if (valid_count > (CALIBRATION_COUNT / 3)) {baseline = valid_sum / valid_count;rt_kprintf("Calibrated baseline: %d us\n", baseline);} else {baseline = (min + max) / 2;rt_kprintf("Fallback baseline: %d us\n", baseline);}
}/*** @brief 触摸检测定时器回调函数*/
static void touch_timer_handler(void *parameter)
{rt_uint32_t current_value = measure_charge_time();// 调试输出static int count = 0;if (++count % 1000 == 0) { // 每1000次采样输出一次rt_kprintf("[%d] Current: %d us, Baseline: %d us, Threshold: %d us\n",count, current_value, baseline, (rt_uint32_t)(baseline * TOUCH_THRESHOLD));}// 检测触摸if (current_value > (rt_uint32_t)(baseline * TOUCH_THRESHOLD) &&current_value < MAX_CHARGE_TIME) {if (!touched) {touched = RT_TRUE;rt_kprintf("TOUCH detected! Value: %d us\n", current_value);// 调用按键按下回调if (key_press_callback) {rt_kprintf("Calling press callback\n");key_press_callback();}}} else {if (touched) {touched = RT_FALSE;rt_kprintf("Touch released\n");// 调用按键释放回调if (key_release_callback) {rt_kprintf("Calling release callback\n");key_release_callback();}}}
}/*** @brief 初始化电容按键功能*/
int cap_key_init(void)
{// 初始化TIM2捕获功能tim2_capture_init();// 初始校准advanced_calibration();// 创建触摸检测定时器rt_timer_init(&touch_timer,"touch_timer",touch_timer_handler,RT_NULL,rt_tick_from_millisecond(SAMPLE_INTERVAL),RT_TIMER_FLAG_PERIODIC | RT_TIMER_FLAG_HARD_TIMER);rt_timer_start(&touch_timer);rt_kprintf("Capacitive key (PA5) initialized!\n");return RT_EOK;
}
INIT_APP_EXPORT(cap_key_init);// 调试命令
static void cap_debug(int argc, char *argv[])
{if (argc > 1) {if (rt_strcmp(argv[1], "measure") == 0) {rt_kprintf("Current value: %d us\n", measure_charge_time());}else if (rt_strcmp(argv[1], "calibrate") == 0) {advanced_calibration();}} else {rt_kprintf("Usage:\n");rt_kprintf("cap_debug measure    - Get current value\n");rt_kprintf("cap_debug calibrate - Recalibrate sensor\n");}
}
MSH_CMD_EXPORT(cap_debug, Capacitive key debug tool);/*** @brief TIM2中断处理函数*/
//void TIM2_IRQHandler(void)
//{
//    HAL_TIM_IRQHandler(&htim2);
//}

Tpad_tim2.h:

/** Copyright (c) 2006-2021, RT-Thread Development Team** SPDX-License-Identifier: Apache-2.0** Change Logs:* Date           Author       Notes* 2025-06-26     c       the first version*/
#ifndef APPLICATIONS_TPAD_TIM2_H_
#define APPLICATIONS_TPAD_TIM2_H_void cap_key_set_callback(void (*press_cb)(void), void (*release_cb)(void));#endif /* APPLICATIONS_TPAD_TIM2_H_ */

main.c:

#include <rtthread.h>
#include <rtdevice.h>
#include <board.h>
#include <Tpad_tim2.h>#define DBG_TAG "main"
#define DBG_LVL DBG_LOG
#include <rtdbg.h>#define LED_G_PIN    GET_PIN(H, 11)// 按键按下处理
static void on_key_press(void)
{rt_kprintf("PRESS\n");rt_pin_mode(LED_G_PIN, PIN_MODE_OUTPUT);rt_pin_write(LED_G_PIN, PIN_LOW);// 执行操作,如点亮LED
}// 按键释放处理
static void on_key_release(void)
{rt_kprintf("RELEASE\n");rt_pin_mode(LED_G_PIN, PIN_MODE_OUTPUT);rt_pin_write(LED_G_PIN, PIN_HIGH);
}int main(void)
{// 设置回调函数cap_key_set_callback(on_key_press, on_key_release);while (1) {rt_thread_mdelay(1000);rt_kprintf("on...\n");}return 0;
}

运行下载后,当手指放上去时,会亮绿灯,手指离开会熄灭。终端也会打印对应的信息。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/88945.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/88945.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华为云Flexus+DeepSeek征文|单机部署 与 CCE 高可用部署下 Dify 性能实测

引言 在当今的 AI 应用开发领域&#xff0c;选择合适的部署方式对于应用的性能表现、资源利用和成本控制至关重要。华为云为开发者提供了多样化的部署选择&#xff0c;其中基于单机 Flexus 实例的基础版部署和基于 CCE 容器的高可用版部署是两种常见的方式。本文将深入对比这两…

钉钉小程序框架:Pinia 状态管理与持久化存储封装

上一篇文章完成了 Pinia 在钉钉小程序中的引入与基础配置 文章地址&#xff1a;钉钉小程序框架引入 Pinia 状态管理-CSDN博客 本文将深入探讨如何通过Pinia 结合持久化存储 实现用户状态 在上一章节中&#xff0c;我们已经完成了 Pinia 在钉钉小程序中的引入与基础配置。本章将…

云计算产业链

一、云计算定义与分类体系 本质特征 按需服务模式&#xff1a;以网络化方式提供可配置的计算资源共享池&#xff08;网络/服务器/存储/应用&#xff09;。核心能力&#xff1a;快速弹性扩容、资源池化共享、按使用量付费、低管理开销。技术原理&#xff1a;通过分布式计算将大型…

git使用详解和示例

什么是 Git&#xff1f; Git 是一个 分布式版本控制系统&#xff08;DVCS&#xff09;&#xff0c;用于跟踪文件的变化&#xff0c;协调多人协作开发。由 Linus Torvalds 开发&#xff0c;用于管理 Linux 内核代码。 Git 的核心概念 名称说明工作区 (Working Directory)你看到…

深度学习的引出

虽然我们的神经⽹络给出了令⼈印象深刻的表现&#xff0c;但这样的表现带有⼏分神秘 ⽹络中的权重和偏置是被⾃动发现的。这意味着我们不能⽴即解释⽹络怎么做的、做了什么。我们能否找 到⼀些⽅法来理解我们的⽹络通过什么原理分类⼿写数字&#xff1f;并且&#xff0c;在知道…

GEO(生成式引擎优化)—— 内容创作者与企业的生死新战场

在搜索引擎优化&#xff08;SEO&#xff09;定义了互联网信息获取规则数十年后&#xff0c;一场由生成式人工智能&#xff08;AIGC&#xff09;驱动的风暴正悄然重塑整个格局。当ChatGPT、Claude、Gemini等AI助手能够直接生成整合后的答案&#xff0c;而非仅仅提供链接列表时&a…

混合密度模型GMM的似然函数(二)

设 Θ { π k , θ k } k 1 K \varTheta \{ \pi_k, \boldsymbol {\theta}_k \}_{k1}^{K} Θ{πk​,θk​}k1K​为参数向量&#xff0c; X { x 1 , ⋯ , x n } \mathcal {X} \{ {\bm x}_1, \cdots, {\bm x}_n \} X{x1​,⋯,xn​}为观测数据&#xff0c;给定数据点的独立性&a…

selenium元素定位

当我们可以打开浏览器后我们如果想要进行web测试我们自然要对网页的一些功能进行单独拿出来进行测试&#xff0c;但是我们要怎么才能拿到我们想要的元素&#xff0c;并且对其进行操作呢。 我们就以百度主页的输入框为例&#xff0c;如果我们想要王输入框中输入一些内容我们就需…

2025第十五届上海生物发酵展:江苏健达干燥盛装赴会

2025 年 8 月 7 - 9 日&#xff0c;上海新国际博览中心将迎来一场生物发酵行业的盛会 —— 第 15 届上海国际生物发酵产品与技术装备展览会&#xff08;BIOCHINA 2025&#xff09;。作为国内干燥设备领域的领军企业&#xff0c;江苏健达干燥工程有限公司受邀盛装参展&#xff0…

【效率工具】单机游戏修改方案:轻量管理器+全能平台组合

大家好&#xff01;今天我要给大家介绍两款超级实用的软件&#xff0c;专门为喜欢玩单机游戏的小伙伴们准备。 一、风灵月影管理器 不想满网翻修改器&#xff1f;这个 27M 的小工具直接帮你一键搞定&#xff0c;这款软件是由B站UP鸦无量 开发。 收录上千款游戏补丁&#xff0c;…

七天学会SpringCloud分布式微服务——01——基础概念

重点是复习体系&#xff0c;从今天6.24开始&#xff0c;确保转化为自己的东西心平气和&#xff0c;脚踏实地学习的是尚硅谷微服务 1、从单体架构到集群架构再到分布式架构 单体架构 就是 所有的功能&#xff08;服务&#xff09;模块 都部署在同一台服务器&#xff08;一台服…

三分钟学会利用deepseek将复杂信息转换成可视化图表

数据可视化是传达复杂信息的重要手段。通过将数据转化为直观的图表、图形和交互式界面,我们可以更高效地理解信息、发现趋势并做出决策。对于普通人来说,要将数据可视化可谓千难万难。但在AI工具飞速发展的今天,这个过程将会变得非常简单。今天分享的内容就是如何使用生成式…

PDF处理控件Spire.PDF系列教程:Python中快速提取PDF文本、表格、图像及文档信息

在 Python 中读取 PDF 文档是实现文档自动化、内容分析和数据提取的基础操作之一。无论你处理的是合同、报告、发票&#xff0c;还是科研论文&#xff0c;能够通过代码访问 PDF 内容&#xff0c;不仅能节省时间&#xff0c;还能带来更高效的处理流程。 要在 Python 中准确提取…

微软人工智能证书AI-102 | 如何快速通过?

微软 AI-102 考试&#xff0c;全称 “Designing and Implementing a Microsoft Azure AI Solution”&#xff0c;是微软推出的用于验证考生在 Azure 平台上设计和实施 AI 解决方案核心能力的认证考试。以下是具体介绍&#xff1a; 考试描述&#xff1a; 考试主要衡量考生实施计…

github使用指南

1、生成SSH密钥对 ssh-keygen -t ed25519 -C "你的github邮箱"然后根据提示保存路径&#xff0c;设置密码 2、将公钥添加到github cat ~/.ssh/id_ed25519.pub复制输出内容。 在gihub中点击New SSH Key&#xff0c;添加密钥 3、配置git使用SSH地址 git remote se…

AD22以上的基础操作

1.检测创建的原理图器件库 2.原理图页加大 Size&#xff1a;常规和自定义 推荐可视化栅格100mil 快捷键VG 3.原理图器件器件号排序 自动排序&#xff1a;快捷键TAA 先解锁 4.BOM(Bill of Material)物料表导出 description描述&#xff1a;类似精度。 导出各种类型bom表 5…

FastAPI技术深度解析与实战指南

导读&#xff1a;在Python Web开发领域经历了Django和Flask多年统治后&#xff0c;FastAPI的崛起正在重新定义API开发的技术标准。这篇深度技术解析将为开发者揭示FastAPI如何通过独特的架构设计解决传统框架的核心痛点。 传统Python Web框架在面对高并发场景时暴露出明显的性能…

Python 可迭代的对象、迭代器 和生成器(何时使用生成器表达式)

何时使用生成器表达式 在示例 10-16 中&#xff0c;为了实现 Vector 类&#xff0c;我用了几个生成器表达 式&#xff0c;eq、hash、abs、angle、angles、format、add 和 __mul__ 方法中各有一个生成器表达式。在这些方法中使用列表推 导也行&#xff0c;不过立即返回的列表要…

复习和预习(C++)答案解析

填空题答案及解释 在 for 循环实现累加时&#xff0c;通常在循环上方初始化累加器变量&#xff0c;如 int m ______。 答案&#xff1a;0 解释&#xff1a;累加器需从 0 开始&#xff0c;才能正确累积后续值的总和。 switch 语句根据表达式的值与各个______后的常量表达式进行…

uniapp处理后端返回的html字符串

前言&#xff1a;采用v-html方法处理 1.处理前 <html><head><meta http‐equiv"Content‐Type" content"text/html; charsetUTF-8"></head><body><form ↵<input type"submit" value"立刻提交"…