Bladeren bron

driver: add parallel IO TX driver

morris 3 jaren geleden
bovenliggende
commit
f534247a00
41 gewijzigde bestanden met toevoegingen van 3125 en 4 verwijderingen
  1. 4 0
      components/driver/.build-test-rules.yml
  2. 6 0
      components/driver/CMakeLists.txt
  3. 20 0
      components/driver/Kconfig
  4. 187 0
      components/driver/parlio/include/driver/parlio_tx.h
  5. 26 0
      components/driver/parlio/include/driver/parlio_types.h
  6. 89 0
      components/driver/parlio/parlio_common.c
  7. 70 0
      components/driver/parlio/parlio_private.h
  8. 628 0
      components/driver/parlio/parlio_tx.c
  9. 2 2
      components/driver/rmt/include/driver/rmt_types.h
  10. 18 0
      components/driver/test_apps/parlio/CMakeLists.txt
  11. 2 0
      components/driver/test_apps/parlio/README.md
  12. 7 0
      components/driver/test_apps/parlio/main/CMakeLists.txt
  13. 51 0
      components/driver/test_apps/parlio/main/test_app_main.c
  14. 40 0
      components/driver/test_apps/parlio/main/test_board.h
  15. 280 0
      components/driver/test_apps/parlio/main/test_parlio_tx.c
  16. 22 0
      components/driver/test_apps/parlio/pytest_parlio_unity.py
  17. 7 0
      components/driver/test_apps/parlio/sdkconfig.ci.iram_safe
  18. 5 0
      components/driver/test_apps/parlio/sdkconfig.ci.release
  19. 5 0
      components/driver/test_apps/parlio/sdkconfig.defaults
  20. 4 0
      components/hal/CMakeLists.txt
  21. 8 0
      components/hal/esp32c6/include/hal/clk_gate_ll.h
  22. 612 0
      components/hal/esp32c6/include/hal/parlio_ll.h
  23. 8 0
      components/hal/esp32h2/include/hal/clk_gate_ll.h
  24. 614 0
      components/hal/esp32h2/include/hal/parlio_ll.h
  25. 1 0
      components/hal/include/hal/gdma_types.h
  26. 46 0
      components/hal/include/hal/parlio_hal.h
  27. 68 0
      components/hal/include/hal/parlio_types.h
  28. 19 0
      components/hal/parlio_hal.c
  29. 4 0
      components/soc/CMakeLists.txt
  30. 28 0
      components/soc/esp32c6/include/soc/Kconfig.soc_caps.in
  31. 16 0
      components/soc/esp32c6/include/soc/clk_tree_defs.h
  32. 1 0
      components/soc/esp32c6/include/soc/periph_defs.h
  33. 9 0
      components/soc/esp32c6/include/soc/soc_caps.h
  34. 66 0
      components/soc/esp32c6/parlio_periph.c
  35. 32 0
      components/soc/esp32h2/include/soc/Kconfig.soc_caps.in
  36. 16 0
      components/soc/esp32h2/include/soc/clk_tree_defs.h
  37. 2 2
      components/soc/esp32h2/include/soc/parl_io_struct.h
  38. 1 0
      components/soc/esp32h2/include/soc/periph_defs.h
  39. 10 0
      components/soc/esp32h2/include/soc/soc_caps.h
  40. 50 0
      components/soc/esp32h2/parlio_periph.c
  41. 41 0
      components/soc/include/soc/parlio_periph.h

+ 4 - 0
components/driver/.build-test-rules.yml

@@ -64,6 +64,10 @@ components/driver/test_apps/mcpwm:
   disable:
     - if: SOC_MCPWM_SUPPORTED != 1
 
+components/driver/test_apps/parlio:
+  disable:
+    - if: SOC_PARLIO_SUPPORTED != 1
+
 components/driver/test_apps/pulse_cnt:
   disable:
     - if: SOC_PCNT_SUPPORTED != 1

+ 6 - 0
components/driver/CMakeLists.txt

@@ -17,6 +17,7 @@ set(includes "include"
              "i2s/include"
              "ledc/include"
              "mcpwm/include"
+             "parlio/include"
              "pcnt/include"
              "rmt/include"
              "sdio_slave/include"
@@ -49,6 +50,11 @@ if(CONFIG_SOC_DAC_SUPPORTED)
                      "deprecated/${target}/dac_legacy.c")
 endif()
 
+# Parallel IO related source files
+if(CONFIG_SOC_PARLIO_SUPPORTED)
+    list(APPEND srcs "parlio/parlio_common.c" "parlio/parlio_tx.c")
+endif()
+
 # GPIO related source files
 if(CONFIG_SOC_DEDICATED_GPIO_SUPPORTED)
     list(APPEND srcs "gpio/dedic_gpio.c")

+ 20 - 0
components/driver/Kconfig

@@ -478,4 +478,24 @@ menu "Driver Configurations"
                 USB Serial/JTAG is in use.
     endmenu # USB Serial/JTAG Configuration
 
+    menu "Parallel IO Configuration"
+        depends on SOC_PARLIO_SUPPORTED
+
+        config PARLIO_ENABLE_DEBUG_LOG
+            bool "Enable debug log"
+            default n
+            help
+                Wether to enable the debug log message for parallel IO driver.
+                Note that, this option only controls the parallel IO driver log, won't affect other drivers.
+
+        config PARLIO_ISR_IRAM_SAFE
+            bool "Parallel IO ISR IRAM-Safe"
+            default n
+            select GDMA_CTRL_FUNC_IN_IRAM # the driver needs to start the GDMA in the interrupt
+            help
+                Ensure the Parallel IO interrupt is IRAM-Safe by allowing the interrupt handler to be
+                executable when the cache is disabled (e.g. SPI Flash write).
+
+    endmenu # Parallel IO Configuration
+
 endmenu  # Driver configurations

+ 187 - 0
components/driver/parlio/include/driver/parlio_tx.h

@@ -0,0 +1,187 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#pragma once
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "esp_err.h"
+#include "driver/parlio_types.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief Parallel IO TX unit configuration
+ */
+typedef struct {
+    parlio_clock_source_t clk_src;  /*!< Parallel IO internal clock source */
+    gpio_num_t clk_in_gpio_num;     /*!< If the clock source is input from external, set the corresponding GPIO number.
+                                         Otherwise, set to `-1` and the driver will use the internal `clk_src` as clock source.
+                                         This option has higher priority than `clk_src` */
+    uint32_t input_clk_src_freq_hz; /*!< Frequency of the input clock source, valid only if `clk_in_gpio_num` is not `-1` */
+    uint32_t output_clk_freq_hz;    /*!< Frequency of the output clock. It's divided from either internal `clk_src` or external clock source */
+    size_t data_width;              /*!< Parallel IO data width, can set to 1/2/4/8/..., but can't bigger than PARLIO_TX_UNIT_MAX_DATA_WIDTH */
+    gpio_num_t data_gpio_nums[PARLIO_TX_UNIT_MAX_DATA_WIDTH]; /*!< Parallel IO data GPIO numbers, if any GPIO is not used, you can set it to `-1` */
+    gpio_num_t clk_out_gpio_num; /*!< GPIO number of the output clock signal, the clock is synced with TX data */
+    gpio_num_t valid_gpio_num;   /*!< GPIO number of the valid signal, which stays high when transferring data.
+                                      Note that, the valid signal will always occupy the MSB data bit */
+    size_t trans_queue_depth; /*!< Depth of internal transaction queue */
+    size_t max_transfer_size; /*!< Maximum transfer size in one transaction, in bytes. This decides the number of DMA nodes will be used for each transaction */
+    parlio_sample_edge_t sample_edge;       /*!< Parallel IO sample edge */
+    parlio_bit_pack_order_t bit_pack_order; /*!< Set the order of packing the bits into bytes (only works when `data_width` < 8) */
+    struct {
+        uint32_t clk_gate_en: 1;  /*!< Enable TX clock gating,
+                                       the output clock will be controlled by the MSB bit of the data bus,
+                                       i.e. by data_gpio_nums[PARLIO_TX_UNIT_MAX_DATA_WIDTH-1]. High level to enable the clock output, low to disable */
+        uint32_t io_loop_back: 1; /*!< For debug/test, the signal output from the GPIO will be fed to the input path as well */
+    } flags;                      /*!< Extra configuration flags */
+} parlio_tx_unit_config_t;
+
+/**
+ * @brief Create a Parallel IO TX unit
+ *
+ * @param[in] config Parallel IO TX unit configuration
+ * @param[out] ret_unit Returned Parallel IO TX unit handle
+ * @return
+ *      - ESP_OK: Create Parallel IO TX unit successfully
+ *      - ESP_ERR_INVALID_ARG: Create Parallel IO TX unit failed because of invalid argument
+ *      - ESP_ERR_NO_MEM: Create Parallel IO TX unit failed because of out of memory
+ *      - ESP_ERR_NOT_FOUND: Create Parallel IO TX unit failed because all TX units are used up and no more free one
+ *      - ESP_ERR_NOT_SUPPORTED: Create Parallel IO TX unit failed because some feature is not supported by hardware, e.g. clock gating
+ *      - ESP_FAIL: Create Parallel IO TX unit failed because of other error
+ */
+esp_err_t parlio_new_tx_unit(const parlio_tx_unit_config_t *config, parlio_tx_unit_handle_t *ret_unit);
+
+/**
+ * @brief Delete a Parallel IO TX unit
+ *
+ * @param[in] unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @return
+ *      - ESP_OK: Delete Parallel IO TX unit successfully
+ *      - ESP_ERR_INVALID_ARG: Delete Parallel IO TX unit failed because of invalid argument
+ *      - ESP_ERR_INVALID_STATE: Delete Parallel IO TX unit failed because it is still in working
+ *      - ESP_FAIL: Delete Parallel IO TX unit failed because of other error
+ */
+esp_err_t parlio_del_tx_unit(parlio_tx_unit_handle_t unit);
+
+/**
+ * @brief Enable the Parallel IO TX unit
+ *
+ * @note This function will transit the driver state from init to enable
+ * @note This function will acquire a PM lock that might be installed during channel allocation
+ * @note If there're transaction pending in the queue, this function will pick up the first one and start the transfer
+ *
+ * @param[in] unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @return
+ *      - ESP_OK: Enable Parallel IO TX unit successfully
+ *      - ESP_ERR_INVALID_ARG: Enable Parallel IO TX unit failed because of invalid argument
+ *      - ESP_ERR_INVALID_STATE: Enable Parallel IO TX unit failed because it is already enabled
+ *      - ESP_FAIL: Enable Parallel IO TX unit failed because of other error
+ */
+esp_err_t parlio_tx_unit_enable(parlio_tx_unit_handle_t unit);
+
+/**
+ * @brief Disable the Parallel IO TX unit
+ *
+ * @note This function will transit the driver state from enable to init
+ * @note This function will release the PM lock that might be installed during channel allocation
+ * @note If one transaction is undergoing, this function will terminate it immediately
+ *
+ * @param[in] unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @return
+ *      - ESP_OK: Disable Parallel IO TX unit successfully
+ *      - ESP_ERR_INVALID_ARG: Disable Parallel IO TX unit failed because of invalid argument
+ *      - ESP_ERR_INVALID_STATE: Disable Parallel IO TX unit failed because it's not enabled yet
+ *      - ESP_FAIL: Disable Parallel IO TX unit failed because of other error
+ */
+esp_err_t parlio_tx_unit_disable(parlio_tx_unit_handle_t unit);
+
+/**
+ * @brief Type of Parallel IO TX done event data
+ */
+typedef struct {
+} parlio_tx_done_event_data_t;
+
+/**
+ * @brief Prototype of parlio tx event callback
+ * @param[in] tx_unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @param[in] edata Point to Parallel IO TX event data. The lifecycle of this pointer memory is inside this function,
+ *                  user should copy it into static memory if used outside this function.
+ * @param[in] user_ctx User registered context, passed from `parlio_tx_unit_register_event_callbacks`
+ *
+ * @return Whether a high priority task has been waken up by this callback function
+ */
+typedef bool (*parlio_tx_done_callback_t)(parlio_tx_unit_handle_t tx_unit, const parlio_tx_done_event_data_t *edata, void *user_ctx);
+
+/**
+ * @brief Group of Parallel IO TX callbacks
+ * @note The callbacks are all running under ISR environment
+ * @note When CONFIG_PARLIO_ISR_IRAM_SAFE is enabled, the callback itself and functions called by it should be placed in IRAM.
+ *       The variables used in the function should be in the SRAM as well.
+ */
+typedef struct {
+    parlio_tx_done_callback_t on_trans_done; /*!< Event callback, invoked when one transmission is finished */
+} parlio_tx_event_callbacks_t;
+
+/**
+ * @brief Set event callbacks for Parallel IO TX unit
+ *
+ * @note User can deregister a previously registered callback by calling this function and setting the callback member in the `cbs` structure to NULL.
+ * @note When CONFIG_PARLIO_ISR_IRAM_SAFE is enabled, the callback itself and functions called by it should be placed in IRAM.
+ *       The variables used in the function should be in the SRAM as well. The `user_data` should also reside in SRAM.
+ *
+ * @param[in] tx_unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @param[in] cbs Group of callback functions
+ * @param[in] user_data User data, which will be passed to callback functions directly
+ * @return
+ *      - ESP_OK: Set event callbacks successfully
+ *      - ESP_ERR_INVALID_ARG: Set event callbacks failed because of invalid argument
+ *      - ESP_FAIL: Set event callbacks failed because of other error
+ */
+esp_err_t parlio_tx_unit_register_event_callbacks(parlio_tx_unit_handle_t tx_unit, const parlio_tx_event_callbacks_t *cbs, void *user_data);
+
+/**
+ * @brief Parallel IO transmit configuration
+ */
+typedef struct {
+    uint32_t idle_value; /*!< The value on the data line when the parallel IO is in idle state */
+} parlio_transmit_config_t;
+
+/**
+ * @brief Transmit data on by Parallel IO TX unit
+ *
+ * @note After the function returns, it doesn't mean the transaction is finished. This function only constructs a transcation structure and push into a queue.
+ *
+ * @param[in] tx_unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @param[in] payload Pointer to the data to be transmitted
+ * @param[in] payload_bits Length of the data to be transmitted, in bits
+ * @param[in] config Transmit configuration
+ * @return
+ *      - ESP_OK: Transmit data successfully
+ *      - ESP_ERR_INVALID_ARG: Transmit data failed because of invalid argument
+ *      - ESP_ERR_INVALID_STATE: Transmit data failed because the Parallel IO TX unit is not enabled
+ *      - ESP_FAIL: Transmit data failed because of other error
+ */
+esp_err_t parlio_tx_unit_transmit(parlio_tx_unit_handle_t tx_unit, const void *payload, size_t payload_bits, const parlio_transmit_config_t *config);
+
+/**
+ * @brief Wait for all pending TX transactions done
+ *
+ * @param[in] tx_unit Parallel IO TX unit that created by `parlio_new_tx_unit`
+ * @param[in] timeout_ms Timeout in milliseconds, `-1` means to wait forever
+ * @return
+ *      - ESP_OK: All pending TX transactions is finished and recycled
+ *      - ESP_ERR_INVALID_ARG: Wait for all pending TX transactions done failed because of invalid argument
+ *      - ESP_ERR_TIMEOUT: Wait for all pending TX transactions done timeout
+ *      - ESP_FAIL: Wait for all pending TX transactions done failed because of other error
+ */
+esp_err_t parlio_tx_unit_wait_all_done(parlio_tx_unit_handle_t tx_unit, int timeout_ms);
+
+#ifdef __cplusplus
+}
+#endif

+ 26 - 0
components/driver/parlio/include/driver/parlio_types.h

@@ -0,0 +1,26 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#pragma once
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "esp_err.h"
+#include "hal/parlio_types.h"
+#include "hal/gpio_types.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief Type of Parallel IO TX unit handle
+ */
+typedef struct parlio_tx_unit_t *parlio_tx_unit_handle_t;
+
+#ifdef __cplusplus
+}
+#endif

+ 89 - 0
components/driver/parlio/parlio_common.c

