diff --git a/arch/esp32/esp32s2.ini b/arch/esp32/esp32s2.ini index 40fdc461aa..fe92954a4e 100644 --- a/arch/esp32/esp32s2.ini +++ b/arch/esp32/esp32s2.ini @@ -16,4 +16,7 @@ build_flags = lib_ignore = ${esp32_base.lib_ignore} NimBLE-Arduino - libpax \ No newline at end of file + libpax + +lib_deps = ${esp32_base.lib_deps} + tanakamasayuki/EspUsbHost@^1.0.2 diff --git a/arch/esp32/esp32s3.ini b/arch/esp32/esp32s3.ini index 1cd0e20331..c53e2f225c 100644 --- a/arch/esp32/esp32s3.ini +++ b/arch/esp32/esp32s3.ini @@ -4,3 +4,6 @@ custom_esp32_kind = esp32s3 monitor_speed = 115200 +lib_deps = ${esp32_base.lib_deps} + tanakamasayuki/EspUsbHost@^1.0.2 + diff --git a/src/input/kbUsbBase.cpp b/src/input/kbUsbBase.cpp new file mode 100644 index 0000000000..9aedbf70cb --- /dev/null +++ b/src/input/kbUsbBase.cpp @@ -0,0 +1,72 @@ +#include "kbUsbBase.h" +#include "configuration.h" + +#if CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32S2 + +KbUsbBase::KbUsbBase(const char *name) : concurrency::OSThread(name) +{ + this->_originName = name; +} + +int32_t KbUsbBase::runOnce() +{ + if (firstTime) { + // This is the first time the OSThread library has called this function, so init the USB HID routines + begin(); + firstTime = 0; + } else { + + task(); + } + return 100; +} + +void KbUsbBase::onKeyboardKey(uint8_t ascii, uint8_t keycode, uint8_t modifier) +{ + + if (ascii != 0) { + LOG_DEBUG("Key 0x%x Code 0x%x Mod 0x%x pressed\n", ascii, keycode, modifier); + // reset shift now that we have a keypress + InputEvent e; + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_NONE; + e.source = this->_originName; + switch (ascii) { + case 0x1b: // ESC + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_CANCEL; + break; + case 0x08: // Back + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_BACK; + e.kbchar = ascii; + break; + case 0xb5: // Up + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_UP; + break; + case 0xb6: // Down + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_DOWN; + break; + case 0xb4: // Left + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_LEFT; + e.kbchar = ascii; + break; + case 0xb7: // Right + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_RIGHT; + e.kbchar = ascii; + break; + case 0x0d: // Enter + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_SELECT; + break; + case 0x00: // nopress + e.inputEvent = meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_NONE; + break; + default: // all other keys + e.inputEvent = ANYKEY; + e.kbchar = ascii; + break; + } + if (e.inputEvent != meshtastic_ModuleConfig_CannedMessageConfig_InputEventChar_NONE) { + this->notifyObservers(&e); + } + } +} + +#endif \ No newline at end of file diff --git a/src/input/kbUsbBase.h b/src/input/kbUsbBase.h new file mode 100644 index 0000000000..15194c035b --- /dev/null +++ b/src/input/kbUsbBase.h @@ -0,0 +1,21 @@ +#pragma once + +#include "InputBroker.h" +#include "concurrency/OSThread.h" +#if CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32S2 +#include "EspUsbHost.h" + +class KbUsbBase : public Observable, public concurrency::OSThread, public EspUsbHost +{ + public: + explicit KbUsbBase(const char *name); + + protected: + virtual int32_t runOnce() override; + + private: + void onKeyboardKey(uint8_t ascii, uint8_t keycode, uint8_t modifier); + const char *_originName; + bool firstTime = 1; +}; +#endif \ No newline at end of file diff --git a/src/input/kbUsbImpl.cpp b/src/input/kbUsbImpl.cpp new file mode 100644 index 0000000000..2d16088936 --- /dev/null +++ b/src/input/kbUsbImpl.cpp @@ -0,0 +1,15 @@ +#include "kbUsbImpl.h" +#include "InputBroker.h" + +#if CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32S2 + +KbUsbImpl *kbUsbImpl; + +KbUsbImpl::KbUsbImpl() : KbUsbBase("usbKB") {} + +void KbUsbImpl::init() +{ + inputBroker->registerSource(this); +} + +#endif // INPUTBROKER_MATRIX_TYPE \ No newline at end of file diff --git a/src/input/kbUsbImpl.h b/src/input/kbUsbImpl.h new file mode 100644 index 0000000000..6715d60a3a --- /dev/null +++ b/src/input/kbUsbImpl.h @@ -0,0 +1,23 @@ +#pragma once +#include "kbUsbBase.h" +#include "main.h" + +#if CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32S2 + +/** + * @brief The idea behind this class to have static methods for the event handlers. + * Check attachInterrupt() at RotaryEncoderInteruptBase.cpp + * Technically you can have as many rotary encoders hardver attached + * to your device as you wish, but you always need to have separate event + * handlers, thus you need to have a RotaryEncoderInterrupt implementation. + */ +class KbUsbImpl : public KbUsbBase +{ + public: + KbUsbImpl(); + void init(); +}; + +extern KbUsbImpl *kbUsbImpl; + +#endif \ No newline at end of file diff --git a/src/modules/Modules.cpp b/src/modules/Modules.cpp index f386147d07..f6a5598864 100644 --- a/src/modules/Modules.cpp +++ b/src/modules/Modules.cpp @@ -9,7 +9,9 @@ #include "input/UpDownInterruptImpl1.h" #include "input/cardKbI2cImpl.h" #include "input/kbMatrixImpl.h" +#include "input/kbUsbImpl.h" #endif + #if !MESHTASTIC_EXCLUDE_ADMIN #include "modules/AdminModule.h" #endif @@ -168,10 +170,14 @@ void setupModules() kbMatrixImpl = new KbMatrixImpl(); kbMatrixImpl->init(); #endif // INPUTBROKER_MATRIX_TYPE +#if CONFIG_IDF_TARGET_ESP32S3 || CONFIG_IDF_TARGET_ESP32S2 + kbUsbImpl = new KbUsbImpl(); + kbUsbImpl->init(); +#endif #ifdef INPUTBROKER_SERIAL_TYPE aSerialKeyboardImpl = new SerialKeyboardImpl(); aSerialKeyboardImpl->init(); -#endif // INPUTBROKER_MATRIX_TYPE +#endif // INPUTBROKER_SERIAL_TYPE #endif // HAS_BUTTON #if ARCH_PORTDUINO && !HAS_TFT aLinuxInputImpl = new LinuxInputImpl(); diff --git a/variants/bpi_picow_esp32_s3/platformio.ini b/variants/bpi_picow_esp32_s3/platformio.ini index 7e94cc97e2..992e14e150 100644 --- a/variants/bpi_picow_esp32_s3/platformio.ini +++ b/variants/bpi_picow_esp32_s3/platformio.ini @@ -8,7 +8,7 @@ board_level = extra upload_protocol = esptool ;upload_port = /dev/ttyACM2 lib_deps = - ${esp32_base.lib_deps} + ${esp32s3_base.lib_deps} caveman99/ESP32 Codec2@^1.0.1 build_flags = - ${esp32_base.build_flags} -D PRIVATE_HW -I variants/bpi_picow_esp32_s3 \ No newline at end of file + ${esp32s3_base.build_flags} -D PRIVATE_HW -I variants/bpi_picow_esp32_s3 \ No newline at end of file diff --git a/variants/heltec_esp32c3/platformio.ini b/variants/heltec_esp32c3/platformio.ini index 6fe5c3c691..61aa94b6a3 100644 --- a/variants/heltec_esp32c3/platformio.ini +++ b/variants/heltec_esp32c3/platformio.ini @@ -2,7 +2,7 @@ extends = esp32c3_base board = esp32-c3-devkitm-1 build_flags = - ${esp32_base.build_flags} + ${esp32c3_base.build_flags} -D HELTEC_HT62 -I variants/heltec_esp32c3 monitor_speed = 115200 diff --git a/variants/m5stack-stamp-c3/platformio.ini b/variants/m5stack-stamp-c3/platformio.ini index bab65b6213..a48d5cc263 100644 --- a/variants/m5stack-stamp-c3/platformio.ini +++ b/variants/m5stack-stamp-c3/platformio.ini @@ -3,7 +3,7 @@ extends = esp32c3_base board = esp32-c3-devkitm-1 board_level = extra build_flags = - ${esp32_base.build_flags} + ${esp32c3_base.build_flags} -D PRIVATE_HW -I variants/m5stack-stamp-c3 monitor_speed = 115200 diff --git a/variants/my_esp32s3_diy_eink/platformio.ini b/variants/my_esp32s3_diy_eink/platformio.ini index e81f2c1abe..ee99ce7920 100644 --- a/variants/my_esp32s3_diy_eink/platformio.ini +++ b/variants/my_esp32s3_diy_eink/platformio.ini @@ -11,13 +11,13 @@ upload_speed = 921600 platform_packages = tool-esptoolpy@^1.40500.0 lib_deps = - ${esp32_base.lib_deps} + ${esp32s3_base.lib_deps} zinggjm/GxEPD2@^1.5.1 adafruit/Adafruit NeoPixel @ ^1.12.0 build_unflags = -DARDUINO_USB_MODE=1 build_flags = - ;${esp32_base.build_flags} -D MY_ESP32S3_DIY -I variants/my_esp32s3_diy_eink - ${esp32_base.build_flags} -D PRIVATE_HW -I variants/my_esp32s3_diy_eink + ;${esp32s3_base.build_flags} -D MY_ESP32S3_DIY -I variants/my_esp32s3_diy_eink + ${esp32s3_base.build_flags} -D PRIVATE_HW -I variants/my_esp32s3_diy_eink -Dmy -DEINK_DISPLAY_MODEL=GxEPD2_290_T5D -DEINK_WIDTH=296 diff --git a/variants/my_esp32s3_diy_oled/platformio.ini b/variants/my_esp32s3_diy_oled/platformio.ini index 2d7a5cd910..0cce9a3d12 100644 --- a/variants/my_esp32s3_diy_oled/platformio.ini +++ b/variants/my_esp32s3_diy_oled/platformio.ini @@ -11,12 +11,12 @@ upload_speed = 921600 platform_packages = tool-esptoolpy@^1.40500.0 lib_deps = - ${esp32_base.lib_deps} + ${esp32s3_base.lib_deps} adafruit/Adafruit NeoPixel @ ^1.12.0 build_unflags = -DARDUINO_USB_MODE=1 build_flags = - ;${esp32_base.build_flags} -D MY_ESP32S3_DIY -I variants/my_esp32s3_diy_oled - ${esp32_base.build_flags} -D PRIVATE_HW -I variants/my_esp32s3_diy_oled + ;${esp32s3_base.build_flags} -D MY_ESP32S3_DIY -I variants/my_esp32s3_diy_oled + ${esp32s3_base.build_flags} -D PRIVATE_HW -I variants/my_esp32s3_diy_oled -DBOARD_HAS_PSRAM -mfix-esp32-psram-cache-issue -DARDUINO_USB_MODE=0 \ No newline at end of file diff --git a/variants/tlora_t3s3_v1/platformio.ini b/variants/tlora_t3s3_v1/platformio.ini index 0a57972803..2ff827b8ec 100644 --- a/variants/tlora_t3s3_v1/platformio.ini +++ b/variants/tlora_t3s3_v1/platformio.ini @@ -5,5 +5,5 @@ board_check = true upload_protocol = esptool build_flags = - ${esp32_base.build_flags} -D TLORA_T3S3_V1 -I variants/tlora_t3s3_v1 - -DGPS_POWER_TOGGLE ; comment this line to disable triple press function on the user button to turn off gps entirely. \ No newline at end of file + ${esp32s3_base.build_flags} -D TLORA_T3S3_V1 -I variants/tlora_t3s3_v1 + -DGPS_POWER_TOGGLE ; comment this line to disable triple press function on the user button to turn off gps entirely.