#include #include #include #include #include #include "devices/Tab5Keyboard.h" #include #include #include #define TAG "Tab5" constexpr auto GPIO_EXP0_PIN_SPEAKER_ENABLE = 1; constexpr auto GPIO_EXP0_PIN_HEADPHONE_DETECT = 7; constexpr auto HP_DETECT_POLL_MS = 1000; constexpr auto KB_DETECT_POLL_MS = 1000; // hp_detect_timer and kb_detect_timer are only touched from start()/stop(), which are called // serially by the module manager — no atomic needed for the handles themselves. static TimerHandle_t hp_detect_timer = nullptr; static TimerHandle_t kb_detect_timer = nullptr; static std::atomic io_expander0_cached { nullptr }; // Flags are written by the timer daemon task and read by start()/stop() — use atomics. static std::atomic hp_detect_last { false }; static std::atomic hp_detect_initialized { false }; static std::atomic kb_late_started { false }; static void headphoneDetectCallback(TimerHandle_t /*timer*/) { Device* cached = io_expander0_cached.load(std::memory_order_acquire); if (!cached) { cached = device_find_by_name("io_expander0"); io_expander0_cached.store(cached, std::memory_order_release); } auto* io_expander0 = cached; if (!io_expander0) { return; // Not ready yet, will retry on next tick } auto* hp_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_HEADPHONE_DETECT, GPIO_OWNER_GPIO); if (!hp_pin) { LOG_W(TAG, "hp_detect: HP_DET pin busy"); return; } bool hp = false; error_t err = gpio_descriptor_get_level(hp_pin, &hp); gpio_descriptor_release(hp_pin); if (err != ERROR_NONE) { LOG_W(TAG, "hp_detect: HP_DET read error: %s", error_to_string(err)); return; } LOG_D(TAG, "hp_detect: HP_DET=%d", (int)hp); if (!hp_detect_initialized || hp != hp_detect_last) { auto* spk_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_SPEAKER_ENABLE, GPIO_OWNER_GPIO); if (!spk_pin) { LOG_W(TAG, "hp_detect: SPK_EN pin busy, will retry"); return; } error_t spk_err = gpio_descriptor_set_level(spk_pin, !hp); gpio_descriptor_release(spk_pin); if (spk_err != ERROR_NONE) { LOG_W(TAG, "hp_detect: SPK_EN set error: %s, will retry", error_to_string(spk_err)); return; } hp_detect_last = hp; hp_detect_initialized = true; LOG_I(TAG, "Headphones %s, speaker %s", hp ? "detected" : "removed", hp ? "disabled" : "enabled"); } } // Detects a Tab5 Keyboard add-on that was plugged in after boot (so it wasn't started by // Lvgl.cpp's attachDevices()). Once lateStart() succeeds, this stops polling for good — there's // no support for re-detecting after the indev is torn down again. static void keyboardDetectCallback(TimerHandle_t timer) { if (kb_late_started.load(std::memory_order_acquire)) { xTimerStop(timer, 0); return; } using namespace tt::hal; auto keyboard = findFirstDevice(tt::hal::Device::Type::Keyboard); if (!keyboard) { return; // Not registered yet, will retry on next tick } if (keyboard->getLvglIndev() != nullptr) { // Already started (boot-time attach) — nothing left to do. kb_late_started.store(true, std::memory_order_release); xTimerStop(timer, 0); return; } if (!keyboard->isAttached()) { return; // Not plugged in yet, will retry on next tick } auto tab5_keyboard = std::static_pointer_cast(keyboard); if (tab5_keyboard->lateStart()) { LOG_I(TAG, "kb_detect: keyboard attached post-boot, LVGL input started"); kb_late_started.store(true, std::memory_order_release); xTimerStop(timer, 0); } } extern "C" { static error_t start() { if (hp_detect_timer != nullptr) { LOG_W(TAG, "hp_detect timer already running"); return ERROR_NONE; } hp_detect_initialized = false; hp_detect_last = false; hp_detect_timer = xTimerCreate("hp_detect", pdMS_TO_TICKS(HP_DETECT_POLL_MS), pdTRUE, nullptr, headphoneDetectCallback); if (!hp_detect_timer) { LOG_E(TAG, "Failed to create hp_detect timer"); return ERROR_RESOURCE; } if (xTimerStart(hp_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_E(TAG, "Failed to start hp_detect timer"); xTimerDelete(hp_detect_timer, pdMS_TO_TICKS(100)); hp_detect_timer = nullptr; return ERROR_RESOURCE; } kb_late_started = false; kb_detect_timer = xTimerCreate("kb_detect", pdMS_TO_TICKS(KB_DETECT_POLL_MS), pdTRUE, nullptr, keyboardDetectCallback); if (!kb_detect_timer) { LOG_E(TAG, "Failed to create kb_detect timer"); return ERROR_RESOURCE; } if (xTimerStart(kb_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_E(TAG, "Failed to start kb_detect timer"); xTimerDelete(kb_detect_timer, pdMS_TO_TICKS(100)); kb_detect_timer = nullptr; return ERROR_RESOURCE; } return ERROR_NONE; } static error_t stop() { if (hp_detect_timer != nullptr) { if (xTimerStop(hp_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_W(TAG, "Failed to stop hp_detect timer"); } if (xTimerDelete(hp_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_E(TAG, "Failed to delete hp_detect timer"); } // Always clear the handle — stale non-null handle is worse than a resource leak, // as it would cause start() to silently skip re-creating the timer. hp_detect_timer = nullptr; io_expander0_cached.store(nullptr, std::memory_order_release); } if (kb_detect_timer != nullptr) { if (xTimerStop(kb_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_W(TAG, "Failed to stop kb_detect timer"); } if (xTimerDelete(kb_detect_timer, pdMS_TO_TICKS(100)) != pdPASS) { LOG_E(TAG, "Failed to delete kb_detect timer"); } kb_detect_timer = nullptr; } return ERROR_NONE; } Module m5stack_tab5_module = { .name = "m5stack-tab5", .start = start, .stop = stop, .symbols = nullptr, .internal = nullptr }; }