async_memcpy.rst 5.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  1. 异步内存复制
  2. ========================
  3. :link_to_translation:`en:[English]`
  4. 概述
  5. --------
  6. {IDF_TARGET_NAME} 有一个 DMA 引擎,能够以异步方式帮助 CPU 完成内部内存复制操作。
  7. 异步 memcpy API 中封装了所有 DMA 配置和操作,:cpp:func:`esp_async_memcpy` 的签名与标准 C 库的 ``memcpy`` 函数基本相同。
  8. DMA 允许多个内存复制请求在首个请求完成之前排队,即允许计算和内存复制的重叠。此外,通过注册事件回调函数,还可以知道内存复制请求完成的准确时间。
  9. .. only:: SOC_AHB_GDMA_SUPPORT_PSRAM
  10. 如果异步 memcpy 是基于 AHB GDMA 构建的,那么也可以用适合的对齐方式从 PSRAM 复制数据或向其中复制数据。
  11. .. only:: SOC_AXI_GDMA_SUPPORT_PSRAM
  12. 如果异步 memcpy 是基于 AXI GDMA 构建的,那么也可以用适合的对齐方式从 PSRAM 复制数据或向其中复制数据。
  13. 配置并安装驱动
  14. ----------------------------
  15. 安装异步 memcpy 驱动的方法取决于底层 DMA 引擎:
  16. .. list::
  17. :SOC_CP_DMA_SUPPORTED: - :cpp:func:`esp_async_memcpy_install_cpdma` 用于安装基于 CP DMA 引擎的异步 memcpy 驱动。
  18. :SOC_AHB_GDMA_SUPPORTED: - :cpp:func:`esp_async_memcpy_install_gdma_ahb` 用于安装基于 AHB GDMA 引擎的异步 memcpy 驱动。
  19. :SOC_AXI_GDMA_SUPPORTED: - :cpp:func:`esp_async_memcpy_install_gdma_axi` 用于安装基于 AXI GDMA 引擎的异步 memcpy 驱动。
  20. - :cpp:func:`esp_async_memcpy_install` 是一个通用 API,用于安装带有默认 DMA 引擎的异步 memcpy 驱动。如果 SoC 具有 CP DMA 引擎,则默认 DMA 引擎为 CP DMA,否则,默认 DMA 引擎为 AHB GDMA。
  21. 在 :cpp:type:`async_memcpy_config_t` 中设置驱动配置:
  22. * :cpp:member:`backlog`:此项用于配置首个请求完成前可以排队的最大内存复制事务数量。如果将此字段设置为零,会应用默认值 4。
  23. * :cpp:member:`sram_trans_align`:声明 SRAM 中数据地址和复制大小的对齐方式,如果数据没有对齐限制,则设置为零。如果设置为四的倍数值(即 4X),驱动程序将内部启用突发模式,这有利于某些和性能相关的应用程序。
  24. * :cpp:member:`psram_trans_align`:声明 PSRAM 中数据地址和复制大小的对齐方式。如果 memcpy 的目标地址位于 PSRAM 中,用户必须给出一个有效值(只支持 16、32、64)。如果设置为零,会默认采用 16 位对齐。在内部,驱动程序会根据对齐方式来配置 DMA 访问 PSRAM 时所用的块大小。
  25. * :cpp:member:`flags`:此项可以启用一些特殊的驱动功能。
  26. .. code-block:: c
  27. async_memcpy_config_t config = ASYNC_MEMCPY_DEFAULT_CONFIG();
  28. // 更新底层 DMA 引擎支持的最大数据流
  29. config.backlog = 8;
  30. async_memcpy_handle_t driver = NULL;
  31. ESP_ERROR_CHECK(esp_async_memcpy_install(&config, &driver)); // 使用默认 DMA 引擎安装驱动
  32. 发送内存复制请求
  33. ------------------------
  34. 使用 :cpp:func:`esp_async_memcpy` API 将内存复制请求发送到 DMA 引擎。在驱动程序成功安装后才能调用该 API。此 API 是线程安全的,因此可以从不同的任务中调用。
  35. 与 libc 版本的 ``memcpy`` 不同,你可以选择给 :cpp:func:`esp_async_memcpy` 设置一个回调函数,以便在内存复制完成时收到通知。注意,回调是在 ISR 上下文中执行的,请不要在回调中调用任何阻塞函数。
  36. 回调函数的原型是 :cpp:type:`async_memcpy_isr_cb_t`。回调函数只有在借助 RTOS API(如 :cpp:func:`xSemaphoreGiveFromISR`)唤醒了高优先级任务后才能返回 true。
  37. .. code-block:: c
  38. // 回调实现,在 ISR 上下文中运行
  39. static bool my_async_memcpy_cb(async_memcpy_handle_t mcp_hdl, async_memcpy_event_t *event, void *cb_args)
  40. {
  41. SemaphoreHandle_t sem = (SemaphoreHandle_t)cb_args;
  42. BaseType_t high_task_wakeup = pdFALSE;
  43. xSemaphoreGiveFromISR(semphr, &high_task_wakeup); // 如果解锁了一些高优先级任务,则将 high_task_wakeup 设置为 pdTRUE
  44. return high_task_wakeup == pdTRUE;
  45. }
  46. // 创建一个信号量,在异步 memcpy 完成时进行报告
  47. SemaphoreHandle_t semphr = xSemaphoreCreateBinary();
  48. // 从用户的上下文中调用
  49. ESP_ERROR_CHECK(esp_async_memcpy(driver_handle, to, from, copy_len, my_async_memcpy_cb, my_semaphore));
  50. // 其他事项
  51. xSemaphoreTake(my_semaphore, portMAX_DELAY); // 等待 buffer 复制完成
  52. 卸载驱动
  53. ----------------
  54. 使用 :cpp:func:`esp_async_memcpy_uninstall` 卸载异步 memcpy 驱动。无需在每次 memcpy 操作后手动卸载。如果你的应用程序不再需要此驱动,此 API 可以帮助回收内存和其他硬件资源。
  55. .. only:: SOC_ETM_SUPPORTED and SOC_GDMA_SUPPORT_ETM
  56. ETM 事件
  57. ---------
  58. 在异步 memcpy 操作完成时会生成一个事件,此事件能够与 :doc:`ETM </api-reference/peripherals/etm>` 模块进行交互。可以调用 :cpp:func:`esp_async_memcpy_new_etm_event` 获取 ETM 事件句柄。
  59. 如需了解如何将此事件连接到 ETM 通道,请参考文档 :doc:`ETM </api-reference/peripherals/etm>`。
  60. API 参考
  61. -------------
  62. .. include-build-file:: inc/esp_async_memcpy.inc