#include "badge/gpio.h" #include "badge/config.h" #include #define BUTTON_PIN 0 #if UVOK_EPAP_BOARD == BOARD_ESP32_CROWPANEL // Elecrow #define EXIT_KEY 1 #define HOME_KEY 2 #define NEXT_KEY 4 #define OK_KEY 5 #define PRV_KEY 6 #endif static void gpio_loop(void *); typedef uint32_t pin_notification_t; #define GPIO_STACK 2048 #define GPIO_QUEUE_LEN 4 static unsigned long pressedTime = 0; static unsigned long releasedTime = 0; typedef struct { uint32_t last_change; uint32_t last_level; uint32_t pressed; } InputInfo_t; static volatile InputInfo_t inputs[NUM_DIGITAL_PINS]; static struct { StaticTask_t task; StackType_t task_stack[GPIO_STACK / sizeof(StackType_t)]; StaticQueue_t queue; uint8_t queue_storage[GPIO_QUEUE_LEN * sizeof(pin_notification_t)]; QueueHandle_t queue_handle; } gpio_task_stuff; static ARDUINO_ISR_ATTR void gpio_pin_irq(void *); void de::uvok::badge::gpio_init(void) { xTaskCreateStatic(gpio_loop, "gpio", GPIO_STACK, NULL, configMAX_PRIORITIES - 1, gpio_task_stuff.task_stack, &gpio_task_stuff.task); gpio_task_stuff.queue_handle = xQueueCreateStatic(4, sizeof(pin_notification_t), &(gpio_task_stuff.queue_storage[0]), &gpio_task_stuff.queue); #if UVOK_EPAP_BOARD == BOARD_ESP32_CROWPANEL uint8_t inPins[] = {EXIT_KEY, HOME_KEY, NEXT_KEY, OK_KEY, PRV_KEY}; for (uint8_t p : inPins) { pinMode(p, GPIO_MODE_INPUT); // simply use pin number as arg attachInterruptArg(p, gpio_pin_irq, (void *)(ptrdiff_t)p, CHANGE); inputs[p].last_level = digitalRead(p); } #endif } long de::uvok::badge::gpio_poll(void) { int x = 0; static int lastState = HIGH; int buttonState = digitalRead(BUTTON_PIN); long pressDuration = 0; if (lastState == HIGH && buttonState == LOW) { Serial.println("``\\__"); pressedTime = millis(); lastState = LOW; } else if (lastState == LOW && buttonState == HIGH) { lastState = HIGH; Serial.println("__/``"); releasedTime = millis(); pressDuration = releasedTime - pressedTime; }; return pressDuration; } static void gpio_loop(void *ctx) { Serial.println("Starting GPIO loop"); while (1) { pin_notification_t received; if (xQueueReceive(gpio_task_stuff.queue_handle, &received, portMAX_DELAY) == pdTRUE) { const int level = (received & 0x100) ? HIGH : LOW; const bool pressed = !level; const int pin = received & 0xff; Serial.printf("Pin %d was %s\n", pin, pressed ? "pressed" : "released"); } } } static ARDUINO_ISR_ATTR void gpio_pin_irq(void *arg) { uint8_t pin_no = (uint8_t)(ptrdiff_t)arg; uint32_t now = millis(); // Elecrow inputs - the selector - are very dirty... const bool is_dirty = now - inputs[pin_no].last_change < 100; inputs[pin_no].last_change = now; if (is_dirty) return; int pinLevel = digitalRead(pin_no); pin_notification_t send = (pinLevel << 8) | (pin_no); BaseType_t wakeup = false; xQueueSendFromISR(gpio_task_stuff.queue_handle, &send, &wakeup); if (wakeup) { portYIELD_FROM_ISR(); } }