@@ -0,0 +1,89 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include <sys/lock.h>
+#include "sdkconfig.h"
+#if CONFIG_PARLIO_ENABLE_DEBUG_LOG
+// The local log level must be defined before including esp_log.h
+// Set the maximum log level for this source file
+#define LOG_LOCAL_LEVEL ESP_LOG_DEBUG
+#endif
+#include "esp_log.h"
+#include "esp_check.h"
+#include "clk_ctrl_os.h"
+#include "soc/rtc.h"
+#include "soc/parlio_periph.h"
+#include "hal/parlio_ll.h"
+#include "esp_private/esp_clk.h"
+#include "esp_private/periph_ctrl.h"
+#include "parlio_private.h"
+
+static const char *TAG = "parlio";
+
+typedef struct parlio_platform_t {
+    _lock_t mutex;                             // platform level mutex lock
+    parlio_group_t *groups[SOC_PARLIO_GROUPS]; // array of parallel IO group instances
+    int group_ref_counts[SOC_PARLIO_GROUPS];   // reference count used to protect group install/uninstall
+} parlio_platform_t;
+
+static parlio_platform_t s_platform; // singleton platform
+
+parlio_group_t *parlio_acquire_group_handle(int group_id)
+{
+    bool new_group = false;
+    parlio_group_t *group = NULL;
+
+    // prevent install parlio group concurrently
+    _lock_acquire(&s_platform.mutex);
+    if (!s_platform.groups[group_id]) {
+        group = heap_caps_calloc(1, sizeof(parlio_group_t), PARLIO_MEM_ALLOC_CAPS);
+        if (group) {
+            new_group = true;
+            s_platform.groups[group_id] = group;
+            group->group_id = group_id;
+            group->spinlock = (portMUX_TYPE)portMUX_INITIALIZER_UNLOCKED;
+            // enable APB access PARLIO registers
+            periph_module_enable(parlio_periph_signals.groups[group_id].module);
+            periph_module_reset(parlio_periph_signals.groups[group_id].module);
+            // hal layer initialize
+            parlio_hal_init(&group->hal);
+        }
+    } else { // group already install
+        group = s_platform.groups[group_id];
+    }
+    if (group) {
+        // someone acquired the group handle means we have a new object that refer to this group
+        s_platform.group_ref_counts[group_id]++;
+    }
+    _lock_release(&s_platform.mutex);
+
+    if (new_group) {
+        ESP_LOGD(TAG, "new group(%d) at %p", group_id, group);
+    }
+    return group;
+}
+
+void parlio_release_group_handle(parlio_group_t *group)
+{
+    int group_id = group->group_id;
+    bool do_deinitialize = false;
+
+    _lock_acquire(&s_platform.mutex);
+    s_platform.group_ref_counts[group_id]--;
+    if (s_platform.group_ref_counts[group_id] == 0) {
+        do_deinitialize = true;
+        s_platform.groups[group_id] = NULL;
+        // hal layer deinitialize
+        parlio_hal_deinit(&group->hal);
+        periph_module_disable(parlio_periph_signals.groups[group_id].module);
+        free(group);
+    }
+    _lock_release(&s_platform.mutex);
+
+    if (do_deinitialize) {
+        ESP_LOGD(TAG, "del group(%d)", group_id);
+    }
+}

+ 70 - 0
components/driver/parlio/parlio_private.h

@@ -0,0 +1,70 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#pragma once
+
+#include "sdkconfig.h"
+#include "freertos/FreeRTOS.h"
+#include "soc/soc_caps.h"
+#include "hal/parlio_types.h"
+#include "hal/parlio_hal.h"
+#include "esp_heap_caps.h"
+#include "driver/parlio_types.h"
+
+#if CONFIG_PARLIO_ISR_IRAM_SAFE
+#define PARLIO_MEM_ALLOC_CAPS    (MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT)
+#else
+#define PARLIO_MEM_ALLOC_CAPS    MALLOC_CAP_DEFAULT
+#endif
+
+#if SOC_PARLIO_TX_RX_SHARE_INTERRUPT
+#define PARLIO_INTR_ALLOC_FLAG_SHARED ESP_INTR_FLAG_SHARED
+#else
+#define PARLIO_INTR_ALLOC_FLAG_SHARED 0
+#endif
+
+#if CONFIG_PARLIO_ISR_IRAM_SAFE
+#define PARLIO_INTR_ALLOC_FLAG   (ESP_INTR_FLAG_LOWMED | PARLIO_INTR_ALLOC_FLAG_SHARED | ESP_INTR_FLAG_IRAM)
+#else
+#define PARLIO_INTR_ALLOC_FLAG   (ESP_INTR_FLAG_LOWMED | PARLIO_INTR_ALLOC_FLAG_SHARED)
+#endif
+
+#define PARLIO_PM_LOCK_NAME_LEN_MAX 16
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+enum {
+    PARLIO_TX_QUEUE_READY,
+    PARLIO_TX_QUEUE_PROGRESS,
+    PARLIO_TX_QUEUE_COMPLETE,
+    PARLIO_TX_QUEUE_MAX,
+};
+
+typedef enum {
+    PARLIO_TX_FSM_INIT_WAIT,
+    PARLIO_TX_FSM_INIT,
+    PARLIO_TX_FSM_ENABLE_WAIT,
+    PARLIO_TX_FSM_ENABLE,
+    PARLIO_TX_FSM_RUN_WAIT,
+    PARLIO_TX_FSM_RUN,
+} parlio_tx_fsm_t;
+
+typedef struct parlio_group_t {
+    int group_id;             // group ID, index from 0
+    portMUX_TYPE spinlock;    // to protect per-group register level concurrent access
+    parlio_hal_context_t hal; // hal layer for each group
+    parlio_tx_unit_handle_t tx_units[SOC_PARLIO_TX_UNITS_PER_GROUP]; // tx unit handles
+} parlio_group_t;
+
+parlio_group_t *parlio_acquire_group_handle(int group_id);
+
+void parlio_release_group_handle(parlio_group_t *group);
+
+#ifdef __cplusplus
+}
+#endif

+ 628 - 0
components/driver/parlio/parlio_tx.c

