零基礎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
英文地址,請注明出處:http://en.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"立刻提交"…