/* * SPDX-FileCopyrightText: 2015-2021 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: Apache-2.0 */ /******************************************************************************* * NOTICE * The ll is not public api, don't use in application code. * See readme.md in hal/include/hal/readme.md ******************************************************************************/ // The Lowlevel layer for Touch Sensor #pragma once #include #include #include "hal/misc.h" #include "soc/touch_sensor_periph.h" #include "soc/rtc_cntl_struct.h" #include "soc/rtc_io_struct.h" #include "soc/sens_struct.h" #include "soc/soc_caps.h" #include "hal/touch_sensor_types.h" #ifdef __cplusplus extern "C" { #endif #define TOUCH_LL_READ_RAW 0x0 #define TOUCH_LL_READ_BENCHMARK 0x2 #define TOUCH_LL_READ_SMOOTH 0x3 #define TOUCH_LL_TIMER_FORCE_DONE 0x3 #define TOUCH_LL_TIMER_DONE 0x0 /** * Set touch sensor touch sensor times of charge and discharge. * * @param meas_timers The times of charge and discharge in each measure process of touch channels. * The timer frequency is 8Mhz. Range: 0 ~ 0xffff. */ static inline void touch_ll_set_meas_times(uint16_t meas_time) { //The times of charge and discharge in each measure process of touch channels. HAL_FORCE_MODIFY_U32_REG_FIELD(RTCCNTL.touch_ctrl1, touch_meas_num, meas_time); //the waiting cycles (in 8MHz) between TOUCH_START and TOUCH_XPD HAL_FORCE_MODIFY_U32_REG_FIELD(RTCCNTL.touch_ctrl2, touch_xpd_wait, SOC_TOUCH_PAD_MEASURE_WAIT_MAX); //wait volt stable } /** * Get touch sensor times of charge and discharge. * * @param meas_times Pointer to accept times count of charge and discharge. */ static inline void touch_ll_get_measure_times(uint16_t *meas_time) { *meas_time = HAL_FORCE_READ_U32_REG_FIELD(RTCCNTL.touch_ctrl1, touch_meas_num); } /** * Set touch sensor sleep time. * * @param sleep_cycle The touch sensor will sleep after each measurement. * sleep_cycle decide the interval between each measurement. * t_sleep = sleep_cycle / (RTC_SLOW_CLK frequency). * The approximate frequency value of RTC_SLOW_CLK can be obtained using rtc_clk_slow_freq_get_hz function. */ static inline void touch_ll_set_sleep_time(uint16_t sleep_time) { // touch sensor sleep cycle Time = sleep_cycle / RTC_SLOW_CLK(150k) HAL_FORCE_MODIFY_U32_REG_FIELD(RTCCNTL.touch_ctrl1, touch_sleep_cycles, sleep_time); } /** * Get touch sensor sleep time. * * @param sleep_cycle Pointer to accept sleep cycle number. */ static inline void touch_ll_get_sleep_time(uint16_t *sleep_time) { *sleep_time = HAL_FORCE_READ_U32_REG_FIELD(RTCCNTL.touch_ctrl1, touch_sleep_cycles); } /** * Set touch sensor high voltage threshold of chanrge. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * So the high threshold should be less than the supply voltage. * * @param refh The high voltage threshold of chanrge. */ static inline void touch_ll_set_voltage_high(touch_high_volt_t refh) { RTCCNTL.touch_ctrl2.touch_drefh = refh; } /** * Get touch sensor high voltage threshold of chanrge. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * So the high threshold should be less than the supply voltage. * * @param refh The high voltage threshold of chanrge. */ static inline void touch_ll_get_voltage_high(touch_high_volt_t *refh) { *refh = (touch_high_volt_t)RTCCNTL.touch_ctrl2.touch_drefh; } /** * Set touch sensor low voltage threshold of discharge. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * * @param refl The low voltage threshold of discharge. */ static inline void touch_ll_set_voltage_low(touch_low_volt_t refl) { RTCCNTL.touch_ctrl2.touch_drefl = refl; } /** * Get touch sensor low voltage threshold of discharge. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * * @param refl The low voltage threshold of discharge. */ static inline void touch_ll_get_voltage_low(touch_low_volt_t *refl) { *refl = (touch_low_volt_t)RTCCNTL.touch_ctrl2.touch_drefl; } /** * Set touch sensor high voltage attenuation of chanrge. The actual charge threshold is high voltage threshold minus attenuation value. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * So the high threshold should be less than the supply voltage. * * @param refh The high voltage threshold of chanrge. */ static inline void touch_ll_set_voltage_attenuation(touch_volt_atten_t atten) { RTCCNTL.touch_ctrl2.touch_drange = atten; } /** * Get touch sensor high voltage attenuation of chanrge. The actual charge threshold is high voltage threshold minus attenuation value. * The touch sensor measures the channel capacitance value by charging and discharging the channel. * So the high threshold should be less than the supply voltage. * * @param refh The high voltage threshold of chanrge. */ static inline void touch_ll_get_voltage_attenuation(touch_volt_atten_t *atten) { *atten = (touch_volt_atten_t)RTCCNTL.touch_ctrl2.touch_drange; } /** * Set touch sensor charge/discharge speed(currents) for each pad. * If the slope is 0, the counter would always be zero. * If the slope is 1, the charging and discharging would be slow. The measurement time becomes longer. * If the slope is set 7, which is the maximum value, the charging and discharging would be fast. * The measurement time becomes shorter. * * @note The higher the charge and discharge current, the greater the immunity of the touch channel, * but it will increase the system power consumption. * @param touch_num Touch pad index. * @param slope touch pad charge/discharge speed(currents). */ static inline void touch_ll_set_slope(touch_pad_t touch_num, touch_cnt_slope_t slope) { if (touch_num < TOUCH_PAD_NUM10) { SET_PERI_REG_BITS(RTC_CNTL_TOUCH_DAC_REG, RTC_CNTL_TOUCH_PAD0_DAC_V, slope, (RTC_CNTL_TOUCH_PAD0_DAC_S - touch_num * 3)); } else { SET_PERI_REG_BITS(RTC_CNTL_TOUCH_DAC1_REG, RTC_CNTL_TOUCH_PAD10_DAC_V, slope, (RTC_CNTL_TOUCH_PAD10_DAC_S - (touch_num - TOUCH_PAD_NUM10) * 3)); } } /** * Get touch sensor charge/discharge speed(currents) for each pad. * If the slope is 0, the counter would always be zero. * If the slope is 1, the charging and discharging would be slow. The measurement time becomes longer. * If the slope is set 7, which is the maximum value, the charging and discharging would be fast. * The measurement time becomes shorter. * * @param touch_num Touch pad index. * @param slope touch pad charge/discharge speed(currents). */ static inline void touch_ll_get_slope(touch_pad_t touch_num, touch_cnt_slope_t *slope) { if (touch_num < TOUCH_PAD_NUM10) { *slope = GET_PERI_REG_BITS2(RTC_CNTL_TOUCH_DAC_REG, RTC_CNTL_TOUCH_PAD0_DAC_V, (RTC_CNTL_TOUCH_PAD0_DAC_S - touch_num * 3)); } else { *slope = GET_PERI_REG_BITS2(RTC_CNTL_TOUCH_DAC1_REG, RTC_CNTL_TOUCH_PAD10_DAC_V, (RTC_CNTL_TOUCH_PAD10_DAC_S - (touch_num - TOUCH_PAD_NUM10) * 3)); } } /** * Set initial voltage state of touch channel for each measurement. * * @param touch_num Touch pad index. * @param opt Initial voltage state. */ static inline void touch_ll_set_tie_option(touch_pad_t touch_num, touch_tie_opt_t opt) { RTCIO.touch_pad[touch_num].tie_opt = opt; } /** * Get initial voltage state of touch channel for each measurement. * * @param touch_num Touch pad index. * @param opt Initial voltage state. */ static inline void touch_ll_get_tie_option(touch_pad_t touch_num, touch_tie_opt_t *opt) { *opt = (touch_tie_opt_t)RTCIO.touch_pad[touch_num].tie_opt; } /** * Set touch sensor FSM mode. * The measurement action can be triggered by the hardware timer, as well as by the software instruction. * * @param mode FSM mode. */ static inline void touch_ll_set_fsm_mode(touch_fsm_mode_t mode) { RTCCNTL.touch_ctrl2.touch_start_force = mode; } /** * Get touch sensor FSM mode. * The measurement action can be triggered by the hardware timer, as well as by the software instruction. * * @param mode FSM mode. */ static inline void touch_ll_get_fsm_mode(touch_fsm_mode_t *mode) { *mode = (touch_fsm_mode_t)RTCCNTL.touch_ctrl2.touch_start_force; } /** * Enable/disable clock gate of touch sensor. * * @param enable true/false. */ static inline void touch_ll_clkgate(bool enable) { RTCCNTL.touch_ctrl2.touch_clkgate_en = enable; //enable touch clock for FSM. or force enable. } /** * Get touch sensor FSM state. * @return * - true: fsm state is open. * - false: fsm state is close. */ static inline bool touch_ll_clkgate_get_state(void) { return RTCCNTL.touch_ctrl2.touch_clkgate_en; } /** * Touch timer trigger measurement and always wait measurement done. * Force done for touch timer ensures that the timer always can get the measurement done signal. */ static inline void touch_ll_timer_force_done(void) { RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_FORCE_DONE; RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_DONE; } /** * Start touch sensor FSM timer. * The measurement action can be triggered by the hardware timer, as well as by the software instruction. */ static inline void touch_ll_start_fsm(void) { /** * Touch timer trigger measurement and always wait measurement done. * Force done for touch timer ensures that the timer always can get the measurement done signal. */ RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_FORCE_DONE; RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_DONE; RTCCNTL.touch_ctrl2.touch_slp_timer_en = (RTCCNTL.touch_ctrl2.touch_start_force == TOUCH_FSM_MODE_TIMER ? 1 : 0); } /** * Stop touch sensor FSM timer. * The measurement action can be triggered by the hardware timer, as well as by the software instruction. */ static inline void touch_ll_stop_fsm(void) { RTCCNTL.touch_ctrl2.touch_start_en = 0; //stop touch fsm RTCCNTL.touch_ctrl2.touch_slp_timer_en = 0; RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_FORCE_DONE; RTCCNTL.touch_ctrl2.touch_timer_force_done = TOUCH_LL_TIMER_DONE; } /** * Get touch sensor FSM timer state. * @return * - true: FSM enabled * - false: FSM disabled */ static inline bool touch_ll_get_fsm_state(void) { return (bool)RTCCNTL.touch_ctrl2.touch_slp_timer_en; } /** * Trigger a touch sensor measurement, only support in SW mode of FSM. */ static inline void touch_ll_start_sw_meas(void) { RTCCNTL.touch_ctrl2.touch_start_en = 0; RTCCNTL.touch_ctrl2.touch_start_en = 1; } /** * Set the trigger threshold of touch sensor. * The threshold determines the sensitivity of the touch sensor. * The threshold is the original value of the trigger state minus the benchmark value. * * @note If set "TOUCH_PAD_THRESHOLD_MAX", the touch is never be triggered. * @param touch_num touch pad index * @param threshold threshold of touch sensor. */ static inline void touch_ll_set_threshold(touch_pad_t touch_num, uint32_t threshold) { SENS.touch_thresh[touch_num - 1].thresh = threshold; } /** * Get the trigger threshold of touch sensor. * The threshold determines the sensitivity of the touch sensor. * The threshold is the original value of the trigger state minus the benchmark value. * * @param touch_num touch pad index. * @param threshold pointer to accept threshold. */ static inline void touch_ll_get_threshold(touch_pad_t touch_num, uint32_t *threshold) { *threshold = SENS.touch_thresh[touch_num - 1].thresh; } /** * Enable touch sensor channel. Register touch channel into touch sensor measurement group. * The working mode of the touch sensor is simultaneous measurement. * This function will set the measure bits according to the given bitmask. * * @note If set this mask, the FSM timer should be stop firsty. * @note The touch sensor that in scan map, should be deinit GPIO function firstly. * @param enable_mask bitmask of touch sensor scan group. * e.g. TOUCH_PAD_NUM1 -> BIT(1) * @return * - ESP_OK on success */ static inline void touch_ll_set_channel_mask(uint16_t enable_mask) { RTCCNTL.touch_scan_ctrl.touch_scan_pad_map |= (enable_mask & TOUCH_PAD_BIT_MASK_ALL); SENS.sar_touch_conf.touch_outen |= (enable_mask & TOUCH_PAD_BIT_MASK_ALL); } /** * Get touch sensor channel mask. * * @param enable_mask bitmask of touch sensor scan group. * e.g. TOUCH_PAD_NUM1 -> BIT(1) */ static inline void touch_ll_get_channel_mask(uint16_t *enable_mask) { *enable_mask = SENS.sar_touch_conf.touch_outen \ & RTCCNTL.touch_scan_ctrl.touch_scan_pad_map \ & TOUCH_PAD_BIT_MASK_ALL; } /** * Disable touch sensor channel by bitmask. * * @param enable_mask bitmask of touch sensor scan group. * e.g. TOUCH_PAD_NUM1 -> BIT(1) */ static inline void touch_ll_clear_channel_mask(uint16_t disable_mask) { SENS.sar_touch_conf.touch_outen &= ~(disable_mask & TOUCH_PAD_BIT_MASK_ALL); RTCCNTL.touch_scan_ctrl.touch_scan_pad_map &= ~(disable_mask & TOUCH_PAD_BIT_MASK_ALL); } /** * Get the touch sensor trigger status, usually used in ISR to decide which pads are 'touched'. * * @param status_mask The touch sensor status. e.g. Touch1 trigger status is `status_mask & (BIT1)`. */ static inline void touch_ll_read_trigger_status_mask(uint32_t *status_mask) { *status_mask = SENS.sar_touch_chn_st.touch_pad_active; } /** * Clear all touch sensor status. * * @note Generally no manual removal is required. */ static inline void touch_ll_clear_trigger_status_mask(void) { SENS.sar_touch_conf.touch_status_clr = 1; } /** * Get touch sensor raw data (touch sensor counter value) from register. No block. * * @param touch_num touch pad index. * @return touch_value pointer to accept touch sensor value. */ static inline uint32_t IRAM_ATTR touch_ll_read_raw_data(touch_pad_t touch_num) { SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_RAW; return SENS.sar_touch_status[touch_num - 1].touch_pad_data; } /** * Get touch sensor measure status. No block. * * @return * - If touch sensors measure done. */ static inline bool touch_ll_meas_is_done(void) { return (bool)SENS.sar_touch_chn_st.touch_meas_done; } /************************* esp32s2 only *************************/ /** * Reset the whole of touch module. * * @note Call this function after `touch_pad_fsm_stop`. */ static inline void touch_ll_reset(void) { RTCCNTL.touch_ctrl2.touch_reset = 0; RTCCNTL.touch_ctrl2.touch_reset = 1; RTCCNTL.touch_ctrl2.touch_reset = 0; // Should be set 0. } /** * Set connection type of touch channel in idle status. * When a channel is in measurement mode, other initialized channels are in idle mode. * The touch channel is generally adjacent to the trace, so the connection state of the idle channel * affects the stability and sensitivity of the test channel. * The `CONN_HIGHZ`(high resistance) setting increases the sensitivity of touch channels. * The `CONN_GND`(grounding) setting increases the stability of touch channels. * * @param type Select idle channel connect to high resistance state or ground. */ static inline void touch_ll_set_idle_channel_connect(touch_pad_conn_type_t type) { RTCCNTL.touch_scan_ctrl.touch_inactive_connection = type; } /** * Set connection type of touch channel in idle status. * When a channel is in measurement mode, other initialized channels are in idle mode. * The touch channel is generally adjacent to the trace, so the connection state of the idle channel * affects the stability and sensitivity of the test channel. * The `CONN_HIGHZ`(high resistance) setting increases the sensitivity of touch channels. * The `CONN_GND`(grounding) setting increases the stability of touch channels. * * @param type Select idle channel connect to high resistance state or ground. */ static inline void touch_ll_get_idle_channel_connect(touch_pad_conn_type_t *type) { *type = RTCCNTL.touch_scan_ctrl.touch_inactive_connection; } /** * Get the current measure channel. Touch sensor measurement is cyclic scan mode. * * @return * - touch channel number */ static inline touch_pad_t IRAM_ATTR touch_ll_get_current_meas_channel(void) { return (touch_pad_t)(SENS.sar_touch_status0.touch_scan_curr); } /** * Enable touch sensor interrupt by bitmask. * * @param type interrupt type */ static inline void touch_ll_intr_enable(touch_pad_intr_mask_t int_mask) { if (int_mask & TOUCH_PAD_INTR_MASK_DONE) { RTCCNTL.int_ena_w1ts.rtc_touch_done_w1ts = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_ACTIVE) { RTCCNTL.int_ena_w1ts.rtc_touch_active_w1ts = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_INACTIVE) { RTCCNTL.int_ena_w1ts.rtc_touch_inactive_w1ts = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_SCAN_DONE) { RTCCNTL.int_ena_w1ts.rtc_touch_scan_done_w1ts = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_TIMEOUT) { RTCCNTL.int_ena_w1ts.rtc_touch_timeout_w1ts = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_PROXI_MEAS_DONE) { RTCCNTL.int_ena_w1ts.rtc_touch_approach_loop_done_w1ts = 1; } } /** * Disable touch sensor interrupt by bitmask. * * @param type interrupt type */ static inline void touch_ll_intr_disable(touch_pad_intr_mask_t int_mask) { if (int_mask & TOUCH_PAD_INTR_MASK_DONE) { RTCCNTL.int_ena_w1tc.rtc_touch_done_w1tc = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_ACTIVE) { RTCCNTL.int_ena_w1tc.rtc_touch_active_w1tc = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_INACTIVE) { RTCCNTL.int_ena_w1tc.rtc_touch_inactive_w1tc = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_SCAN_DONE) { RTCCNTL.int_ena_w1tc.rtc_touch_scan_done_w1tc = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_TIMEOUT) { RTCCNTL.int_ena_w1tc.rtc_touch_timeout_w1tc = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_PROXI_MEAS_DONE) { RTCCNTL.int_ena_w1tc.rtc_touch_approach_loop_done_w1tc = 1; } } /** * Clear touch sensor interrupt by bitmask. * * @param int_mask Pad mask to clear interrupts */ static inline void touch_ll_intr_clear(touch_pad_intr_mask_t int_mask) { if (int_mask & TOUCH_PAD_INTR_MASK_DONE) { RTCCNTL.int_clr.rtc_touch_done = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_ACTIVE) { RTCCNTL.int_clr.rtc_touch_active = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_INACTIVE) { RTCCNTL.int_clr.rtc_touch_inactive = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_SCAN_DONE) { RTCCNTL.int_clr.rtc_touch_scan_done = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_TIMEOUT) { RTCCNTL.int_clr.rtc_touch_timeout = 1; } if (int_mask & TOUCH_PAD_INTR_MASK_PROXI_MEAS_DONE) { RTCCNTL.int_clr.rtc_touch_approach_loop_done = 1; } } /** * Get the bitmask of touch sensor interrupt status. * * @return type interrupt type */ static inline uint32_t touch_ll_read_intr_status_mask(void) { uint32_t intr_st = RTCCNTL.int_st.val; uint32_t intr_msk = 0; if (intr_st & RTC_CNTL_TOUCH_DONE_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_DONE; } if (intr_st & RTC_CNTL_TOUCH_ACTIVE_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_ACTIVE; } if (intr_st & RTC_CNTL_TOUCH_INACTIVE_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_INACTIVE; } if (intr_st & RTC_CNTL_TOUCH_SCAN_DONE_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_SCAN_DONE; } if (intr_st & RTC_CNTL_TOUCH_TIMEOUT_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_TIMEOUT; } if (intr_st & RTC_CNTL_TOUCH_APPROACH_LOOP_DONE_INT_ST_M) { intr_msk |= TOUCH_PAD_INTR_MASK_PROXI_MEAS_DONE; } return (intr_msk & TOUCH_PAD_INTR_MASK_ALL); } /** * Enable the timeout check for all touch sensor channels measurements. * When the touch reading of a touch channel exceeds the measurement threshold, * If enable: a timeout interrupt will be generated and it will go to the next channel measurement. * If disable: the FSM is always on the channel, until the measurement of this channel is over. * * @note Set the timeout threshold correctly before enabling it. */ static inline void touch_ll_timeout_enable(void) { RTCCNTL.touch_timeout_ctrl.touch_timeout_en = 1; } /** * Disable the timeout check for all touch sensor channels measurements. * When the touch reading of a touch channel exceeds the measurement threshold, * If enable: a timeout interrupt will be generated and it will go to the next channel measurement. * If disable: the FSM is always on the channel, until the measurement of this channel is over. * * @note Set the timeout threshold correctly before enabling it. */ static inline void touch_ll_timeout_disable(void) { RTCCNTL.touch_timeout_ctrl.touch_timeout_en = 0; } /** * Set timeout threshold for all touch sensor channels measurements. * Compared with touch readings. * * @param threshold Set to the maximum time measured on one channel. */ static inline void touch_ll_timeout_set_threshold(uint32_t threshold) { RTCCNTL.touch_timeout_ctrl.touch_timeout_num = threshold; } /** * Get timeout threshold for all touch sensor channels measurements. * Compared with touch readings. * * @param threshold Point to timeout threshold. */ static inline void touch_ll_timeout_get_threshold(uint32_t *threshold) { *threshold = RTCCNTL.touch_timeout_ctrl.touch_timeout_num; } /************************ Filter register setting ************************/ /** * Get smoothed data that obtained by filtering the raw data. * * @param touch_num touch pad index * @param smooth_data pointer to smoothed data */ static inline void IRAM_ATTR touch_ll_filter_read_smooth(touch_pad_t touch_num, uint32_t *smooth_data) { SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_SMOOTH; *smooth_data = SENS.sar_touch_status[touch_num - 1].touch_pad_data; } /** * Get benchmark value of touch sensor. * * @note After initialization, the benchmark value is the maximum during the first measurement period. * @param touch_num touch pad index * @param touch_value pointer to accept touch sensor value */ static inline void IRAM_ATTR touch_ll_read_benchmark(touch_pad_t touch_num, uint32_t *benchmark) { SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_BENCHMARK; *benchmark = SENS.sar_touch_status[touch_num - 1].touch_pad_data; } /** * Force reset benchmark to raw data of touch sensor. * * @note If call this API, make sure enable clock gate(`touch_ll_clkgate`) first. * @param touch_num touch pad index * - TOUCH_PAD_MAX Reset basaline of all channels. */ static inline void touch_ll_reset_benchmark(touch_pad_t touch_num) { /* Clear touch channels to initialize the channel value (benchmark, raw_data). */ if (touch_num == TOUCH_PAD_MAX) { SENS.sar_touch_chn_st.touch_channel_clr = TOUCH_PAD_BIT_MASK_ALL; } else { SENS.sar_touch_chn_st.touch_channel_clr = (1U << touch_num); } } /** * Set filter mode. The input of the filter is the raw value of touch reading, * and the output of the filter is involved in the judgment of the touch state. * * @param mode Filter mode type. Refer to ``touch_filter_mode_t``. */ static inline void touch_ll_filter_set_filter_mode(touch_filter_mode_t mode) { RTCCNTL.touch_filter_ctrl.touch_filter_mode = mode; } /** * Get filter mode. The input of the filter is the raw value of touch reading, * and the output of the filter is involved in the judgment of the touch state. * * @param mode Filter mode type. Refer to ``touch_filter_mode_t``. */ static inline void touch_ll_filter_get_filter_mode(touch_filter_mode_t *mode) { *mode = (touch_filter_mode_t)RTCCNTL.touch_filter_ctrl.touch_filter_mode; } /** * Set filter mode. The input to the filter is raw data and the output is the smooth data. * The smooth data is used to determine the touch status. * * @param mode Filter mode type. Refer to ``touch_smooth_mode_t``. */ static inline void touch_ll_filter_set_smooth_mode(touch_smooth_mode_t mode) { RTCCNTL.touch_filter_ctrl.touch_smooth_lvl = mode; } /** * Get filter mode. The smooth data is used to determine the touch status. * * @param mode Filter mode type. Refer to ``touch_smooth_mode_t``. */ static inline void touch_ll_filter_get_smooth_mode(touch_smooth_mode_t *mode) { *mode = RTCCNTL.touch_filter_ctrl.touch_smooth_lvl; } /** * Set debounce count, such as `n`. If the measured values continue to exceed * the threshold for `n+1` times, it is determined that the touch sensor state changes. * * @param dbc_cnt Debounce count value. */ static inline void touch_ll_filter_set_debounce(uint32_t dbc_cnt) { RTCCNTL.touch_filter_ctrl.touch_debounce = dbc_cnt; } /** * Get debounce count. * * @param dbc_cnt Debounce count value. */ static inline void touch_ll_filter_get_debounce(uint32_t *dbc_cnt) { *dbc_cnt = RTCCNTL.touch_filter_ctrl.touch_debounce; } /** * Set noise threshold coefficient. Higher = More noise resistance. * The actual noise should be less than (noise coefficient * touch threshold). * Range: 0 ~ 3. The coefficient is 0: 4/8; 1: 3/8; 2: 2/8; 3: 1; * * @param hys_thr Noise threshold coefficient. */ static inline void touch_ll_filter_set_noise_thres(uint32_t noise_thr) { RTCCNTL.touch_filter_ctrl.touch_noise_thres = noise_thr; RTCCNTL.touch_filter_ctrl.config2 = noise_thr; RTCCNTL.touch_filter_ctrl.config1 = 0xF; RTCCNTL.touch_filter_ctrl.config3 = 2; } /** * Get noise threshold coefficient. Higher = More noise resistance. * The actual noise should be less than (noise coefficient * touch threshold). * Range: 0 ~ 3. The coefficient is 0: 4/8; 1: 3/8; 2: 2/8; 3: 1; * * @param noise_thr Noise threshold coefficient. */ static inline void touch_ll_filter_get_noise_thres(uint32_t *noise_thr) { *noise_thr = RTCCNTL.touch_filter_ctrl.touch_noise_thres; } /** * Set jitter filter step size. * If filter mode is jitter, should set filter step for jitter. * Range: 0 ~ 15 * * @param step The step size of the data change. */ static inline void touch_ll_filter_set_jitter_step(uint32_t step) { RTCCNTL.touch_filter_ctrl.touch_jitter_step = step; } /** * Get jitter filter step size. * If filter mode is jitter, should set filter step for jitter. * Range: 0 ~ 15 * * @param step The step size of the data change. */ static inline void touch_ll_filter_get_jitter_step(uint32_t *step) { *step = RTCCNTL.touch_filter_ctrl.touch_jitter_step; } /** * Enable touch sensor filter and detection algorithm. * For more details on the detection algorithm, please refer to the application documentation. */ static inline void touch_ll_filter_enable(void) { RTCCNTL.touch_filter_ctrl.touch_filter_en = 1; } /** * Disable touch sensor filter and detection algorithm. * For more details on the detection algorithm, please refer to the application documentation. */ static inline void touch_ll_filter_disable(void) { RTCCNTL.touch_filter_ctrl.touch_filter_en = 0; } /************************ Denoise register setting ************************/ /** * Enable denoise function. * T0 is an internal channel that does not have a corresponding external GPIO. * T0 will work simultaneously with the measured channel Tn. Finally, the actual * measured value of Tn is the value after subtracting lower bits of T0. * This denoise function filters out interference introduced on all channels, * such as noise introduced by the power supply and external EMI. */ static inline void touch_ll_denoise_enable(void) { RTCCNTL.touch_scan_ctrl.touch_denoise_en = 1; } /** * Enable denoise function. * T0 is an internal channel that does not have a corresponding external GPIO. * T0 will work simultaneously with the measured channel Tn. Finally, the actual * measured value of Tn is the value after subtracting lower bits of T0. * This denoise function filters out interference introduced on all channels, * such as noise introduced by the power supply and external EMI. */ static inline void touch_ll_denoise_disable(void) { RTCCNTL.touch_scan_ctrl.touch_denoise_en = 0; } /** * Set internal reference capacitance of denoise channel. * Select the appropriate internal reference capacitance value so that * the reading of denoise channel is closest to the reading of the channel being measured. * * @param cap_level Capacitance level. */ static inline void touch_ll_denoise_set_cap_level(touch_pad_denoise_cap_t cap_level) { RTCCNTL.touch_ctrl2.touch_refc = cap_level; } /** * Get internal reference capacitance of denoise channel. * Select the appropriate internal reference capacitance value so that * the reading of denoise channel is closest to the reading of the channel being measured. * * @param cap_level Capacitance level. */ static inline void touch_ll_denoise_get_cap_level(touch_pad_denoise_cap_t *cap_level) { *cap_level = RTCCNTL.touch_ctrl2.touch_refc; } /** * Set denoise range of denoise channel. * Determined by measuring the noise amplitude of the denoise channel. * * @param grade Denoise range of denoise channel. */ static inline void touch_ll_denoise_set_grade(touch_pad_denoise_grade_t grade) { RTCCNTL.touch_scan_ctrl.touch_denoise_res = grade; } /** * Set denoise range of denoise channel. * Determined by measuring the noise amplitude of the denoise channel. * * @param grade Denoise range of denoise channel. */ static inline void touch_ll_denoise_get_grade(touch_pad_denoise_grade_t *grade) { *grade = RTCCNTL.touch_scan_ctrl.touch_denoise_res; } /** * Read denoise measure value (TOUCH_PAD_NUM0). * * @param denoise value of denoise. */ static inline void touch_ll_denoise_read_data(uint32_t *data) { *data = SENS.sar_touch_denoise.touch_denoise_data; } /************************ Waterproof register setting ************************/ /** * Set touch channel use for guard pad. * * @param pad_num Touch sensor channel number. */ static inline void touch_ll_waterproof_set_guard_pad(touch_pad_t pad_num) { RTCCNTL.touch_scan_ctrl.touch_out_ring = pad_num; } /** * Get touch channel use for guard pad. * * @param pad_num Touch sensor channel number. */ static inline void touch_ll_waterproof_get_guard_pad(touch_pad_t *pad_num) { *pad_num = RTCCNTL.touch_scan_ctrl.touch_out_ring; } /** * Set max equivalent capacitance for sheild channel. * The equivalent capacitance of the shielded channel can be calculated * from the reading of denoise channel. * * @param pad_num Touch sensor channel number. */ static inline void touch_ll_waterproof_set_sheild_driver(touch_pad_shield_driver_t driver_level) { RTCCNTL.touch_scan_ctrl.touch_bufdrv = driver_level; } /** * Get max equivalent capacitance for sheild channel. * The equivalent capacitance of the shielded channel can be calculated * from the reading of denoise channel. * * @param pad_num Touch sensor channel number. */ static inline void touch_ll_waterproof_get_sheild_driver(touch_pad_shield_driver_t *driver_level) { *driver_level = RTCCNTL.touch_scan_ctrl.touch_bufdrv; } /** * Enable parameter of waterproof function. * * The waterproof function includes a shielded channel (TOUCH_PAD_NUM14) and a guard channel. * Guard pad is used to detect the large area of water covering the touch panel. * Shield pad is used to shield the influence of water droplets covering the touch panel. * It is generally designed as a grid and is placed around the touch buttons. */ static inline void touch_ll_waterproof_enable(void) { RTCCNTL.touch_scan_ctrl.touch_shield_pad_en = 1; } /** * Disable parameter of waterproof function. */ static inline void touch_ll_waterproof_disable(void) { RTCCNTL.touch_scan_ctrl.touch_shield_pad_en = 0; } /************************ Proximity register setting ************************/ /** * Set touch channel number for proximity pad. * If disable the proximity pad, point this pad to `TOUCH_PAD_NUM0` * * @param prox_pad The array of three proximity pads. */ static inline void touch_ll_proximity_set_channel_num(const touch_pad_t prox_pad[]) { SENS.sar_touch_conf.touch_approach_pad0 = prox_pad[0]; SENS.sar_touch_conf.touch_approach_pad1 = prox_pad[1]; SENS.sar_touch_conf.touch_approach_pad2 = prox_pad[2]; } /** * Get touch channel number for proximity pad. * If disable the proximity pad, point this pad to `TOUCH_PAD_NUM0` * * @param prox_pad The array of three proximity pads. */ static inline void touch_ll_proximity_get_channel_num(touch_pad_t prox_pad[]) { prox_pad[0] = SENS.sar_touch_conf.touch_approach_pad0; prox_pad[1] = SENS.sar_touch_conf.touch_approach_pad1; prox_pad[2] = SENS.sar_touch_conf.touch_approach_pad2; } /** * Set cumulative measurement times for proximity pad. * * @param times The cumulative number of measurement cycles. */ static inline void touch_ll_proximity_set_meas_times(uint32_t times) { HAL_FORCE_MODIFY_U32_REG_FIELD(RTCCNTL.touch_approach, touch_approach_meas_time, times); } /** * Get cumulative measurement times for proximity pad. * * @param times The cumulative number of measurement cycles. */ static inline void touch_ll_proximity_get_meas_times(uint32_t *times) { *times = HAL_FORCE_READ_U32_REG_FIELD(RTCCNTL.touch_approach, touch_approach_meas_time); } /** * Read current cumulative measurement times for proximity pad. * * @param times The cumulative number of measurement cycles. */ static inline void touch_ll_proximity_read_meas_cnt(touch_pad_t touch_num, uint32_t *cnt) { if (SENS.sar_touch_conf.touch_approach_pad0 == touch_num) { *cnt = SENS.sar_touch_appr_status.touch_approach_pad0_cnt; } else if (SENS.sar_touch_conf.touch_approach_pad1 == touch_num) { *cnt = SENS.sar_touch_appr_status.touch_approach_pad1_cnt; } else if (SENS.sar_touch_conf.touch_approach_pad2 == touch_num) { *cnt = SENS.sar_touch_appr_status.touch_approach_pad2_cnt; } } /** * Check if the touch sensor channel is the proximity pad. * * @param touch_num The touch sensor channel number. */ static inline bool touch_ll_proximity_pad_check(touch_pad_t touch_num) { if ((SENS.sar_touch_conf.touch_approach_pad0 != touch_num) && (SENS.sar_touch_conf.touch_approach_pad1 != touch_num) && (SENS.sar_touch_conf.touch_approach_pad2 != touch_num)) { return false; } else { return true; } } /************** sleep pad setting ***********************/ /** * Set touch channel number for sleep pad. * * @note Only one touch sensor channel is supported in deep sleep mode. * @param touch_num Touch sensor channel number. */ static inline void touch_ll_sleep_set_channel_num(touch_pad_t touch_num) { RTCCNTL.touch_slp_thres.touch_slp_pad = touch_num; } /** * Get touch channel number for sleep pad. * * @note Only one touch sensor channel is supported in deep sleep mode. * @param touch_num Touch sensor channel number. */ static inline void touch_ll_sleep_get_channel_num(touch_pad_t *touch_num) { *touch_num = RTCCNTL.touch_slp_thres.touch_slp_pad; } /** * Set the trigger threshold of touch sensor in deep sleep. * The threshold determines the sensitivity of the touch sensor. * The threshold is the original value of the trigger state minus the benchmark value. * * @note In general, the touch threshold during sleep can use the threshold parameter parameters before sleep. */ static inline void touch_ll_sleep_set_threshold(uint32_t touch_thres) { RTCCNTL.touch_slp_thres.touch_slp_th = touch_thres; } /** * Get the trigger threshold of touch sensor in deep sleep. * The threshold determines the sensitivity of the touch sensor. * The threshold is the original value of the trigger state minus the benchmark value. * * @note In general, the touch threshold during sleep can use the threshold parameter parameters before sleep. */ static inline void touch_ll_sleep_get_threshold(uint32_t *touch_thres) { *touch_thres = RTCCNTL.touch_slp_thres.touch_slp_th; } /** * Enable proximity function for sleep pad. */ static inline void touch_ll_sleep_enable_approach(void) { RTCCNTL.touch_slp_thres.touch_slp_approach_en = 1; } /** * Disable proximity function for sleep pad. */ static inline void touch_ll_sleep_disable_approach(void) { RTCCNTL.touch_slp_thres.touch_slp_approach_en = 0; } /** * Get proximity function status for sleep pad. */ static inline bool touch_ll_sleep_get_approach_status(void) { return (bool)RTCCNTL.touch_slp_thres.touch_slp_approach_en; } /** * Read benchmark of touch sensor for sleep pad. * * @param benchmark Pointer to accept touch sensor benchmark value. */ static inline void touch_ll_sleep_read_benchmark(uint32_t *benchmark) { SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_BENCHMARK; *benchmark = SENS.sar_touch_slp_status.touch_slp_data; } static inline void touch_ll_sleep_read_smooth(uint32_t *smooth_data) { SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_SMOOTH; *smooth_data = SENS.sar_touch_slp_status.touch_slp_data; } /* Workaround: Note: sleep pad raw data is not in `sar_touch_slp_status` */ static inline void touch_ll_sleep_read_data(uint32_t *raw_data) { uint32_t touch_num = RTCCNTL.touch_slp_thres.touch_slp_pad; SENS.sar_touch_conf.touch_data_sel = TOUCH_LL_READ_RAW; *raw_data = SENS.sar_touch_status[touch_num - 1].touch_pad_data; } static inline void touch_ll_sleep_reset_benchmark(void) { RTCCNTL.touch_approach.touch_slp_channel_clr = 1; } /** * Select touch sensor dbias to save power in sleep mode. * * @note If change the dbias, the reading of touch sensor will changed. Users should make sure the threshold. */ static inline void touch_ll_sleep_low_power(bool is_low_power) { RTCCNTL.touch_ctrl2.touch_dbias = is_low_power; } /** * Read debounce of touch sensor for sleep pad. * * @param debounce Pointer to accept touch sensor debounce value. */ static inline void touch_ll_sleep_read_debounce(uint32_t *debounce) { *debounce = SENS.sar_touch_slp_status.touch_slp_debounce; } /** * Read proximity count of touch sensor for sleep pad. * @param proximity_cnt Pointer to accept touch sensor proximity count value. */ static inline void touch_ll_sleep_read_proximity_cnt(uint32_t *approach_cnt) { *approach_cnt = SENS.sar_touch_appr_status.touch_slp_approach_cnt; } /** * Get the touch pad which caused wakeup from deep sleep. * * @param pad_num pointer to touch pad which caused wakeup. */ static inline void touch_ll_get_wakeup_status(touch_pad_t *pad_num) { *pad_num = (touch_pad_t)RTCCNTL.touch_slp_thres.touch_slp_pad; } #ifdef __cplusplus } #endif