@@ -0,0 +1,628 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <stdatomic.h>
+#include <sys/cdefs.h>
+#include <sys/param.h>
+#include "sdkconfig.h"
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "freertos/queue.h"
+#if CONFIG_PARLIO_ENABLE_DEBUG_LOG
+// The local log level must be defined before including esp_log.h
+// Set the maximum log level for this source file
+#define LOG_LOCAL_LEVEL ESP_LOG_DEBUG
+#endif
+#include "esp_log.h"
+#include "esp_check.h"
+#include "esp_attr.h"
+#include "esp_err.h"
+#include "esp_rom_gpio.h"
+#include "esp_intr_alloc.h"
+#include "esp_pm.h"
+#include "soc/parlio_periph.h"
+#include "hal/parlio_ll.h"
+#include "hal/gpio_hal.h"
+#include "hal/dma_types.h"
+#include "driver/gpio.h"
+#include "driver/parlio_tx.h"
+#include "parlio_private.h"
+#include "esp_memory_utils.h"
+#include "clk_tree.h"
+#include "esp_private/gdma.h"
+
+static const char *TAG = "parlio-tx";
+
+typedef struct {
+    uint32_t idle_value; // Parallel IO bus idle value
+    const void *payload; // payload to be transmitted
+    size_t payload_bits; // payload size in bits
+} parlio_tx_trans_desc_t;
+
+typedef struct parlio_tx_unit_t {
+    int unit_id;           // unit id
+    size_t data_width;     // data width
+    parlio_group_t *group; // group handle
+    intr_handle_t intr;    // allocated interrupt handle
+    esp_pm_lock_handle_t pm_lock;   // power management lock
+    gdma_channel_handle_t dma_chan; // DMA channel
+#if CONFIG_PM_ENABLE
+    char pm_lock_name[PARLIO_PM_LOCK_NAME_LEN_MAX]; // pm lock name
+#endif
+    portMUX_TYPE spinlock;     // prevent resource accessing by user and interrupt concurrently
+    uint32_t out_clk_freq_hz;  // output clock frequency
+    size_t max_transfer_bits;  // maximum transfer size in bits
+    size_t queue_depth;        // size of transaction queue
+    size_t num_trans_inflight; // indicates the number of transactions that are undergoing but not recycled to ready_queue
+    void *queues_storage;      // storage of transaction queues
+    QueueHandle_t trans_queues[PARLIO_TX_QUEUE_MAX]; // transaction queues
+    StaticQueue_t trans_queue_structs[PARLIO_TX_QUEUE_MAX]; // memory to store the static structure for trans_queues
+    parlio_tx_trans_desc_t *cur_trans; // points to current transaction
+    uint32_t idle_value_mask;          // mask of idle value
+    _Atomic parlio_tx_fsm_t fsm;       // Driver FSM state
+    parlio_tx_done_callback_t on_trans_done; // callback function when the transmission is done
+    void *user_data;                   // user data passed to the callback function
+    dma_descriptor_t *dma_nodes; // DMA descriptor nodes
+    parlio_tx_trans_desc_t trans_desc_pool[];   // transaction descriptor pool
+} parlio_tx_unit_t;
+
+static void parlio_tx_default_isr(void *args);
+
+static esp_err_t parlio_tx_register_to_group(parlio_tx_unit_t *unit)
+{
+    parlio_group_t *group = NULL;
+    int unit_id = -1;
+    for (int i = 0; i < SOC_PARLIO_GROUPS; i++) {
+        group = parlio_acquire_group_handle(i);
+        ESP_RETURN_ON_FALSE(group, ESP_ERR_NO_MEM, TAG, "no memory for group (%d)", i);
+        portENTER_CRITICAL(&group->spinlock);
+        for (int j = 0; j < SOC_PARLIO_TX_UNITS_PER_GROUP; j++) {
+            if (group->tx_units[j] == NULL) {
+                group->tx_units[j] = unit;
+                unit_id = j;
+                break;
+            }
+        }
+        portEXIT_CRITICAL(&group->spinlock);
+        if (unit_id < 0) {
+            // didn't find a free unit slot in the group
+            parlio_release_group_handle(group);
+            group = NULL;
+        } else {
+            unit->unit_id = unit_id;
+            unit->group = group;
+            break;
+        }
+    }
+    ESP_RETURN_ON_FALSE(unit_id >= 0, ESP_ERR_NOT_FOUND, TAG, "no free tx unit");
+    return ESP_OK;
+}
+
+static void parlio_tx_unregister_to_group(parlio_tx_unit_t *unit, parlio_group_t *group)
+{
+    portENTER_CRITICAL(&group->spinlock);
+    group->tx_units[unit->unit_id] = NULL;
+    portEXIT_CRITICAL(&group->spinlock);
+    // the tx unit has a reference of the group, release it now
+    parlio_release_group_handle(group);
+}
+
+static esp_err_t parlio_tx_create_trans_queue(parlio_tx_unit_t *tx_unit, const parlio_tx_unit_config_t *config)
+{
+    tx_unit->queue_depth = config->trans_queue_depth;
+    // the queue only saves transaction description pointers
+    tx_unit->queues_storage = heap_caps_calloc(config->trans_queue_depth * PARLIO_TX_QUEUE_MAX, sizeof(parlio_tx_trans_desc_t *), PARLIO_MEM_ALLOC_CAPS);
+    ESP_RETURN_ON_FALSE(tx_unit->queues_storage, ESP_ERR_NO_MEM, TAG, "no mem for queue storage");
+    parlio_tx_trans_desc_t **pp_trans_desc = (parlio_tx_trans_desc_t **)tx_unit->queues_storage;
+    for (int i = 0; i < PARLIO_TX_QUEUE_MAX; i++) {
+        tx_unit->trans_queues[i] = xQueueCreateStatic(config->trans_queue_depth, sizeof(parlio_tx_trans_desc_t *),
+                                   (uint8_t *)pp_trans_desc, &tx_unit->trans_queue_structs[i]);
+        pp_trans_desc += config->trans_queue_depth;
+        // because trans_queue_structs is guaranteed to be non-NULL, so the trans_queues will also not be NULL
+        assert(tx_unit->trans_queues[i]);
+    }
+    // initialize the ready queue
+    parlio_tx_trans_desc_t *p_trans_desc = NULL;
+    for (int i = 0; i < config->trans_queue_depth; i++) {
+        p_trans_desc = &tx_unit->trans_desc_pool[i];
+        ESP_RETURN_ON_FALSE(xQueueSend(tx_unit->trans_queues[PARLIO_TX_QUEUE_READY], &p_trans_desc, 0) == pdTRUE,
+                            ESP_ERR_INVALID_STATE, TAG, "ready queue full");
+    }
+    return ESP_OK;
+}
+
+static esp_err_t parlio_destroy_tx_unit(parlio_tx_unit_t *tx_unit)
+{
+    if (tx_unit->intr) {
+        ESP_RETURN_ON_ERROR(esp_intr_free(tx_unit->intr), TAG, "delete interrupt service failed");
+    }
+    if (tx_unit->pm_lock) {
+        ESP_RETURN_ON_ERROR(esp_pm_lock_delete(tx_unit->pm_lock), TAG, "delete pm lock failed");
+    }
+    if (tx_unit->dma_chan) {
+        ESP_RETURN_ON_ERROR(gdma_disconnect(tx_unit->dma_chan), TAG, "disconnect dma channel failed");
+        ESP_RETURN_ON_ERROR(gdma_del_channel(tx_unit->dma_chan), TAG, "delete dma channel failed");
+    }
+    for (int i = 0; i < PARLIO_TX_QUEUE_MAX; i++) {
+        if (tx_unit->trans_queues[i]) {
+            vQueueDelete(tx_unit->trans_queues[i]);
+        }
+    }
+    if (tx_unit->group) {
+        // de-register from group
+        parlio_tx_unregister_to_group(tx_unit, tx_unit->group);
+    }
+    free(tx_unit->queues_storage);
+    free(tx_unit->dma_nodes);
+    free(tx_unit);
+    return ESP_OK;
+}
+
+static esp_err_t parlio_tx_unit_configure_gpio(parlio_tx_unit_t *tx_unit, const parlio_tx_unit_config_t *config)
+{
+    int group_id = tx_unit->group->group_id;
+    int unit_id = tx_unit->unit_id;
+    gpio_config_t gpio_conf = {
+        .intr_type = GPIO_INTR_DISABLE,
+        .mode = config->flags.io_loop_back ? GPIO_MODE_INPUT_OUTPUT : GPIO_MODE_OUTPUT,
+        .pull_down_en = false,
+        .pull_up_en = true,
+    };
+
+    // connect peripheral signals via GPIO matrix
+    for (size_t i = 0; i < config->data_width; i++) {
+        if (config->data_gpio_nums[i] >= 0) {
+            gpio_conf.pin_bit_mask = BIT64(config->data_gpio_nums[i]);
+            ESP_RETURN_ON_ERROR(gpio_config(&gpio_conf), TAG, "config data GPIO failed");
+            esp_rom_gpio_connect_out_signal(config->data_gpio_nums[i],
+                                            parlio_periph_signals.groups[group_id].tx_units[unit_id].data_sigs[i], false, false);
+            gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[config->data_gpio_nums[i]], PIN_FUNC_GPIO);
+        }
+    }
+    // Note: the valid signal will override TXD[PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG]
+    if (config->valid_gpio_num >= 0) {
+        gpio_conf.pin_bit_mask = BIT64(config->valid_gpio_num);
+        ESP_RETURN_ON_ERROR(gpio_config(&gpio_conf), TAG, "config valid GPIO failed");
+        esp_rom_gpio_connect_out_signal(config->valid_gpio_num,
+                                        parlio_periph_signals.groups[group_id].tx_units[unit_id].data_sigs[PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG],
+                                        false, false);
+        gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[config->valid_gpio_num], PIN_FUNC_GPIO);
+    }
+    if (config->clk_out_gpio_num >= 0) {
+        gpio_conf.pin_bit_mask = BIT64(config->clk_out_gpio_num);
+        ESP_RETURN_ON_ERROR(gpio_config(&gpio_conf), TAG, "config clk out GPIO failed");
+        esp_rom_gpio_connect_out_signal(config->clk_out_gpio_num,
+                                        parlio_periph_signals.groups[group_id].tx_units[unit_id].clk_out_sig, false, false);
+        gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[config->clk_out_gpio_num], PIN_FUNC_GPIO);
+    }
+    if (config->clk_in_gpio_num >= 0) {
+        gpio_conf.mode = config->flags.io_loop_back ? GPIO_MODE_INPUT_OUTPUT : GPIO_MODE_INPUT;
+        gpio_conf.pin_bit_mask = BIT64(config->clk_in_gpio_num);
+        ESP_RETURN_ON_ERROR(gpio_config(&gpio_conf), TAG, "config clk in GPIO failed");
+        esp_rom_gpio_connect_in_signal(config->clk_in_gpio_num,
+                                       parlio_periph_signals.groups[group_id].tx_units[unit_id].clk_in_sig, false);
+        gpio_hal_iomux_func_sel(GPIO_PIN_MUX_REG[config->clk_in_gpio_num], PIN_FUNC_GPIO);
+    }
+    return ESP_OK;
+}
+
+static esp_err_t parlio_tx_unit_init_dma(parlio_tx_unit_t *tx_unit)
+{
+    gdma_channel_alloc_config_t dma_chan_config = {
+        .direction = GDMA_CHANNEL_DIRECTION_TX,
+    };
+    ESP_RETURN_ON_ERROR(gdma_new_channel(&dma_chan_config, &tx_unit->dma_chan), TAG, "allocate TX DMA channel failed");
+    gdma_connect(tx_unit->dma_chan, GDMA_MAKE_TRIGGER(GDMA_TRIG_PERIPH_PARLIO, 0));
+    gdma_strategy_config_t gdma_strategy_conf = {
+        .auto_update_desc = true,
+        .owner_check = true,
+    };
+    gdma_apply_strategy(tx_unit->dma_chan, &gdma_strategy_conf);
+    return ESP_OK;
+}
+
+static esp_err_t parlio_select_periph_clock(parlio_tx_unit_t *tx_unit, const parlio_tx_unit_config_t *config)
+{
+    parlio_hal_context_t *hal = &tx_unit->group->hal;
+    // parlio_ll_clock_source_t and parlio_clock_source_t are binary compatible if the clock source is from internal
+    parlio_ll_clock_source_t clk_src = (parlio_ll_clock_source_t)(config->clk_src);
+    uint32_t periph_src_clk_hz = 0;
+    // if the source clock is input from the GPIO, then we're in the slave mode
+    if (config->clk_in_gpio_num >= 0) {
+        clk_src = PARLIO_LL_CLK_SRC_PAD;
+        periph_src_clk_hz = config->input_clk_src_freq_hz;
+    } else {
+        // get the internal clock source frequency
+        clk_tree_src_get_freq_hz((soc_module_clk_t)clk_src, CLK_TREE_SRC_FREQ_PRECISION_CACHED, &periph_src_clk_hz);
+    }
+    ESP_RETURN_ON_FALSE(periph_src_clk_hz, ESP_ERR_INVALID_ARG, TAG, "invalid clock source frequency");
+
+#if CONFIG_PM_ENABLE
+    if (clk_src != PARLIO_LL_CLK_SRC_PAD) {
+        // XTAL and PLL clock source will be turned off in light sleep, so we need to create a NO_LIGHT_SLEEP lock
+        sprintf(tx_unit->pm_lock_name, "parlio_tx_%d_%d", tx_unit->group->group_id, tx_unit->unit_id); // e.g. parlio_tx_0_0
+        esp_err_t ret  = esp_pm_lock_create(ESP_PM_NO_LIGHT_SLEEP, 0, tx_unit->pm_lock_name, &tx_unit->pm_lock);
+        ESP_RETURN_ON_ERROR(ret, TAG, "create NO_LIGHT_SLEEP lock failed");
+    }
+#endif
+
+    parlio_ll_tx_set_clock_source(hal->regs, clk_src);
+    // set clock division, round up
+    uint32_t div = (periph_src_clk_hz + config->output_clk_freq_hz - 1) / config->output_clk_freq_hz;
+    parlio_ll_tx_set_clock_div(hal->regs, div);
+    // precision lost due to division, calculate the real frequency
+    tx_unit->out_clk_freq_hz = periph_src_clk_hz / div;
+    if (tx_unit->out_clk_freq_hz != config->output_clk_freq_hz) {
+        ESP_LOGW(TAG, "precision loss, real output frequency: %"PRIu32, tx_unit->out_clk_freq_hz);
+    }
+
+    return ESP_OK;
+}
+
+esp_err_t parlio_new_tx_unit(const parlio_tx_unit_config_t *config, parlio_tx_unit_handle_t *ret_unit)
+{
+#if CONFIG_PARLIO_ENABLE_DEBUG_LOG
+    esp_log_level_set(TAG, ESP_LOG_DEBUG);
+#endif
+    esp_err_t ret = ESP_OK;
+    parlio_tx_unit_t *unit = NULL;
+    ESP_GOTO_ON_FALSE(config && ret_unit, ESP_ERR_INVALID_ARG, err, TAG, "invalid argument");
+    size_t data_width = config->data_width;
+    // data_width must be power of 2 and less than or equal to SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH
+    ESP_GOTO_ON_FALSE(data_width && (data_width <= SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH) && ((data_width & (data_width - 1)) == 0),
+                      ESP_ERR_INVALID_ARG, err, TAG, "invalid data width");
+    // data_width must not conflict with the valid signal
+    ESP_GOTO_ON_FALSE(!(config->valid_gpio_num >= 0 && data_width > PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG),
+                      ESP_ERR_INVALID_ARG, err, TAG, "valid signal conflicts with data signal");
+    ESP_GOTO_ON_FALSE(config->max_transfer_size && config->max_transfer_size <= PARLIO_LL_TX_MAX_BITS_PER_FRAME / 8,
+                      ESP_ERR_INVALID_ARG, err, TAG, "invalid max transfer size");
+#if SOC_PARLIO_TX_CLK_SUPPORT_GATING
+    // clock gating is controlled by either the MSB bit of data bus or the valid signal
+    ESP_GOTO_ON_FALSE(!(config->flags.clk_gate_en && config->valid_gpio_num < 0 && config->data_width <= PARLIO_LL_TX_DATA_LINE_AS_CLK_GATE),
+                      ESP_ERR_INVALID_ARG, err, TAG, "no gpio can control the clock gating");
+#else
+    ESP_GOTO_ON_FALSE(config->flags.clk_gate_en == 0, ESP_ERR_NOT_SUPPORTED, err, TAG, "clock gating is not supported");
+#endif // SOC_PARLIO_TX_CLK_SUPPORT_GATING
+
+    // malloc unit memory
+    unit = heap_caps_calloc(1, sizeof(parlio_tx_unit_t) + sizeof(parlio_tx_trans_desc_t) * config->trans_queue_depth, PARLIO_MEM_ALLOC_CAPS);
+    ESP_GOTO_ON_FALSE(unit, ESP_ERR_NO_MEM, err, TAG, "no memory for tx unit");
+    size_t dma_nodes_num = config->max_transfer_size / DMA_DESCRIPTOR_BUFFER_MAX_SIZE + 1;
+    // DMA descriptors must be placed in internal SRAM
+    unit->dma_nodes = heap_caps_calloc(dma_nodes_num, sizeof(dma_descriptor_t), MALLOC_CAP_INTERNAL | MALLOC_CAP_DMA);
+    ESP_GOTO_ON_FALSE(unit->dma_nodes, ESP_ERR_NO_MEM, err, TAG, "no memory for DMA nodes");
+    unit->max_transfer_bits = config->max_transfer_size * 8;
+
+    unit->data_width = data_width;
+    //create transaction queue
+    ESP_GOTO_ON_ERROR(parlio_tx_create_trans_queue(unit, config), err, TAG, "create transaction queue failed");
+
+    // register the unit to a group
+    ESP_GOTO_ON_ERROR(parlio_tx_register_to_group(unit), err, TAG, "register unit to group failed");
+    parlio_group_t *group = unit->group;
+    parlio_hal_context_t *hal = &group->hal;
+    // select the clock source
+    ESP_GOTO_ON_ERROR(parlio_select_periph_clock(unit, config), err, TAG, "set clock source failed");
+
+    // install interrupt service
+    int isr_flags = PARLIO_INTR_ALLOC_FLAG;
+    ret = esp_intr_alloc_intrstatus(parlio_periph_signals.groups[group->group_id].tx_irq_id, isr_flags,
+                                    (uint32_t)parlio_ll_get_interrupt_status_reg(hal->regs),
+                                    PARLIO_LL_EVENT_TX_EOF, parlio_tx_default_isr, unit, &unit->intr);
+    ESP_GOTO_ON_ERROR(ret, err, TAG, "install interrupt failed");
+
+    // install DMA service
+    ESP_GOTO_ON_ERROR(parlio_tx_unit_init_dma(unit), err, TAG, "install tx DMA failed");
+
+    // reset fifo and core clock domain
+    parlio_ll_tx_reset_clock(hal->regs);
+    parlio_ll_tx_reset_fifo(hal->regs);
+    // stop output clock
+    parlio_ll_tx_enable_clock(hal->regs, false);
+    // clock gating
+    parlio_ll_tx_enable_clock_gating(hal->regs, config->flags.clk_gate_en);
+    // set data width
+    parlio_ll_tx_set_bus_width(hal->regs, data_width);
+    unit->idle_value_mask = (1 << data_width) - 1;
+    // whether to use the valid signal
+    if (config->valid_gpio_num >= 0) {
+        parlio_ll_tx_treat_msb_as_valid(hal->regs, true);
+        unit->idle_value_mask &= ~(1 << PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG);
+    } else {
+        parlio_ll_tx_treat_msb_as_valid(hal->regs, false);
+    }
+    // set data byte packing order
+    if (data_width < 8) {
+        parlio_ll_tx_set_bit_pack_order(hal->regs, config->bit_pack_order);
+    }
+    // set sample clock edge
+    parlio_ll_tx_set_sample_clock_edge(hal->regs, config->sample_edge);
+
+    // clear any pending interrupt
+    parlio_ll_clear_interrupt_status(hal->regs, PARLIO_LL_EVENT_TX_MASK);
+
+    // GPIO Matrix/MUX configuration
+    ESP_GOTO_ON_ERROR(parlio_tx_unit_configure_gpio(unit, config), err, TAG, "configure gpio failed");
+
+    portMUX_INITIALIZE(&unit->spinlock);
+    atomic_init(&unit->fsm, PARLIO_TX_FSM_INIT);
+    // return TX unit handle
+    *ret_unit = unit;
+    ESP_LOGD(TAG, "new tx unit(%d,%d) at %p, out clk=%"PRIu32"Hz, queue_depth=%zu, idle_mask=%"PRIx32,
+             group->group_id, unit->unit_id, unit, unit->out_clk_freq_hz, unit->queue_depth, unit->idle_value_mask);
+    return ESP_OK;
+
+err:
+    if (unit) {
+        parlio_destroy_tx_unit(unit);
+    }
+    return ret;
+}
+
+esp_err_t parlio_del_tx_unit(parlio_tx_unit_handle_t unit)
+{
+    ESP_RETURN_ON_FALSE(unit, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+    ESP_RETURN_ON_FALSE(atomic_load(&unit->fsm) == PARLIO_TX_FSM_INIT, ESP_ERR_INVALID_STATE, TAG, "unit not in init state");
+    ESP_LOGD(TAG, "del tx unit(%d,%d)", unit->group->group_id, unit->unit_id);
+    return parlio_destroy_tx_unit(unit);
+}
+
+static void IRAM_ATTR parlio_tx_mount_dma_data(dma_descriptor_t *desc_head, const void *buffer, size_t len)
+{
+    size_t prepared_length = 0;
+    uint8_t *data = (uint8_t *)buffer;
+    dma_descriptor_t *desc = desc_head;
+    while (len > DMA_DESCRIPTOR_BUFFER_MAX_SIZE) {
+        desc->dw0.suc_eof = 0; // not the end of the transaction
+        desc->dw0.size = DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
+        desc->dw0.length = DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
+        desc->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
+        desc->buffer = &data[prepared_length];
+        desc = desc->next; // move to next descriptor
+        prepared_length += DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
+        len -= DMA_DESCRIPTOR_BUFFER_MAX_SIZE;
+    }
+    if (len) {
+        desc->dw0.suc_eof = 1; // end of the transaction
+        desc->dw0.size = len;
+        desc->dw0.length = len;
+        desc->dw0.owner = DMA_DESCRIPTOR_BUFFER_OWNER_DMA;
+        desc->buffer = &data[prepared_length];
+        desc = desc->next; // move to next descriptor
+        prepared_length += len;
+    }
+}
+
+esp_err_t parlio_tx_unit_wait_all_done(parlio_tx_unit_handle_t tx_unit, int timeout_ms)
+{
+    ESP_RETURN_ON_FALSE(tx_unit, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+    TickType_t wait_ticks = timeout_ms < 0 ? portMAX_DELAY : pdMS_TO_TICKS(timeout_ms);
+    // recycle all pending transactions
+    parlio_tx_trans_desc_t *t = NULL;
+    size_t num_trans_inflight = tx_unit->num_trans_inflight;
+    for (size_t i = 0; i < num_trans_inflight; i++) {
+        ESP_RETURN_ON_FALSE(xQueueReceive(tx_unit->trans_queues[PARLIO_TX_QUEUE_COMPLETE], &t, wait_ticks) == pdTRUE,
+                            ESP_ERR_TIMEOUT, TAG, "flush timeout");
+        ESP_RETURN_ON_FALSE(xQueueSend(tx_unit->trans_queues[PARLIO_TX_QUEUE_READY], &t, 0) == pdTRUE,
+                            ESP_ERR_INVALID_STATE, TAG, "ready queue full");
+        tx_unit->num_trans_inflight--;
+    }
+    return ESP_OK;
+}
+
+esp_err_t parlio_tx_unit_register_event_callbacks(parlio_tx_unit_handle_t tx_unit, const parlio_tx_event_callbacks_t *cbs, void *user_data)
+{
+    ESP_RETURN_ON_FALSE(tx_unit && cbs, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+
+#if CONFIG_PARLIO_ISR_IRAM_SAFE
+    if (cbs->on_trans_done) {
+        ESP_RETURN_ON_FALSE(esp_ptr_in_iram(cbs->on_trans_done), ESP_ERR_INVALID_ARG, TAG, "on_trans_done callback not in IRAM");
+    }
+    if (user_data) {
+        ESP_RETURN_ON_FALSE(esp_ptr_internal(user_data), ESP_ERR_INVALID_ARG, TAG, "user context not in internal RAM");
+    }
+#endif
+
+    tx_unit->on_trans_done = cbs->on_trans_done;
+    tx_unit->user_data = user_data;
+    return ESP_OK;
+}
+
+static void IRAM_ATTR parlio_tx_do_transaction(parlio_tx_unit_t *tx_unit, parlio_tx_trans_desc_t *t)
+{
+    parlio_hal_context_t *hal = &tx_unit->group->hal;
+
+    tx_unit->cur_trans = t;
+
+    // DMA transfer data based on bytes not bits, so convert the bit length to bytes, round up
+    parlio_tx_mount_dma_data(tx_unit->dma_nodes, t->payload, (t->payload_bits + 7) / 8);
+
+    parlio_ll_tx_reset_fifo(hal->regs);
+    parlio_ll_tx_reset_clock(hal->regs);
+    parlio_ll_tx_set_idle_data_value(hal->regs, t->idle_value);
+    parlio_ll_tx_set_trans_bit_len(hal->regs, t->payload_bits);
+
+    gdma_start(tx_unit->dma_chan, (intptr_t)tx_unit->dma_nodes);
+    // wait until the data goes from the DMA to TX unit's FIFO
+    while (parlio_ll_tx_is_ready(hal->regs) == false);
+    // turn on the core clock after we start the TX unit
+    parlio_ll_tx_start(hal->regs, true);
+    parlio_ll_tx_enable_clock(hal->regs, true);
+}
+
+esp_err_t parlio_tx_unit_enable(parlio_tx_unit_handle_t tx_unit)
+{
+    ESP_RETURN_ON_FALSE(tx_unit, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+    parlio_tx_fsm_t expected_fsm = PARLIO_TX_FSM_INIT;
+    if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_ENABLE_WAIT)) {
+        // acquire power management lock
+        if (tx_unit->pm_lock) {
+            esp_pm_lock_acquire(tx_unit->pm_lock);
+        }
+        parlio_hal_context_t *hal = &tx_unit->group->hal;
+        parlio_ll_enable_interrupt(hal->regs, PARLIO_LL_EVENT_TX_EOF, true);
+        atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_ENABLE);
+    } else {
+        ESP_RETURN_ON_FALSE(false, ESP_ERR_INVALID_STATE, TAG, "unit not in init state");
+    }
+
+    // check if we need to start one pending transaction
+    parlio_tx_trans_desc_t *t = NULL;
+    expected_fsm = PARLIO_TX_FSM_ENABLE;
+    if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_RUN_WAIT)) {
+        // check if we need to start one transaction
+        if (xQueueReceive(tx_unit->trans_queues[PARLIO_TX_QUEUE_PROGRESS], &t, 0) == pdTRUE) {
+            atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_RUN);
+            parlio_tx_do_transaction(tx_unit, t);
+        } else {
+            atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_ENABLE);
+        }
+    }
+
+    return ESP_OK;
+}
+
+esp_err_t parlio_tx_unit_disable(parlio_tx_unit_handle_t tx_unit)
+{
+    ESP_RETURN_ON_FALSE(tx_unit, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+    bool valid_state = false;
+    // check the supported states, and switch to intermediate state: INIT_WAIT
+    parlio_tx_fsm_t expected_fsm = PARLIO_TX_FSM_ENABLE;
+    if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_INIT_WAIT)) {
+        valid_state = true;
+    }
+    expected_fsm = PARLIO_TX_FSM_RUN;
+    if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_INIT_WAIT)) {
+        valid_state = true;
+        assert(tx_unit->cur_trans);
+        // recycle the interrupted transaction
+        if (xQueueSend(tx_unit->trans_queues[PARLIO_TX_QUEUE_COMPLETE], &tx_unit->cur_trans, 0) == pdFALSE) {
+            // this should never happen
+            valid_state = false;
+        }
+        tx_unit->cur_trans = NULL;
+    }
+    ESP_RETURN_ON_FALSE(valid_state, ESP_ERR_INVALID_STATE, TAG, "unit can't be disabled in state %d", expected_fsm);
+
+    // stop the TX engine
+    parlio_hal_context_t *hal = &tx_unit->group->hal;
+    gdma_stop(tx_unit->dma_chan);
+    parlio_ll_tx_start(hal->regs, false);
+    parlio_ll_enable_interrupt(hal->regs, PARLIO_LL_EVENT_TX_EOF, false);
+
+    // release power management lock
+    if (tx_unit->pm_lock) {
+        esp_pm_lock_release(tx_unit->pm_lock);
+    }
+
+    // finally we switch to the INIT state
+    atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_INIT);
+
+    return ESP_OK;
+}
+
+esp_err_t parlio_tx_unit_transmit(parlio_tx_unit_handle_t tx_unit, const void *payload, size_t payload_bits, const parlio_transmit_config_t *config)
+{
+    ESP_RETURN_ON_FALSE(tx_unit && payload && payload_bits, ESP_ERR_INVALID_ARG, TAG, "invalid argument");
+    ESP_RETURN_ON_FALSE((payload_bits % tx_unit->data_width) == 0, ESP_ERR_INVALID_ARG, TAG, "payload bit length must align to bus width");
+    ESP_RETURN_ON_FALSE(payload_bits <= tx_unit->max_transfer_bits, ESP_ERR_INVALID_ARG, TAG, "payload bit length too large");
+#if !SOC_PARLIO_TRANS_BIT_ALIGN
+    ESP_RETURN_ON_FALSE((payload_bits % 8) == 0, ESP_ERR_INVALID_ARG, TAG, "payload bit length must be multiple of 8");
+#endif // !SOC_PARLIO_TRANS_BIT_ALIGN
+
+    // acquire one transaction description from ready queue or complete queue
+    parlio_tx_trans_desc_t *t = NULL;
+    if (xQueueReceive(tx_unit->trans_queues[PARLIO_TX_QUEUE_READY], &t, 0) != pdTRUE) {
+        if (xQueueReceive(tx_unit->trans_queues[PARLIO_TX_QUEUE_COMPLETE], &t, 0) == pdTRUE) {
+            tx_unit->num_trans_inflight--;
+        }
+    }
+    ESP_RETURN_ON_FALSE(t, ESP_ERR_INVALID_STATE, TAG, "no free transaction descriptor, please consider increasing trans_queue_depth");
+
+    // fill in the transaction descriptor
+    memset(t, 0, sizeof(parlio_tx_trans_desc_t));
+    t->payload = payload;
+    t->payload_bits = payload_bits;
+    t->idle_value = config->idle_value & tx_unit->idle_value_mask;
+
+    // send the transaction descriptor to progress queue
+    ESP_RETURN_ON_FALSE(xQueueSend(tx_unit->trans_queues[PARLIO_TX_QUEUE_PROGRESS], &t, 0) == pdTRUE,
+                        ESP_ERR_INVALID_STATE, TAG, "failed to send transaction descriptor to progress queue");
+    tx_unit->num_trans_inflight++;
+
+    // check if we need to start one pending transaction
+    parlio_tx_fsm_t expected_fsm = PARLIO_TX_FSM_ENABLE;
+    if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_RUN_WAIT)) {
+        // check if we need to start one transaction
+        if (xQueueReceive(tx_unit->trans_queues[PARLIO_TX_QUEUE_PROGRESS], &t, 0) == pdTRUE) {
+            atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_RUN);
+            parlio_tx_do_transaction(tx_unit, t);
+        } else {
+            atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_ENABLE);
+        }
+    }
+
+    return ESP_OK;
+}
+
+static void IRAM_ATTR parlio_tx_default_isr(void *args)
+{
+    parlio_tx_unit_t *tx_unit = (parlio_tx_unit_t *)args;
+    parlio_group_t *group = tx_unit->group;
+    parlio_hal_context_t *hal = &group->hal;
+    BaseType_t high_task_woken = pdFALSE;
+    bool need_yield = false;
+
+    uint32_t status = parlio_ll_tx_get_interrupt_status(hal->regs);
+
+    if (status & PARLIO_LL_EVENT_TX_EOF) {
+        parlio_ll_clear_interrupt_status(hal->regs, PARLIO_LL_EVENT_TX_EOF);
+        parlio_ll_tx_enable_clock(hal->regs, false);
+        parlio_ll_tx_start(hal->regs, false);
+
+        parlio_tx_trans_desc_t *trans_desc = NULL;
+
+        parlio_tx_fsm_t expected_fsm = PARLIO_TX_FSM_RUN;
+        if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_ENABLE_WAIT)) {
+            trans_desc = tx_unit->cur_trans;
+            // move current finished transaction to the complete queue
+            xQueueSendFromISR(tx_unit->trans_queues[PARLIO_TX_QUEUE_COMPLETE], &trans_desc, &high_task_woken);
+            if (high_task_woken == pdTRUE) {
+                need_yield = true;
+            }
+            tx_unit->cur_trans = NULL;
+            atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_ENABLE);
+        }
+
+        // invoke callback
+        parlio_tx_done_callback_t done_cb = tx_unit->on_trans_done;
+        if (done_cb) {
+            if (done_cb(tx_unit, NULL, tx_unit->user_data)) {
+                need_yield = true;
+            }
+        }
+
+        // if the tx unit is till in enable state (i.e. not disabled by user), let's try start the next pending transaction
+        expected_fsm = PARLIO_TX_FSM_ENABLE;
+        if (atomic_compare_exchange_strong(&tx_unit->fsm, &expected_fsm, PARLIO_TX_FSM_RUN_WAIT)) {
+            if (xQueueReceiveFromISR(tx_unit->trans_queues[PARLIO_TX_QUEUE_PROGRESS], &trans_desc, &high_task_woken) == pdTRUE) {
+                atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_RUN);
+                parlio_tx_do_transaction(tx_unit, trans_desc);
+                if (high_task_woken == pdTRUE) {
+                    need_yield = true;
+                }
+            } else {
+                atomic_store(&tx_unit->fsm, PARLIO_TX_FSM_ENABLE);
+            }
+        }
+
+        if (need_yield) {
+            portYIELD_FROM_ISR();
+        }
+    }
+}

