|
|
@@ -1,240 +0,0 @@
|
|
|
-/* Touch Pad Interrupt Example
|
|
|
-
|
|
|
- This example code is in the Public Domain (or CC0 licensed, at your option.)
|
|
|
-
|
|
|
- Unless required by applicable law or agreed to in writing, this
|
|
|
- software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
|
|
|
- CONDITIONS OF ANY KIND, either express or implied.
|
|
|
-*/
|
|
|
-#include <stdio.h>
|
|
|
-#include "freertos/FreeRTOS.h"
|
|
|
-#include "freertos/task.h"
|
|
|
-#include "freertos/queue.h"
|
|
|
-#include "esp_log.h"
|
|
|
-#include "driver/touch_pad.h"
|
|
|
-#include "soc/rtc_periph.h"
|
|
|
-#include "soc/sens_periph.h"
|
|
|
-#include "esp_sleep.h"
|
|
|
-
|
|
|
-static const char *TAG = "Touch pad";
|
|
|
-
|
|
|
-static QueueHandle_t que_touch = NULL;
|
|
|
-typedef struct touch_msg {
|
|
|
- touch_pad_intr_mask_t intr_mask;
|
|
|
- uint32_t pad_status_msk;
|
|
|
- uint32_t curr_pad;
|
|
|
-} touch_event_t;
|
|
|
-
|
|
|
-#define TOUCH_BUTTON_NUM 4
|
|
|
-#define TOUCH_BUTTON_WATERPROOF_ENABLE 1
|
|
|
-#define TOUCH_BUTTON_DENOISE_ENABLE 1
|
|
|
-#define TOUCH_CHANGE_CONFIG 0
|
|
|
-
|
|
|
-static const touch_pad_t button[TOUCH_BUTTON_NUM] = {
|
|
|
- TOUCH_PAD_NUM7, // 'SELECT' button.
|
|
|
- TOUCH_PAD_NUM9, // 'MENU' button.
|
|
|
- TOUCH_PAD_NUM11, // 'BACK' button.
|
|
|
- TOUCH_PAD_NUM13, // Guard ring for waterproof design.
|
|
|
- // If this pad be touched, other pads no response.
|
|
|
-};
|
|
|
-
|
|
|
-/*
|
|
|
- * Touch threshold. The threshold determines the sensitivity of the touch.
|
|
|
- * This threshold is derived by testing changes in readings from different touch channels.
|
|
|
- * If (raw_data - benchmark) > benchmark * threshold, the pad be activated.
|
|
|
- * If (raw_data - benchmark) < benchmark * threshold, the pad be inactivated.
|
|
|
- */
|
|
|
-static const float button_threshold[TOUCH_BUTTON_NUM] = {
|
|
|
- 0.2, // 20%.
|
|
|
- 0.2, // 20%.
|
|
|
- 0.2, // 20%.
|
|
|
- 0.1, // 10%.
|
|
|
-};
|
|
|
-
|
|
|
-/*
|
|
|
- Handle an interrupt triggered when a pad is touched.
|
|
|
- Recognize what pad has been touched and save it in a table.
|
|
|
- */
|
|
|
-static void touchsensor_interrupt_cb(void *arg)
|
|
|
-{
|
|
|
- int task_awoken = pdFALSE;
|
|
|
- touch_event_t evt;
|
|
|
-
|
|
|
- evt.intr_mask = touch_pad_read_intr_status_mask();
|
|
|
- evt.pad_status_msk = touch_pad_get_status();
|
|
|
- /* Note: Obtaining channel information in the interrupt callback function as the channel that triggers the interrupt is risky.
|
|
|
- If the execution of the interrupt callback function is delayed by a channel measurement time,
|
|
|
- the channel information obtained is wrong. Both light sleep and SPI FLASH reading and
|
|
|
- writing may delay the response of the interrupt function. */
|
|
|
- evt.curr_pad = touch_pad_get_current_meas_channel();
|
|
|
-
|
|
|
- xQueueSendFromISR(que_touch, &evt, &task_awoken);
|
|
|
- if (task_awoken == pdTRUE) {
|
|
|
- portYIELD_FROM_ISR();
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-static void tp_example_set_thresholds(void)
|
|
|
-{
|
|
|
- uint32_t touch_value;
|
|
|
- for (int i = 0; i < TOUCH_BUTTON_NUM; i++) {
|
|
|
- //read benchmark value
|
|
|
- touch_pad_read_benchmark(button[i], &touch_value);
|
|
|
- //set interrupt threshold.
|
|
|
- touch_pad_set_thresh(button[i], touch_value * button_threshold[i]);
|
|
|
- ESP_LOGI(TAG, "touch pad [%d] base %d, thresh %d", \
|
|
|
- button[i], touch_value, (uint32_t)(touch_value * button_threshold[i]));
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-static void touchsensor_filter_set(touch_filter_mode_t mode)
|
|
|
-{
|
|
|
- /* Filter function */
|
|
|
- touch_filter_config_t filter_info = {
|
|
|
- .mode = mode, // Test jitter and filter 1/4.
|
|
|
- .debounce_cnt = 1, // 1 time count.
|
|
|
- .noise_thr = 0, // 50%
|
|
|
- .jitter_step = 4, // use for jitter mode.
|
|
|
- .smh_lvl = TOUCH_PAD_SMOOTH_IIR_2,
|
|
|
- };
|
|
|
- touch_pad_filter_set_config(&filter_info);
|
|
|
- touch_pad_filter_enable();
|
|
|
- ESP_LOGI(TAG, "touch pad filter init");
|
|
|
-}
|
|
|
-
|
|
|
-static void tp_example_read_task(void *pvParameter)
|
|
|
-{
|
|
|
- uint8_t pad_num = 0;
|
|
|
- uint32_t touch_trig_diff = 0;
|
|
|
- touch_event_t evt = {0};
|
|
|
- static uint8_t guard_mode_flag = 0;
|
|
|
- /* Wait touch sensor init done */
|
|
|
- vTaskDelay(50 / portTICK_RATE_MS);
|
|
|
- uint32_t last_pad_status_msk = touch_pad_get_status();
|
|
|
- tp_example_set_thresholds();
|
|
|
-
|
|
|
- while (1) {
|
|
|
- int ret = xQueueReceive(que_touch, &evt, (portTickType)portMAX_DELAY);
|
|
|
- if (ret != pdTRUE) {
|
|
|
- continue;
|
|
|
- }
|
|
|
- if (evt.intr_mask & (TOUCH_PAD_INTR_MASK_ACTIVE | TOUCH_PAD_INTR_MASK_INACTIVE)) {
|
|
|
- if (last_pad_status_msk == evt.pad_status_msk) {
|
|
|
- ESP_LOGW(TAG, "TouchSensor status no changes, pad mask 0x%x", evt.pad_status_msk);
|
|
|
- continue;
|
|
|
- }
|
|
|
- pad_num = 0;
|
|
|
- touch_trig_diff = evt.pad_status_msk ^ last_pad_status_msk; // Record changes in channel status. Each bit represents a channel
|
|
|
- last_pad_status_msk = evt.pad_status_msk; // Update the pad active status
|
|
|
-
|
|
|
- /* Traverse all channels and find out the channel where the state changes, and determine the state type */
|
|
|
- while (touch_trig_diff) {
|
|
|
- if (touch_trig_diff & BIT(pad_num)) {
|
|
|
- if (evt.pad_status_msk & BIT(pad_num)) { // Touch channel active
|
|
|
- /* if guard pad be touched, other pads no response. */
|
|
|
- if (pad_num == button[3]) {
|
|
|
- guard_mode_flag = 1;
|
|
|
- ESP_LOGW(TAG, "TouchSensor [%d] be activated, enter guard mode", pad_num);
|
|
|
- } else {
|
|
|
- if (guard_mode_flag == 0) {
|
|
|
- ESP_LOGI(TAG, "TouchSensor [%d] be activated, status mask 0x%x", pad_num, evt.pad_status_msk);
|
|
|
- } else {
|
|
|
- ESP_LOGW(TAG, "In guard mode. No response");
|
|
|
- }
|
|
|
- }
|
|
|
- } else { // Touch channel inactive
|
|
|
- /* if guard pad be touched, other pads no response. */
|
|
|
- if (pad_num == button[3]) {
|
|
|
- guard_mode_flag = 0;
|
|
|
- ESP_LOGW(TAG, "TouchSensor [%d] be inactivated, exit guard mode", pad_num);
|
|
|
- } else {
|
|
|
- if (guard_mode_flag == 0) {
|
|
|
- ESP_LOGI(TAG, "TouchSensor [%d] be inactivated, status mask 0x%x", pad_num, evt.pad_status_msk);
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- touch_trig_diff &= (~(BIT(pad_num))); // Clear the channels that have been checked
|
|
|
- }
|
|
|
- pad_num ++;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- if (evt.intr_mask & TOUCH_PAD_INTR_MASK_SCAN_DONE) {
|
|
|
- ESP_LOGI(TAG, "The touch sensor group measurement is done [%d].", evt.curr_pad);
|
|
|
- }
|
|
|
- if (evt.intr_mask & TOUCH_PAD_INTR_MASK_TIMEOUT) {
|
|
|
- /* Add your exception handling in here. */
|
|
|
- ESP_LOGI(TAG, "Touch sensor channel %d measure timeout. Skip this exception channel!!", evt.curr_pad);
|
|
|
- touch_pad_timeout_resume(); // Point on the next channel to measure.
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-void app_main(void)
|
|
|
-{
|
|
|
- if (que_touch == NULL) {
|
|
|
- que_touch = xQueueCreate(TOUCH_BUTTON_NUM, sizeof(touch_event_t));
|
|
|
- }
|
|
|
- // Initialize touch pad peripheral, it will start a timer to run a filter
|
|
|
- ESP_LOGI(TAG, "Initializing touch pad");
|
|
|
- /* Initialize touch pad peripheral. */
|
|
|
- touch_pad_init();
|
|
|
- for (int i = 0; i < TOUCH_BUTTON_NUM; i++) {
|
|
|
- touch_pad_config(button[i]);
|
|
|
- }
|
|
|
-
|
|
|
-#if TOUCH_CHANGE_CONFIG
|
|
|
- /* If you want change the touch sensor default setting, please write here(after initialize). There are examples: */
|
|
|
- touch_pad_set_meas_time(TOUCH_PAD_SLEEP_CYCLE_DEFAULT, TOUCH_PAD_MEASURE_CYCLE_DEFAULT);
|
|
|
- touch_pad_set_voltage(TOUCH_PAD_HIGH_VOLTAGE_THRESHOLD, TOUCH_PAD_LOW_VOLTAGE_THRESHOLD, TOUCH_PAD_ATTEN_VOLTAGE_THRESHOLD);
|
|
|
- touch_pad_set_idle_channel_connect(TOUCH_PAD_IDLE_CH_CONNECT_DEFAULT);
|
|
|
- for (int i = 0; i < TOUCH_BUTTON_NUM; i++) {
|
|
|
- touch_pad_set_cnt_mode(i, TOUCH_PAD_SLOPE_DEFAULT, TOUCH_PAD_TIE_OPT_DEFAULT);
|
|
|
- }
|
|
|
-#endif
|
|
|
-
|
|
|
-#if TOUCH_BUTTON_DENOISE_ENABLE
|
|
|
- /* Denoise setting at TouchSensor 0. */
|
|
|
- touch_pad_denoise_t denoise = {
|
|
|
- /* The bits to be cancelled are determined according to the noise level. */
|
|
|
- .grade = TOUCH_PAD_DENOISE_BIT4,
|
|
|
- /* By adjusting the parameters, the reading of T0 should be approximated to the reading of the measured channel. */
|
|
|
- .cap_level = TOUCH_PAD_DENOISE_CAP_L4,
|
|
|
- };
|
|
|
- touch_pad_denoise_set_config(&denoise);
|
|
|
- touch_pad_denoise_enable();
|
|
|
- ESP_LOGI(TAG, "Denoise function init");
|
|
|
-#endif
|
|
|
-
|
|
|
-#if TOUCH_BUTTON_WATERPROOF_ENABLE
|
|
|
- /* Waterproof function */
|
|
|
- touch_pad_waterproof_t waterproof = {
|
|
|
- .guard_ring_pad = button[3], // If no ring pad, set 0;
|
|
|
- /* It depends on the number of the parasitic capacitance of the shield pad.
|
|
|
- Based on the touch readings of T14 and T0, estimate the size of the parasitic capacitance on T14
|
|
|
- and set the parameters of the appropriate hardware. */
|
|
|
- .shield_driver = TOUCH_PAD_SHIELD_DRV_L2,
|
|
|
- };
|
|
|
- touch_pad_waterproof_set_config(&waterproof);
|
|
|
- touch_pad_waterproof_enable();
|
|
|
- ESP_LOGI(TAG, "touch pad waterproof init");
|
|
|
-#endif
|
|
|
-
|
|
|
- /* Filter setting */
|
|
|
- touchsensor_filter_set(TOUCH_PAD_FILTER_IIR_16);
|
|
|
- touch_pad_timeout_set(true, SOC_TOUCH_PAD_THRESHOLD_MAX);
|
|
|
- /* Register touch interrupt ISR, enable intr type. */
|
|
|
- touch_pad_isr_register(touchsensor_interrupt_cb, NULL, TOUCH_PAD_INTR_MASK_ALL);
|
|
|
- /* If you have other touch algorithm, you can get the measured value after the `TOUCH_PAD_INTR_MASK_SCAN_DONE` interrupt is generated. */
|
|
|
- touch_pad_intr_enable(TOUCH_PAD_INTR_MASK_ACTIVE | TOUCH_PAD_INTR_MASK_INACTIVE | TOUCH_PAD_INTR_MASK_TIMEOUT);
|
|
|
-
|
|
|
- /* In ESP32-S3 If the touch wakeup not enable, the touch action of the touch sensor during light sleep may be lost. */
|
|
|
- esp_sleep_enable_touchpad_wakeup();
|
|
|
-
|
|
|
- /* Enable touch sensor clock. Work mode is "timer trigger". */
|
|
|
- touch_pad_set_fsm_mode(TOUCH_FSM_MODE_TIMER);
|
|
|
- touch_pad_fsm_start();
|
|
|
-
|
|
|
- // Start a task to show what pads have been touched
|
|
|
- xTaskCreate(&tp_example_read_task, "touch_pad_read_task", 2048, NULL, 5, NULL);
|
|
|
-}
|