123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177 |
- /*
- * infrared receiver (using espressif's example)
- *
- * (c) Philippe G. 2020, philippe_44@outlook.com
- *
- * This software is released under the MIT License.
- * https://opensource.org/licenses/MIT
- *
- */
- #include <stdio.h>
- #include <string.h>
- #include "freertos/FreeRTOS.h"
- #include "freertos/task.h"
- #include "esp_err.h"
- #include "esp_log.h"
- #include "driver/rmt.h"
- #include "infrared.h"
- static const char* TAG = "IR";
- #define RMT_RX_ACTIVE_LEVEL 0 /*!< If we connect with a IR receiver, the data is active low */
- #define RMT_RX_CHANNEL 0 /*!< RMT channel for receiver */
- #define RMT_CLK_DIV 100 /*!< RMT counter clock divider */
- #define RMT_TICK_10_US (80000000/RMT_CLK_DIV/100000) /*!< RMT counter value for 10 us.(Source clock is APB clock) */
- #define NEC_HEADER_HIGH_US 9000 /*!< NEC protocol header: positive 9ms */
- #define NEC_HEADER_LOW_US 4500 /*!< NEC protocol header: negative 4.5ms*/
- #define NEC_BIT_ONE_HIGH_US 560 /*!< NEC protocol data bit 1: positive 0.56ms */
- #define NEC_BIT_ONE_LOW_US (2250-NEC_BIT_ONE_HIGH_US) /*!< NEC protocol data bit 1: negative 1.69ms */
- #define NEC_BIT_ZERO_HIGH_US 560 /*!< NEC protocol data bit 0: positive 0.56ms */
- #define NEC_BIT_ZERO_LOW_US (1120-NEC_BIT_ZERO_HIGH_US) /*!< NEC protocol data bit 0: negative 0.56ms */
- #define NEC_BIT_MARGIN 150 /*!< NEC parse margin time */
- #define NEC_ITEM_DURATION(d) ((d & 0x7fff)*10/RMT_TICK_10_US) /*!< Parse duration time from memory register value */
- #define NEC_DATA_ITEM_NUM 34 /*!< NEC code item number: header + 32bit data + end */
- #define rmt_item32_tIMEOUT_US 9500 /*!< RMT receiver timeout value(us) */
- /****************************************************************************************
- *
- */
- inline bool nec_check_in_range(int duration_ticks, int target_us, int margin_us) {
- if(( NEC_ITEM_DURATION(duration_ticks) < (target_us + margin_us))
- && ( NEC_ITEM_DURATION(duration_ticks) > (target_us - margin_us))) {
- return true;
- } else {
- return false;
- }
- }
- /****************************************************************************************
- *
- */
- static bool nec_header_if(rmt_item32_t* item) {
- if((item->level0 == RMT_RX_ACTIVE_LEVEL && item->level1 != RMT_RX_ACTIVE_LEVEL)
- && nec_check_in_range(item->duration0, NEC_HEADER_HIGH_US, NEC_BIT_MARGIN)
- && nec_check_in_range(item->duration1, NEC_HEADER_LOW_US, NEC_BIT_MARGIN)) {
- return true;
- }
- return false;
- }
- /****************************************************************************************
- *
- */
- static bool nec_bit_one_if(rmt_item32_t* item) {
- if((item->level0 == RMT_RX_ACTIVE_LEVEL && item->level1 != RMT_RX_ACTIVE_LEVEL)
- && nec_check_in_range(item->duration0, NEC_BIT_ONE_HIGH_US, NEC_BIT_MARGIN)
- && nec_check_in_range(item->duration1, NEC_BIT_ONE_LOW_US, NEC_BIT_MARGIN)) {
- return true;
- }
- return false;
- }
- /****************************************************************************************
- *
- */
- static bool nec_bit_zero_if(rmt_item32_t* item) {
- if((item->level0 == RMT_RX_ACTIVE_LEVEL && item->level1 != RMT_RX_ACTIVE_LEVEL)
- && nec_check_in_range(item->duration0, NEC_BIT_ZERO_HIGH_US, NEC_BIT_MARGIN)
- && nec_check_in_range(item->duration1, NEC_BIT_ZERO_LOW_US, NEC_BIT_MARGIN)) {
- return true;
- }
- return false;
- }
- /****************************************************************************************
- *
- */
- static int nec_parse_items(rmt_item32_t* item, int item_num, uint16_t* addr, uint16_t* data) {
- int w_len = item_num;
- if(w_len < NEC_DATA_ITEM_NUM) {
- return -1;
- }
- int i = 0, j = 0;
- if(!nec_header_if(item++)) {
- return -1;
- }
- uint16_t addr_t = 0;
- for(j = 15; j >= 0; j--) {
- if(nec_bit_one_if(item)) {
- addr_t |= (1 << j);
- } else if(nec_bit_zero_if(item)) {
- addr_t |= (0 << j);
- } else {
- return -1;
- }
- item++;
- i++;
- }
- uint16_t data_t = 0;
- for(j = 15; j >= 0; j--) {
- if(nec_bit_one_if(item)) {
- data_t |= (1 << j);
- } else if(nec_bit_zero_if(item)) {
- data_t |= (0 << j);
- } else {
- return -1;
- }
- item++;
- i++;
- }
- *addr = addr_t;
- *data = data_t;
- return i;
- }
- /****************************************************************************************
- *
- */
- void infrared_receive(RingbufHandle_t rb, infrared_handler handler) {
- size_t rx_size = 0;
- rmt_item32_t* item = (rmt_item32_t*) xRingbufferReceive(rb, &rx_size, 10 / portTICK_RATE_MS);
-
- if (item) {
- uint16_t addr, cmd;
- int offset = 0;
-
- while (1) {
- // parse data value from ringbuffer.
- int res = nec_parse_items(item + offset, rx_size / 4 - offset, &addr, &cmd);
- if (res > 0) {
- offset += res + 1;
- handler(addr, cmd);
- ESP_LOGD(TAG, "RMT RCV --- addr: 0x%04x cmd: 0x%04x", addr, cmd);
- } else break;
- }
-
- // after parsing the data, return spaces to ringbuffer.
- vRingbufferReturnItem(rb, (void*) item);
- }
- }
- /****************************************************************************************
- *
- */
- void infrared_init(RingbufHandle_t *rb, int gpio) {
- rmt_config_t rmt_rx;
-
- ESP_LOGI(TAG, "Starting Infrared Receiver on gpio %d", gpio);
-
- // initialize RMT driver
- rmt_rx.channel = RMT_RX_CHANNEL;
- rmt_rx.gpio_num = gpio;
- rmt_rx.clk_div = RMT_CLK_DIV;
- rmt_rx.mem_block_num = 1;
- rmt_rx.rmt_mode = RMT_MODE_RX;
- rmt_rx.rx_config.filter_en = true;
- rmt_rx.rx_config.filter_ticks_thresh = 100;
- rmt_rx.rx_config.idle_threshold = rmt_item32_tIMEOUT_US / 10 * (RMT_TICK_10_US);
- rmt_config(&rmt_rx);
- rmt_driver_install(rmt_rx.channel, 1000, 0);
-
- // get RMT RX ringbuffer
- rmt_get_ringbuf_handle(RMT_RX_CHANNEL, rb);
- rmt_rx_start(RMT_RX_CHANNEL, 1);
- }
|