+ 2 - 2
components/driver/rmt/include/driver/rmt_types.h

@@ -42,7 +42,7 @@ typedef struct {
  * @brief Prototype of RMT event callback
  * @param[in] tx_chan RMT channel handle, created from `rmt_new_tx_channel()`
  * @param[in] edata Point to RMT event data. The lifecycle of this pointer memory is inside this function,
- *                  user should copy it into static memory if used outside this funcion.
+ *                  user should copy it into static memory if used outside this function.
  * @param[in] user_ctx User registered context, passed from `rmt_tx_register_event_callbacks()`
  *
  * @return Whether a high priority task has been waken up by this callback function
@@ -62,7 +62,7 @@ typedef struct {
  *
  * @param[in] rx_chan RMT channel handle, created from `rmt_new_rx_channel()`
  * @param[in] edata Point to RMT event data. The lifecycle of this pointer memory is inside this function,
- *                  user should copy it into static memory if used outside this funcion.
+ *                  user should copy it into static memory if used outside this function.
  * @param[in] user_ctx User registered context, passed from `rmt_rx_register_event_callbacks()`
  * @return Whether a high priority task has been waken up by this function
  */

+ 18 - 0
components/driver/test_apps/parlio/CMakeLists.txt

@@ -0,0 +1,18 @@
+# This is the project CMakeLists.txt file for the test subproject
+cmake_minimum_required(VERSION 3.16)
+
+include($ENV{IDF_PATH}/tools/cmake/project.cmake)
+project(parlio_test)
+
+if(CONFIG_COMPILER_DUMP_RTL_FILES)
+    add_custom_target(check_test_app_sections ALL
+                      COMMAND ${PYTHON} $ENV{IDF_PATH}/tools/ci/check_callgraph.py
+                      --rtl-dir ${CMAKE_BINARY_DIR}/esp-idf/driver/
+                      --elf-file ${CMAKE_BINARY_DIR}/parlio_test.elf
+                      find-refs
+                      --from-sections=.iram0.text
+                      --to-sections=.flash.text,.flash.rodata
+                      --exit-code
+                      DEPENDS ${elf}
+                      )
+endif()

+ 2 - 0
components/driver/test_apps/parlio/README.md

@@ -0,0 +1,2 @@
+| Supported Targets | ESP32-C6 | ESP32-H2 |
+| ----------------- | -------- | -------- |

+ 7 - 0
components/driver/test_apps/parlio/main/CMakeLists.txt

@@ -0,0 +1,7 @@
+set(srcs "test_app_main.c"
+         "test_parlio_tx.c")
+
+# In order for the cases defined by `TEST_CASE` to be linked into the final elf,
+# the component can be registered as WHOLE_ARCHIVE
+idf_component_register(SRCS ${srcs}
+                       WHOLE_ARCHIVE)

+ 51 - 0
components/driver/test_apps/parlio/main/test_app_main.c

@@ -0,0 +1,51 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "unity.h"
+#include "unity_test_runner.h"
+#include "esp_heap_caps.h"
+
+// Some resources are lazy allocated in pulse_cnt driver, the threshold is left for that case
+#define TEST_MEMORY_LEAK_THRESHOLD (-300)
+
+static size_t before_free_8bit;
+static size_t before_free_32bit;
+
+static void check_leak(size_t before_free, size_t after_free, const char *type)
+{
+    ssize_t delta = after_free - before_free;
+    printf("MALLOC_CAP_%s: Before %u bytes free, After %u bytes free (delta %d)\n", type, before_free, after_free, delta);
+    TEST_ASSERT_MESSAGE(delta >= TEST_MEMORY_LEAK_THRESHOLD, "memory leak");
+}
+
+void setUp(void)
+{
+    before_free_8bit = heap_caps_get_free_size(MALLOC_CAP_8BIT);
+    before_free_32bit = heap_caps_get_free_size(MALLOC_CAP_32BIT);
+}
+
+void tearDown(void)
+{
+    size_t after_free_8bit = heap_caps_get_free_size(MALLOC_CAP_8BIT);
+    size_t after_free_32bit = heap_caps_get_free_size(MALLOC_CAP_32BIT);
+    check_leak(before_free_8bit, after_free_8bit, "8BIT");
+    check_leak(before_free_32bit, after_free_32bit, "32BIT");
+}
+
+void app_main(void)
+{
+    //  ____                 _ _      _   ___ ___    _____         _
+    // |  _ \ __ _ _ __ __ _| | | ___| | |_ _/ _ \  |_   _|__  ___| |_
+    // | |_) / _` | '__/ _` | | |/ _ \ |  | | | | |   | |/ _ \/ __| __|
+    // |  __/ (_| | | | (_| | | |  __/ |  | | |_| |   | |  __/\__ \ |_
+    // |_|   \__,_|_|  \__,_|_|_|\___|_| |___\___/    |_|\___||___/\__|
+    printf(" ____                 _ _      _   ___ ___    _____         _\r\n");
+    printf("|  _ \\ __ _ _ __ __ _| | | ___| | |_ _/ _ \\  |_   _|__  ___| |_\r\n");
+    printf("| |_) / _` | '__/ _` | | |/ _ \\ |  | | | | |   | |/ _ \\/ __| __|\r\n");
+    printf("|  __/ (_| | | | (_| | | |  __/ |  | | |_| |   | |  __/\\__ \\ |_\r\n");
+    printf("|_|   \\__,_|_|  \\__,_|_|_|\\___|_| |___\\___/    |_|\\___||___/\\__|\r\n");
+    unity_run_menu();
+}

+ 40 - 0
components/driver/test_apps/parlio/main/test_board.h

@@ -0,0 +1,40 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+#pragma once
+
+#include "sdkconfig.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+#if CONFIG_IDF_TARGET_ESP32C6
+#define TEST_CLK_GPIO       10
+#define TEST_DATA0_GPIO     0
+#define TEST_DATA1_GPIO     1
+#define TEST_DATA2_GPIO     2
+#define TEST_DATA3_GPIO     3
+#define TEST_DATA4_GPIO     4
+#define TEST_DATA5_GPIO     5
+#define TEST_DATA6_GPIO     6
+#define TEST_DATA7_GPIO     7
+#elif CONFIG_IDF_TARGET_ESP32H2
+#define TEST_CLK_GPIO       10
+#define TEST_DATA0_GPIO     0
+#define TEST_DATA1_GPIO     1
+#define TEST_DATA2_GPIO     2
+#define TEST_DATA3_GPIO     3
+#define TEST_DATA4_GPIO     4
+#define TEST_DATA5_GPIO     5
+#define TEST_DATA6_GPIO     8
+#define TEST_DATA7_GPIO     9
+#else
+#error "Unsupported target"
+#endif
+
+#ifdef __cplusplus
+}
+#endif

+ 280 - 0
components/driver/test_apps/parlio/main/test_parlio_tx.c

@@ -0,0 +1,280 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include <stdio.h>
+#include "sdkconfig.h"
+#include "freertos/FreeRTOS.h"
+#include "freertos/task.h"
+#include "unity.h"
+#include "driver/parlio_tx.h"
+#include "driver/gpio.h"
+#include "soc/soc_caps.h"
+#include "esp_attr.h"
+#include "test_board.h"
+
+#if CONFIG_PARLIO_ISR_IRAM_SAFE
+#define TEST_PARLIO_CALLBACK_ATTR IRAM_ATTR
+#else
+#define TEST_PARLIO_CALLBACK_ATTR
+#endif
+
+TEST_CASE("parallel_tx_unit_install_uninstall", "[parlio_tx]")
+{
+    printf("install tx units exhaustively\r\n");
+    parlio_tx_unit_handle_t units[SOC_PARLIO_GROUPS * SOC_PARLIO_TX_UNITS_PER_GROUP];
+    int k = 0;
+    parlio_tx_unit_config_t config = {
+        .clk_src = PARLIO_CLK_SRC_DEFAULT,
+        .data_width = SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH,
+        .clk_in_gpio_num = -1, // clock source from internal
+        .clk_out_gpio_num = 0,
+        .output_clk_freq_hz = 1 * 1000 * 1000,
+        .trans_queue_depth = 4,
+        .max_transfer_size = 64,
+        .valid_gpio_num = -1,
+    };
+    for (int i = 0; i < SOC_PARLIO_GROUPS; i++) {
+        for (int j = 0; j < SOC_PARLIO_TX_UNITS_PER_GROUP; j++) {
+            TEST_ESP_OK(parlio_new_tx_unit(&config, &units[k++]));
+        }
+    }
+    TEST_ESP_ERR(ESP_ERR_NOT_FOUND, parlio_new_tx_unit(&config, &units[0]));
+
+    for (int i = 0; i < k; i++) {
+        TEST_ESP_OK(parlio_del_tx_unit(units[i]));
+    }
+
+    printf("install tx unit with valid signal and external core clock\r\n");
+    // clock from external
+    config.clk_in_gpio_num = 2;
+    // failed because of invalid clock source frequency
+    TEST_ESP_ERR(ESP_ERR_INVALID_ARG, parlio_new_tx_unit(&config, &units[0]));
+    config.input_clk_src_freq_hz = 1000000;
+
+    config.valid_gpio_num = 0;
+    // failed because of data line conflict with valid signal
+    TEST_ESP_ERR(ESP_ERR_INVALID_ARG, parlio_new_tx_unit(&config, &units[0]));
+
+    config.data_width = 4;
+    TEST_ESP_OK(parlio_new_tx_unit(&config, &units[0]));
+    TEST_ESP_OK(parlio_tx_unit_enable(units[0]));
+    // delete unit before it's disabled is not allowed
+    TEST_ESP_ERR(ESP_ERR_INVALID_STATE, parlio_del_tx_unit(units[0]));
+    TEST_ESP_OK(parlio_tx_unit_disable(units[0]));
+    TEST_ESP_OK(parlio_del_tx_unit(units[0]));
+}
+
+TEST_PARLIO_CALLBACK_ATTR
+static bool test_parlio_tx_done_callback(parlio_tx_unit_handle_t tx_unit, const parlio_tx_done_event_data_t *edata, void *user_ctx)
+{
+    BaseType_t high_task_wakeup = pdFALSE;
+    TaskHandle_t task = (TaskHandle_t)user_ctx;
+    vTaskNotifyGiveFromISR(task, &high_task_wakeup);
+    return high_task_wakeup == pdTRUE;
+}
+
+TEST_CASE("parallel_tx_unit_trans_done_event", "[parlio_tx]")
+{
+    printf("install parlio tx unit\r\n");
+    parlio_tx_unit_handle_t tx_unit = NULL;
+    parlio_tx_unit_config_t config = {
+        .clk_src = PARLIO_CLK_SRC_DEFAULT,
+        .data_width = 8,
+        .clk_in_gpio_num = -1,  // use internal clock source
+        .valid_gpio_num = -1,   // don't generate valid signal
+        .clk_out_gpio_num = TEST_CLK_GPIO,
+        .data_gpio_nums = {
+            TEST_DATA0_GPIO,
+            TEST_DATA1_GPIO,
+            TEST_DATA2_GPIO,
+            TEST_DATA3_GPIO,
+            TEST_DATA4_GPIO,
+            TEST_DATA5_GPIO,
+            TEST_DATA6_GPIO,
+            TEST_DATA7_GPIO,
+        },
+        .output_clk_freq_hz = 1 * 1000 * 1000,
+        .trans_queue_depth = 8,
+        .max_transfer_size = 128,
+        .bit_pack_order = PARLIO_BIT_PACK_ORDER_LSB,
+        .sample_edge = PARLIO_SAMPLE_EDGE_POS,
+    };
+    TEST_ESP_OK(parlio_new_tx_unit(&config, &tx_unit));
+    TEST_ESP_OK(parlio_tx_unit_enable(tx_unit));
+
+    printf("register trans_done event callback\r\n");
+    parlio_tx_event_callbacks_t cbs = {
+        .on_trans_done = test_parlio_tx_done_callback,
+    };
+    TEST_ESP_OK(parlio_tx_unit_register_event_callbacks(tx_unit, &cbs, xTaskGetCurrentTaskHandle()));
+
+    printf("send packets and check event is fired\r\n");
+    parlio_transmit_config_t transmit_config = {
+        .idle_value = 0x00,
+    };
+    uint8_t payload[64] = {0};
+    for (int i = 0; i < 64; i++) {
+        payload[i] = i;
+    }
+    TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, 64 * sizeof(uint8_t) * 8, &transmit_config));
+    TEST_ASSERT_NOT_EQUAL(0, ulTaskNotifyTake(pdTRUE, portMAX_DELAY));
+    TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, 64 * sizeof(uint8_t) * 8, &transmit_config));
+    TEST_ASSERT_NOT_EQUAL(0, ulTaskNotifyTake(pdTRUE, portMAX_DELAY));
+
+    TEST_ESP_OK(parlio_tx_unit_disable(tx_unit));
+    TEST_ESP_OK(parlio_del_tx_unit(tx_unit));
+};
+
+TEST_CASE("parallel_tx_unit_enable_disable", "[parlio_tx]")
+{
+    printf("install parlio tx unit\r\n");
+    parlio_tx_unit_handle_t tx_unit = NULL;
+    parlio_tx_unit_config_t config = {
+        .clk_src = PARLIO_CLK_SRC_DEFAULT,
+        .data_width = 8,
+        .clk_in_gpio_num = -1,  // use internal clock source
+        .valid_gpio_num = -1,   // don't generate valid signal
+        .clk_out_gpio_num = TEST_CLK_GPIO,
+        .data_gpio_nums = {
+            TEST_DATA0_GPIO,
+            TEST_DATA1_GPIO,
+            TEST_DATA2_GPIO,
+            TEST_DATA3_GPIO,
+            TEST_DATA4_GPIO,
+            TEST_DATA5_GPIO,
+            TEST_DATA6_GPIO,
+            TEST_DATA7_GPIO,
+        },
+        .output_clk_freq_hz = 1 * 1000 * 1000,
+        .trans_queue_depth = 64,
+        .max_transfer_size = 256,
+        .bit_pack_order = PARLIO_BIT_PACK_ORDER_LSB,
+        .sample_edge = PARLIO_SAMPLE_EDGE_POS,
+    };
+    TEST_ESP_OK(parlio_new_tx_unit(&config, &tx_unit));
+    TEST_ESP_OK(parlio_tx_unit_enable(tx_unit));
+
+    printf("send packets for multiple times\r\n");
+    parlio_transmit_config_t transmit_config = {
+        .idle_value = 0x00,
+    };
+    uint8_t payload[128] = {0};
+    for (int i = 0; i < 128; i++) {
+        payload[i] = i;
+    }
+    for (int j = 0; j < 64; j++) {
+        TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, 128 * sizeof(uint8_t) * 8, &transmit_config));
+    }
+
+    printf("disable the transaction in the middle\r\n");
+    while (parlio_tx_unit_disable(tx_unit) != ESP_OK) {
+        esp_rom_delay_us(1000);
+    }
+    vTaskDelay(pdMS_TO_TICKS(100));
+
+    printf("resume the transaction and pending packets should continue\r\n");
+    TEST_ESP_OK(parlio_tx_unit_enable(tx_unit));
+    TEST_ESP_OK(parlio_tx_unit_wait_all_done(tx_unit, -1));
+    TEST_ESP_OK(parlio_tx_unit_disable(tx_unit));
+    TEST_ESP_OK(parlio_del_tx_unit(tx_unit));
+}
+
+TEST_CASE("parallel_tx_unit_idle_value", "[parlio_tx]")
+{
+    printf("install parlio tx unit\r\n");
+    parlio_tx_unit_handle_t tx_unit = NULL;
+    parlio_tx_unit_config_t config = {
+        .clk_src = PARLIO_CLK_SRC_DEFAULT,
+        .data_width = 8,
+        .clk_in_gpio_num = -1,  // use internal clock source
+        .valid_gpio_num = -1,   // don't generate valid signal
+        .clk_out_gpio_num = TEST_CLK_GPIO,
+        .data_gpio_nums = {
+            TEST_DATA0_GPIO,
+            TEST_DATA1_GPIO,
+            TEST_DATA2_GPIO,
+            TEST_DATA3_GPIO,
+            TEST_DATA4_GPIO,
+            TEST_DATA5_GPIO,
+            TEST_DATA6_GPIO,
+            TEST_DATA7_GPIO,
+        },
+        .output_clk_freq_hz = 1 * 1000 * 1000,
+        .trans_queue_depth = 4,
+        .max_transfer_size = 64,
+        .bit_pack_order = PARLIO_BIT_PACK_ORDER_LSB,
+        .sample_edge = PARLIO_SAMPLE_EDGE_POS,
+        .flags.io_loop_back = 1,   // enable loop back by GPIO matrix, so that we can read the level of the data line by gpio driver
+    };
+    TEST_ESP_OK(parlio_new_tx_unit(&config, &tx_unit));
+    TEST_ESP_OK(parlio_tx_unit_enable(tx_unit));
+
+    printf("send packet with different idle_value\r\n");
+    parlio_transmit_config_t transmit_config = {
+        .idle_value = 0x00,
+    };
+    uint8_t payload[8] = {0};
+    for (int i = 0; i < 8; i++) {
+        payload[i] = i;
+    }
+    for (int j = 0; j < 16; j++) {
+        transmit_config.idle_value = j;
+        TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, sizeof(payload) * 8, &transmit_config));
+        TEST_ESP_OK(parlio_tx_unit_wait_all_done(tx_unit, 100));
+        TEST_ASSERT_EQUAL(j & 0x01, gpio_get_level(TEST_DATA0_GPIO));
+    }
+
+    TEST_ESP_OK(parlio_tx_unit_disable(tx_unit));
+    TEST_ESP_OK(parlio_del_tx_unit(tx_unit));
+}
+
+#if SOC_PARLIO_TX_CLK_SUPPORT_GATING
+TEST_CASE("parallel_tx_clock_gating", "[paralio_tx]")
+{
+    printf("install parlio tx unit\r\n");
+    parlio_tx_unit_handle_t tx_unit = NULL;
+    parlio_tx_unit_config_t config = {
+        .clk_src = PARLIO_CLK_SRC_DEFAULT,
+        .data_width = 2,
+        .clk_in_gpio_num = -1,  // use internal clock source
+        .valid_gpio_num = TEST_DATA7_GPIO, // generate the valid signal
+        .clk_out_gpio_num = TEST_CLK_GPIO,
+        .data_gpio_nums = {
+            TEST_DATA0_GPIO,
+            TEST_DATA1_GPIO,
+        },
+        .output_clk_freq_hz = 1 * 1000 * 1000,
+        .trans_queue_depth = 4,
+        .max_transfer_size = 64,
+        .bit_pack_order = PARLIO_BIT_PACK_ORDER_MSB,
+        .sample_edge = PARLIO_SAMPLE_EDGE_POS,
+        .flags.clk_gate_en = true, // enable clock gating, controlled by the level of TEST_DATA7_GPIO
+        .flags.io_loop_back = true, // for reading the level of the clock line in IDLE state
+    };
+    TEST_ESP_OK(parlio_new_tx_unit(&config, &tx_unit));
+    TEST_ESP_OK(parlio_tx_unit_enable(tx_unit));
+
+    printf("send packets and see if the clock is gated when there's no transaction on line\r\n");
+    parlio_transmit_config_t transmit_config = {
+        .idle_value = 0x00,
+    };
+    uint8_t payload[8] = {0};
+    for (int i = 0; i < 8; i++) {
+        payload[i] = 0x1B; // 8'b00011011, in PARLIO_BIT_PACK_ORDER_MSB, you should see 2'b00, 2'b01, 2'b10, 2'b11 on the data line
+    }
+    TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, 8 * sizeof(uint8_t) * 8, &transmit_config));
+    TEST_ESP_OK(parlio_tx_unit_wait_all_done(tx_unit, -1));
+    // check if the level on the clock line is low
+    TEST_ASSERT_EQUAL(0, gpio_get_level(TEST_CLK_GPIO));
+    TEST_ESP_OK(parlio_tx_unit_transmit(tx_unit, payload, 8 * sizeof(uint8_t) * 8, &transmit_config));
+    TEST_ESP_OK(parlio_tx_unit_wait_all_done(tx_unit, -1));
+    TEST_ASSERT_EQUAL(0, gpio_get_level(TEST_CLK_GPIO));
+    TEST_ASSERT_EQUAL(0, gpio_get_level(TEST_CLK_GPIO));
+
+    TEST_ESP_OK(parlio_tx_unit_disable(tx_unit));
+    TEST_ESP_OK(parlio_del_tx_unit(tx_unit));
+}
+#endif // SOC_PARLIO_TX_CLK_SUPPORT_GATING

