mirror of
https://github.com/ByteWelder/Tactility.git
synced 2026-04-18 09:25:06 +00:00
* **New Features** * BMI270 6-axis IMU driver added; new unified filesystem abstraction for mounted filesystems. * Public Wi‑Fi API surface (no implementation yet) * SDMMC driver added (kernel drive$) * expanded GPIO interrupt/callback support * **Improvements** * M5Stack Tab5: revamped GPIO/power initialization and IMU integration. * LVGL updates including device fontSize configuration. * Updated all code related to SD card device/fs handling * Rename LilyGO T-HMI S3 to LilyGO T-HMI * **Bug Fixes** * Simplified and consolidated SD card handling and mount discovery.
216 lines
8.5 KiB
C++
216 lines
8.5 KiB
C++
#include "devices/Display.h"
|
|
#include "devices/SdCard.h"
|
|
|
|
#include <tactility/drivers/gpio_controller.h>
|
|
#include <tactility/drivers/i2c_controller.h>
|
|
|
|
#include <Tactility/hal/Configuration.h>
|
|
#include <Tactility/hal/i2c/I2c.h>
|
|
|
|
using namespace tt::hal;
|
|
|
|
static constexpr auto* TAG = "Tab5";
|
|
|
|
static DeviceVector createDevices() {
|
|
return {
|
|
createDisplay(),
|
|
createSdCard(),
|
|
};
|
|
}
|
|
|
|
/*
|
|
PI4IOE5V6408-0 (0x43)
|
|
- Bit 0: RF internal/external switch
|
|
- Bit 1: Speaker enable
|
|
- Bit 2: External 5V bus enable
|
|
- Bit 3: /
|
|
- Bit 4: LCD reset
|
|
- Bit 5: Touch reset
|
|
- Bit 6: Camera reset
|
|
- Bit 7: Headphone detect
|
|
*/
|
|
constexpr auto GPIO_EXP0_PIN_RF_INTERNAL_EXTERNAL = 0;
|
|
constexpr auto GPIO_EXP0_PIN_SPEAKER_ENABLE = 1;
|
|
constexpr auto GPIO_EXP0_PIN_EXTERNAL_5V_BUS_ENABLE = 2;
|
|
constexpr auto GPIO_EXP0_PIN_LCD_RESET = 4;
|
|
constexpr auto GPIO_EXP0_PIN_TOUCH_RESET = 5;
|
|
constexpr auto GPIO_EXP0_PIN_CAMERA_RESET = 6;
|
|
constexpr auto GPIO_EXP0_PIN_HEADPHONE_DETECT = 7;
|
|
|
|
/*
|
|
PI4IOE5V6408-1 (0x44)
|
|
- Bit 0: C6 WLAN enable
|
|
- Bit 1: /
|
|
- Bit 2: /
|
|
- Bit 3: USB-A 5V enable
|
|
- Bit 4: Device power: PWROFF_PLUSE
|
|
- Bit 5: IP2326: nCHG_QC_EN
|
|
- Bit 6: IP2326: CHG_STAT_LED
|
|
- Bit 7: IP2326: CHG_EN
|
|
*/
|
|
constexpr auto GPIO_EXP1_PIN_C6_WLAN_ENABLE = 0;
|
|
constexpr auto GPIO_EXP1_PIN_USB_A_5V_ENABLE = 3;
|
|
constexpr auto GPIO_EXP1_PIN_DEVICE_POWER = 4;
|
|
constexpr auto GPIO_EXP1_PIN_IP2326_NCHG_QC_EN = 5;
|
|
constexpr auto GPIO_EXP1_PIN_IP2326_CHG_STAT_LED = 6;
|
|
constexpr auto GPIO_EXP1_PIN_IP2326_CHG_EN = 7;
|
|
|
|
static void initExpander0(::Device* io_expander0) {
|
|
auto* rf_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_RF_INTERNAL_EXTERNAL, GPIO_OWNER_GPIO);
|
|
check(rf_pin);
|
|
auto* speaker_enable_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_SPEAKER_ENABLE, GPIO_OWNER_GPIO);
|
|
check(speaker_enable_pin);
|
|
auto* external_5v_bus_enable_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_EXTERNAL_5V_BUS_ENABLE, GPIO_OWNER_GPIO);
|
|
check(external_5v_bus_enable_pin);
|
|
auto* lcd_reset_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_LCD_RESET, GPIO_OWNER_GPIO);
|
|
check(lcd_reset_pin);
|
|
auto* touch_reset_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_TOUCH_RESET, GPIO_OWNER_GPIO);
|
|
check(touch_reset_pin);
|
|
auto* camera_reset_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_CAMERA_RESET, GPIO_OWNER_GPIO);
|
|
check(camera_reset_pin);
|
|
auto* headphone_detect_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_HEADPHONE_DETECT, GPIO_OWNER_GPIO);
|
|
check(headphone_detect_pin);
|
|
|
|
gpio_descriptor_set_flags(rf_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(speaker_enable_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(external_5v_bus_enable_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(lcd_reset_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(touch_reset_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(camera_reset_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(headphone_detect_pin, GPIO_FLAG_DIRECTION_INPUT);
|
|
|
|
gpio_descriptor_set_level(rf_pin, false);
|
|
gpio_descriptor_set_level(speaker_enable_pin, false);
|
|
gpio_descriptor_set_level(external_5v_bus_enable_pin, true);
|
|
gpio_descriptor_set_level(lcd_reset_pin, false);
|
|
gpio_descriptor_set_level(touch_reset_pin, false);
|
|
gpio_descriptor_set_level(camera_reset_pin, true);
|
|
vTaskDelay(pdMS_TO_TICKS(10));
|
|
// Enable touch and lcd, but not the camera
|
|
gpio_descriptor_set_level(lcd_reset_pin, true);
|
|
gpio_descriptor_set_level(touch_reset_pin, true);
|
|
|
|
gpio_descriptor_release(rf_pin);
|
|
gpio_descriptor_release(speaker_enable_pin);
|
|
gpio_descriptor_release(external_5v_bus_enable_pin);
|
|
gpio_descriptor_release(lcd_reset_pin);
|
|
gpio_descriptor_release(touch_reset_pin);
|
|
gpio_descriptor_release(camera_reset_pin);
|
|
gpio_descriptor_release(headphone_detect_pin);
|
|
}
|
|
|
|
static void initExpander1(::Device* io_expander1) {
|
|
|
|
auto* c6_wlan_enable_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_C6_WLAN_ENABLE, GPIO_OWNER_GPIO);
|
|
auto* usb_a_5v_enable_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_USB_A_5V_ENABLE, GPIO_OWNER_GPIO);
|
|
auto* device_power_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_DEVICE_POWER, GPIO_OWNER_GPIO);
|
|
auto* ip2326_ncharge_qc_enable_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_IP2326_NCHG_QC_EN, GPIO_OWNER_GPIO);
|
|
auto* ip2326_charge_state_led_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_IP2326_CHG_STAT_LED, GPIO_OWNER_GPIO);
|
|
auto* ip2326_charge_enable_pin = gpio_descriptor_acquire(io_expander1, GPIO_EXP1_PIN_IP2326_CHG_EN, GPIO_OWNER_GPIO);
|
|
|
|
gpio_descriptor_set_flags(c6_wlan_enable_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(usb_a_5v_enable_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(device_power_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(ip2326_ncharge_qc_enable_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(ip2326_charge_state_led_pin, GPIO_FLAG_DIRECTION_OUTPUT);
|
|
gpio_descriptor_set_flags(ip2326_charge_enable_pin, GPIO_FLAG_DIRECTION_INPUT | GPIO_FLAG_PULL_UP);
|
|
|
|
gpio_descriptor_set_level(c6_wlan_enable_pin, true);
|
|
gpio_descriptor_set_level(usb_a_5v_enable_pin, true);
|
|
gpio_descriptor_set_level(device_power_pin, false);
|
|
gpio_descriptor_set_level(ip2326_ncharge_qc_enable_pin, false);
|
|
gpio_descriptor_set_level(ip2326_charge_state_led_pin, false);
|
|
|
|
gpio_descriptor_release(c6_wlan_enable_pin);
|
|
gpio_descriptor_release(usb_a_5v_enable_pin);
|
|
gpio_descriptor_release(device_power_pin);
|
|
gpio_descriptor_release(ip2326_ncharge_qc_enable_pin);
|
|
gpio_descriptor_release(ip2326_charge_state_led_pin);
|
|
gpio_descriptor_release(ip2326_charge_enable_pin);
|
|
}
|
|
|
|
static error_t initSound(::Device* i2c_controller, ::Device* io_expander0 = nullptr) {
|
|
// Init data from M5Unified:
|
|
// https://github.com/m5stack/M5Unified/blob/master/src/M5Unified.cpp
|
|
static constexpr uint8_t ES8388_I2C_ADDR = 0x10;
|
|
static constexpr uint8_t ENABLED_BULK_DATA[] = {
|
|
0, 0x80, // RESET/ CSM POWER ON
|
|
0, 0x00,
|
|
0, 0x00,
|
|
0, 0x0E,
|
|
1, 0x00,
|
|
2, 0x0A, // CHIP POWER: power up all
|
|
3, 0xFF, // ADC POWER: power down all
|
|
4, 0x3C, // DAC POWER: power up and LOUT1/ROUT1/LOUT2/ROUT2 enable
|
|
5, 0x00, // ChipLowPower1
|
|
6, 0x00, // ChipLowPower2
|
|
7, 0x7C, // VSEL
|
|
8, 0x00, // set I2S slave mode
|
|
// reg9-22 == adc
|
|
23, 0x18, // I2S format (16bit)
|
|
24, 0x00, // I2S MCLK ratio (128)
|
|
25, 0x20, // DAC unmute
|
|
26, 0x00, // LDACVOL 0x00~0xC0
|
|
27, 0x00, // RDACVOL 0x00~0xC0
|
|
28, 0x08, // enable digital click free power up and down
|
|
29, 0x00,
|
|
38, 0x00, // DAC CTRL16
|
|
39, 0xB8, // LEFT Ch MIX
|
|
42, 0xB8, // RIGHTCh MIX
|
|
43, 0x08, // ADC and DAC separate
|
|
45, 0x00, // 0x00=1.5k VREF analog output / 0x10=40kVREF analog output
|
|
46, 0x21,
|
|
47, 0x21,
|
|
48, 0x21,
|
|
49, 0x21
|
|
};
|
|
|
|
error_t error = i2c_controller_write_register_array(
|
|
i2c_controller,
|
|
ES8388_I2C_ADDR,
|
|
ENABLED_BULK_DATA,
|
|
sizeof(ENABLED_BULK_DATA),
|
|
pdMS_TO_TICKS(1000)
|
|
);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to enable ES8388: %s", error_to_string(error));
|
|
return error;
|
|
}
|
|
|
|
auto* speaker_enable_pin = gpio_descriptor_acquire(io_expander0, GPIO_EXP0_PIN_SPEAKER_ENABLE, GPIO_OWNER_GPIO);
|
|
check(speaker_enable_pin, "Failed to acquire speaker enable pin");
|
|
error = gpio_descriptor_set_level(speaker_enable_pin, true);
|
|
gpio_descriptor_release(speaker_enable_pin);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to enable amplifier: %s", error_to_string(error));
|
|
return ERROR_RESOURCE;
|
|
}
|
|
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
static bool initBoot() {
|
|
auto* i2c0 = device_find_by_name("i2c0");
|
|
check(i2c0, "i2c0 not found");
|
|
|
|
auto* io_expander0 = device_find_by_name("io_expander0");
|
|
check(io_expander0, "io_expander0 not found");
|
|
auto* io_expander1 = device_find_by_name("io_expander1");
|
|
check(io_expander1, "io_expander1 not found");
|
|
|
|
initExpander0(io_expander0);
|
|
initExpander1(io_expander1);
|
|
|
|
error_t error = initSound(i2c0, io_expander0);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to enable ES8388");
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
extern const Configuration hardwareConfiguration = {
|
|
.initBoot = initBoot,
|
|
.createDevices = createDevices
|
|
};
|