From 7e2be83f27e83077f4af2b3bd763d28a6652eb5a Mon Sep 17 00:00:00 2001 From: Tinyu Date: Wed, 4 Sep 2024 15:25:30 +0800 Subject: [PATCH] bsp: New BSP for M5Core2 (BSP-514) (#344) * bsp: New BSP for M5Core2 --------- Signed-off-by: Tinyu --- .build-test-rules.yml | 2 +- .github/workflows/upload_component.yml | 2 +- .github/workflows/upload_component_noglib.yml | 2 +- README.md | 59 +- SquareLine/boards/v9/m5stack_core_2/image.png | Bin 0 -> 43600 bytes .../v9/m5stack_core_2/main/idf_component.yml | 6 + .../boards/v9/m5stack_core_2/manifest.json | 15 + bsp/m5stack_core_2/CMakeLists.txt | 7 + bsp/m5stack_core_2/Kconfig | 102 +++ bsp/m5stack_core_2/LICENSE | 202 ++++++ bsp/m5stack_core_2/README.md | 47 ++ bsp/m5stack_core_2/idf_component.yml | 23 + bsp/m5stack_core_2/include/bsp/config.h | 16 + bsp/m5stack_core_2/include/bsp/display.h | 121 ++++ bsp/m5stack_core_2/include/bsp/esp-bsp.h | 8 + .../include/bsp/m5stack_core_2.h | 361 +++++++++++ bsp/m5stack_core_2/include/bsp/touch.h | 51 ++ bsp/m5stack_core_2/m5stack_core_2.c | 603 ++++++++++++++++++ bsp/m5stack_core_2/m5stack_core_2_idf5.c | 100 +++ .../priv_include/bsp_err_check.h | 58 ++ docu/pics/m5stack_core2.webp | Bin 0 -> 15222 bytes examples/bsp_ext.py | 3 +- examples/display/sdkconfig.bsp.m5stack_core_2 | 25 + .../display_audio_photo/main/app_disp_fs.c | 6 + .../sdkconfig.bsp.m5stack_core_2 | 26 + 25 files changed, 1812 insertions(+), 33 deletions(-) create mode 100644 SquareLine/boards/v9/m5stack_core_2/image.png create mode 100644 SquareLine/boards/v9/m5stack_core_2/main/idf_component.yml create mode 100644 SquareLine/boards/v9/m5stack_core_2/manifest.json create mode 100644 bsp/m5stack_core_2/CMakeLists.txt create mode 100644 bsp/m5stack_core_2/Kconfig create mode 100644 bsp/m5stack_core_2/LICENSE create mode 100644 bsp/m5stack_core_2/README.md create mode 100644 bsp/m5stack_core_2/idf_component.yml create mode 100644 bsp/m5stack_core_2/include/bsp/config.h create mode 100644 bsp/m5stack_core_2/include/bsp/display.h create mode 100644 bsp/m5stack_core_2/include/bsp/esp-bsp.h create mode 100644 bsp/m5stack_core_2/include/bsp/m5stack_core_2.h create mode 100644 bsp/m5stack_core_2/include/bsp/touch.h create mode 100644 bsp/m5stack_core_2/m5stack_core_2.c create mode 100644 bsp/m5stack_core_2/m5stack_core_2_idf5.c create mode 100644 bsp/m5stack_core_2/priv_include/bsp_err_check.h create mode 100644 docu/pics/m5stack_core2.webp create mode 100644 examples/display/sdkconfig.bsp.m5stack_core_2 create mode 100644 examples/display_audio_photo/sdkconfig.bsp.m5stack_core_2 diff --git a/.build-test-rules.yml b/.build-test-rules.yml index 0be46653..d58f6056 100644 --- a/.build-test-rules.yml +++ b/.build-test-rules.yml @@ -2,7 +2,7 @@ examples: disable: - if: CONFIG_NAME in ["esp-box", "esp-box-lite"] reason: Do not build examples for deprecated BSPs - - if: IDF_VERSION_MAJOR < 5 and CONFIG_NAME in ["esp32_c3_lcdkit", "esp32_s3_lcd_ev_board", "esp32_s3_usb_otg", "m5stack_core_s3", "m5dial"] + - if: IDF_VERSION_MAJOR < 5 and CONFIG_NAME in ["esp32_c3_lcdkit", "esp32_s3_lcd_ev_board", "esp32_s3_usb_otg", "m5stack_core_s3", "m5stack_core_2", "m5dial"] reason: Example depends on BSP, which is supported only for IDF >= 5.0 - if: IDF_VERSION_MAJOR < 5 and IDF_TARGET in ["esp32c2", "esp32p4", "esp32c5", "esp32c6"] reason: Example depends on target, which is supported only for IDF >= 5.0 diff --git a/.github/workflows/upload_component.yml b/.github/workflows/upload_component.yml index f52ddd01..b8985005 100644 --- a/.github/workflows/upload_component.yml +++ b/.github/workflows/upload_component.yml @@ -17,7 +17,7 @@ jobs: uses: espressif/upload-components-ci-action@v1 with: directories: > - bsp/esp32_azure_iot_kit;bsp/esp32_s2_kaluga_kit;bsp/esp_wrover_kit;bsp/esp-box;bsp/esp32_s3_usb_otg;bsp/esp32_s3_eye;bsp/esp32_s3_lcd_ev_board;bsp/esp32_s3_korvo_2;bsp/esp-box-lite;bsp/esp32_lyrat;bsp/esp32_c3_lcdkit;bsp/esp-box-3;bsp/esp_bsp_generic;bsp/esp32_s3_korvo_1;bsp/esp32_p4_function_ev_board;bsp/m5stack_core_s3;bsp/m5dial; + bsp/esp32_azure_iot_kit;bsp/esp32_s2_kaluga_kit;bsp/esp_wrover_kit;bsp/esp-box;bsp/esp32_s3_usb_otg;bsp/esp32_s3_eye;bsp/esp32_s3_lcd_ev_board;bsp/esp32_s3_korvo_2;bsp/esp-box-lite;bsp/esp32_lyrat;bsp/esp32_c3_lcdkit;bsp/esp-box-3;bsp/esp_bsp_generic;bsp/esp32_s3_korvo_1;bsp/esp32_p4_function_ev_board;bsp/m5stack_core_s3;bsp/m5dial;bsp/m5stack_core_2; components/bh1750;components/ds18b20;components/es8311;components/es7210;components/fbm320;components/hts221;components/mag3110;components/mpu6050;components/esp_lvgl_port;components/icm42670; components/lcd_touch/esp_lcd_touch;components/lcd_touch/esp_lcd_touch_ft5x06;components/lcd_touch/esp_lcd_touch_gt911;components/lcd_touch/esp_lcd_touch_tt21100;components/lcd_touch/esp_lcd_touch_gt1151;components/lcd_touch/esp_lcd_touch_cst816s; components/lcd/esp_lcd_gc9a01;components/lcd/esp_lcd_ili9341;components/lcd/esp_lcd_ra8875;components/lcd_touch/esp_lcd_touch_stmpe610;components/lcd/esp_lcd_sh1107;components/lcd/esp_lcd_st7796;components/lcd/esp_lcd_gc9503;components/lcd/esp_lcd_ssd1681;components/lcd/esp_lcd_ili9881c; diff --git a/.github/workflows/upload_component_noglib.yml b/.github/workflows/upload_component_noglib.yml index c0beb8b9..9c09bcc9 100644 --- a/.github/workflows/upload_component_noglib.yml +++ b/.github/workflows/upload_component_noglib.yml @@ -16,7 +16,7 @@ jobs: - name: Upload noglib version of BSPs # TODO: Extend this part to all BSPs env: - BSPs: "bsp/esp_wrover_kit bsp/esp32_s3_eye bsp/esp32_p4_function_ev_board bsp/m5stack_core_s3 bsp/m5dial" + BSPs: "bsp/esp_wrover_kit bsp/esp32_s3_eye bsp/esp32_p4_function_ev_board bsp/m5stack_core_s3 bsp/m5stack_core_2 bsp/m5dial" run: | pip install idf-component-manager==1.* py-markdown-table --upgrade python .github/ci/bsp_noglib.py ${BSPs} diff --git a/README.md b/README.md index 744c91a3..b2bcb3a5 100644 --- a/README.md +++ b/README.md @@ -4,28 +4,29 @@ Board support packages for development boards using Espressif's SoCs, written in C. ## Supported boards -| Board name | SoC | Features | Photo | -|---|---|---|---| -| [ESP-WROVER-KIT](bsp/esp_wrover_kit) | ESP32 | LCD display, uSD card slot | | -| [ESP-BOX](bsp/esp-box) | ESP32-S3 | LCD display with touch, audio codec + power amplifier,
accelerometer and gyroscope | | -| [ESP-BOX-Lite](bsp/esp-box-lite) | ESP32-S3 | LCD display, audio codec + power amplifier | | -| [ESP32-Azure IoT Kit](bsp/esp32_azure_iot_kit) | ESP32 | OLED display, uSD card slot, accelerometer,
magnetometer, humidity, pressure, light
and temperature sensors | | -| [ESP32-S2-Kaluga Kit](bsp/esp32_s2_kaluga_kit) | ESP32-S2 | LCD display, audio codec + power amplifier,
smart LED and camera | | -| [ESP32-S3-USB-OTG](bsp/esp32_s3_usb_otg) | ESP32-S3 | LCD display, uSD card slot, USB-OTG | | -| [ESP32-S3-EYE](bsp/esp32_s3_eye) | ESP32-S3 | LCD display, camera, uSD card slot, microphone and accelerometer | | -| [ESP32-S3-LCD-EV-Board](bsp/esp32_s3_lcd_ev_board) | ESP32-S3 | LCD display with touch, audio codec + power amplifier | | -| [ESP32-S3-LCD-EV-Board-2](bsp/esp32_s3_lcd_ev_board) | ESP32-S3 | LCD display with touch, audio codec + power amplifier | | -| [ESP32-S3-Korvo-2](bsp/esp32_s3_korvo_2) | ESP32-S3 | LCD display, camera, uSD card slot, microphone, audio codec + power amplifier | | -| [ESP32-LyraT](bsp/esp32_lyrat) | ESP32 | uSD card slot, microphone, audio codec + power amplifier | | -| [ESP32-C3-LCDKit](bsp/esp32_c3_lcdkit) | ESP32-C3 | LCD display with encoder, IR, PDM audio| | -| [ESP-BOX-3](bsp/esp-box-3) | ESP32-S3 | LCD display with touch, audio codec + power amplifier,
accelerometer and gyroscope | | -| [ESP32-S3-KORVO-1](bsp/esp32_s3_korvo_1) | ESP32-S3 | uSD card slot, microphone, audio codec + power amplifier, RGB led strip | | -| [M5Stack CoreS3](bsp/m5stack_core_s3) | ESP32-S3 | LCD display with touch, uSD card slot, microphone, audio codec | | -| [M5Dial](bsp/m5dial) | ESP32-S3 | LCD display with touch and encoder | | +| Board name | SoC | Features | Photo | +| ---------------------------------------------------- | -------- | ----------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------- | +| [ESP-WROVER-KIT](bsp/esp_wrover_kit) | ESP32 | LCD display, uSD card slot | | +| [ESP-BOX](bsp/esp-box) | ESP32-S3 | LCD display with touch, audio codec + power amplifier,
accelerometer and gyroscope | | +| [ESP-BOX-Lite](bsp/esp-box-lite) | ESP32-S3 | LCD display, audio codec + power amplifier | | +| [ESP32-Azure IoT Kit](bsp/esp32_azure_iot_kit) | ESP32 | OLED display, uSD card slot, accelerometer,
magnetometer, humidity, pressure, light
and temperature sensors | | +| [ESP32-S2-Kaluga Kit](bsp/esp32_s2_kaluga_kit) | ESP32-S2 | LCD display, audio codec + power amplifier,
smart LED and camera | | +| [ESP32-S3-USB-OTG](bsp/esp32_s3_usb_otg) | ESP32-S3 | LCD display, uSD card slot, USB-OTG | | +| [ESP32-S3-EYE](bsp/esp32_s3_eye) | ESP32-S3 | LCD display, camera, uSD card slot, microphone and accelerometer | | +| [ESP32-S3-LCD-EV-Board](bsp/esp32_s3_lcd_ev_board) | ESP32-S3 | LCD display with touch, audio codec + power amplifier | | +| [ESP32-S3-LCD-EV-Board-2](bsp/esp32_s3_lcd_ev_board) | ESP32-S3 | LCD display with touch, audio codec + power amplifier | | +| [ESP32-S3-Korvo-2](bsp/esp32_s3_korvo_2) | ESP32-S3 | LCD display, camera, uSD card slot, microphone, audio codec + power amplifier | | +| [ESP32-LyraT](bsp/esp32_lyrat) | ESP32 | uSD card slot, microphone, audio codec + power amplifier | | +| [ESP32-C3-LCDKit](bsp/esp32_c3_lcdkit) | ESP32-C3 | LCD display with encoder, IR, PDM audio | | +| [ESP-BOX-3](bsp/esp-box-3) | ESP32-S3 | LCD display with touch, audio codec + power amplifier,
accelerometer and gyroscope | | +| [ESP32-S3-KORVO-1](bsp/esp32_s3_korvo_1) | ESP32-S3 | uSD card slot, microphone, audio codec + power amplifier, RGB led strip | | +| [M5Stack CoreS3](bsp/m5stack_core_s3) | ESP32-S3 | LCD display with touch, uSD card slot, microphone, audio codec | | +| [M5Stack Core2](bsp/m5stack_core_2) | ESP32 | LCD display with touch, uSD card slot | | +| [M5Dial](bsp/m5dial) | ESP32-S3 | LCD display with touch and encoder | | ## LCD displays and TOUCH | [LVGL port](components/esp_lvgl_port) | [LCD drivers](LCD.md) | -| :---: | :---: | +| :-----------------------------------: | :-------------------: | The BSP repository includes a lot of LCD and Touch driver components. The list of available and planned LCDs is [here](LCD.md). @@ -39,16 +40,16 @@ Moreover, LVGL port includes recommendations and tips for [increasing graphical Best way to start with ESP-BSP is trying one of the [examples](examples) on your board. Every example contains `README.md` with a list of supported boards. Here is the summary of the available examples: -| Example | Supported boards | Try with ESP Launchpad | -|---|---|---| -| [audio](examples/audio) | ESP32-S3-Korvo-2 | [Flash audio](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=audio) | -| [display](examples/display) | WROVER-KIT | [Flash display](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display) | -| [display_camera](examples/display_camera) | Kaluga-kit | [Flash display_camera](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_camera) | -| [display_audio_photo](examples/display_audio_photo) | ESP-BOX | [Flash display_audio_photo](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_audio_photo) | -| [display_rotation](examples/display_rotation) | ESP-BOX | [Flash display_rotation](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_rotation) | -| [display_lvgl_demos](examples/display_lvgl_demos) | ESP32-S3-LCD-EV-Board | [Flash display_lvgl_demos](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_lvgl_demos) | -| [display_sensors](examples/display_sensors) | Azure-IoT-kit | [Flash display_sensors](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_sensors) | -| [mqtt_example](examples/mqtt_example) | Azure-IoT-kit | - | +| Example | Supported boards | Try with ESP Launchpad | +| --------------------------------------------------- | --------------------- | -------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| [audio](examples/audio) | ESP32-S3-Korvo-2 | [Flash audio](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=audio) | +| [display](examples/display) | WROVER-KIT | [Flash display](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display) | +| [display_camera](examples/display_camera) | Kaluga-kit | [Flash display_camera](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_camera) | +| [display_audio_photo](examples/display_audio_photo) | ESP-BOX | [Flash display_audio_photo](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_audio_photo) | +| [display_rotation](examples/display_rotation) | ESP-BOX | [Flash display_rotation](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_rotation) | +| [display_lvgl_demos](examples/display_lvgl_demos) | ESP32-S3-LCD-EV-Board | [Flash display_lvgl_demos](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_lvgl_demos) | +| [display_sensors](examples/display_sensors) | Azure-IoT-kit | [Flash display_sensors](https://espressif.github.io/esp-launchpad/?flashConfigURL=https://espressif.github.io/esp-bsp/config.toml&app=display_sensors) | +| [mqtt_example](examples/mqtt_example) | Azure-IoT-kit | - | ### BSP headers and options * `bsp/name-of-the-bsp.h`: Main include file of the BSP with public API diff --git a/SquareLine/boards/v9/m5stack_core_2/image.png b/SquareLine/boards/v9/m5stack_core_2/image.png new file mode 100644 index 0000000000000000000000000000000000000000..735b347138189653f0f2f17309ea35e5e1098951 GIT binary patch literal 43600 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.0" + m5stack_core_2: ">=1.0.0" diff --git a/SquareLine/boards/v9/m5stack_core_2/manifest.json b/SquareLine/boards/v9/m5stack_core_2/manifest.json new file mode 100644 index 00000000..d382db56 --- /dev/null +++ b/SquareLine/boards/v9/m5stack_core_2/manifest.json @@ -0,0 +1,15 @@ +{ + "name": "M5Stack Core2", + "version": "1.0.0", + "mcu": "ESP32", + "screen_width": "320", + "screen_height": "240", + "screen_color_swap": true, + "supported_lvgl_version": "9.1.*", + "short_description": "M5Core2 is the second generation core device in the M5Stack development kit series, which further enhances the functions of the original generation of cores.The MCU is an ESP32 model D0WDQ6-V3 and has dual core Xtensa® 32-bit 240Mhz LX6 processors that can be controlled separately. Wi-Fi are supported as standard and it includes an on board 16MB Flash and 8MB PSRAM, USB TYPE-C interface for charging, downloading of programs and serial communication, a 2.0-inch integrated capacitive touch screen, and a built-in vibration motor.", + "long_description": "M5Core2 also features a built-in RTC module which can provide accurate timing. The power supply is managed by an AXP192 power management chip, which can effectively control the power consumption of the base and a built-in green LED power indicator helps to notify the user of battery level. The battery capacity has been upgraded to 390mAh, which can power the core for much longer than the previous model.The M5Core2 retains the TF-card(microSD) slot and speakers. However, in order to ensure higher quality sound output, the I2S digital audio interface power amplifier chip is used to effectively prevent signal distortion. There are independent power and reset buttons on the left side and bottom of the base.", + "placeholders": { + "__ESP_BOARD_INCLUDE__": "bsp/esp-bsp.h", + "__ESP_BOARD_I2C_INIT__": "/* Initialize I2C (for touch) */\n bsp_i2c_init();" + } +} \ No newline at end of file diff --git a/bsp/m5stack_core_2/CMakeLists.txt b/bsp/m5stack_core_2/CMakeLists.txt new file mode 100644 index 00000000..674756f5 --- /dev/null +++ b/bsp/m5stack_core_2/CMakeLists.txt @@ -0,0 +1,7 @@ +idf_component_register( + SRCS "m5stack_core_2.c" "m5stack_core_2_idf5.c" + INCLUDE_DIRS "include" + PRIV_INCLUDE_DIRS "priv_include" + REQUIRES driver spiffs + PRIV_REQUIRES fatfs esp_lcd +) diff --git a/bsp/m5stack_core_2/Kconfig b/bsp/m5stack_core_2/Kconfig new file mode 100644 index 00000000..e539bcfd --- /dev/null +++ b/bsp/m5stack_core_2/Kconfig @@ -0,0 +1,102 @@ +menu "Board Support Package" + + config BSP_ERROR_CHECK + bool "Enable error check in BSP" + default y + help + Error check assert the application before returning the error code. + + menu "PMU" + choice PMU_VERSION + prompt "PMU Version" + default BSP_PMU_AXP192 + help + Core2 v1.0 PMU version is AXP192, Core2 v1.1 PMU version is AXP2101, you can determine the version of Core2 by the backplane sticker or PCB. + + config BSP_PMU_AXP192 + bool "AXP192" + + config BSP_PMU_AXP2101 + bool "AXP2101" + + endchoice + endmenu + + menu "I2C" + config BSP_I2C_NUM + int "I2C peripheral index" + default 1 + range 0 1 + help + ESP32 has two I2C peripherals, pick the one you want to use. + + config BSP_I2C_FAST_MODE + bool "Enable I2C fast mode" + default y + help + I2C has two speed modes: normal (100kHz) and fast (400kHz). + + config BSP_I2C_CLK_SPEED_HZ + int + default 400000 if BSP_I2C_FAST_MODE + default 100000 + endmenu + + menu "SPIFFS - Virtual File System" + config BSP_SPIFFS_FORMAT_ON_MOUNT_FAIL + bool "Format SPIFFS if mounting fails" + default n + help + Format SPIFFS if it fails to mount the filesystem. + + config BSP_SPIFFS_MOUNT_POINT + string "SPIFFS mount point" + default "/spiffs" + help + Mount point of SPIFFS in the Virtual File System. + + config BSP_SPIFFS_PARTITION_LABEL + string "Partition label of SPIFFS" + default "storage" + help + Partition label which stores SPIFFS. + + config BSP_SPIFFS_MAX_FILES + int "Max files supported for SPIFFS VFS" + default 5 + help + Supported max files for SPIFFS in the Virtual File System. + endmenu + + menu "SD card - Virtual File System" + config BSP_SD_FORMAT_ON_MOUNT_FAIL + bool "Format SD card if mounting fails" + default n + help + The SDMMC host will format (FAT) the SD card if it fails to mount the filesystem. + + config BSP_SD_MOUNT_POINT + string "SD card mount point" + default "/sdcard" + help + Mount point of the SD card in the Virtual File System + + endmenu + + menu "Display" + config BSP_DISPLAY_BRIGHTNESS_LEDC_CH + int "LEDC channel index" + default 1 + range 0 7 + help + LEDC channel is used to generate PWM signal that controls display brightness. + Set LEDC index that should be used. + endmenu + + config BSP_I2S_NUM + int "I2S peripheral index" + default 0 + range 0 1 + help + ESP32 has two I2S peripherals, pick the one you want to use. +endmenu diff --git a/bsp/m5stack_core_2/LICENSE b/bsp/m5stack_core_2/LICENSE new file mode 100644 index 00000000..d6456956 --- /dev/null +++ b/bsp/m5stack_core_2/LICENSE @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/bsp/m5stack_core_2/README.md b/bsp/m5stack_core_2/README.md new file mode 100644 index 00000000..89967d1f --- /dev/null +++ b/bsp/m5stack_core_2/README.md @@ -0,0 +1,47 @@ +# BSP: M5Stack Core2 + +> [!WARNING] +> The SD card is not working simultaneously with the LCD screen. We are working on a fix. + +[![Component Registry](https://components.espressif.com/components/espressif/m5stack_core_2/badge.svg)](https://components.espressif.com/components/espressif/m5stack_core_2) + +* [Hardware Reference](https://docs.m5stack.com/en/core/Core2) + +basicgray + +M5Core2 is the second generation core device in the M5Stack development kit series, which further enhances the functions of the original generation of cores. It is a powerful and user-friendly development board with the following features: + +- **MCU**: Equipped with an ESP32-D0WDQ6-V3, featuring dual core Xtensa® 32-bit 240Mhz LX6 processors that can be controlled separately. +- **Memory**: Onboard 16MB Flash and 8MB PSRAM. +- **Interface**: USB TYPE-C interface for charging, program downloading, and serial communication. +- **Display**: 2.0-inch integrated capacitive touch screen with three programmable capacitive buttons on the front. +- **Power Management:** Managed by an AXP192 power management chip (upgraded to AXP2101 in Core2 V1.1), which effectively controls power consumption. It includes a built-in green LED power indicator for battery level notification. The battery capacity is 390mAh, providing longer power duration than the previous model. +- **Audio**: I2S digital audio interface power amplifier chip to prevent signal distortion, along with a built-in speaker. +- **Expansion**: Retains a TF-card (microSD) slot, and an expansion board on the back with a 6-axis IMU sensor and a microphone. +- **RTC Module**: Built-in RTC module for accurate timing, with a dedicated battery for RTC power supply in Core2 V1.1. +Additional Features: Built-in vibration motor for haptic feedback, independent power, and reset buttons on the left side and bottom of the base. + +#### Core2 V1.1 Enhancements +Core2 V1.1 is an iterative version of Core2 with the following upgrades and additional features: + +- **Power Management**: Uses the AXP2101 power management chip for enhanced power control. +- **Indicators**: Built-in blue power indicator light for specific functions or status indications. +- RTC Battery: Dedicated battery for RTC power supply for accurate timing. +- **User Interaction**: Enhanced touch screen experience with programmable virtual buttons for diverse human-machine interaction. + + + + +### Capabilities and dependencies +| Capability | Available | Component |Version| +|-------------|------------------|------------------------------------------------------------------------------------------------------------|-------| +| DISPLAY |:heavy_check_mark:| [espressif/esp_lcd_ili9341](https://components.espressif.com/components/espressif/esp_lcd_ili9341) | ^1 | +| LVGL_PORT |:heavy_check_mark:| [espressif/esp_lvgl_port](https://components.espressif.com/components/espressif/esp_lvgl_port) | ^2 | +| TOUCH |:heavy_check_mark:|[espressif/esp_lcd_touch_ft5x06](https://components.espressif.com/components/espressif/esp_lcd_touch_ft5x06)| ^1 | +| BUTTONS | :x: | | | +| AUDIO |:heavy_check_mark:| [espressif/esp_codec_dev](https://components.espressif.com/components/espressif/esp_codec_dev) | ~1.1 | +|AUDIO_SPEAKER|:heavy_check_mark:| | | +| AUDIO_MIC | :x: | | | +| SDCARD |:heavy_check_mark:| idf | >=5.0 | +| IMU | :x: | | | + diff --git a/bsp/m5stack_core_2/idf_component.yml b/bsp/m5stack_core_2/idf_component.yml new file mode 100644 index 00000000..353132c8 --- /dev/null +++ b/bsp/m5stack_core_2/idf_component.yml @@ -0,0 +1,23 @@ +version: "1.0.0" +description: Board Support Package (BSP) for M5Stack Core2 +url: https://github.com/espressif/esp-bsp/tree/master/bsp/m5stack_core_2 + +targets: + - esp32 + +tags: + - bsp + +dependencies: + idf: ">=5.0" + esp_lcd_ili9341: "^1" + esp_lcd_touch_ft5x06: "^1" + + espressif/esp_lvgl_port: + version: "^2" + public: true + override_path: "../../components/esp_lvgl_port" + + esp_codec_dev: + version: "~1.1" + public: true diff --git a/bsp/m5stack_core_2/include/bsp/config.h b/bsp/m5stack_core_2/include/bsp/config.h new file mode 100644 index 00000000..f3cb5ebe --- /dev/null +++ b/bsp/m5stack_core_2/include/bsp/config.h @@ -0,0 +1,16 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +/************************************************************************************************** + * BSP configuration + **************************************************************************************************/ +// By default, this BSP is shipped with LVGL graphical library. Enabling this option will exclude it. +// If you want to use BSP without LVGL, select BSP version with 'noglib' suffix. +#if !defined(BSP_CONFIG_NO_GRAPHIC_LIB) // Check if the symbol is not coming from compiler definitions (-D...) +#define BSP_CONFIG_NO_GRAPHIC_LIB (0) +#endif diff --git a/bsp/m5stack_core_2/include/bsp/display.h b/bsp/m5stack_core_2/include/bsp/display.h new file mode 100644 index 00000000..1963bbd6 --- /dev/null +++ b/bsp/m5stack_core_2/include/bsp/display.h @@ -0,0 +1,121 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +/** + * @file + * @brief BSP LCD + * + * This file offers API for basic LCD control. + * It is useful for users who want to use the LCD without the default Graphical Library LVGL. + * + * For standard LCD initialization with LVGL graphical library, you can call all-in-one function bsp_display_start(). + */ + +#pragma once +#include "esp_lcd_types.h" + +/* LCD color formats */ +#define ESP_LCD_COLOR_FORMAT_RGB565 (1) +#define ESP_LCD_COLOR_FORMAT_RGB888 (2) + +/* LCD display color format */ +#define BSP_LCD_COLOR_FORMAT (ESP_LCD_COLOR_FORMAT_RGB565) +/* LCD display color bytes endianess */ +#define BSP_LCD_BIGENDIAN (1) +/* LCD display color bits */ +#define BSP_LCD_BITS_PER_PIXEL (16) +/* LCD display color space */ +#define BSP_LCD_COLOR_SPACE (ESP_LCD_COLOR_SPACE_BGR) +/* LCD definition */ +#define BSP_LCD_H_RES (320) +#define BSP_LCD_V_RES (240) + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief BSP display configuration structure + * + */ +typedef struct { + int max_transfer_sz; /*!< Maximum transfer size, in bytes. */ +} bsp_display_config_t; + +/** + * @brief Create new display panel + * + * For maximum flexibility, this function performs only reset and initialization of the display. + * You must turn on the display explicitly by calling esp_lcd_panel_disp_on_off(). + * The display's backlight is not turned on either. You can use bsp_display_backlight_on/off(), + * bsp_display_brightness_set() (on supported boards) or implement your own backlight control. + * + * If you want to free resources allocated by this function, you can use esp_lcd API, ie.: + * + * \code{.c} + * esp_lcd_panel_del(panel); + * esp_lcd_panel_io_del(io); + * spi_bus_free(spi_num_from_configuration); + * \endcode + * + * @param[in] config display configuration + * @param[out] ret_panel esp_lcd panel handle + * @param[out] ret_io esp_lcd IO handle + * @return + * - ESP_OK On success + * - Else esp_lcd failure + */ +esp_err_t bsp_display_new(const bsp_display_config_t *config, esp_lcd_panel_handle_t *ret_panel, esp_lcd_panel_io_handle_t *ret_io); + +/** + * @brief Initialize display's brightness + * + * Brightness is controlled with AXP2101 via I2C. + * + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG Parameter error + */ +esp_err_t bsp_display_brightness_init(void); + +/** + * @brief Set display's brightness + * + * Brightness is controlled with AXP2101 via I2C. + * Backlight must be already initialized by calling bsp_display_brightness_init() or bsp_display_start() + * + * @param[in] brightness_percent Brightness in [%] + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG Parameter error + */ +esp_err_t bsp_display_brightness_set(int brightness_percent); + +/** + * @brief Turn on display backlight + * + * Backlight must be already initialized by calling bsp_display_brightness_init() or bsp_display_start() + * + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG Parameter error + */ +esp_err_t bsp_display_backlight_on(void); + +/** + * @brief Turn off display backlight + * + * Backlight must be already initialized by calling bsp_display_brightness_init() or bsp_display_start() + * + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG Parameter error + */ +esp_err_t bsp_display_backlight_off(void); + +#ifdef __cplusplus +} +#endif diff --git a/bsp/m5stack_core_2/include/bsp/esp-bsp.h b/bsp/m5stack_core_2/include/bsp/esp-bsp.h new file mode 100644 index 00000000..642f9e17 --- /dev/null +++ b/bsp/m5stack_core_2/include/bsp/esp-bsp.h @@ -0,0 +1,8 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once +#include "bsp/m5stack_core_2.h" diff --git a/bsp/m5stack_core_2/include/bsp/m5stack_core_2.h b/bsp/m5stack_core_2/include/bsp/m5stack_core_2.h new file mode 100644 index 00000000..ac827867 --- /dev/null +++ b/bsp/m5stack_core_2/include/bsp/m5stack_core_2.h @@ -0,0 +1,361 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +/** + * @file + * @brief ESP BSP: M5Stack Core2 + */ + +#pragma once + +#include "sdkconfig.h" +#include "driver/gpio.h" +#include "driver/i2c.h" +#include "driver/sdmmc_host.h" +#include "esp_codec_dev.h" +#include "bsp/config.h" +#include "bsp/display.h" + +#include "driver/i2s_std.h" + +#if (BSP_CONFIG_NO_GRAPHIC_LIB == 0) +#include "lvgl.h" +#include "esp_lvgl_port.h" +#endif // BSP_CONFIG_NO_GRAPHIC_LIB == 0 + +/************************************************************************************************** + * BSP Capabilities + **************************************************************************************************/ + +#define BSP_CAPS_DISPLAY 1 +#define BSP_CAPS_TOUCH 1 +#define BSP_CAPS_BUTTONS 0 +#define BSP_CAPS_AUDIO 1 +#define BSP_CAPS_AUDIO_SPEAKER 1 +#define BSP_CAPS_AUDIO_MIC 0 +#define BSP_CAPS_SDCARD 1 +#define BSP_CAPS_IMU 0 + +/************************************************************************************************** + * M5Stack-Core-2 pinout + **************************************************************************************************/ +/* I2C */ +#define BSP_I2C_SCL (GPIO_NUM_22) +#define BSP_I2C_SDA (GPIO_NUM_21) +/* Audio */ +#define BSP_I2S_SCLK (GPIO_NUM_12) //I2S_BCK +#define BSP_I2S_MCLK (GPIO_NUM_NC) +#define BSP_I2S_LCLK (GPIO_NUM_0) //I2S_WCK +#define BSP_I2S_DOUT (GPIO_NUM_2) +#define BSP_I2S_DSIN (GPIO_NUM_34) +#define BSP_POWER_AMP_IO (GPIO_NUM_NC) +#define BSP_MUTE_STATUS (GPIO_NUM_NC) +/* Display */ +#define BSP_LCD_MOSI (GPIO_NUM_23) +#define BSP_LCD_MISO (GPIO_NUM_38) +#define BSP_LCD_PCLK (GPIO_NUM_18) +#define BSP_LCD_CS (GPIO_NUM_5) +#define BSP_LCD_DC (GPIO_NUM_15) +#define BSP_LCD_RST (GPIO_NUM_NC) +#define BSP_LCD_BACKLIGHT (GPIO_NUM_NC) +#define BSP_LCD_TOUCH_INT (GPIO_NUM_39) + +/* SD card */ +#define BSP_SD_MOSI (GPIO_NUM_23) +#define BSP_SD_MISO (GPIO_NUM_38) +#define BSP_SD_SCK (GPIO_NUM_18) +#define BSP_SD_CS (GPIO_NUM_4) + + +#ifdef __cplusplus +extern "C" { +#endif + +/************************************************************************************************** + * + * I2S audio interface + * + * For speaker initialization use bsp_audio_codec_speaker_init() which is inside initialize I2S with bsp_audio_init(). + * After speaker initialization, use functions from esp_codec_dev for play/record audio. + * Example audio play: + * \code{.c} + * esp_codec_dev_set_out_vol(spk_codec_dev, DEFAULT_VOLUME); + * esp_codec_dev_open(spk_codec_dev, &fs); + * esp_codec_dev_write(spk_codec_dev, wav_bytes, bytes_read_from_spiffs); + * esp_codec_dev_close(spk_codec_dev); + * \endcode + **************************************************************************************************/ + +/** + * @brief Init audio + * + * @note There is no deinit audio function. Users can free audio resources by calling i2s_del_channel() + * @warning The type of i2s_config param is depending on IDF version. + * @param[in] i2s_config I2S configuration. Pass NULL to use default values (Mono, duplex, 16bit, 22050 Hz) + * @return + * - ESP_OK On success + * - ESP_ERR_NOT_SUPPORTED The communication mode is not supported on the current chip + * - ESP_ERR_INVALID_ARG NULL pointer or invalid configuration + * - ESP_ERR_NOT_FOUND No available I2S channel found + * - ESP_ERR_NO_MEM No memory for storing the channel information + * - ESP_ERR_INVALID_STATE This channel has not initialized or already started + */ +#if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 0, 0) +esp_err_t bsp_audio_init(const i2s_config_t *i2s_config); +#else +esp_err_t bsp_audio_init(const i2s_std_config_t *i2s_config); +#endif + +/** + * @brief Get codec I2S interface (initialized in bsp_audio_init) + * + * @return + * - Pointer to codec I2S interface handle or NULL when error occured + */ +const audio_codec_data_if_t *bsp_audio_get_codec_itf(void); + +/** + * @brief Initialize speaker codec device + * + * @return Pointer to codec device handle or NULL when error occured + */ +esp_codec_dev_handle_t bsp_audio_codec_speaker_init(void); + +/************************************************************************************************** + * + * I2C interface + * + * There are multiple devices connected to I2C peripheral: + * - AXP192 / AXP2101 PMU + * - LCD Touch controller + **************************************************************************************************/ +#define BSP_I2C_NUM CONFIG_BSP_I2C_NUM + +/** + * @brief Init I2C driver + * + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG I2C parameter error + * - ESP_FAIL I2C driver installation error + * + */ +esp_err_t bsp_i2c_init(void); + +/** + * @brief Deinit I2C driver and free its resources + * + * @return + * - ESP_OK On success + * - ESP_ERR_INVALID_ARG I2C parameter error + * + */ +esp_err_t bsp_i2c_deinit(void); + +/************************************************************************************************** + * + * SPIFFS + * + * After mounting the SPIFFS, it can be accessed with stdio functions ie.: + * \code{.c} + * FILE* f = fopen(BSP_SPIFFS_MOUNT_POINT"/hello.txt", "w"); + * fprintf(f, "Hello World!\n"); + * fclose(f); + * \endcode + **************************************************************************************************/ +#define BSP_SPIFFS_MOUNT_POINT CONFIG_BSP_SPIFFS_MOUNT_POINT + +/** + * @brief Mount SPIFFS to virtual file system + * + * @return + * - ESP_OK on success + * - ESP_ERR_INVALID_STATE if esp_vfs_spiffs_register was already called + * - ESP_ERR_NO_MEM if memory can not be allocated + * - ESP_FAIL if partition can not be mounted + * - other error codes + */ +esp_err_t bsp_spiffs_mount(void); + +/** + * @brief Unmount SPIFFS from virtual file system + * + * @return + * - ESP_OK on success + * - ESP_ERR_NOT_FOUND if the partition table does not contain SPIFFS partition with given label + * - ESP_ERR_INVALID_STATE if esp_vfs_spiffs_unregister was already called + * - ESP_ERR_NO_MEM if memory can not be allocated + * - ESP_FAIL if partition can not be mounted + * - other error codes + */ +esp_err_t bsp_spiffs_unmount(void); + +/************************************************************************************************** + * + * SD card + * + * After mounting the SD card, it can be accessed with stdio functions ie.: + * \code{.c} + * FILE* f = fopen(BSP_SD_MOUNT_POINT"/hello.txt", "w"); + * fprintf(f, "Hello %s!\n", bsp_sdcard->cid.name); + * fclose(f); + * \endcode + * + * @attention IO2 is also routed to RGB LED and push button + **************************************************************************************************/ +#define BSP_SD_MOUNT_POINT CONFIG_BSP_SD_MOUNT_POINT +extern sdmmc_card_t *bsp_sdcard; + +/** + * @brief Mount microSD card to virtual file system + * + * @return + * - ESP_OK on success + * - ESP_ERR_INVALID_STATE if esp_vfs_fat_sdmmc_mount was already called + * - ESP_ERR_NO_MEM if memory can not be allocated + * - ESP_FAIL if partition can not be mounted + * - other error codes from SDMMC or SPI drivers, SDMMC protocol, or FATFS drivers + */ +esp_err_t bsp_sdcard_mount(void); + +/** + * @brief Unmount micorSD card from virtual file system + * + * @return + * - ESP_OK on success + * - ESP_ERR_NOT_FOUND if the partition table does not contain FATFS partition with given label + * - ESP_ERR_INVALID_STATE if esp_vfs_fat_spiflash_mount was already called + * - ESP_ERR_NO_MEM if memory can not be allocated + * - ESP_FAIL if partition can not be mounted + * - other error codes from wear levelling library, SPI flash driver, or FATFS drivers + */ +esp_err_t bsp_sdcard_unmount(void); + +/************************************************************************************************** + * + * LCD interface + * + * M5Stack-Core-2 is shipped with 2.0inch ILI9341C display controller. + * It features 16-bit colors, 320x240 resolution and capacitive touch controller. + * + * LVGL is used as graphics library. LVGL is NOT thread safe, therefore the user must take LVGL mutex + * by calling bsp_display_lock() before calling and LVGL API (lv_...) and then give the mutex with + * bsp_display_unlock(). + * + * Display's backlight must be enabled explicitly by calling bsp_display_backlight_on() + **************************************************************************************************/ +#define BSP_LCD_PIXEL_CLOCK_HZ (40 * 1000 * 1000) +#define BSP_LCD_SPI_NUM (SPI2_HOST) + + +#if (BSP_CONFIG_NO_GRAPHIC_LIB == 0) +#define BSP_LCD_DRAW_BUFF_SIZE (BSP_LCD_H_RES * 50) +#define BSP_LCD_DRAW_BUFF_DOUBLE (1) + +/** + * @brief BSP display configuration structure + */ +typedef struct { + lvgl_port_cfg_t lvgl_port_cfg; /*!< LVGL port configuration */ + uint32_t buffer_size; /*!< Size of the buffer for the screen in pixels */ + bool double_buffer; /*!< True, if should be allocated two buffers */ + struct { + unsigned int buff_dma: 1; /*!< Allocated LVGL buffer will be DMA capable */ + unsigned int buff_spiram: 1; /*!< Allocated LVGL buffer will be in PSRAM */ + } flags; +} bsp_display_cfg_t; +/** + * @brief Initialize display + * + * This function initializes SPI, display controller and starts LVGL handling task. + * LCD backlight must be enabled separately by calling bsp_display_brightness_set() + * + * @return Pointer to LVGL display or NULL when error occured + */ +lv_display_t *bsp_display_start(void); + +/** + * @brief Initialize display + * + * This function initializes SPI, display controller and starts LVGL handling task. + * LCD backlight must be enabled separately by calling bsp_display_brightness_set() + * + * @param cfg display configuration + * + * @return Pointer to LVGL display or NULL when error occured + */ +lv_display_t *bsp_display_start_with_config(const bsp_display_cfg_t *cfg); + +/** + * @brief Get pointer to input device (touch, buttons, ...) + * + * @note The LVGL input device is initialized in bsp_display_start() function. + * + * @return Pointer to LVGL input device or NULL when not initialized + */ +lv_indev_t *bsp_display_get_input_dev(void); + +/** + * @brief Take LVGL mutex + * + * @param timeout_ms Timeout in [ms]. 0 will block indefinitely. + * @return true Mutex was taken + * @return false Mutex was NOT taken + */ +bool bsp_display_lock(uint32_t timeout_ms); + +/** + * @brief Give LVGL mutex + * + */ +void bsp_display_unlock(void); + +/** + * @brief Rotate screen + * + * Display must be already initialized by calling bsp_display_start() + * + * @param[in] disp Pointer to LVGL display + * @param[in] rotation Angle of the display rotation + */ +void bsp_display_rotate(lv_display_t *disp, lv_display_rotation_t rotation); +#endif // BSP_CONFIG_NO_GRAPHIC_LIB == 0 + +/************************************************************************************************** + * + * BSP Features + * + * This module provides an interface to enable and disable various features on the M5Stack-Core-2 board. + * Supported features include the LCD display, touch screen, SD card, and speaker. + * + * The bsp_feature_enable function is used to enable or disable a specified feature. + * It communicates with the Power Management Unit (PMU) via the I2C interface to control the power of each feature. + * + * Supported PMU models include AXP2101 and AXP192. + * Depending on the configuration, the function selects the appropriate PMU registers and values to enable or disable the feature. + * + * Parameters: + * - feature: The feature to enable or disable, of type bsp_feature_t enum. + * - enable: A boolean value, true to enable the feature, false to disable it. + * + * Return value: + * - esp_err_t: Error code indicating the result of the operation. + * + **************************************************************************************************/ +typedef enum { + BSP_FEATURE_LCD, + BSP_FEATURE_TOUCH, + BSP_FEATURE_SD, + BSP_FEATURE_SPEAKER, + BSP_FEATURE_BATTERY, + BSP_FEATURE_VIBRATION +} bsp_feature_t; + +esp_err_t bsp_feature_enable(bsp_feature_t feature, bool enable); + +#ifdef __cplusplus +} +#endif diff --git a/bsp/m5stack_core_2/include/bsp/touch.h b/bsp/m5stack_core_2/include/bsp/touch.h new file mode 100644 index 00000000..095e8ce1 --- /dev/null +++ b/bsp/m5stack_core_2/include/bsp/touch.h @@ -0,0 +1,51 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +/** + * @file + * @brief BSP Touchscreen + * + * This file offers API for basic touchscreen initialization. + * It is useful for users who want to use the touchscreen without the default Graphical Library LVGL. + * + * For standard LCD initialization with LVGL graphical library, you can call all-in-one function bsp_display_start(). + */ + +#pragma once +#include "esp_lcd_touch.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief BSP touch configuration structure + * + */ +typedef struct { + void *dummy; /*!< Prepared for future use. */ +} bsp_touch_config_t; + +/** + * @brief Create new touchscreen + * + * If you want to free resources allocated by this function, you can use esp_lcd_touch API, ie.: + * + * \code{.c} + * esp_lcd_touch_del(tp); + * \endcode + * + * @param[in] config touch configuration + * @param[out] ret_touch esp_lcd_touch touchscreen handle + * @return + * - ESP_OK On success + * - Else esp_lcd_touch failure + */ +esp_err_t bsp_touch_new(const bsp_touch_config_t *config, esp_lcd_touch_handle_t *ret_touch); + +#ifdef __cplusplus +} +#endif diff --git a/bsp/m5stack_core_2/m5stack_core_2.c b/bsp/m5stack_core_2/m5stack_core_2.c new file mode 100644 index 00000000..fd7f8c6b --- /dev/null +++ b/bsp/m5stack_core_2/m5stack_core_2.c @@ -0,0 +1,603 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "driver/gpio.h" +#include "driver/spi_master.h" +#include "esp_err.h" +#include "esp_log.h" +#include "esp_check.h" +#include "esp_spiffs.h" +#include "esp_lcd_panel_io.h" +#include "esp_lcd_panel_vendor.h" +#include "esp_lcd_panel_ops.h" +#include "esp_vfs_fat.h" +#include "driver/sdmmc_host.h" +#include "driver/sdspi_host.h" +#include "driver/i2c.h" + +#include "bsp/m5stack_core_2.h" +#include "bsp/display.h" +#include "bsp/touch.h" +#include "esp_lcd_ili9341.h" +#include "esp_lcd_touch_ft5x06.h" +#include "bsp_err_check.h" +#include "esp_codec_dev_defaults.h" + +static const char *TAG = "M5Stack"; + +#define BSP_AXP192_ADDR 0x34 +#define BSP_AXP2101_ADDR 0x34 + +#if (BSP_CONFIG_NO_GRAPHIC_LIB == 0) +static lv_display_t *disp; +static lv_indev_t *disp_indev = NULL; +#endif // (BSP_CONFIG_NO_GRAPHIC_LIB == 0) +static esp_lcd_touch_handle_t tp; // LCD touch handle +sdmmc_card_t *bsp_sdcard = NULL; // Global SD card handler +static bool i2c_initialized = false; +static bool spi_initialized = false; + +esp_err_t bsp_i2c_init(void) +{ + /* I2C was initialized before */ + if (i2c_initialized) { + return ESP_OK; + } + + const i2c_config_t i2c_conf = {.mode = I2C_MODE_MASTER, + .sda_io_num = BSP_I2C_SDA, + .sda_pullup_en = GPIO_PULLUP_DISABLE, + .scl_io_num = BSP_I2C_SCL, + .scl_pullup_en = GPIO_PULLUP_DISABLE, + .master.clk_speed = CONFIG_BSP_I2C_CLK_SPEED_HZ + }; + BSP_ERROR_CHECK_RETURN_ERR(i2c_param_config(BSP_I2C_NUM, &i2c_conf)); + BSP_ERROR_CHECK_RETURN_ERR(i2c_driver_install(BSP_I2C_NUM, i2c_conf.mode, 0, 0, 0)); + + i2c_initialized = true; + + return ESP_OK; +} + +esp_err_t bsp_i2c_deinit(void) +{ + BSP_ERROR_CHECK_RETURN_ERR(i2c_driver_delete(BSP_I2C_NUM)); + i2c_initialized = false; + return ESP_OK; +} + +uint8_t read8bit(uint8_t sub_addr) +{ + // Read register data + uint8_t reg_data[1] = {0}; + esp_err_t err = i2c_master_write_read_device(BSP_I2C_NUM, BSP_AXP192_ADDR, &sub_addr, 1, reg_data, sizeof(reg_data), + 1000 / portTICK_PERIOD_MS); + if (err != ESP_OK) { + ESP_LOGE(TAG, "Failed to write & read register address: %s", esp_err_to_name(err)); + } + ESP_LOGD(TAG, "AXP192 register %x: 0x%x", sub_addr, reg_data[0]); + + return reg_data[0]; +} + +esp_err_t bsp_feature_enable(bsp_feature_t feature, bool enable) +{ + esp_err_t err = ESP_OK; + + /* Initilize I2C */ + BSP_ERROR_CHECK_RETURN_ERR(bsp_i2c_init()); + + switch (feature) { + case BSP_FEATURE_LCD: // ldo2 + case BSP_FEATURE_TOUCH: + case BSP_FEATURE_SD: +#if defined(CONFIG_BSP_PMU_AXP2101) + /* AXP ALDO4 voltage / SD Card / Touch Pad / 3V3 */ + const uint8_t aldo4_value = enable ? 0x1C : 0x00; + const uint8_t feature_ctr[] = {0x95, aldo4_value}; // axp: lcd logic and sdcard voltage preset to 3.3v + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, feature_ctr, sizeof(feature_ctr), + 1000 / portTICK_PERIOD_MS); + +#elif defined(CONFIG_BSP_PMU_AXP192) + const uint8_t pmu_value = enable ? (read8bit(0x28) & 0x0f) | 0xf0 : (read8bit(0x28) & 0x0f); + const uint8_t pmu_ldo2[] = { + 0x28, pmu_value + }; + + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_ldo2, sizeof(pmu_ldo2), + 1000 / portTICK_PERIOD_MS); +#endif + break; + case BSP_FEATURE_SPEAKER: +#if defined(CONFIG_BSP_PMU_AXP2101) + /* AXP ALDO3 voltage / Codec+Mic / 3V3 */ + const uint8_t aldo3_value = enable ? 0x1C : 0x00; + const uint8_t spk_ctr[] = {0x94, aldo3_value}; // axp: lcd logic and sdcard voltage preset to 3.3v + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, spk_ctr, sizeof(spk_ctr), + 1000 / portTICK_PERIOD_MS); +#elif defined(CONFIG_BSP_PMU_AXP192) + const uint8_t led_gpio_value = enable ? (read8bit(0x94) | 0x04) | 0xf0 : (read8bit(0x94) & ~0x04); + const uint8_t led_gpio_set[] = {0x94, led_gpio_value}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, led_gpio_set, + sizeof(led_gpio_set), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#endif + break; + case BSP_FEATURE_BATTERY: +#if defined(CONFIG_BSP_PMU_AXP2101) + // Battery detection enabled. + const uint8_t axp_bat_val = enable ? 0x01 : 0x00; + const uint8_t axp_bat_ctr[] = {0x68, axp_bat_val}; + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, axp_bat_ctr, sizeof(axp_bat_ctr), + 1000 / portTICK_PERIOD_MS); +#endif + break; + case BSP_FEATURE_VIBRATION: +#if defined(CONFIG_BSP_PMU_AXP2101) + const uint8_t dldo1_vol[] = {0x99, 0x1C}; // AXP DLDO1 Voltage set + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, dldo1_vol, sizeof(dldo1_vol), + 1000 / portTICK_PERIOD_MS); + const uint8_t dldo1_en_val = enable ? (read8bit(0x90) | 0x80) : (read8bit(0x90) & ~0x80); + const uint8_t dldo1_en[] = {0x90, dldo1_en_val}; // AXP DLDO1 Enable + err |= i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, dldo1_en, sizeof(dldo1_en), + 1000 / portTICK_PERIOD_MS); +#elif defined(CONFIG_BSP_PMU_AXP192) + const uint8_t pmu_ldo3[] = {0x28, (read8bit(0x28) & 0xf0) | 0x0f}; // Vibrator power voltage preset + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_ldo3, sizeof(pmu_ldo3), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_ldo3_en_val = enable ? read8bit(0x12) | 0x08 : read8bit(0x12) & ~0x08; + const uint8_t pmu_ldo3_en[] = {0x12, pmu_ldo3_en_val}; // ldo3 enable + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_ldo3_en, sizeof(pmu_ldo3_en), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#endif + break; + } + return err; +} + +static esp_err_t bsp_spi_init(uint32_t max_transfer_sz) +{ + /* SPI was initialized before */ + if (spi_initialized) { + return ESP_OK; + } + + ESP_LOGD(TAG, "Initialize SPI bus"); + const spi_bus_config_t buscfg = { + .sclk_io_num = BSP_LCD_PCLK, + .mosi_io_num = BSP_LCD_MOSI, + .miso_io_num = BSP_LCD_MISO, + .quadwp_io_num = GPIO_NUM_NC, + .quadhd_io_num = GPIO_NUM_NC, + .max_transfer_sz = max_transfer_sz, + }; + ESP_RETURN_ON_ERROR(spi_bus_initialize(BSP_LCD_SPI_NUM, &buscfg, SPI_DMA_CH_AUTO), TAG, "SPI init failed"); + + spi_initialized = true; + + return ESP_OK; +} + +esp_err_t bsp_spiffs_mount(void) +{ + esp_vfs_spiffs_conf_t conf = { + .base_path = CONFIG_BSP_SPIFFS_MOUNT_POINT, + .partition_label = CONFIG_BSP_SPIFFS_PARTITION_LABEL, + .max_files = CONFIG_BSP_SPIFFS_MAX_FILES, +#ifdef CONFIG_BSP_SPIFFS_FORMAT_ON_MOUNT_FAIL + .format_if_mount_failed = true, +#else + .format_if_mount_failed = false, +#endif + }; + + esp_err_t ret_val = esp_vfs_spiffs_register(&conf); + + BSP_ERROR_CHECK_RETURN_ERR(ret_val); + + size_t total = 0, used = 0; + ret_val = esp_spiffs_info(conf.partition_label, &total, &used); + if (ret_val != ESP_OK) { + ESP_LOGE(TAG, "Failed to get SPIFFS partition information (%s)", esp_err_to_name(ret_val)); + } else { + ESP_LOGI(TAG, "Partition size: total: %d, used: %d", total, used); + } + + return ret_val; +} + +esp_err_t bsp_spiffs_unmount(void) +{ + return esp_vfs_spiffs_unregister(CONFIG_BSP_SPIFFS_PARTITION_LABEL); +} + +esp_err_t bsp_sdcard_mount(void) +{ + BSP_ERROR_CHECK_RETURN_ERR(bsp_feature_enable(BSP_FEATURE_SD, true)); + + const esp_vfs_fat_sdmmc_mount_config_t mount_config = { +#ifdef CONFIG_BSP_SD_FORMAT_ON_MOUNT_FAIL + .format_if_mount_failed = true, +#else + .format_if_mount_failed = false, +#endif + .max_files = 5, + .allocation_unit_size = 16 * 1024 + }; + + sdmmc_host_t host = SDSPI_HOST_DEFAULT(); + host.slot = BSP_LCD_SPI_NUM; + sdspi_device_config_t slot_config = SDSPI_DEVICE_CONFIG_DEFAULT(); + slot_config.gpio_cs = BSP_SD_CS; + slot_config.host_id = host.slot; + + ESP_RETURN_ON_ERROR(bsp_spi_init((BSP_LCD_H_RES * BSP_LCD_V_RES) * sizeof(uint16_t)), TAG, ""); + + return esp_vfs_fat_sdspi_mount(BSP_SD_MOUNT_POINT, &host, &slot_config, &mount_config, &bsp_sdcard); +} + +esp_err_t bsp_sdcard_unmount(void) +{ + return esp_vfs_fat_sdcard_unmount(BSP_SD_MOUNT_POINT, bsp_sdcard); +} + +esp_codec_dev_handle_t bsp_audio_codec_speaker_init(void) +{ + const audio_codec_data_if_t *i2s_data_if = bsp_audio_get_codec_itf(); + if (i2s_data_if == NULL) { + /* Initilize I2C */ + BSP_ERROR_CHECK_RETURN_NULL(bsp_i2c_init()); + /* Configure I2S peripheral and Power Amplifier */ + BSP_ERROR_CHECK_RETURN_NULL(bsp_audio_init(NULL)); + i2s_data_if = bsp_audio_get_codec_itf(); + } + assert(i2s_data_if); + + BSP_ERROR_CHECK_RETURN_ERR(bsp_feature_enable(BSP_FEATURE_SPEAKER, true)); + + esp_codec_dev_cfg_t codec_dev_cfg = { + .dev_type = ESP_CODEC_DEV_TYPE_OUT, + .codec_if = NULL, + .data_if = i2s_data_if, + }; + return esp_codec_dev_new(&codec_dev_cfg); +} + +// Bit number used to represent command and parameter +#define LCD_CMD_BITS 8 +#define LCD_PARAM_BITS 8 +#define LCD_LEDC_CH CONFIG_BSP_DISPLAY_BRIGHTNESS_LEDC_CH + +esp_err_t bsp_display_brightness_init(void) +{ + /* Initilize I2C */ + BSP_ERROR_CHECK_RETURN_ERR(bsp_i2c_init()); +#if defined(CONFIG_BSP_PMU_AXP2101) + const uint8_t lcd_bl_en[] = {0x90, 0x3F}; // AXP ALDO1~4 BLDO1~2 Enable + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, lcd_bl_en, sizeof(lcd_bl_en), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t lcd_bl_val[] = {0x96, 0b000011000}; // AXP BLDO1 voltage + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, lcd_bl_val, sizeof(lcd_bl_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + + // AXP2101 + // PowerKey Hold=1sec / PowerOff=4sec IRQLEVEL/OFFLEVEL/ONLEVEL setting + const uint8_t axp_pwr_val[] = {0x27, 0b00000000}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, axp_pwr_val, sizeof(axp_pwr_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + // Internal off-discharge enable for DCDC & LDO & SWITCH + const uint8_t axp_en_val[] = {0x10, 0b00110000}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, axp_en_val, sizeof(axp_en_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + // BATFET disable + const uint8_t axp_bat_val[] = {0x12, 0b00000000}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, axp_bat_val, sizeof(axp_bat_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + // CHGLED setting + const uint8_t axp_hgled_val[] = {0x69, 0b00010011}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, axp_hgled_val, sizeof(axp_hgled_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#elif defined(CONFIG_BSP_PMU_AXP192) + read8bit(0x12); + + const uint8_t vbus_limit[] = {0x30, (read8bit(0x30) & 0x04) | 0x02}; // axp: vbus limit off + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, vbus_limit, sizeof(vbus_limit), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_gpio1[] = {0x92, read8bit(0x92) & 0xf8}; // AXP192 GPIO1:OD OUTPUT + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_gpio1, sizeof(pmu_gpio1), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_gpio2[] = {0x93, read8bit(0x93) & 0xf8}; // AXP192 GPIO2:OD OUTPUT + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_gpio2, sizeof(pmu_gpio2), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_gpio3[] = {0x35, (read8bit(0x35) & 0x1c) | 0xa2}; // AXP192 RTC CHG + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_gpio3, sizeof(pmu_gpio3), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t esp_vol[] = {0x26, 0x6a}; // AXP192 RTC CHG + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, esp_vol, sizeof(esp_vol), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t lcd_bl_val[] = {0x27, 0x68}; // Lcd backlight voltage + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, lcd_bl_val, sizeof(lcd_bl_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_ldo2[] = {0x28, + (read8bit(0x28) & 0x0f) | 0xf0 + }; // axp: lcd logic and sdcard voltage preset to 3.3v + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_ldo2, sizeof(pmu_ldo2), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t pmu_ldo2_en[] = {0x12, (read8bit(0x12) & 0xfb) | 0x04}; // ldo2 enable + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pmu_ldo2_en, sizeof(pmu_ldo2_en), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t lcd_bl_en[] = {0x12, (read8bit(0x12) & 0xfd) | 0x02}; // Lcd backlight enable + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, lcd_bl_en, sizeof(lcd_bl_en), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + + const uint8_t chg_current[] = {0x33, ((read8bit(0x33) & 0xf0)) | 0x0}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, chg_current, sizeof(chg_current), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + + const uint8_t apx_gpio4[] = {0x95, (read8bit(0x95) & 0x72) | 0x84}; + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, apx_gpio4, sizeof(apx_gpio4), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + read8bit(0x95); + const uint8_t pek_key[] = {0x36, 0x4c}; // Lcd backlight enable + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, pek_key, sizeof(pek_key), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t adc_en[] = {0x82, 0xff}; // Lcd backlight enable + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, adc_en, sizeof(adc_en), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + const uint8_t lcd_rset2[] = {0x96, read8bit(0x96) &(~0x02)}; // Lcd backlight enable + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, lcd_rset2, sizeof(lcd_rset2), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); + vTaskDelay(pdMS_TO_TICKS(100)); // 延迟100ms + const uint8_t lcd_rset[] = {0x96, read8bit(0x96) | 0x02}; // Lcd backlight enable + ESP_RETURN_ON_ERROR( + i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, lcd_rset, sizeof(lcd_rset), 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#endif + return ESP_OK; +} + +esp_err_t bsp_display_brightness_set(int brightness_percent) +{ + if (brightness_percent > 100) { + brightness_percent = 100; + } + if (brightness_percent < 0) { + brightness_percent = 0; + } + + ESP_LOGI(TAG, "Setting LCD backlight: %d%%", brightness_percent); +#if defined(CONFIG_BSP_PMU_AXP2101) + const uint8_t reg_val = 20 + ((8 * brightness_percent) / 100); // 0b00000 ~ 0b11100; under 20, it is too dark + const uint8_t lcd_bl_val[] = {0x96, reg_val}; // AXP BLDO1 voltage + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP2101_ADDR, lcd_bl_val, sizeof(lcd_bl_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#elif defined(CONFIG_BSP_PMU_AXP192) + const uint8_t reg_val = 90 + ((8 * brightness_percent) / 100); // 0b00000 ~ 0b11100; under 20, it is too dark + const uint8_t lcd_bl_val[] = {0x27, reg_val}; // AXP DCDC3 voltage + ESP_RETURN_ON_ERROR(i2c_master_write_to_device(BSP_I2C_NUM, BSP_AXP192_ADDR, lcd_bl_val, sizeof(lcd_bl_val), + 1000 / portTICK_PERIOD_MS), + TAG, "I2C write failed"); +#endif + + return ESP_OK; +} + +esp_err_t bsp_display_backlight_off(void) +{ + return bsp_display_brightness_set(0); +} + +esp_err_t bsp_display_backlight_on(void) +{ + return bsp_display_brightness_set(100); +} + +esp_err_t bsp_display_new(const bsp_display_config_t *config, esp_lcd_panel_handle_t *ret_panel, + esp_lcd_panel_io_handle_t *ret_io) +{ + esp_err_t ret = ESP_OK; + assert(config != NULL && config->max_transfer_sz > 0); + + BSP_ERROR_CHECK_RETURN_ERR(bsp_feature_enable(BSP_FEATURE_LCD, true)); + + /* Initialize SPI */ + ESP_RETURN_ON_ERROR(bsp_spi_init(config->max_transfer_sz), TAG, ""); + + ESP_LOGI(TAG, "Install panel IO"); + const esp_lcd_panel_io_spi_config_t io_config = { + .dc_gpio_num = BSP_LCD_DC, + .cs_gpio_num = BSP_LCD_CS, + .pclk_hz = BSP_LCD_PIXEL_CLOCK_HZ, + .lcd_cmd_bits = LCD_CMD_BITS, + .lcd_param_bits = LCD_PARAM_BITS, + .spi_mode = 0, + .trans_queue_depth = 10, + }; + ESP_GOTO_ON_ERROR(esp_lcd_new_panel_io_spi((esp_lcd_spi_bus_handle_t)BSP_LCD_SPI_NUM, &io_config, ret_io), err, TAG, + "New panel IO failed"); + + ESP_LOGI(TAG, "Install LCD driver"); + const esp_lcd_panel_dev_config_t panel_config = { + .reset_gpio_num = BSP_LCD_RST, // Shared with Touch reset + .color_space = BSP_LCD_COLOR_SPACE, + .bits_per_pixel = BSP_LCD_BITS_PER_PIXEL, + }; + + ESP_GOTO_ON_ERROR(esp_lcd_new_panel_ili9341(*ret_io, &panel_config, ret_panel), err, TAG, "New panel failed"); + + esp_lcd_panel_reset(*ret_panel); + esp_lcd_panel_init(*ret_panel); + esp_lcd_panel_invert_color(*ret_panel, true); + return ret; + +err: + if (*ret_panel) { + esp_lcd_panel_del(*ret_panel); + } + if (*ret_io) { + esp_lcd_panel_io_del(*ret_io); + } + spi_bus_free(BSP_LCD_SPI_NUM); + return ret; +} + +esp_err_t bsp_touch_new(const bsp_touch_config_t *config, esp_lcd_touch_handle_t *ret_touch) +{ + BSP_ERROR_CHECK_RETURN_ERR(bsp_feature_enable(BSP_FEATURE_TOUCH, true)); + + /* Initialize touch */ + const esp_lcd_touch_config_t tp_cfg = { + .x_max = BSP_LCD_H_RES, + .y_max = BSP_LCD_V_RES, + .rst_gpio_num = GPIO_NUM_NC, // Shared with LCD reset + .int_gpio_num = BSP_LCD_TOUCH_INT, + .levels = + { + .reset = 0, + .interrupt = 0, + }, + .flags = + { + .swap_xy = 0, + .mirror_x = 0, + .mirror_y = 0, + }, + }; + esp_lcd_panel_io_handle_t tp_io_handle = NULL; + const esp_lcd_panel_io_i2c_config_t tp_io_config = ESP_LCD_TOUCH_IO_I2C_FT5x06_CONFIG(); + ESP_RETURN_ON_ERROR(esp_lcd_new_panel_io_i2c((esp_lcd_i2c_bus_handle_t)BSP_I2C_NUM, &tp_io_config, &tp_io_handle), + TAG, ""); + return esp_lcd_touch_new_i2c_ft5x06(tp_io_handle, &tp_cfg, ret_touch); +} + +#if (BSP_CONFIG_NO_GRAPHIC_LIB == 0) +static lv_display_t *bsp_display_lcd_init(const bsp_display_cfg_t *cfg) +{ + assert(cfg != NULL); + esp_lcd_panel_io_handle_t io_handle = NULL; + esp_lcd_panel_handle_t panel_handle = NULL; + const bsp_display_config_t bsp_disp_cfg = { + .max_transfer_sz = BSP_LCD_DRAW_BUFF_SIZE * sizeof(uint16_t), + }; + BSP_ERROR_CHECK_RETURN_NULL(bsp_display_new(&bsp_disp_cfg, &panel_handle, &io_handle)); + + esp_lcd_panel_disp_on_off(panel_handle, true); + + /* Add LCD screen */ + ESP_LOGD(TAG, "Add LCD screen"); + const lvgl_port_display_cfg_t disp_cfg = { + .io_handle = io_handle, + .panel_handle = panel_handle, + .buffer_size = cfg->buffer_size, + .double_buffer = cfg->double_buffer, + .hres = BSP_LCD_H_RES, + .vres = BSP_LCD_V_RES, + .monochrome = false, + /* Rotation values must be same as used in esp_lcd for initial settings of the screen */ + .rotation = + { + .swap_xy = false, + .mirror_x = false, + .mirror_y = false, + }, + .flags = { + .buff_dma = cfg->flags.buff_dma, + .buff_spiram = cfg->flags.buff_spiram, +#if LVGL_VERSION_MAJOR >= 9 + .swap_bytes = (BSP_LCD_BIGENDIAN ? true : false), +#endif + } + }; + + return lvgl_port_add_disp(&disp_cfg); +} + +static lv_indev_t *bsp_display_indev_init(lv_display_t *disp) +{ + BSP_ERROR_CHECK_RETURN_NULL(bsp_touch_new(NULL, &tp)); + assert(tp); + + /* Add touch input (for selected screen) */ + const lvgl_port_touch_cfg_t touch_cfg = { + .disp = disp, + .handle = tp, + }; + + return lvgl_port_add_touch(&touch_cfg); +} + +lv_display_t *bsp_display_start(void) +{ + bsp_display_cfg_t cfg = {.lvgl_port_cfg = ESP_LVGL_PORT_INIT_CONFIG(), + .buffer_size = BSP_LCD_DRAW_BUFF_SIZE, + .double_buffer = BSP_LCD_DRAW_BUFF_DOUBLE, + .flags = { + .buff_dma = true, + .buff_spiram = false, + } + }; + cfg.lvgl_port_cfg.task_affinity = 1; /* For camera */ + return bsp_display_start_with_config(&cfg); +} + +lv_display_t *bsp_display_start_with_config(const bsp_display_cfg_t *cfg) +{ + assert(cfg != NULL); + BSP_ERROR_CHECK_RETURN_NULL(lvgl_port_init(&cfg->lvgl_port_cfg)); + + BSP_ERROR_CHECK_RETURN_NULL(bsp_display_brightness_init()); + + BSP_NULL_CHECK(disp = bsp_display_lcd_init(cfg), NULL); + + BSP_NULL_CHECK(disp_indev = bsp_display_indev_init(disp), NULL); + + return disp; +} + +lv_indev_t *bsp_display_get_input_dev(void) +{ + return disp_indev; +} + +void bsp_display_rotate(lv_display_t *disp, lv_display_rotation_t rotation) +{ + lv_disp_set_rotation(disp, rotation); +} + +bool bsp_display_lock(uint32_t timeout_ms) +{ + return lvgl_port_lock(timeout_ms); +} + +void bsp_display_unlock(void) +{ + lvgl_port_unlock(); +} + +#endif // (BSP_CONFIG_NO_GRAPHIC_LIB == 0) diff --git a/bsp/m5stack_core_2/m5stack_core_2_idf5.c b/bsp/m5stack_core_2/m5stack_core_2_idf5.c new file mode 100644 index 00000000..a79d1ad7 --- /dev/null +++ b/bsp/m5stack_core_2/m5stack_core_2_idf5.c @@ -0,0 +1,100 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "esp_err.h" +#include "bsp/m5stack_core_2.h" +#include "bsp_err_check.h" +#include "esp_codec_dev_defaults.h" + +static const char *TAG = "M5Stack"; + +static i2s_chan_handle_t i2s_tx_chan = NULL; +static i2s_chan_handle_t i2s_rx_chan = NULL; +static const audio_codec_data_if_t *i2s_data_if = NULL; /* Codec data interface */ + +/* Can be used for i2s_std_gpio_config_t and/or i2s_std_config_t initialization */ +#define BSP_I2S_GPIO_CFG \ + { \ + .bclk = BSP_I2S_SCLK, .ws = BSP_I2S_LCLK, .dout = BSP_I2S_DOUT, .din = BSP_I2S_DSIN, \ + .invert_flags = { \ + .mclk_inv = false, \ + .bclk_inv = false, \ + .ws_inv = true, \ + }, \ + .mclk = BSP_I2S_MCLK, \ + } +/* This configuration is used by default in bsp_audio_init() */ +#define BSP_I2S_DUPLEX_MONO_CFG(_sample_rate) \ + { \ + .clk_cfg = I2S_STD_CLK_DEFAULT_CONFIG(_sample_rate), \ + .slot_cfg = I2S_STD_PHILIP_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_MONO), \ + .gpio_cfg = BSP_I2S_GPIO_CFG, \ + } + +esp_err_t bsp_audio_init(const i2s_std_config_t *i2s_config) +{ + esp_err_t ret = ESP_FAIL; + if (i2s_tx_chan && i2s_rx_chan) { + ESP_LOGW(TAG, "Audio was initialized before"); + return ESP_OK; + } + + /* Setup I2S peripheral */ + i2s_chan_config_t chan_cfg = I2S_CHANNEL_DEFAULT_CONFIG(CONFIG_BSP_I2S_NUM, I2S_ROLE_MASTER); + chan_cfg.auto_clear = true; // Auto clear the legacy data in the DMA buffer + BSP_ERROR_CHECK_RETURN_ERR(i2s_new_channel(&chan_cfg, &i2s_tx_chan, &i2s_rx_chan)); + + /* Setup I2S channels */ + i2s_std_config_t std_cfg_default = BSP_I2S_DUPLEX_MONO_CFG(44100); + + const i2s_std_config_t *p_i2s_cfg = &std_cfg_default; + if (i2s_config != NULL) { + p_i2s_cfg = i2s_config; + } + + if (i2s_tx_chan != NULL) { + ESP_GOTO_ON_ERROR(i2s_channel_init_std_mode(i2s_tx_chan, p_i2s_cfg), err, TAG, + "I2S TX channel initialization failed"); + ESP_GOTO_ON_ERROR(i2s_channel_enable(i2s_tx_chan), err, TAG, "I2S TX enabling failed"); + } + if (i2s_rx_chan != NULL) { + ESP_GOTO_ON_ERROR(i2s_channel_init_std_mode(i2s_rx_chan, p_i2s_cfg), err, TAG, + "I2S RX channel initialization failed"); + ESP_GOTO_ON_ERROR(i2s_channel_enable(i2s_rx_chan), err, TAG, "I2S RX enabling failed"); + } + + audio_codec_i2s_cfg_t i2s_cfg = { + .port = CONFIG_BSP_I2S_NUM, + .rx_handle = i2s_rx_chan, + .tx_handle = i2s_tx_chan, + }; + i2s_data_if = audio_codec_new_i2s_data(&i2s_cfg); + BSP_NULL_CHECK_GOTO(i2s_data_if, err); + + ESP_LOGI(TAG, "I2S initialized successfully"); + + ESP_LOGI(TAG, "I2S config: sample_rate=%lu, data_bit_width=%d, slot_mode=%ld", + (unsigned long)p_i2s_cfg->clk_cfg.sample_rate_hz, p_i2s_cfg->slot_cfg.data_bit_width, + (long)p_i2s_cfg->slot_cfg.slot_mode); + + return ESP_OK; + +err: + if (i2s_tx_chan) { + i2s_del_channel(i2s_tx_chan); + } + if (i2s_rx_chan) { + i2s_del_channel(i2s_rx_chan); + } + + ESP_LOGE(TAG, "Initialization failed, ret: %d", ret); + return ret; +} + +const audio_codec_data_if_t *bsp_audio_get_codec_itf(void) +{ + return i2s_data_if; +} diff --git a/bsp/m5stack_core_2/priv_include/bsp_err_check.h b/bsp/m5stack_core_2/priv_include/bsp_err_check.h new file mode 100644 index 00000000..cf2f36eb --- /dev/null +++ b/bsp/m5stack_core_2/priv_include/bsp_err_check.h @@ -0,0 +1,58 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "esp_check.h" +#include "sdkconfig.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/* Assert on error, if selected in menuconfig. Otherwise return error code. */ +#if CONFIG_BSP_ERROR_CHECK +#define BSP_ERROR_CHECK_RETURN_ERR(x) ESP_ERROR_CHECK(x) +#define BSP_ERROR_CHECK_RETURN_NULL(x) ESP_ERROR_CHECK(x) +#define BSP_ERROR_CHECK(x, ret) ESP_ERROR_CHECK(x) +#define BSP_NULL_CHECK(x, ret) assert(x) +#define BSP_NULL_CHECK_GOTO(x, goto_tag) assert(x) +#else +#define BSP_ERROR_CHECK_RETURN_ERR(x) do { \ + esp_err_t err_rc_ = (x); \ + if (unlikely(err_rc_ != ESP_OK)) { \ + return err_rc_; \ + } \ + } while(0) + +#define BSP_ERROR_CHECK_RETURN_NULL(x) do { \ + if (unlikely((x) != ESP_OK)) { \ + return NULL; \ + } \ + } while(0) + +#define BSP_NULL_CHECK(x, ret) do { \ + if ((x) == NULL) { \ + return ret; \ + } \ + } while(0) + +#define BSP_ERROR_CHECK(x, ret) do { \ + if (unlikely((x) != ESP_OK)) { \ + return ret; \ + } \ + } while(0) + +#define BSP_NULL_CHECK_GOTO(x, goto_tag) do { \ + if ((x) == NULL) { \ + goto goto_tag; \ + } \ + } while(0) +#endif + +#ifdef __cplusplus +} +#endif diff --git a/docu/pics/m5stack_core2.webp b/docu/pics/m5stack_core2.webp new file mode 100644 index 0000000000000000000000000000000000000000..b91f44f2979f40ceea59529fe805c79721d50901 GIT binary patch literal 15222 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 ep*^Ekvs8DbGdIx zqcf;M%;g>VMAJV)dZ_HkLHg^yrkHnZvA?!Z^Eq6P@c;&H#95v}) zSf15xn7$5jr6wYT)`Ks;L06`jqMd`G#!55d)hm}Nou{Y>@5uh!S@qW;td5JKoOL)K z4zpCU|E=cgy^Ns%@fSdna75Zs5L}^a%{*ipp;}AokH~mp^p@icjsb4+`)loibY6P+ zNefChMvJdTW;F#V?lX_i;BKJ@LDjPNuNgU<3aZCy%{@#>TVYg@lM-cvwOA9Yk@Zw; z{!Saf?zHoAVV?aAJ;P0sF|ch?^p?<##P4$VmO+EiOre?vTNFHy=jp%&dQ6_=u;4b( z5(BaHaeV*SW8Dmj3f!O>B$Pa}nas-_o2`VUcA|IIx2@*=-nK1<=vh(FVR=)PsxpH0 z-Xjqf1!z8Hoj4VsTM^=c5Hvc8X_g&~r{Yy=DYpxM8dMZ{89-@95(Q{2mR&TtaPOMF zfxtcvNU}mYb}C*GvtkSC4U)W9EwxseZb-hZ#C08!JVM=2E#8g$;*PDSlBt{oOzDw+ zG`ZGU_mJZVIAmSq0mgLi8P^sj=v> zugZ&O0RC6x!Kdy*vI8w~^gCHq)(jwpzlSG#s_>Yp5)L1l;_^mNJg2Z`vpDe1>fxij z5Ek8OL?@ye!tHhUWde{k_YvurIu!~znstEog&sMwY9Szy*X69ur3T(bv`G_@U6{+` zfGvbEdWSuf9h_~|5MQoqPeCU(c$vAF%1NZ5rKB}mo*t8d;;X6?hJ?W7>cEyO0_L_v_ zo60l6KuEM^aC+Brl(8nsn2WU8Fn865x-83XvAm3yr6uE6XD`y38~CMOL-A0Xzvsxj zi_dN-{>`813#Pf_b)Y#KV#vf?+qyaNv4I-$Jv7t|9xL=o5x!@dr6s7ppd}u%Den(k zKh?<`J23&FP@54p#@bW5W)Pn8CKtr}kTrR(oo#)f!Gkp}?iR9GD9fN8UBkcFNu zh^>(UgZMi+(TVT|BKMA-p^NWV9B zMHr)*W^vmTUa_$f_8d`S^tB)dkSK*3-*X&O_O88LYww^`rCtCa%zQP{ik)BB{BgBn z4s*WX645Kl7WAOj_Ve#3jXM$%I4oL_O7jB+5?WGv@HVt%7kp_@GRa=*02M2W?4~%4 z@$IV!8`PLPp8*iTJ=-DEv2pvJ`}ET2?g7Fm3N3@CBNqPrXD8xJO<#+?#b8~*v_W+N z-nRsJUG+ofm!YhD;q>|{Sx z3(`I+=(NC}h0eR9b(VgcfZ^j8+B=HO7&$Hm1|gwPk!0#v<)cKD*-;#i%Fa(EXY7og zYh$||?2A{cGJb#Q>X17d=9v!W;S(G)wF`vyr7l`g(*>xaz%BdZT}DatI49RDu$r)^Jc^}}Q67v#WLYNe03CP#rw`0;p9was zx^#ds!Q>of^D(jEc@7p}G$}_DjXPegOGzu2%Fyi?AF&frgAHmN7g@iE8M{JiDTDr5 zeny#m8O{uLAJs+(VJJ z%)_&_ej_C=jx4epEgol~NA>S4u;o#x$d}kgSEnZo1Scf4l4WC+Cyp=Ns+QuP@%&Kx zP7+{Od!{2=#U4t+N^KUo?uj3K&k!_~iK%oNA}EVWAADEt0KKGPWu;cs2K9$m`Bd&d zg13}CN|KjUys*oZGEay9d|mePqOu9$l!H2=5Xu@vL@)89D51F;NtcP_I|3YV5!g|6!%3B5YhdNg+MLwT2YN$)LZpRWsq$^(T z9l$q6a^n(&WVIKst1}H4ckR@s3OtyD^@aS(KQX^802EK%!K#6^OfhjI>{uC|cD!aA zZjegt;Zi5^0&@V`rmUk~i|268+f-%o{`(0_hXfZ)BF^;`TJZZX z<}E$xuZB?4i4+zmYB6Kot3mYVVSigmuCQM z3x}AA?N*Wz2%OWI4P>CwO^;n#z|9(?b0Ey}hP(7v-jqOe7P!gc>Ul=p0|L zDR;zWBF5Ok2H>7K<$zsx9x1Ua`=urjWf~JD5U|BuA|S}3l#+1GZ%o0yM3)0csIy(E zwzw{Xv%vIKd28MRy~#+_(c^$QnmrZdoDc{ihL+rk-|5vJlJj}d{QdKe`z;*x#JVk! zlzofAlvPo#JJm9a@I@3Qp0NZ1b&n)am7t+`EGTU0UTwMYb2O#%^F3DJb%bGYVg?p7 zX~08}8y#AL{n1VX$Qz0y{28d=pkQT`Q^%>&}c!Njb#-|69}+2jK4aty^nXfa)jJrhsx zm}L6X0ZC(J@d2%JuYK0CY)#7?HuW}M1nW=|=`%)|Jq$o%mOj^ZCqx})xhachhLX3& zLVNuH0Nxdj-%xteQS)J`pshl8H()G(O7P%tYY#(TiGhH6icvg^cu}3iKFt5pNYN=3 z;ZArfa6eKnCx&{i14gyM+MrkK$`YjAZX;=9J+kGfDi)(4mR?ovrR1aLqCE#tZTCs3 z@m}F)+7T(ybbdpDb82%QMXw>(r)#bp`rGl&a@`r zIv|fRP5u0P4mk1$VB=U&HT)+F9Dpd2qh4VmH=5P}z%4e@v%L8*1I?xBB97{FV~%WJ zazczXj@k>o`40E^b4%n^zN<7hN|eby{-iMflrbGdr)#HRWgEcKeF^;%5U%pLVhx*y zUG(6|pWFN(g1D7c=yuE*j&wbhlVw<)+%X0;^9g3?M2^qzqU@w;FvM_v>IS5{EE8){ zcem21Z1rX>g5B+sf#1T3xKRynC7pQ}2YU2>PmD8kF;GavB3bM$}RKtsKRzoh+NPyc=5 F{{Xp*b8`Ry literal 0 HcmV?d00001 diff --git a/examples/bsp_ext.py b/examples/bsp_ext.py index 1ed1b2c9..c135cea2 100644 --- a/examples/bsp_ext.py +++ b/examples/bsp_ext.py @@ -59,7 +59,8 @@ def set_bsp_callback(action: str, ctx: Context, args: PropertyDict, **kwargs: st 'esp32_s3_korvo_1', 'esp32_p4_function_ev_board', 'm5stack_core_s3', - 'm5dial' + 'm5dial', + 'm5stack_core_2' } if bsp == '': diff --git a/examples/display/sdkconfig.bsp.m5stack_core_2 b/examples/display/sdkconfig.bsp.m5stack_core_2 new file mode 100644 index 00000000..2394a8df --- /dev/null +++ b/examples/display/sdkconfig.bsp.m5stack_core_2 @@ -0,0 +1,25 @@ +# This file was generated using idf.py save-defconfig. It can be edited manually. +# Espressif IoT Development Framework (ESP-IDF) Project Minimal Configuration +# +CONFIG_IDF_TARGET="esp32" +CONFIG_ESPTOOLPY_FLASHSIZE_16MB=y +CONFIG_ESP_DEFAULT_CPU_FREQ_240=y + +## LVGL8 ## +CONFIG_LV_USE_PERF_MONITOR=y +CONFIG_LV_COLOR_16_SWAP=y +CONFIG_LV_MEM_CUSTOM=y +CONFIG_LV_MEMCPY_MEMSET_STD=y + +## LVGL9 ## +CONFIG_LV_CONF_SKIP=y + +#CLIB default +CONFIG_LV_USE_CLIB_MALLOC=y +CONFIG_LV_USE_CLIB_SPRINTF=y +CONFIG_LV_USE_CLIB_STRING=y + +# Performance monitor +CONFIG_LV_USE_OBSERVER=y +CONFIG_LV_USE_SYSMON=y +CONFIG_LV_USE_PERF_MONITOR=y diff --git a/examples/display_audio_photo/main/app_disp_fs.c b/examples/display_audio_photo/main/app_disp_fs.c index 51287975..79c93085 100644 --- a/examples/display_audio_photo/main/app_disp_fs.c +++ b/examples/display_audio_photo/main/app_disp_fs.c @@ -164,10 +164,12 @@ void app_audio_init(void) esp_codec_dev_set_out_vol(spk_codec_dev, DEFAULT_VOLUME); /* Initialize microphone */ +#if BSP_CAPS_AUDIO_MIC mic_codec_dev = bsp_audio_codec_microphone_init(); assert(mic_codec_dev); /* Microphone input gain */ esp_codec_dev_set_in_gain(mic_codec_dev, 50.0); +#endif } void app_disp_fs_init(void) @@ -766,6 +768,7 @@ static void rec_stop_event_cb(lv_event_t *e) static void rec_file(void *arg) { +#if BSP_CAPS_AUDIO_MIC char *path = arg; FILE *record_file = NULL; int16_t *recording_buffer = heap_caps_malloc(BUFFER_SIZE, MALLOC_CAP_DEFAULT); @@ -834,6 +837,9 @@ static void rec_file(void *arg) } vTaskDelete(NULL); +#else + ESP_LOGI(TAG, "Recording not supported!"); +#endif } /* Stop playing recorded audio file */ diff --git a/examples/display_audio_photo/sdkconfig.bsp.m5stack_core_2 b/examples/display_audio_photo/sdkconfig.bsp.m5stack_core_2 new file mode 100644 index 00000000..e885cc57 --- /dev/null +++ b/examples/display_audio_photo/sdkconfig.bsp.m5stack_core_2 @@ -0,0 +1,26 @@ +# This file was generated using idf.py save-defconfig. It can be edited manually. +# Espressif IoT Development Framework (ESP-IDF) Project Minimal Configuration +# +CONFIG_IDF_TARGET="esp32" +CONFIG_ESPTOOLPY_FLASHFREQ_80M=y +CONFIG_ESPTOOLPY_FLASHSIZE_4MB=y +CONFIG_PARTITION_TABLE_CUSTOM=y +CONFIG_SPIRAM=y +CONFIG_SPIRAM_SPEED_80M=y +CONFIG_ESP_INT_WDT_TIMEOUT_MS=300 +CONFIG_LV_USE_CLIB_MALLOC=y +CONFIG_LV_USE_CLIB_STRING=y +CONFIG_LV_USE_CLIB_SPRINTF=y +CONFIG_LV_ATTRIBUTE_FAST_MEM_USE_IRAM=y +CONFIG_LV_FONT_MONTSERRAT_12=y +CONFIG_LV_FONT_MONTSERRAT_16=y +CONFIG_LV_USE_OBSERVER=y +CONFIG_LV_USE_SYSMON=y +CONFIG_LV_USE_PERF_MONITOR=y +CONFIG_LV_BUILD_EXAMPLES=n +CONFIG_LV_USE_DEMO_WIDGETS=y +CONFIG_LV_USE_DEMO_STRESS=y +CONFIG_LV_USE_DEMO_MUSIC=y +CONFIG_LV_DEMO_MUSIC_ROUND=y +CONFIG_LV_DEMO_MUSIC_AUTO_PLAY=y +