+ 22 - 0
components/driver/test_apps/parlio/pytest_parlio_unity.py

@@ -0,0 +1,22 @@
+# SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+# SPDX-License-Identifier: CC0-1.0
+
+import pytest
+from pytest_embedded import Dut
+
+
+@pytest.mark.esp32c6
+@pytest.mark.esp32h2
+@pytest.mark.generic
+@pytest.mark.parametrize(
+    'config',
+    [
+        'iram_safe',
+        'release',
+    ],
+    indirect=True,
+)
+def test_parlio(dut: Dut) -> None:
+    dut.expect_exact('Press ENTER to see the list of tests')
+    dut.write('*')
+    dut.expect_unity_test_output()

+ 7 - 0
components/driver/test_apps/parlio/sdkconfig.ci.iram_safe

@@ -0,0 +1,7 @@
+CONFIG_COMPILER_DUMP_RTL_FILES=y
+CONFIG_COMPILER_OPTIMIZATION_NONE=y
+CONFIG_PARLIO_ISR_IRAM_SAFE=y
+# place non-ISR FreeRTOS functions in Flash
+CONFIG_FREERTOS_PLACE_FUNCTIONS_INTO_FLASH=y
+# silent the error check, as the error string are stored in rodata, causing RTL check failure
+CONFIG_COMPILER_OPTIMIZATION_CHECKS_SILENT=y

+ 5 - 0
components/driver/test_apps/parlio/sdkconfig.ci.release

@@ -0,0 +1,5 @@
+CONFIG_PM_ENABLE=y
+CONFIG_FREERTOS_USE_TICKLESS_IDLE=y
+CONFIG_COMPILER_OPTIMIZATION_SIZE=y
+CONFIG_BOOTLOADER_COMPILER_OPTIMIZATION_SIZE=y
+CONFIG_COMPILER_OPTIMIZATION_ASSERTIONS_SILENT=y

+ 5 - 0
components/driver/test_apps/parlio/sdkconfig.defaults

@@ -0,0 +1,5 @@
+# This file was generated using idf.py save-defconfig. It can be edited manually.
+# Espressif IoT Development Framework (ESP-IDF) Project Minimal Configuration
+#
+CONFIG_ESP_TASK_WDT_INIT=n
+CONFIG_FREERTOS_HZ=1000

+ 4 - 0
components/hal/CMakeLists.txt

@@ -105,6 +105,10 @@ if(NOT BOOTLOADER_BUILD)
         list(APPEND srcs "etm_hal.c")
     endif()
 
+    if(CONFIG_SOC_PARLIO_SUPPORTED)
+        list(APPEND srcs "parlio_hal.c")
+    endif()
+
     if(CONFIG_SOC_ADC_DMA_SUPPORTED)
         list(APPEND srcs "adc_hal.c")
     endif()

+ 8 - 0
components/hal/esp32c6/include/hal/clk_gate_ll.h

@@ -58,6 +58,8 @@ static inline uint32_t periph_ll_get_clk_en_mask(periph_module_t periph)
             return PCR_PWM_CLK_EN;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CLK_EN;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_CLK_EN;
         case PERIPH_AES_MODULE:
             return PCR_AES_CLK_EN;
         case PERIPH_SHA_MODULE:
@@ -136,6 +138,8 @@ static inline uint32_t periph_ll_get_rst_en_mask(periph_module_t periph, bool en
             return PCR_PWM_RST_EN;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_RST_EN;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_RST_EN;
         case PERIPH_ECC_MODULE:
             return PCR_ECC_RST_EN;
         case PERIPH_TEMPSENSOR_MODULE:
@@ -233,6 +237,8 @@ static uint32_t periph_ll_get_clk_en_reg(periph_module_t periph)
             return PCR_PWM_CONF_REG;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CONF_REG;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_IO_CONF_REG;
         case PERIPH_AES_MODULE:
             return PCR_AES_CONF_REG;
         case PERIPH_SHA_MODULE:
@@ -297,6 +303,8 @@ static uint32_t periph_ll_get_rst_en_reg(periph_module_t periph)
             return PCR_PWM_CONF_REG;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CONF_REG;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_IO_CONF_REG;
         case PERIPH_AES_MODULE:
             return PCR_AES_CONF_REG;
         case PERIPH_SHA_MODULE:

+ 612 - 0
components/hal/esp32c6/include/hal/parlio_ll.h

@@ -0,0 +1,612 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+// Note that most of the register operations in this layer are non-atomic operations.
+
+#pragma once
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "hal/assert.h"
+#include "hal/misc.h"
+#include "soc/pcr_struct.h"
+#include "soc/parl_io_struct.h"
+#include "hal/parlio_types.h"
+
+#define PARLIO_LL_RX_MAX_BYTES_PER_FRAME 0xFFFF
+#define PARLIO_LL_RX_MAX_CLOCK_DIV       0x10000
+#define PARLIO_LL_RX_MAX_TIMEOUT         0xFFFF
+
+#define PARLIO_LL_TX_MAX_BYTES_PER_FRAME 0xFFFF
+#define PARLIO_LL_TX_MAX_BITS_PER_FRAME  (PARLIO_LL_TX_MAX_BYTES_PER_FRAME * 8)
+#define PARLIO_LL_TX_MAX_CLOCK_DIV       0x10000
+
+#define PARLIO_LL_EVENT_TX_FIFO_EMPTY    (1 << 0)
+#define PARLIO_LL_EVENT_RX_FIFO_FULL     (1 << 1)
+#define PARLIO_LL_EVENT_TX_EOF           (1 << 2)
+#define PARLIO_LL_EVENT_TX_MASK          (PARLIO_LL_EVENT_TX_FIFO_EMPTY | PARLIO_LL_EVENT_TX_EOF)
+#define PARLIO_LL_EVENT_RX_MASK          (PARLIO_LL_EVENT_RX_FIFO_FULL)
+
+#define PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG 15 // TXD[15] can be used a valid signal
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef enum {
+    PARLIO_LL_CLK_SRC_XTAL = PARLIO_CLK_SRC_XTAL,
+    PARLIO_LL_CLK_SRC_PLL_F240M = PARLIO_CLK_SRC_PLL_F240M,
+    PARLIO_LL_CLK_SRC_PAD, // clock source from GPIO pad
+} parlio_ll_clock_source_t;
+
+typedef enum {
+    PARLIO_LL_RX_EOF_COND_RX_FULL,     /*!< RX unit generates EOF event when it receives enough data */
+    PARLIO_LL_RX_EOF_COND_EN_INACTIVE, /*!< RX unit generates EOF event when the external enable signal becomes inactive */
+} parlio_ll_rx_eof_cond_t;
+
+///////////////////////////////////////RX Unit///////////////////////////////////////
+
+/**
+ * @brief Set the clock source for the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param src Clock source
+ */
+static inline void parlio_ll_rx_set_clock_source(parl_io_dev_t *dev, parlio_ll_clock_source_t src)
+{
+    (void)dev;
+    uint32_t clk_sel = 0;
+    switch (src) {
+    case PARLIO_LL_CLK_SRC_XTAL:
+        clk_sel = 0;
+        break;
+    case PARLIO_LL_CLK_SRC_PLL_F240M:
+        clk_sel = 1;
+        break;
+    case PARLIO_LL_CLK_SRC_PAD:
+        clk_sel = 3;
+        break;
+
+    default: // unsupported clock source
+        HAL_ASSERT(false);
+        break;
+    }
+    PCR.parl_clk_rx_conf.parl_clk_rx_sel = clk_sel;
+}
+
+/**
+ * @brief Set the clock divider for the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param div Clock divider
+ */
+static inline void parlio_ll_rx_set_clock_div(parl_io_dev_t *dev, uint32_t div)
+{
+    (void)dev;
+    HAL_ASSERT(div > 0 && div <= PARLIO_LL_RX_MAX_CLOCK_DIV);
+    PCR.parl_clk_rx_conf.parl_clk_rx_div_num = div - 1;
+}
+
+/**
+ * @brief Reset the RX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_rx_reset_clock(parl_io_dev_t *dev)
+{
+    (void)dev;
+    PCR.parl_clk_rx_conf.parl_rx_rst_en = 1;
+    PCR.parl_clk_rx_conf.parl_rx_rst_en = 0;
+}
+
+/**
+ * @brief Enable the RX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_rx_enable_clock(parl_io_dev_t *dev, bool en)
+{
+    (void)dev;
+    PCR.parl_clk_rx_conf.parl_clk_rx_en = en;
+}
+
+/**
+ * @brief Set the condition to generate the RX EOF event
+ *
+ * @param dev Parallel IO register base address
+ * @param cond RX EOF condition
+ */
+static inline void parlio_ll_rx_set_eof_condition(parl_io_dev_t *dev, parlio_ll_rx_eof_cond_t cond)
+{
+    dev->rx_cfg0.rx_eof_gen_sel = cond;
+}
+
+/**
+ * @brief Start RX unit to sample the input data
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to start, False to stop
+ */
+static inline void parlio_ll_rx_start(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_cfg0.rx_start = en;
+}
+
+/**
+ * @brief Set the receive length
+ *
+ * @note The receive length can be used to generate DMA EOF signal, or to work as a frame end delimiter
+ *
+ * @param dev Parallel IO register base address
+ * @param bitlen Number of bits to receive in the next transaction, bitlen must be a multiple of 8
+ */
+static inline void parlio_ll_rx_set_recv_bit_len(parl_io_dev_t *dev, uint32_t bitlen)
+{
+    HAL_FORCE_MODIFY_U32_REG_FIELD(dev->rx_cfg0, rx_data_bytelen, bitlen / 8);
+}
+
+/**
+ * @brief Set the sub mode of the level controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param active_level Level of the external enable signal, true for active high, false for active low
+ */
+static inline void parlio_ll_rx_set_level_recv_mode(parl_io_dev_t *dev, bool active_level)
+{
+    dev->rx_cfg0.rx_smp_mode_sel = 0;
+    dev->rx_cfg0.rx_level_submode_sel = !active_level; // 0: active low, 1: active high
+}
+
+/**
+ * @brief Set the sub mode of the pulse controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param start_inc Whether the start pulse is counted
+ * @param end_inc Whether the end pulse is counted
+ * @param end_by_len Whether to use the frame length to determine the end of the frame
+ * @param pulse_inv Whether the pulse is inverted
+ */
+static inline void parlio_ll_rx_set_pulse_recv_mode(parl_io_dev_t *dev, bool start_inc, bool end_inc, bool end_by_len, bool pulse_inv)
+{
+    uint32_t submode = 0;
+    uint32_t step = 1;
+    if (end_by_len) {
+        submode += 4;
+    } else {
+        step = 2;
+        if (!end_inc) {
+            submode += 1;
+        }
+    }
+    if (!start_inc) {
+        submode += step;
+    }
+    if (pulse_inv) {
+        submode += 6;
+    }
+    dev->rx_cfg0.rx_smp_mode_sel = 1;
+    dev->rx_cfg0.rx_pulse_submode_sel = submode;
+}
+
+/**
+ * @brief Set the receive mode to software controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_set_soft_recv_mode(parl_io_dev_t *dev)
+{
+    dev->rx_cfg0.rx_smp_mode_sel = 2;
+}
+
+/**
+ * @brief Whether to start the software controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_start_soft_recv(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_cfg0.rx_sw_en = en;
+}
+
+/**
+ * @brief Set the sample clock edge
+ *
+ * @param dev Parallel IO register base address
+ * @param edge Sample clock edge
+ */
+static inline void parlio_ll_rx_set_sample_clock_edge(parl_io_dev_t *dev, parlio_sample_edge_t edge)
+{
+    dev->rx_cfg0.rx_clk_edge_sel = edge;
+}
+
+/**
+ * @brief Set the order to pack bits into one byte
+ *
+ * @param dev Parallel IO register base address
+ * @param order Packing order
+ */
+static inline void parlio_ll_rx_set_bit_pack_order(parl_io_dev_t *dev, parlio_bit_pack_order_t order)
+{
+    dev->rx_cfg0.rx_bit_pack_order = order;
+}
+
+/**
+ * @brief Set the bus width of the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param width Bus width
+ */
+static inline void parlio_ll_rx_set_bus_width(parl_io_dev_t *dev, uint32_t width)
+{
+    uint32_t width_sel = 0;
+    switch (width) {
+    case 16:
+        width_sel = 0;
+        break;
+    case 8:
+        width_sel = 1;
+        break;
+    case 4:
+        width_sel = 2;
+        break;
+    case 2:
+        width_sel = 3;
+        break;
+    case 1:
+        width_sel = 4;
+        break;
+    default:
+        HAL_ASSERT(false);
+    }
+    dev->rx_cfg0.rx_bus_wid_sel = width_sel;
+}
+
+/**
+ * @brief Reset RX Async FIFO
+ *
+ * @note During the reset of the asynchronous FIFO, it takes two clock cycles to synchronize within AHB clock domain (GDMA) and Core clock domain.
+ *       The reset synchronization must be performed two clock cycles in advance.
+ * @note If the next frame transfer needs to be reset, you need to first switch to the internal free-running clock,
+ *       and then switch to the actual clock after the reset is completed.
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_rx_reset_fifo(parl_io_dev_t *dev)
+{
+    dev->rx_cfg0.rx_fifo_srst = 1;
+    dev->rx_cfg0.rx_fifo_srst = 0;
+}
+
+/**
+ * @brief Set which data line as the enable signal
+ *
+ * @param dev Parallel IO register base address
+ * @param line_num Data line number (0-15)
+ */
+static inline void parlio_ll_rx_treat_data_line_as_en(parl_io_dev_t *dev, uint32_t line_num)
+{
+    dev->rx_cfg1.rx_ext_en_sel = line_num;
+}
+
+/**
+ * @brief Enable RX timeout feature
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_enable_timeout(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_cfg1.rx_timeout_en = en;
+}
+
+/**
+ * @brief Set the threshold of RX timeout
+ *
+ * @param dev Parallel IO register base address
+ * @param thres Threshold of RX timeout
+ */
+static inline void parlio_ll_rx_set_timeout_thres(parl_io_dev_t *dev, uint32_t thres)
+{
+    HAL_FORCE_MODIFY_U32_REG_FIELD(dev->rx_cfg1, rx_timeout_threshold, thres);
+}
+
+/**
+ * @brief Update the RX configuration, to make the new configuration take effect
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_update_config(parl_io_dev_t *dev)
+{
+    dev->rx_cfg1.rx_reg_update = 1;
+    while (dev->rx_cfg1.rx_reg_update);
+}
+
+///////////////////////////////////TX Unit///////////////////////////////////////
+
+/**
+ * @brief Set the clock source for the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param src Clock source
+ */
+static inline void parlio_ll_tx_set_clock_source(parl_io_dev_t *dev, parlio_ll_clock_source_t src)
+{
+    (void)dev;
+    uint32_t clk_sel = 0;
+    switch (src) {
+    case PARLIO_LL_CLK_SRC_XTAL:
+        clk_sel = 0;
+        break;
+    case PARLIO_LL_CLK_SRC_PLL_F240M:
+        clk_sel = 1;
+        break;
+    case PARLIO_LL_CLK_SRC_PAD:
+        clk_sel = 3;
+        break;
+
+    default: // unsupported clock source
+        HAL_ASSERT(false);
+        break;
+    }
+    PCR.parl_clk_tx_conf.parl_clk_tx_sel = clk_sel;
+}
+
+/**
+ * @brief Set the clock divider for the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param div Clock divider
+ */
+static inline void parlio_ll_tx_set_clock_div(parl_io_dev_t *dev, uint32_t div)
+{
+    (void)dev;
+    HAL_ASSERT(div > 0 && div <= PARLIO_LL_TX_MAX_CLOCK_DIV);
+    PCR.parl_clk_tx_conf.parl_clk_tx_div_num = div - 1;
+}
+
+/**
+ * @brief Reset the TX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_reset_clock(parl_io_dev_t *dev)
+{
+    (void)dev;
+    PCR.parl_clk_tx_conf.parl_tx_rst_en = 1;
+    PCR.parl_clk_tx_conf.parl_tx_rst_en = 0;
+}
+
+/**
+ * @brief Enable the TX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_enable_clock(parl_io_dev_t *dev, bool en)
+{
+    (void)dev;
+    PCR.parl_clk_tx_conf.parl_clk_tx_en = en;
+}
+
+/**
+ * @brief Set the data length to be transmitted
+ *
+ * @param dev Parallel IO register base address
+ * @param bitlen Data length in bits, must be a multiple of 8
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_set_trans_bit_len(parl_io_dev_t *dev, uint32_t bitlen)
+{
+    HAL_FORCE_MODIFY_U32_REG_FIELD(dev->tx_cfg0, tx_bytelen, bitlen / 8);
+}
+
+/**
+ * @brief Wether to enable the TX clock gating
+ *
+ * @note The TXD[7] will be taken as the gating enable signal
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_tx_enable_clock_gating(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_cfg0.tx_gating_en = en;
+}
+
+/**
+ * @brief Start TX unit to transmit data
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to start, False to stop
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_start(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_cfg0.tx_start = en;
+}
+
+/**
+ * @brief Whether to treat the MSB of TXD as the valid signal
+ *
+ * @note If enabled, TXD[15] will work as valid signal, which stay high during data transmission.
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_tx_treat_msb_as_valid(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_cfg0.tx_hw_valid_en = en;
+}
+
+/**
+ * @brief Set the sample clock edge
+ *
+ * @param dev Parallel IO register base address
+ * @param edge Sample clock edge
+ */
+static inline void parlio_ll_tx_set_sample_clock_edge(parl_io_dev_t *dev, parlio_sample_edge_t edge)
+{
+    dev->tx_cfg0.tx_smp_edge_sel = edge;
+}
+
+/**
+ * @brief Set the order to unpack bits from a byte
+ *
+ * @param dev Parallel IO register base address
+ * @param order Packing order
+ */
+static inline void parlio_ll_tx_set_bit_pack_order(parl_io_dev_t *dev, parlio_bit_pack_order_t order)
+{
+    dev->tx_cfg0.tx_bit_unpack_order = order;
+}
+
+/**
+ * @brief Set the bus width of the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param width Bus width
+ */
+static inline void parlio_ll_tx_set_bus_width(parl_io_dev_t *dev, uint32_t width)
+{
+    uint32_t width_sel = 0;
+    switch (width) {
+    case 16:
+        width_sel = 0;
+        break;
+    case 8:
+        width_sel = 1;
+        break;
+    case 4:
+        width_sel = 2;
+        break;
+    case 2:
+        width_sel = 3;
+        break;
+    case 1:
+        width_sel = 4;
+        break;
+    default:
+        HAL_ASSERT(false);
+    }
+    dev->tx_cfg0.tx_bus_wid_sel = width_sel;
+}
+
+/**
+ * @brief Reset TX Async FIFO
+ *
+ * @note During the reset of the asynchronous FIFO, it takes two clock cycles to synchronize within AHB clock domain (GDMA) and Core clock domain.
+ *       The reset synchronization must be performed two clock cycles in advance.
+ * @note If the next frame transfer needs to be reset, you need to first switch to the internal free-running clock,
+ *       and then switch to the actual clock after the reset is completed.
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_reset_fifo(parl_io_dev_t *dev)
+{
+    dev->tx_cfg0.tx_fifo_srst = 1;
+    dev->tx_cfg0.tx_fifo_srst = 0;
+}
+
+/**
+ * @brief Set the value to output on the TXD when the TX unit is in IDLE state
+ *
+ * @param dev Parallel IO register base address
+ * @param value Value to output
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_set_idle_data_value(parl_io_dev_t *dev, uint32_t value)
+{
+    HAL_FORCE_MODIFY_U32_REG_FIELD(dev->tx_cfg1, tx_idle_value, value);
+}
+
+/**
+ * @brief Check whether the TX unit is ready
+ *
+ * @param dev Parallel IO register base address
+ * @return true: ready, false: busy
+ */
+__attribute__((always_inline))
+static inline bool parlio_ll_tx_is_ready(parl_io_dev_t *dev)
+{
+    return dev->st.tx_ready;
+}
+
+////////////////////////////////////Interrupt////////////////////////////////////////////////
+
+/**
+ * @brief Enable Parallel IO interrupt for specific event mask
+ *
+ * @param dev Parallel IO register base address
+ * @param mask Event mask
+ * @param enable True to enable, False to disable
+ */
+static inline void parlio_ll_enable_interrupt(parl_io_dev_t *dev, uint32_t mask, bool enable)
+{
+    if (enable) {
+        dev->int_ena.val |= mask;
+    } else {
+        dev->int_ena.val &= ~mask;
+    }
+}
+
+/**
+ * @brief Get interrupt status for TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @return Interrupt status
+ */
+__attribute__((always_inline))
+static inline uint32_t parlio_ll_tx_get_interrupt_status(parl_io_dev_t *dev)
+{
+    return dev->int_st.val & PARLIO_LL_EVENT_TX_MASK;
+}
+
+/**
+ * @brief Get interrupt status for RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @return Interrupt status
+ */
+__attribute__((always_inline))
+static inline uint32_t parlio_ll_rx_get_interrupt_status(parl_io_dev_t *dev)
+{
+    return dev->int_st.val & PARLIO_LL_EVENT_RX_MASK;
+}
+
+/**
+ * @brief Clear Parallel IO interrupt status by mask
+ *
+ * @param dev Parallel IO register base address
+ * @param mask Interrupt status mask
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_clear_interrupt_status(parl_io_dev_t *dev, uint32_t mask)
+{
+    dev->int_clr.val = mask;
+}
+
+/**
+ * @brief Get interrupt status register address
+ *
+ * @param dev Parallel IO register base address
+ * @return Register address
+ */
+static inline volatile void *parlio_ll_get_interrupt_status_reg(parl_io_dev_t *dev)
+{
+    return &dev->int_st;
+}
+
+#ifdef __cplusplus
+}
+#endif

+ 8 - 0
components/hal/esp32h2/include/hal/clk_gate_ll.h

@@ -57,6 +57,8 @@ static inline uint32_t periph_ll_get_clk_en_mask(periph_module_t periph)
             return PCR_PWM_CLK_EN;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CLK_EN;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_CLK_EN;
         case PERIPH_AES_MODULE:
             return PCR_AES_CLK_EN;
         case PERIPH_SHA_MODULE:
@@ -130,6 +132,8 @@ static inline uint32_t periph_ll_get_rst_en_mask(periph_module_t periph, bool en
             return PCR_PWM_RST_EN;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_RST_EN;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_RST_EN;
         case PERIPH_TEMPSENSOR_MODULE:
             return PCR_TSENS_RST_EN;
         case PERIPH_AES_MODULE:
@@ -221,6 +225,8 @@ static uint32_t periph_ll_get_clk_en_reg(periph_module_t periph)
             return PCR_PWM_CONF_REG;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CONF_REG;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_IO_CONF_REG;
         case PERIPH_AES_MODULE:
             return PCR_AES_CONF_REG;
         case PERIPH_SHA_MODULE:
@@ -280,6 +286,8 @@ static uint32_t periph_ll_get_rst_en_reg(periph_module_t periph)
             return PCR_PWM_CONF_REG;
         case PERIPH_ETM_MODULE:
             return PCR_ETM_CONF_REG;
+        case PERIPH_PARLIO_MODULE:
+            return PCR_PARL_IO_CONF_REG;
         case PERIPH_AES_MODULE:
             return PCR_AES_CONF_REG;
         case PERIPH_SHA_MODULE:

+ 614 - 0
components/hal/esp32h2/include/hal/parlio_ll.h

@@ -0,0 +1,614 @@
+/*
+ * SPDX-FileCopyrightText: 2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+// Note that most of the register operations in this layer are non-atomic operations.
+
+#pragma once
+
+#include <stdbool.h>
+#include <stdint.h>
+#include "hal/assert.h"
+#include "hal/misc.h"
+#include "soc/pcr_struct.h"
+#include "soc/parl_io_struct.h"
+#include "hal/parlio_types.h"
+
+#define PARLIO_LL_RX_MAX_BYTES_PER_FRAME 0xFFFF
+#define PARLIO_LL_RX_MAX_CLOCK_DIV       0x10000
+#define PARLIO_LL_RX_MAX_TIMEOUT         0xFFFF
+
+#define PARLIO_LL_TX_MAX_BITS_PER_FRAME  0x7FFFF
+#define PARLIO_LL_TX_MAX_CLOCK_DIV       0x10000
+
+#define PARLIO_LL_EVENT_TX_FIFO_EMPTY    (1 << 0)
+#define PARLIO_LL_EVENT_RX_FIFO_FULL     (1 << 1)
+#define PARLIO_LL_EVENT_TX_EOF           (1 << 2)
+#define PARLIO_LL_EVENT_TX_MASK          (PARLIO_LL_EVENT_TX_FIFO_EMPTY | PARLIO_LL_EVENT_TX_EOF)
+#define PARLIO_LL_EVENT_RX_MASK          (PARLIO_LL_EVENT_RX_FIFO_FULL)
+
+#define PARLIO_LL_TX_DATA_LINE_AS_VALID_SIG 7 // TXD[7] can be used a valid signal
+#define PARLIO_LL_TX_DATA_LINE_AS_CLK_GATE  7 // TXD[7] can be used as clock gate signal
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef enum {
+    PARLIO_LL_CLK_SRC_XTAL = PARLIO_CLK_SRC_XTAL,
+    PARLIO_LL_CLK_SRC_PLL_F96M = PARLIO_CLK_SRC_PLL_F96M,
+    PARLIO_LL_CLK_SRC_PAD, // clock source from GPIO pad
+} parlio_ll_clock_source_t;
+
+typedef enum {
+    PARLIO_LL_RX_EOF_COND_RX_FULL,     /*!< RX unit generates EOF event when it receives enough data */
+    PARLIO_LL_RX_EOF_COND_EN_INACTIVE, /*!< RX unit generates EOF event when the external enable signal becomes inactive */
+} parlio_ll_rx_eof_cond_t;
+
+///////////////////////////////////////RX Unit///////////////////////////////////////
+
+/**
+ * @brief Set the clock source for the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param src Clock source
+ */
+static inline void parlio_ll_rx_set_clock_source(parl_io_dev_t *dev, parlio_ll_clock_source_t src)
+{
+    (void)dev;
+    uint32_t clk_sel = 0;
+    switch (src) {
+    case PARLIO_LL_CLK_SRC_XTAL:
+        clk_sel = 0;
+        break;
+    case PARLIO_LL_CLK_SRC_PLL_F96M:
+        clk_sel = 1;
+        break;
+    case PARLIO_LL_CLK_SRC_PAD:
+        clk_sel = 3;
+        break;
+
+    default: // unsupported clock source
+        HAL_ASSERT(false);
+        break;
+    }
+    PCR.parl_clk_rx_conf.parl_clk_rx_sel = clk_sel;
+}
+
+/**
+ * @brief Set the clock divider for the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param div Clock divider
+ */
+static inline void parlio_ll_rx_set_clock_div(parl_io_dev_t *dev, uint32_t div)
+{
+    (void)dev;
+    HAL_ASSERT(div > 0 && div <= PARLIO_LL_RX_MAX_CLOCK_DIV);
+    PCR.parl_clk_rx_conf.parl_clk_rx_div_num = div - 1;
+}
+
+/**
+ * @brief Reset the RX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_reset_clock(parl_io_dev_t *dev)
+{
+    (void)dev;
+    PCR.parl_clk_rx_conf.parl_rx_rst_en = 1;
+    PCR.parl_clk_rx_conf.parl_rx_rst_en = 0;
+}
+
+/**
+ * @brief Enable the RX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_enable_clock(parl_io_dev_t *dev, bool en)
+{
+    (void)dev;
+    PCR.parl_clk_rx_conf.parl_clk_rx_en = en;
+}
+
+/**
+ * @brief Set the condition to generate the RX EOF event
+ *
+ * @param dev Parallel IO register base address
+ * @param cond RX EOF condition
+ */
+static inline void parlio_ll_rx_set_eof_condition(parl_io_dev_t *dev, parlio_ll_rx_eof_cond_t cond)
+{
+    dev->rx_genrl_cfg.rx_eof_gen_sel = cond;
+}
+
+/**
+ * @brief Start RX unit to sample the input data
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to start, False to stop
+ */
+static inline void parlio_ll_rx_start(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_start_cfg.rx_start = en;
+}
+
+/**
+ * @brief Set the receive length
+ *
+ * @note The receive length can be used to generate DMA EOF signal, or to work as a frame end delimiter
+ *
+ * @param dev Parallel IO register base address
+ * @param bitlen Number of bits to receive in the next transaction, bitlen must be a multiple of 8
+ */
+static inline void parlio_ll_rx_set_recv_bit_len(parl_io_dev_t *dev, uint32_t bitlen)
+{
+    dev->rx_data_cfg.rx_bitlen = bitlen;
+}
+
+/**
+ * @brief Set the sub mode of the level controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param active_level Level of the external enable signal, true for active high, false for active low
+ */
+static inline void parlio_ll_rx_set_level_recv_mode(parl_io_dev_t *dev, bool active_level)
+{
+    dev->rx_mode_cfg.rx_smp_mode_sel = 0;
+    dev->rx_mode_cfg.rx_ext_en_inv = !active_level; // 0: active low, 1: active high
+}
+
+/**
+ * @brief Set the sub mode of the pulse controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param start_inc Whether the start pulse is counted
+ * @param end_inc Whether the end pulse is counted
+ * @param end_by_len Whether to use the frame length to determine the end of the frame
+ * @param pulse_inv Whether the pulse is inverted
+ */
+static inline void parlio_ll_rx_set_pulse_recv_mode(parl_io_dev_t *dev, bool start_inc, bool end_inc, bool end_by_len, bool pulse_inv)
+{
+    uint32_t submode = 0;
+    uint32_t step = 1;
+    if (end_by_len) {
+        submode += 4;
+    } else { // end by pulse
+        step = 2;
+        if (!end_inc) {
+            submode += 1;
+        }
+    }
+    if (!start_inc) {
+        submode += step;
+    }
+    dev->rx_mode_cfg.rx_smp_mode_sel = 1;
+    dev->rx_mode_cfg.rx_pulse_submode_sel = submode;
+    dev->rx_mode_cfg.rx_ext_en_inv = pulse_inv;
+}
+
+/**
+ * @brief Set the receive mode to software controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_set_soft_recv_mode(parl_io_dev_t *dev)
+{
+    dev->rx_mode_cfg.rx_smp_mode_sel = 2;
+}
+
+/**
+ * @brief Whether to start the software controlled receive mode
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_start_soft_recv(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_mode_cfg.rx_sw_en = en;
+}
+
+/**
+ * @brief Set the sample clock edge
+ *
+ * @param dev Parallel IO register base address
+ * @param edge Sample clock edge
+ */
+static inline void parlio_ll_rx_set_sample_clock_edge(parl_io_dev_t *dev, parlio_sample_edge_t edge)
+{
+    dev->rx_clk_cfg.rx_clk_i_inv = edge;
+    dev->rx_clk_cfg.rx_clk_o_inv = edge;
+}
+
+/**
+ * @brief Set the order to pack bits into one byte
+ *
+ * @param dev Parallel IO register base address
+ * @param order Packing order
+ */
+static inline void parlio_ll_rx_set_bit_pack_order(parl_io_dev_t *dev, parlio_bit_pack_order_t order)
+{
+    dev->rx_data_cfg.rx_data_order_inv = order;
+}
+
+/**
+ * @brief Set the bus width of the RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param width Bus width
+ */
+static inline void parlio_ll_rx_set_bus_width(parl_io_dev_t *dev, uint32_t width)
+{
+    uint32_t width_sel = 0;
+    switch (width) {
+    case 8:
+        width_sel = 3;
+        break;
+    case 4:
+        width_sel = 2;
+        break;
+    case 2:
+        width_sel = 1;
+        break;
+    case 1:
+        width_sel = 0;
+        break;
+    default:
+        HAL_ASSERT(false);
+    }
+    dev->rx_data_cfg.rx_bus_wid_sel = width_sel;
+}
+
+/**
+ * @brief Reset RX Async FIFO
+ *
+ * @note During the reset of the asynchronous FIFO, it takes two clock cycles to synchronize within AHB clock domain (GDMA) and Core clock domain.
+ *       The reset synchronization must be performed two clock cycles in advance.
+ * @note If the next frame transfer needs to be reset, you need to first switch to the internal free-running clock,
+ *       and then switch to the actual clock after the reset is completed.
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_reset_fifo(parl_io_dev_t *dev)
+{
+    dev->fifo_cfg.rx_fifo_srst = 1;
+    dev->fifo_cfg.rx_fifo_srst = 0;
+}
+
+/**
+ * @brief Set which data line as the enable signal
+ *
+ * @param dev Parallel IO register base address
+ * @param line_num Data line number (0-15)
+ */
+static inline void parlio_ll_rx_treat_data_line_as_en(parl_io_dev_t *dev, uint32_t line_num)
+{
+    dev->rx_mode_cfg.rx_ext_en_sel = line_num;
+}
+
+/**
+ * @brief Wether to enable the RX clock gating
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_enable_clock_gating(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_genrl_cfg.rx_gating_en = en;
+}
+
+/**
+ * @brief Enable RX timeout feature
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_rx_enable_timeout(parl_io_dev_t *dev, bool en)
+{
+    dev->rx_genrl_cfg.rx_timeout_en = en;
+}
+
+/**
+ * @brief Set the threshold of RX timeout
+ *
+ * @param dev Parallel IO register base address
+ * @param thres Threshold of RX timeout
+ */
+static inline void parlio_ll_rx_set_timeout_thres(parl_io_dev_t *dev, uint32_t thres)
+{
+    HAL_FORCE_MODIFY_U32_REG_FIELD(dev->rx_genrl_cfg, rx_timeout_thres, thres);
+}
+
+/**
+ * @brief Update the RX configuration, to make the new configuration take effect
+ *
+ * @param dev Parallel IO register base address
+ */
+static inline void parlio_ll_rx_update_config(parl_io_dev_t *dev)
+{
+    dev->reg_update.rx_reg_update = 1;
+    while (dev->reg_update.rx_reg_update);
+}
+
+///////////////////////////////////TX Unit///////////////////////////////////////
+
+/**
+ * @brief Set the clock source for the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param src Clock source
+ */
+static inline void parlio_ll_tx_set_clock_source(parl_io_dev_t *dev, parlio_ll_clock_source_t src)
+{
+    (void)dev;
+    uint32_t clk_sel = 0;
+    switch (src) {
+    case PARLIO_LL_CLK_SRC_XTAL:
+        clk_sel = 0;
+        break;
+    case PARLIO_LL_CLK_SRC_PLL_F96M:
+        clk_sel = 1;
+        break;
+    case PARLIO_LL_CLK_SRC_PAD:
+        clk_sel = 3;
+        break;
+
+    default: // unsupported clock source
+        HAL_ASSERT(false);
+        break;
+    }
+    PCR.parl_clk_tx_conf.parl_clk_tx_sel = clk_sel;
+}
+
+/**
+ * @brief Set the clock divider for the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param div Clock divider
+ */
+static inline void parlio_ll_tx_set_clock_div(parl_io_dev_t *dev, uint32_t div)
+{
+    (void)dev;
+    HAL_ASSERT(div > 0 && div <= PARLIO_LL_TX_MAX_CLOCK_DIV);
+    PCR.parl_clk_tx_conf.parl_clk_tx_div_num = div - 1;
+}
+
+/**
+ * @brief Reset the TX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_reset_clock(parl_io_dev_t *dev)
+{
+    (void)dev;
+    PCR.parl_clk_tx_conf.parl_tx_rst_en = 1;
+    PCR.parl_clk_tx_conf.parl_tx_rst_en = 0;
+}
+
+/**
+ * @brief Enable the TX unit Core clock domain
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_enable_clock(parl_io_dev_t *dev, bool en)
+{
+    (void)dev;
+    PCR.parl_clk_tx_conf.parl_clk_tx_en = en;
+}
+
+/**
+ * @brief Set the data length to be transmitted
+ *
+ * @param dev Parallel IO register base address
+ * @param bitlen Data length in bits, must be a multiple of 8
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_set_trans_bit_len(parl_io_dev_t *dev, uint32_t bitlen)
+{
+    dev->tx_data_cfg.tx_bitlen = bitlen;
+}
+
+/**
+ * @brief Wether to enable the TX clock gating
+ *
+ * @note The MSB of TXD will be taken as the gating enable signal
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_tx_enable_clock_gating(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_genrl_cfg.tx_gating_en = en;
+}
+
+/**
+ * @brief Start TX unit to transmit data
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to start, False to stop
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_start(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_start_cfg.tx_start = en;
+}
+
+/**
+ * @brief Whether to treat the MSB of TXD as the valid signal
+ *
+ * @note If enabled, TXD[15] will work as valid signal, which stay high during data transmission.
+ *
+ * @param dev Parallel IO register base address
+ * @param en True to enable, False to disable
+ */
+static inline void parlio_ll_tx_treat_msb_as_valid(parl_io_dev_t *dev, bool en)
+{
+    dev->tx_genrl_cfg.tx_valid_output_en = en;
+}
+
+/**
+ * @brief Set the sample clock edge
+ *
+ * @param dev Parallel IO register base address
+ * @param edge Sample clock edge
+ */
+static inline void parlio_ll_tx_set_sample_clock_edge(parl_io_dev_t *dev, parlio_sample_edge_t edge)
+{
+    dev->tx_clk_cfg.tx_clk_i_inv = edge;
+    dev->tx_clk_cfg.tx_clk_o_inv = edge;
+}
+
+/**
+ * @brief Set the order to unpack bits from a byte
+ *
+ * @param dev Parallel IO register base address
+ * @param order Packing order
+ */
+static inline void parlio_ll_tx_set_bit_pack_order(parl_io_dev_t *dev, parlio_bit_pack_order_t order)
+{
+    dev->tx_data_cfg.tx_data_order_inv = order;
+}
+
+/**
+ * @brief Set the bus width of the TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @param width Bus width
+ */
+static inline void parlio_ll_tx_set_bus_width(parl_io_dev_t *dev, uint32_t width)
+{
+    uint32_t width_sel = 0;
+    switch (width) {
+    case 8:
+        width_sel = 3;
+        break;
+    case 4:
+        width_sel = 2;
+        break;
+    case 2:
+        width_sel = 1;
+        break;
+    case 1:
+        width_sel = 0;
+        break;
+    default:
+        HAL_ASSERT(false);
+    }
+    dev->tx_data_cfg.tx_bus_wid_sel = width_sel;
+}
+
+/**
+ * @brief Reset TX Async FIFO
+ *
+ * @note During the reset of the asynchronous FIFO, it takes two clock cycles to synchronize within AHB clock domain (GDMA) and Core clock domain.
+ *       The reset synchronization must be performed two clock cycles in advance.
+ * @note If the next frame transfer needs to be reset, you need to first switch to the internal free-running clock,
+ *       and then switch to the actual clock after the reset is completed.
+ *
+ * @param dev Parallel IO register base address
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_reset_fifo(parl_io_dev_t *dev)
+{
+    dev->fifo_cfg.tx_fifo_srst = 1;
+    dev->fifo_cfg.tx_fifo_srst = 0;
+}
+
+/**
+ * @brief Set the value to output on the TXD when the TX unit is in IDLE state
+ *
+ * @param dev Parallel IO register base address
+ * @param value Value to output
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_tx_set_idle_data_value(parl_io_dev_t *dev, uint32_t value)
+{
+    dev->tx_genrl_cfg.tx_idle_value = value;
+}
+
+/**
+ * @brief Check whether the TX unit is ready
+ *
+ * @param dev Parallel IO register base address
+ * @return true: ready, false: busy
+ */
+__attribute__((always_inline))
+static inline bool parlio_ll_tx_is_ready(parl_io_dev_t *dev)
+{
+    return dev->st.tx_ready;
+}
+
+////////////////////////////////////Interrupt////////////////////////////////////////////////
+
+/**
+ * @brief Enable Parallel IO interrupt for specific event mask
+ *
+ * @param dev Parallel IO register base address
+ * @param mask Event mask
+ * @param enable True to enable, False to disable
+ */
+static inline void parlio_ll_enable_interrupt(parl_io_dev_t *dev, uint32_t mask, bool enable)
+{
+    if (enable) {
+        dev->int_ena.val |= mask;
+    } else {
+        dev->int_ena.val &= ~mask;
+    }
+}
+
+/**
+ * @brief Get interrupt status for TX unit
+ *
+ * @param dev Parallel IO register base address
+ * @return Interrupt status
+ */
+__attribute__((always_inline))
+static inline uint32_t parlio_ll_tx_get_interrupt_status(parl_io_dev_t *dev)
+{
+    return dev->int_st.val & PARLIO_LL_EVENT_TX_MASK;
+}
+
+/**
+ * @brief Get interrupt status for RX unit
+ *
+ * @param dev Parallel IO register base address
+ * @return Interrupt status
+ */
+__attribute__((always_inline))
+static inline uint32_t parlio_ll_rx_get_interrupt_status(parl_io_dev_t *dev)
+{
+    return dev->int_st.val & PARLIO_LL_EVENT_RX_MASK;
+}
+
+/**
+ * @brief Clear Parallel IO interrupt status by mask
+ *
+ * @param dev Parallel IO register base address
+ * @param mask Interrupt status mask
+ */
+__attribute__((always_inline))
+static inline void parlio_ll_clear_interrupt_status(parl_io_dev_t *dev, uint32_t mask)
+{
+    dev->int_clr.val = mask;
+}
+
+/**
+ * @brief Get interrupt status register address
+ *
+ * @param dev Parallel IO register base address
+ * @return Register address
+ */
+static inline volatile void *parlio_ll_get_interrupt_status_reg(parl_io_dev_t *dev)
+{
+    return &dev->int_st;
+}
+
+#ifdef __cplusplus
+}
+#endif

+ 1 - 0
components/hal/include/hal/gdma_types.h

@@ -27,6 +27,7 @@ typedef enum {
     GDMA_TRIG_PERIPH_LCD,  /*!< GDMA trigger peripheral: LCD */
     GDMA_TRIG_PERIPH_CAM,  /*!< GDMA trigger peripheral: CAM */
     GDMA_TRIG_PERIPH_RMT,  /*!< GDMA trigger peripheral: RMT */
+    GDMA_TRIG_PERIPH_PARLIO, /*!< GDMA trigger peripheral: PARLIO */
 } gdma_trigger_peripheral_t;
 
 /**

+ 46 - 0
components/hal/include/hal/parlio_hal.h

@@ -0,0 +1,46 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+/*******************************************************************************
+ * NOTICE
+ * The hal is not public api, don't use in application code.
+ * See readme.md in hal/include/hal/readme.md
+ ******************************************************************************/
+
+#pragma once
+
+#include <stdint.h>
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef struct parl_io_dev_t *parlio_soc_handle_t; // Parallel IO SOC layer handle
+
+/**
+ * @brief HAL context type of Parallel IO driver
+ */
+typedef struct {
+    parlio_soc_handle_t regs; /*!< Parallel IO Register base address */
+} parlio_hal_context_t;
+
+/**
+ * @brief Initialize the Parallel IO HAL driver
+ *
+ * @param hal: Parallel IO HAL context
+ */
+void parlio_hal_init(parlio_hal_context_t *hal);
+
+/**
+ * @brief Deinitialize the Parallel IO HAL driver
+ *
+ * @param hal: Parallel IO HAL context
+ */
+void parlio_hal_deinit(parlio_hal_context_t *hal);
+
+#ifdef __cplusplus
+}
+#endif

+ 68 - 0
components/hal/include/hal/parlio_types.h

@@ -0,0 +1,68 @@
+/*
+ * SPDX-FileCopyrightText: 2021-2022 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#pragma once
+
+#include <stdint.h>
+#include <stdbool.h>
+#include "soc/soc_caps.h"
+#include "soc/clk_tree_defs.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/**
+ * @brief Parallel IO sample edge
+ */
+typedef enum {
+    PARLIO_SAMPLE_EDGE_NEG, /*!< Sample data on falling edge of clock */
+    PARLIO_SAMPLE_EDGE_POS, /*!< Sample data on rising edge of clock */
+} parlio_sample_edge_t;
+
+/**
+ * @brief Parallel IO bit packing order
+ *
+ * Data in memory:
+ * Byte 0: MSB < B0.7 B0.6 B0.5 B0.4 B0.3 B0.2 B0.1 B0.0 > LSB
+ * Byte 1: MSB < B1.7 B1.6 B1.5 B1.4 B1.3 B1.2 B1.1 B1.0 > LSB
+ *
+ * Output on line (PARLIO_BIT_PACK_ORDER_LSB):
+ *          Cycle 0   Cycle 1   Cycle 2  ---> time
+ * GPIO 0:   B0.0      B0.4      B1.0
+ * GPIO 1:   B0.1      B0.5      B1.1
+ * GPIO 2:   B0.2      B0.6      B1.2
+ * GPIO 3:   B0.3      B0.7      B1.3
+ *
+ * Output on line (PARLIO_BIT_PACK_ORDER_MSB):
+ *          Cycle 0   Cycle 1   Cycle 2  ---> time
+ * GPIO 0:   B0.4      B0.0      B1.4
+ * GPIO 1:   B0.5      B0.1      B1.5
+ * GPIO 2:   B0.6      B0.2      B1.6
+ * GPIO 3:   B0.7      B0.3      B1.7
+ */
+typedef enum {
+    PARLIO_BIT_PACK_ORDER_LSB, /*!< Bit pack order: LSB */
+    PARLIO_BIT_PACK_ORDER_MSB, /*!< Bit pack order: MSB */
+} parlio_bit_pack_order_t;
+
+#if SOC_PARLIO_SUPPORTED
+/**
+ * @brief Parallel IO clock source
+ * @note User should select the clock source based on the power and resolution requirement
+ */
+typedef soc_periph_parlio_clk_src_t parlio_clock_source_t;
+
+/// Maximum data width of TX unit
+#define PARLIO_TX_UNIT_MAX_DATA_WIDTH SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH
+#else
+typedef int parlio_clock_source_t;
+#define PARLIO_TX_UNIT_MAX_DATA_WIDTH 0
+#endif // SOC_PARLIO_SUPPORTED
+
+#ifdef __cplusplus
+}
+#endif

+ 19 - 0
components/hal/parlio_hal.c

@@ -0,0 +1,19 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include <stddef.h>
+#include "hal/parlio_hal.h"
+#include "hal/parlio_ll.h"
+
+void parlio_hal_init(parlio_hal_context_t *hal)
+{
+    hal->regs = &PARL_IO;
+}
+
+void parlio_hal_deinit(parlio_hal_context_t *hal)
+{
+    hal->regs = NULL;
+}

+ 4 - 0
components/soc/CMakeLists.txt

@@ -78,6 +78,10 @@ if(CONFIG_SOC_LCDCAM_SUPPORTED OR CONFIG_SOC_LCD_I80_SUPPORTED)
     list(APPEND srcs "${target}/lcd_periph.c")
 endif()
 
+if(CONFIG_SOC_PARLIO_SUPPORTED)
+    list(APPEND srcs "${target}/parlio_periph.c")
+endif()
+
 if(CONFIG_SOC_MCPWM_SUPPORTED)
     list(APPEND srcs "${target}/mcpwm_periph.c")
 endif()

+ 28 - 0
components/soc/esp32c6/include/soc/Kconfig.soc_caps.in

@@ -39,6 +39,10 @@ config SOC_ETM_SUPPORTED
     bool
     default y
 
+config SOC_PARLIO_SUPPORTED
+    bool
+    default y
+
 config SOC_BT_SUPPORTED
     bool
     default y
@@ -667,6 +671,30 @@ config SOC_MCPWM_CAPTURE_CLK_FROM_GROUP
     bool
     default y
 
+config SOC_PARLIO_GROUPS
+    int
+    default 1
+
+config SOC_PARLIO_TX_UNITS_PER_GROUP
+    int
+    default 1
+
+config SOC_PARLIO_RX_UNITS_PER_GROUP
+    int
+    default 1
+
+config SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH
+    int
+    default 16
+
+config SOC_PARLIO_RX_UNIT_MAX_DATA_WIDTH
+    int
+    default 16
+
+config SOC_PARLIO_TX_RX_SHARE_INTERRUPT
+    bool
+    default y
+
 config SOC_RSA_MAX_BIT_LEN
     int
     default 3072

+ 16 - 0
components/soc/esp32c6/include/soc/clk_tree_defs.h

@@ -405,6 +405,22 @@ typedef enum {
     LEDC_USE_RTC8M_CLK __attribute__((deprecated("please use 'LEDC_USE_RC_FAST_CLK' instead"))) = LEDC_USE_RC_FAST_CLK,   /*!< Alias of 'LEDC_USE_RC_FAST_CLK' */
 } soc_periph_ledc_clk_src_legacy_t;
 
+//////////////////////////////////////////////////PARLIO////////////////////////////////////////////////////////////////
+
+/**
+ * @brief Array initializer for all supported clock sources of PARLIO
+ */
+#define SOC_PARLIO_CLKS {SOC_MOD_CLK_XTAL, SOC_MOD_CLK_PLL_F240M}
+
+/**
+ * @brief PARLIO clock source
+ */
+typedef enum {
+    PARLIO_CLK_SRC_XTAL = SOC_MOD_CLK_XTAL,           /*!< Select XTAL as the source clock */
+    PARLIO_CLK_SRC_PLL_F240M = SOC_MOD_CLK_PLL_F240M, /*!< Select PLL_F240M as the source clock */
+    PARLIO_CLK_SRC_DEFAULT = SOC_MOD_CLK_PLL_F240M,   /*!< Select PLL_F240M as the default clock choice */
+} soc_periph_parlio_clk_src_t;
+
 #ifdef __cplusplus
 }
 #endif

+ 1 - 0
components/soc/esp32c6/include/soc/periph_defs.h

@@ -37,6 +37,7 @@ typedef enum {
     PERIPH_GDMA_MODULE,
     PERIPH_MCPWM0_MODULE,
     PERIPH_ETM_MODULE,
+    PERIPH_PARLIO_MODULE,
     PERIPH_SYSTIMER_MODULE,
     PERIPH_SARADC_MODULE,
     PERIPH_TEMPSENSOR_MODULE,

+ 9 - 0
components/soc/esp32c6/include/soc/soc_caps.h

@@ -34,6 +34,7 @@
 #define SOC_MCPWM_SUPPORTED             1
 #define SOC_TWAI_SUPPORTED              1
 #define SOC_ETM_SUPPORTED               1
+#define SOC_PARLIO_SUPPORTED            1
 #define SOC_BT_SUPPORTED                1
 #define SOC_IEEE802154_SUPPORTED        1
 #define SOC_ASYNC_MEMCPY_SUPPORTED      1
@@ -279,6 +280,14 @@
 /*------------------------ USB SERIAL JTAG CAPS ------------------------------*/
 // #define SOC_USB_SERIAL_JTAG_SUPPORT_LIGHT_SLEEP     (1)     /*!< Support to maintain minimum usb communication during light sleep */ // TODO: IDF-6395
 
+/*-------------------------- PARLIO CAPS --------------------------------------*/
+#define SOC_PARLIO_GROUPS                    1U  /*!< Number of parallel IO peripherals */
+#define SOC_PARLIO_TX_UNITS_PER_GROUP        1U  /*!< number of TX units in each group */
+#define SOC_PARLIO_RX_UNITS_PER_GROUP        1U  /*!< number of RX units in each group */
+#define SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH    16  /*!< Number of data lines of the TX unit */
+#define SOC_PARLIO_RX_UNIT_MAX_DATA_WIDTH    16  /*!< Number of data lines of the RX unit */
+#define SOC_PARLIO_TX_RX_SHARE_INTERRUPT     1   /*!< TX and RX unit share the same interrupt source number */
+
 /*--------------------------- RSA CAPS ---------------------------------------*/
 #define SOC_RSA_MAX_BIT_LEN    (3072)
 

+ 66 - 0
components/soc/esp32c6/parlio_periph.c

@@ -0,0 +1,66 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "soc/parlio_periph.h"
+#include "soc/gpio_sig_map.h"
+
+const parlio_signal_conn_t parlio_periph_signals = {
+    .groups = {
+        [0] = {
+            .module = PERIPH_PARLIO_MODULE,
+            .tx_irq_id = ETS_PARL_IO_INTR_SOURCE,
+            .rx_irq_id = ETS_PARL_IO_INTR_SOURCE,
+            .tx_units = {
+                [0] = {
+                    .data_sigs = {
+                        PARL_TX_DATA0_IDX,
+                        PARL_TX_DATA1_IDX,
+                        PARL_TX_DATA2_IDX,
+                        PARL_TX_DATA3_IDX,
+                        PARL_TX_DATA4_IDX,
+                        PARL_TX_DATA5_IDX,
+                        PARL_TX_DATA6_IDX,
+                        PARL_TX_DATA7_IDX,
+                        PARL_TX_DATA8_IDX,
+                        PARL_TX_DATA9_IDX,
+                        PARL_TX_DATA10_IDX,
+                        PARL_TX_DATA11_IDX,
+                        PARL_TX_DATA12_IDX,
+                        PARL_TX_DATA13_IDX,
+                        PARL_TX_DATA14_IDX,
+                        PARL_TX_DATA15_IDX,
+                    },
+                    .clk_out_sig = PARL_TX_CLK_OUT_IDX,
+                    .clk_in_sig = PARL_TX_CLK_IN_IDX,
+                }
+            },
+            .rx_units = {
+                [0] = {
+                    .data_sigs = {
+                        PARL_RX_DATA0_IDX,
+                        PARL_RX_DATA1_IDX,
+                        PARL_RX_DATA2_IDX,
+                        PARL_RX_DATA3_IDX,
+                        PARL_RX_DATA4_IDX,
+                        PARL_RX_DATA5_IDX,
+                        PARL_RX_DATA6_IDX,
+                        PARL_RX_DATA7_IDX,
+                        PARL_RX_DATA8_IDX,
+                        PARL_RX_DATA9_IDX,
+                        PARL_RX_DATA10_IDX,
+                        PARL_RX_DATA11_IDX,
+                        PARL_RX_DATA12_IDX,
+                        PARL_RX_DATA13_IDX,
+                        PARL_RX_DATA14_IDX,
+                        PARL_RX_DATA15_IDX,
+                    },
+                    .clk_out_sig = -1,
+                    .clk_in_sig = PARL_RX_CLK_IN_IDX,
+                }
+            }
+        },
+    },
+};

+ 32 - 0
components/soc/esp32h2/include/soc/Kconfig.soc_caps.in

@@ -79,6 +79,10 @@ config SOC_RMT_SUPPORTED
     bool
     default y
 
+config SOC_PARLIO_SUPPORTED
+    bool
+    default y
+
 config SOC_GPSPI_SUPPORTED
     bool
     default y
@@ -615,6 +619,34 @@ config SOC_MCPWM_CAPTURE_CLK_FROM_GROUP
     bool
     default y
 
+config SOC_PARLIO_GROUPS
+    int
+    default 1
+
+config SOC_PARLIO_TX_UNITS_PER_GROUP
+    int
+    default 1
+
+config SOC_PARLIO_RX_UNITS_PER_GROUP
+    int
+    default 1
+
+config SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH
+    int
+    default 8
+
+config SOC_PARLIO_RX_UNIT_MAX_DATA_WIDTH
+    int
+    default 8
+
+config SOC_PARLIO_TX_CLK_SUPPORT_GATING
+    bool
+    default y
+
+config SOC_PARLIO_TRANS_BIT_ALIGN
+    bool
+    default y
+
 config SOC_RTC_CNTL_CPU_PD_DMA_BUS_WIDTH
     int
     default 128

+ 16 - 0
components/soc/esp32h2/include/soc/clk_tree_defs.h

@@ -415,6 +415,22 @@ typedef enum {
     LEDC_USE_RTC8M_CLK __attribute__((deprecated("please use 'LEDC_USE_RC_FAST_CLK' instead"))) = LEDC_USE_RC_FAST_CLK,   /*!< Alias of 'LEDC_USE_RC_FAST_CLK' */
 } soc_periph_ledc_clk_src_legacy_t;
 
+//////////////////////////////////////////////////PARLIO////////////////////////////////////////////////////////////////
+
+/**
+ * @brief Array initializer for all supported clock sources of PARLIO
+ */
+#define SOC_PARLIO_CLKS {SOC_MOD_CLK_XTAL, SOC_MOD_CLK_PLL_F96M}
+
+/**
+ * @brief PARLIO clock source
+ */
+typedef enum {
+    PARLIO_CLK_SRC_XTAL = SOC_MOD_CLK_XTAL,          /*!< Select XTAL as the source clock */
+    PARLIO_CLK_SRC_PLL_F96M = SOC_MOD_CLK_PLL_F96M,  /*!< Select PLL_F96M as the source clock */
+    PARLIO_CLK_SRC_DEFAULT = SOC_MOD_CLK_PLL_F96M,   /*!< Select PLL_F96M as the default clock choice */
+} soc_periph_parlio_clk_src_t;
+
 #ifdef __cplusplus
 }
 #endif

+ 2 - 2
components/soc/esp32h2/include/soc/parl_io_struct.h

@@ -1,5 +1,5 @@
 /**
- * SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
  *
  *  SPDX-License-Identifier: Apache-2.0
  */
@@ -467,7 +467,7 @@ typedef union {
 } parl_io_version_reg_t;
 
 
-typedef struct {
+typedef struct parl_io_dev_t {
     volatile parl_io_rx_mode_cfg_reg_t rx_mode_cfg;
     volatile parl_io_rx_data_cfg_reg_t rx_data_cfg;
     volatile parl_io_rx_genrl_cfg_reg_t rx_genrl_cfg;

+ 1 - 0
components/soc/esp32h2/include/soc/periph_defs.h

@@ -40,6 +40,7 @@ typedef enum {
     PERIPH_GDMA_MODULE,
     PERIPH_MCPWM0_MODULE,
     PERIPH_ETM_MODULE,
+    PERIPH_PARLIO_MODULE,
     PERIPH_SYSTIMER_MODULE,
     PERIPH_SARADC_MODULE,
     PERIPH_TEMPSENSOR_MODULE,

+ 10 - 0
components/soc/esp32h2/include/soc/soc_caps.h

@@ -48,6 +48,7 @@
 #define SOC_SDM_SUPPORTED               1
 #define SOC_ETM_SUPPORTED               1
 #define SOC_RMT_SUPPORTED               1
+#define SOC_PARLIO_SUPPORTED            1
 #define SOC_GPSPI_SUPPORTED             1
 #define SOC_LEDC_SUPPORTED              1
 #define SOC_I2C_SUPPORTED               1
@@ -268,6 +269,15 @@
 /*------------------------ USB SERIAL JTAG CAPS ------------------------------*/
 // #define SOC_USB_SERIAL_JTAG_SUPPORT_LIGHT_SLEEP     (1)     /*!< Support to maintain minimum usb communication during light sleep */ // TODO: IDF-6395
 
+/*-------------------------- PARLIO CAPS --------------------------------------*/
+#define SOC_PARLIO_GROUPS                    1U /*!< Number of parallel IO peripherals */
+#define SOC_PARLIO_TX_UNITS_PER_GROUP        1U /*!< number of TX units in each group */
+#define SOC_PARLIO_RX_UNITS_PER_GROUP        1U /*!< number of RX units in each group */
+#define SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH    8  /*!< Number of data lines of the TX unit */
+#define SOC_PARLIO_RX_UNIT_MAX_DATA_WIDTH    8  /*!< Number of data lines of the RX unit */
+#define SOC_PARLIO_TX_CLK_SUPPORT_GATING     1  /*!< Support gating TX clock */
+#define SOC_PARLIO_TRANS_BIT_ALIGN           1  /*!< Support bit alignment in transaction */
+
 // TODO: IDF-6267 (Copy from esp32c6, need check)
 /*-------------------------- RTC CAPS --------------------------------------*/
 #define SOC_RTC_CNTL_CPU_PD_DMA_BUS_WIDTH       (128)

+ 50 - 0
components/soc/esp32h2/parlio_periph.c

@@ -0,0 +1,50 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#include "soc/parlio_periph.h"
+#include "soc/gpio_sig_map.h"
+
+const parlio_signal_conn_t parlio_periph_signals = {
+    .groups = {
+        [0] = {
+            .module = PERIPH_PARLIO_MODULE,
+            .tx_irq_id = ETS_PARL_IO_TX_INTR_SOURCE,
+            .rx_irq_id = ETS_PARL_IO_RX_INTR_SOURCE,
+            .tx_units = {
+                [0] = {
+                    .data_sigs = {
+                        PARL_TX_DATA0_IDX,
+                        PARL_TX_DATA1_IDX,
+                        PARL_TX_DATA2_IDX,
+                        PARL_TX_DATA3_IDX,
+                        PARL_TX_DATA4_IDX,
+                        PARL_TX_DATA5_IDX,
+                        PARL_TX_DATA6_IDX,
+                        PARL_TX_DATA7_IDX,
+                    },
+                    .clk_out_sig = PARL_TX_CLK_OUT_IDX,
+                    .clk_in_sig = PARL_TX_CLK_IN_IDX,
+                }
+            },
+            .rx_units = {
+                [0] = {
+                    .data_sigs = {
+                        PARL_RX_DATA0_IDX,
+                        PARL_RX_DATA1_IDX,
+                        PARL_RX_DATA2_IDX,
+                        PARL_RX_DATA3_IDX,
+                        PARL_RX_DATA4_IDX,
+                        PARL_RX_DATA5_IDX,
+                        PARL_RX_DATA6_IDX,
+                        PARL_RX_DATA7_IDX,
+                    },
+                    .clk_out_sig = PARL_RX_CLK_OUT_IDX,
+                    .clk_in_sig = PARL_RX_CLK_IN_IDX,
+                }
+            }
+        },
+    },
+};

+ 41 - 0
components/soc/include/soc/parlio_periph.h

@@ -0,0 +1,41 @@
+/*
+ * SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
+ *
+ * SPDX-License-Identifier: Apache-2.0
+ */
+
+#pragma once
+
+#include <stdint.h>
+#include "soc/soc_caps.h"
+#include "soc/periph_defs.h"
+#include "soc/parl_io_reg.h"
+#include "soc/parl_io_struct.h"
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+typedef struct {
+    struct {
+        struct {
+            const int data_sigs[SOC_PARLIO_TX_UNIT_MAX_DATA_WIDTH];
+            const int clk_out_sig;
+            const int clk_in_sig;
+        } tx_units[SOC_PARLIO_TX_UNITS_PER_GROUP];
+        struct {
+            const int data_sigs[SOC_PARLIO_RX_UNIT_MAX_DATA_WIDTH];
+            const int clk_out_sig;
+            const int clk_in_sig;
+        } rx_units[SOC_PARLIO_RX_UNITS_PER_GROUP];
+        const int tx_irq_id;
+        const int rx_irq_id;
+        const periph_module_t module;
+    } groups[SOC_PARLIO_GROUPS];
+} parlio_signal_conn_t;
+
+extern const parlio_signal_conn_t parlio_periph_signals;
+
+#ifdef __cplusplus
+}
+#endif