数据流转
本篇介绍数据在处理单元(element)之间的流转方式,涵盖数据载体(payload)的字段与所有权、数据端口(port)的 acquire-release 协议、外部接口(IO)在处理链首尾的衔接行为,以及字节缓存(cache)。
处理单元(element)的生命周期、端口属性、能力描述(capability)与运行时方法(method)见 GMF 处理单元。处理链(pipeline)编排、执行线程(task)调度、生命周期与控制接口见 GMF 处理链与任务调度。数据总线(data bus)四种实现及其选型见 数据总线。整体架构与对象关系参考 GMF-Core 概览。
数据载体、数据端口
每个处理单元对外只暴露两个数据接口:输入数据端口与输出数据端口。数据在数据端口之间以 esp_gmf_payload_t 的形式传递,处理单元不负责分配缓冲区,也不负责跨线程同步,只需按一套 acquire-release 协议读写数据。
一次 process 中数据的完整流动如下,以一条 Decoder → Resample 处理链为例。
sequenceDiagram
participant T as 执行线程
participant E as resample
participant IP as in_port
participant OP as out_port
participant D as 下游
T->>+E: process()
E->>+IP: acquire_in(size)
IP-->>-E: 输入数据载体 in
E->>+OP: acquire_out(size)
OP-->>-E: 输出数据载体 out
Note over E: 处理输入并生成输出
E->>OP: release_out(out)
OP->>D: 提交输出数据载体
E->>IP: release_in(in)
E-->>-T: ok
以上序列包含两条规则。第一,acquire_in 返回的数据载体在 release_in 之前仍归当前处理单元使用,端口会通过引用计数或下层实现维持这段数据的有效性;处理单元不需要自行处理跨线程同步。第二,release_out 提交输出数据载体。对块型数据端口,提交过程通常不拷贝数据,而是把数据载体交给下游输入端口或下层数据队列,一份数据可以穿越整条处理链而不产生中间拷贝。
acquire-release 是 GMF-Core 数据通路的基础协议,下面分别展开数据载体与数据端口的细节。
数据载体
字段
esp_gmf_payload_t 定义在 esp_gmf_payload.h,字段如下。
字段 |
类型 |
含义 |
|---|---|---|
|
|
指向数据缓冲区 |
|
|
缓冲区总长度 |
|
|
缓冲区中有效数据的长度 |
|
|
流结束标志,最后一块数据置 1,触发处理链进入 |
|
|
显示时间戳,用于音视频同步 |
|
|
1 位位域,指示 |
|
|
7 位位域,目前定义了 |
生命周期与所有权
处理单元代码中的数据载体表现为一个 esp_gmf_payload_t * 指针。所有权在 acquire-release 期间属于当前处理单元,归还后回到数据端口与数据总线管理。使用时遵循三条规则:
不要自行 free:acquire 得到的数据载体由数据端口管理,处理单元只能读写
buf内容,不能调用esp_gmf_payload_delete。valid_size 由写方设置:申请到一块输出数据载体后,处理单元写入数据并把实际写入的字节数填到
valid_size,下游据此判断有多少有效数据。acquire 与 release 成对:即使处理过程出错也必须 release,否则数据端口进入泄漏状态,后续调用会耗尽缓冲区。
处理单元自己分配数据载体的场景(较少见,典型是源头外部接口处理单元)可以用 esp_gmf_payload_new() 或 esp_gmf_payload_new_with_len() 显式创建,在不再引用时用 esp_gmf_payload_delete() 释放。需要满足 PSRAM cache 对齐时用 esp_gmf_payload_realloc_aligned_buf()。
is_done 与 pts
is_done 是处理链从 RUNNING 自然过渡到 FINISHED 的信号。外部接口处理单元读到文件末尾时把 is_done 置 1 随数据载体下发,下游处理单元依次处理完带结束标记的数据载体后,逐级触发 close。
pts 是显示时间戳,通常由源头处理单元填充。中间处理单元按需更新或直接透传。接近播放端的处理单元据此做音视频同步。
数据端口
方向与类型
每个数据端口由两个属性定义。
方向:ESP_GMF_PORT_DIR_IN 表示输入数据端口(从上游读取数据载体),ESP_GMF_PORT_DIR_OUT 表示输出数据端口(向下游写入数据载体)。每个处理单元有一个输入数据端口和一个输出数据端口。
类型:字节型和块型两种,对应两种数据交换策略。
ESP_GMF_PORT_TYPE_BYTE(字节型)让处理单元按任意字节数读写,框架负责必要时的内部拷贝与拼接。适合解码器解析协议头这类需要精确定长访问的场景。ESP_GMF_PORT_TYPE_BLOCK(块型)一次访问一整块,不做拷贝,只在数据端口之间传递缓冲区地址。适合处理整帧视频或批量 PCM,性能更高但可配置性较低。
acquire-release 协议
数据端口对外暴露四个 API,成对使用形成 acquire-release 访问协议。以 acquire_in 为例说明完整签名:
esp_gmf_err_io_t esp_gmf_port_acquire_in(
esp_gmf_port_handle_t handle,
esp_gmf_payload_t **load,
uint32_t wanted_size,
int wait_ticks);
四个参数的作用:
handle是输入数据端口句柄。load是输出参数,调用返回后*load指向一块准备好供读取的数据载体。wanted_size是处理单元期望获取的数据长度。对源头外部接口或绑定数据总线的数据端口,底层实现会按该值准备数据;对已经由上游处理单元提交到输入数据端口的数据载体,端口会直接返回当前数据载体,实际有效长度以(*load)->valid_size为准。wait_ticks是等待超时时间,单位为系统 tick。ESP_GMF_MAX_DELAY表示无限等待,0表示立即返回。
返回值类型是 esp_gmf_err_io_t:
返回码 |
值 |
含义 |
|---|---|---|
|
|
成功,值表示实际读到的字节数 |
|
-1 |
操作失败 |
|
-2 |
等待超时 |
|
-3 |
被 |
处理单元代码处理返回值的模板:
esp_gmf_payload_t *in_load = NULL;
esp_gmf_err_io_t ret = esp_gmf_port_acquire_in(in_port, &in_load, 1024, ESP_GMF_MAX_DELAY);
if (ret == ESP_GMF_IO_ABORT) {
return ESP_GMF_JOB_ERR_ABORT;
} else if (ret < 0) {
return ESP_GMF_JOB_ERR_FAIL;
}
/* 使用 in_load->buf, in_load->valid_size */
四个 API 成对工作:
读入侧:
acquire_in获取可读数据载体,处理单元消费后release_in归还写出侧:
acquire_out获取可写数据载体,处理单元填充后release_out提交
process 中的典型调用顺序是先 acquire_in 获取输入,再 acquire_out 准备输出缓冲区,处理完后先 release_out 提交下游,最后 release_in 归还输入。先申请 out 再处理,可以在下游阻塞时提前等待,避免处理完成后才发现输出端口不可用。
acquire 与 release 必须成对出现,即使处理过程出错也要 release。框架提供了几个辅助宏简化错误路径:
ESP_GMF_PORT_ACQUIRE_IN_CHECK(TAG, ret, err, goto ACQ_FAIL);
ESP_GMF_PORT_RELEASE_IN_CHECK(TAG, ret, err, goto REL_FAIL);
这些宏把返回值检查和错误分支收敛到一行。
共享缓冲区管理
GMF 通过两种机制减少内存占用,避免不必要的数据拷贝。
引用计数
旁路或原地处理的处理单元,可将同一份数据载体直接传递给下游处理单元或用户应用。每增加一个使用者,数据端口的 ref_count 加一;对应 release_in / release_out 完成后减一。计数归零时,端口调用下层 release 回调回收缓冲区。共享行为由数据端口的 is_shared 标志控制,可通过 esp_gmf_port_enable_payload_share() 开启或关闭(默认可共享)。
flowchart LR
Source --> Bypass
Bypass --> User["用户"]
Bypass --> Downstream["下游处理单元"]
上述流程中始终共享同一份数据,无需额外拷贝。
缓冲区复用(AB Buffer)
需要生成新输出数据的处理单元,框架通过 is_shared 在各处理单元之间复用数据载体缓冲区。以处理链 A → B → C 为例:
A → B → C
A 输出 Buffer A。
B 以 Buffer A 为输入,处理后写入 Buffer B。
B 完成后,Buffer A 可被回收并再次分配。
C 以 Buffer B 为输入,其输出可复用已释放的 Buffer A。
数据沿处理链推进时,Buffer A 与 Buffer B 交替复用。处理链包含多个处理单元时,通常只需两块工作缓冲区即可完成流转,降低内存消耗。
处理单元在初始化时会声明输入输出数据端口的约束,例如端口类型、对齐要求和每次 acquire 推荐的数据量。端口属性属于处理单元配置,完整说明见 GMF 处理单元。
数据端口对应的数据队列由数据总线负责。GMF-Core 提供四种数据总线(ringbuffer / fifo / block / pbuf),数据端口在创建时绑定其中一种,处理单元代码看到的接口完全一致。四种实现的取舍与流控接口见 数据总线。
外部接口(IO)
外部接口(esp_gmf_io_t)是处理链首尾的特殊处理单元,承担文件、网络、编解码器等外部数据源的读写。除基础的 open / acquire / release / close 外,还支持异步模式:由 io_process 执行线程将数据写入内部数据总线,调用方从缓存读取,以平滑 HTTP 下载等速率波动;可通过 esp_gmf_io_speed_stats_t 获取即时与平均传输速率(Kbps)。此外还提供数据源切换、中断恢复等高级控制接口。
同步与异步两种工作模式
外部接口在创建时通过 esp_gmf_io_cfg_t 配置工作模式。
同步模式:thread.stack = 0 且不配置 buffer_cfg。esp_gmf_io_acquire_read 直接在调用线程上下文中执行底层外部接口操作(例如 fread)。优点是延迟低、内存占用小;缺点是调用方必须能容忍外部接口操作存在较长阻塞(例如网络读)。
异步模式:thread.stack > 0 且配置 buffer_cfg.buffer_size。框架内部创建一个 io_process 执行线程,独立完成底层外部接口操作并把数据缓存到数据总线。用户调用 acquire_read 时实际从数据总线中读取,与底层外部接口解耦。这种模式适合”读固定字节数”或”生产消费速率差异大”的场景,例如解码器需要稳定的字节流而网络下载存在突发。
esp_gmf_io_cfg_t cfg = {
.thread = {
.stack = 4 * 1024,
.prio = 5,
.core = 0,
},
.buffer_cfg = {
.io_size = 4096, /* 每次底层 IO 读取大小 */
.buffer_size = 16 * 1024,
.read_filter = NULL,
},
.enable_speed_monitor = true,
};
URL 评分策略
esp_gmf_io_t 的 get_score 用于获取匹配度,便于从注册池中的多个外部接口里找到最匹配的外部接口。注册池函数 esp_gmf_pool_get_io_tag_by_url 遍历所有已注册外部接口的 get_score,返回评分最高者。三档评分定义:
评分 |
值 |
含义 |
|---|---|---|
|
0 |
外部接口不支持该 URL |
|
50 |
协议头匹配,例如 |
|
100 |
协议加扩展名都匹配,或专用 IO 高优先级匹配 |
自定义外部接口实现 get_score 时可以返回任意中间值,便于在多个候选外部接口中区分优先级。
无缝衔接与重置
esp_gmf_io_reload() 用一个新 URI 重新开启已有外部接口,不销毁底层连接。主要面向 HLS 切片这类”同主机连续下载多个分片”的场景,复用 HTTP 连接以避免反复握手的开销。重新加载内部对异步外部接口做了三件事:清 data_bus 的 EOF 标志、清 abort 标志、调用外部接口的 reload 回调重新打开新 URI。reload 必须在前一次读取完成后调用。
esp_gmf_io_reset() 做更全面的清理:把位置和文件大小清零、调用外部接口的 reset 回调、重置执行线程并重新加载外部接口 process job。常用于处理链 reset 之后对外部接口做配套清理。
done / abort 控制
IO 层对 done 与 abort 的控制接口与数据总线一一对应,但在异步模式下还会额外管理 IO 执行线程的 hold 状态。
esp_gmf_io_done()标记 EOF:abort 异步外部接口的数据总线并把执行线程 hold 住,下游acquire会获取带is_done的数据载体。调用方在切换到下一个数据源前必须调用esp_gmf_io_clear_done()释放被 hold 的执行线程,否则后续外部接口不会推进。esp_gmf_io_abort()立即中止当前操作:数据总线 abort、执行线程 hold,所有acquire/release返回ESP_GMF_IO_ABORT。配套的esp_gmf_io_clear_abort()恢复正常工作。
播放列表切换曲目的典型流程:
/* 当前曲目播放完毕 */
esp_gmf_io_done(io_handle);
/* 等待处理链处理完剩余数据载体 */
...
/* 切换到下一首 */
esp_gmf_io_set_uri(io_handle, next_uri);
esp_gmf_io_clear_done(io_handle);
/* 处理链继续运行,无需重建 */
自定义数据载体处理
esp_gmf_io_buffer_cfg_t 的 read_filter 回调允许在外部接口读到原始数据后、写入数据总线之前插入自定义处理。常见用途包括解密、字节序转换、协议解封装。回调签名:
esp_gmf_err_t (*read_filter)(esp_gmf_io_handle_t obj,
void *payload,
uint32_t wanted_size,
int block_ticks);
读到的数据载体在回调中可以原地修改 buf 与 valid_size。
速度监控
把 esp_gmf_io_cfg_t 的 enable_speed_monitor 置 true 后,框架会维护当前与平均传输速度(Kbps),通过 esp_gmf_io_get_speed_stats() 取出 esp_gmf_io_speed_stats_t。运行中也可以用 esp_gmf_io_enable_speed_monitor() 动态开关。常用于网络状况监测与传输速率自适应。
字节缓存(cache)
块型数据端口一次 acquire 返回一整块数据,而解码器这类处理单元常常需要”从数据流中精确读 N 个字节”。esp_gmf_cache 模块提供字节级缓存能力:
esp_gmf_cache_handle_t cache;
esp_gmf_cache_new(1024, &cache);
/* 每次 acquire 到一块 block 数据载体后,提交给 cache */
esp_gmf_cache_load(cache, in_load->buf, in_load->valid_size);
/* 然后按任意字节数读 */
int read;
esp_gmf_cache_acquire(cache, 7, out_buf, &read);
esp_gmf_cache_release(cache, 7);
字节缓存内部维护一个可扩展的滑动缓冲区。处理单元可以连续把多块数据载体中的字节加载到字节缓存中,再按需要读取任意长度的数据;即使一次读取跨越两块数据载体的边界,字节缓存也会从内部缓冲区中连续返回。典型使用场景是音频解码器解析协议头或帧头。
API 参考
本篇涉及的头文件:
esp_gmf_payload.h:数据载体结构与生命周期esp_gmf_port.h:数据端口的配置与访问 APIesp_gmf_cache.h:字节缓存esp_gmf_io.h:外部接口对象的公共接口
Header File
Functions
-
esp_gmf_err_t esp_gmf_payload_new(esp_gmf_payload_t **instance)
Create a new payload instance without buffer.
- 参数:
instance – [out] Pointer to store the newly created payload instance
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_MEMORY_LACK Not enough memory to create the payload instance
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
esp_gmf_err_t esp_gmf_payload_new_with_len(uint32_t buf_length, esp_gmf_payload_t **instance)
Create a new payload instance and buffer by specified buffer length.
- 参数:
buf_length – [in] Length of the payload buffer
instance – [out] Pointer to store the newly created payload instance
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_MEMORY_LACK Not enough memory to create the payload instance
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
esp_gmf_err_t esp_gmf_payload_copy_data(esp_gmf_payload_t *src, esp_gmf_payload_t *dest)
Copy data and done flag from one payload instance to another.
- 参数:
src – [in] Source payload instance
dest – [out] Destination payload instance
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
esp_gmf_err_t esp_gmf_payload_realloc_buf(esp_gmf_payload_t *instance, uint32_t new_length)
Reallocate the buffer of a payload instance to the specified length Unlike
realloc,the original valid data in the buffer was not copied to the new buffer.- 参数:
instance – [in] Payload instance to reallocate the buffer for
new_length – [in] New length for the payload buffer
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_MEMORY_LACK Not enough memory to reallocate the buffer
ESP_GMF_ERR_INVALID_ARG Invalid instance or new_length is zero
-
esp_gmf_err_t esp_gmf_payload_realloc_aligned_buf(esp_gmf_payload_t *instance, uint8_t align, uint32_t new_length)
Reallocate the buffer of a payload instance to the specified length with specified byte alignment Behavior same as
esp_gmf_payload_realloc_bufwith an additional alignment request Unlikerealloc,the original valid data in the buffer was not copied to the new buffer.- 参数:
instance – [in] Payload instance to reallocate the buffer for
align – [in] Byte alignment for the new payload buffer
new_length – [in] New length for the payload buffer
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_MEMORY_LACK Not enough memory to reallocate the buffer
ESP_GMF_ERR_INVALID_ARG Invalid instance or new_length is zero
-
esp_gmf_err_t esp_gmf_payload_realloc_buffer_with_separate_alignment(esp_gmf_payload_t *instance, uint8_t addr_align, uint8_t size_align, uint32_t new_length)
Reallocate the payload buffer with separate address and length alignment.
备注
Unlike
realloc, the original valid data in the buffer is not copied to the new buffer.addr_align0 substitutesesp_gmf_oal_get_spiram_cache_align(); use1for natural heap alignment (calloc). Whensize_alignis greater than1, it must be a power of two; the allocated byte length is rounded up to a multiple ofsize_align. Use0or1forsize_alignto keepnew_lengthunchanged.- 参数:
instance – [in] Payload instance to reallocate the buffer for
addr_align – [in] Byte alignment of the buffer base address
size_align – [in] Byte alignment used to round up the requested buffer length
new_length – [in] Minimum usable length for the payload buffer
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_MEMORY_LACK Not enough memory to reallocate the buffer
ESP_GMF_ERR_INVALID_ARG Invalid instance, new_length is zero, or invalid alignment
-
esp_gmf_err_t esp_gmf_payload_set_done(esp_gmf_payload_t *instance)
Set the done flag for a payload instance.
- 参数:
instance – [in] Payload instance to set as done
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
esp_gmf_err_t esp_gmf_payload_clean_done(esp_gmf_payload_t *instance)
Clean the done flag for a payload instance.
- 参数:
instance – [in] Payload instance to clean the done status for
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
void esp_gmf_payload_delete(esp_gmf_payload_t *instance)
Delete a payload instance, if needs_free is set free associated resources.
- 参数:
instance – [in] Payload instance to delete
Structures
-
struct esp_gmf_payload_t
Structure representing a payload in GMF.
Public Members
-
uint8_t *buf
Pointer to the payload buffer
-
size_t buf_length
Length of the payload buffer
-
size_t valid_size
Size of valid data in the payload buffer
-
bool is_done
Flag indicating if this payload buffer marks the end of the stream
-
uint64_t pts
Presentation time stamp
-
uint8_t needs_free
Flag indicating if the payload buffer needs to be freed by esp_gmf_payload_delete or not
-
uint8_t meta_flag
Meta flag for the payload
-
uint8_t *buf
Macros
-
ESP_GMF_META_FLAG_AUD_RECOVERY_PLC
The meta flag for the current payload.
The current frame is recovered through the packet loss concealment (PLC) mechanism
Header File
Functions
-
esp_gmf_err_t esp_gmf_port_init(esp_gmf_port_config_t *cfg, esp_gmf_port_handle_t *out_result)
Initialize a GMF port with the given configuration.
- 参数:
cfg – [in] Pointer to the configuration structure
out_result – [out] Pointer to store the result handle after initialization
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_MEMORY_LACK Memory allocate failed
-
esp_gmf_err_t esp_gmf_port_deinit(esp_gmf_port_handle_t handle)
Deinitialize a GMF port.
- 参数:
handle – [in] The handle of the port to be deinitialized
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_set_payload(esp_gmf_port_handle_t handle, esp_gmf_payload_t *load)
Set the self payload for the specific port.
- 参数:
handle – [in] The handle of the port
load – [in] Pointer to the payload structure
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_clean_payload_done(esp_gmf_port_handle_t handle)
Clean the payload done status of a GMF port.
- 参数:
handle – [in] The handle of the port
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
Enables or disables shared payload for the specified port.
- 参数:
handle – [in] The port handle to enable or disable payload sharing on
enable – [in] Set to true to enable payload sharing, or false to disable it
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument provided
-
esp_gmf_err_t esp_gmf_port_reset(esp_gmf_port_handle_t handle)
Reset the port payload and variable of self payload.
- 参数:
handle – [in] The handle of the port
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_set_wait_ticks(esp_gmf_port_handle_t handle, int wait_ticks_ms)
Set the wait ticks for the specific port.
- 参数:
handle – [in] The handle of the port
wait_ticks_ms – [in] Number of milliseconds to wait
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_set_reader(esp_gmf_port_handle_t handle, void *reader)
Set the esp_gmf_port_acquire_in and esp_gmf_port_release_in caller for the specific port.
- 参数:
handle – [in] The handle of the port
reader – [in] Pointer to the reader
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_set_writer(esp_gmf_port_handle_t handle, void *writer)
Set the esp_gmf_port_acquire_out and esp_gmf_port_acquire_out caller for the specific port.
- 参数:
handle – [in] The handle of the port
writer – [in] Pointer to the writer
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_add_last(esp_gmf_port_handle_t head, esp_gmf_port_handle_t io_inst)
Add a GMF port to the end of the list.
- 参数:
head – [in] The head of the port list
io_inst – [in] The port instance to be added
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_port_del_at(esp_gmf_port_handle_t *head, esp_gmf_port_handle_t io_inst)
Delete a GMF port from the list.
- 参数:
head – [inout] A pointer to the head of the port list
io_inst – [in] The port instance to be deleted
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_io_t esp_gmf_port_acquire_in(esp_gmf_port_handle_t handle, esp_gmf_payload_t **load, uint32_t wanted_size, int wait_ticks)
Acquire the expected valid data into the specified payload, regardless of whether the payload is NULL or not If writer of port is valid, the payload from the previous element stored on the port
payloadpointer is fetched If the*loadpointer is NULL, a new payload will be allocated before calling the acquire operation The actual valid size is stored inload->valid_size- 参数:
handle – [in] The handle of the port
load – [inout] Pointer to store the acquired payload
wanted_size – [in] Size of the payload to be acquired
wait_ticks – [in] Number of ticks to wait, in milliseconds
- 返回:
ESP_GMF_IO_OK Operation successful
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_port_release_in(esp_gmf_port_handle_t handle, esp_gmf_payload_t *load, int wait_ticks)
Call the release operation or clean the payload pointer of the port.
- 参数:
handle – [in] The handle of the port
load – [in] Pointer to the payload to be released
wait_ticks – [in] Number of ticks to wait
- 返回:
ESP_GMF_IO_OK Operation successful
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_port_acquire_out(esp_gmf_port_handle_t handle, esp_gmf_payload_t **load, uint32_t wanted_size, int wait_ticks)
Acquire the buffer of the expected size into the specified payload, If the reader of the port is valid, store the provided or allocated payload to the input port of the next element If reader pointer is NULL, prepare a payload if
*loadis invalid before calling the acquire operation The actual valid size is stored inload->valid_size- 参数:
handle – [in] The handle of the port
load – [inout] Pointer to store the acquired payload
wanted_size – [in] Size of the payload to be acquired
wait_ticks – [in] Number of ticks to wait, in milliseconds
- 返回:
ESP_GMF_IO_OK Operation successful
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_port_release_out(esp_gmf_port_handle_t handle, esp_gmf_payload_t *load, int wait_ticks)
Call the release operation or clean the payload pointer of the port.
- 参数:
handle – [in] The handle of the port
load – [in] Pointer to the payload to be released
wait_ticks – [in] Number of ticks to wait
- 返回:
ESP_GMF_IO_OK Operation successful
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
static inline void *NEW_ESP_GMF_PORT(uint8_t dir, uint8_t type, void *acq, void *release, void *del, void *ctx, int length, int ticks_ms)
-
static inline void *NEW_ESP_GMF_PORT_IN_BYTE(void *acq, void *release, void *del, void *ctx, int length, int ticks_ms)
-
static inline void *NEW_ESP_GMF_PORT_OUT_BYTE(void *acq, void *release, void *del, void *ctx, int length, int ticks_ms)
-
static inline void *NEW_ESP_GMF_PORT_IN_BLOCK(void *acq, void *release, void *del, void *ctx, int length, int ticks_ms)
-
static inline void *NEW_ESP_GMF_PORT_OUT_BLOCK(void *acq, void *release, void *del, void *ctx, int length, int ticks_ms)
Structures
-
struct esp_gmf_port_io_ops_t
Structure defining the I/O operations of a GMF port.
Public Members
-
port_acquire acquire
Function pointer for acquiring data
-
port_release release
Function pointer for releasing data
-
port_acquire acquire
-
struct esp_gmf_port_attr_t
Structure defining the attributes of a GMF port.
-
struct esp_gmf_port_
Structure representing a GMF port The usage of the port in linked elements is as follows.
+———+ +—————+ +——-—+ | In Port +–—> First Element +-—> Out Port | +———+ +—-—+—-—+ +——-—+ | v +———+ +—-—+—-—+ +——-—+ | In Port +–—> More Element +-—> Out Port | +———+ +—-—+—-—+ +——-—+ | v +———+ +—-—+—-—+ +——-—+ | In Port +–—> Last Element +-—> Out Port | +———+ +—————+ +——-—+
Public Members
-
struct esp_gmf_port_ *next
Pointer to the next port
-
void *writer
Acquire out functions caller with the port
-
void *reader
Acquire in functions caller with the port
-
esp_gmf_port_io_ops_t ops
I/O operations of the port
-
esp_gmf_port_attr_t attr
Port attributes
-
int data_length
Data length of the payload
-
void *ctx
User context for the port
-
int wait_ticks
Timeout for port operations
-
esp_gmf_payload_t *payload
Payload pointer to be set
Payload is shared to the next element port or not, 1 for shared (default), 0 for dedicated
-
esp_gmf_payload_t *self_payload
Self payload of the port
-
struct esp_gmf_port_ *ref_port
Pointer to the reference port
-
int8_t ref_count
Reference count indicating the number of active references
-
struct esp_gmf_port_ *next
-
struct esp_gmf_port_config_
Structure defining the configuration of a GMF port.
Macros
-
ESP_GMF_PORT_CHECK(log_tag, ret, ret_value, action, format, ...)
-
ESP_GMF_PORT_ACQUIRE_IN_CHECK(TAG, ret, ret_value, action)
-
ESP_GMF_PORT_ACQUIRE_OUT_CHECK(TAG, ret, ret_value, action)
-
ESP_GMF_PORT_RELEASE_IN_CHECK(TAG, ret, ret_value, action)
-
ESP_GMF_PORT_RELEASE_OUT_CHECK(TAG, ret, ret_value, action)
-
ESP_GMF_PORT_DIR_IN
Definition the direction of a GMF port (input or output)
Input port
-
ESP_GMF_PORT_DIR_OUT
Output port
-
ESP_GMF_PORT_TYPE_BYTE
Definition the bit mask for the type of data handled by a GMF port (byte or block)
A byte-type port transfers values by copying data byte by byte. Users must allocate memory, which means data transmission involves copying. This type of port allows convenient access to arbitrary byte lengths
A block-type port transfers data by passing memory addresses. The memory used by the user is provided by another source, so data transmission does not involve copying. However, accessing arbitrary byte lengths is less flexible and may require data concatenation Bit0 for the byte type of GMF port
-
ESP_GMF_PORT_TYPE_BLOCK
Bit1 for the block type of GMF port
Type Definitions
-
typedef struct esp_gmf_port_ *esp_gmf_port_handle_t
Handle to a GMF port.
-
typedef esp_gmf_err_io_t (*port_acquire)(void *handle, esp_gmf_payload_t *load, uint32_t wanted_size, int wait_ticks)
Function pointer type for acquiring data from a port.
-
typedef esp_gmf_err_io_t (*port_release)(void *handle, esp_gmf_payload_t *load, int wait_ticks)
Function pointer type for releasing data from a port.
-
typedef void (*port_free)(void *p)
Function pointer type for freeing a port.
-
typedef struct esp_gmf_port_ esp_gmf_port_t
Structure representing a GMF port The usage of the port in linked elements is as follows.
+———+ +—————+ +——-—+ | In Port +–—> First Element +-—> Out Port | +———+ +—-—+—-—+ +——-—+ | v +———+ +—-—+—-—+ +——-—+ | In Port +–—> More Element +-—> Out Port | +———+ +—-—+—-—+ +——-—+ | v +———+ +—-—+—-—+ +——-—+ | In Port +–—> Last Element +-—> Out Port | +———+ +—————+ +——-—+
-
typedef struct esp_gmf_port_config_ esp_gmf_port_config_t
Structure defining the configuration of a GMF port.
Header File
Functions
-
static inline esp_gmf_err_t esp_gmf_cache_new(uint32_t len, esp_gmf_cache_t **handle)
Create a new GMF cache instance.
- 参数:
len – [in] The size of the cache buffer to allocate
handle – [out] Pointer to the cache handle to be initialized
- 返回:
ESP_GMF_ERR_OK Success, cache is initialized
ESP_GMF_ERR_INVALID_ARG If
handleis NULLESP_GMF_ERR_MEMORY_LACK If memory allocation fails
-
static inline esp_gmf_err_t esp_gmf_cache_delete(esp_gmf_cache_t *handle)
Delete a GMF cache instance and free associated memory.
- 参数:
handle – [in] Pointer to the cache handle
- 返回:
ESP_GMF_ERR_OK Success, cache is deleted
ESP_GMF_ERR_INVALID_ARG If
handleis NULL
-
static inline esp_gmf_err_t esp_gmf_cache_reset(esp_gmf_cache_t *handle)
Reset a GMF cache instance to its initial state This function clears all cached data and resets the cache state, but preserves the allocated buffer memory.
- 参数:
handle – [in] Pointer to the cache handle
- 返回:
ESP_GMF_ERR_OK Success, cache is reset
ESP_GMF_ERR_INVALID_ARG If
handleis NULL
-
static inline esp_gmf_err_t esp_gmf_cache_ready_for_load(esp_gmf_cache_t *handle, bool *is_ready)
Check if the cache is ready to load new data If the return
is_readyis true,esp_gmf_cache_loadcan be called to load a new payload.- 参数:
handle – [in] Pointer to the cache handle
is_ready – [out] Pointer to a boolean flag indicating whether new data can be loaded
- 返回:
ESP_GMF_ERR_OK Success,
is_readyis setESP_GMF_ERR_INVALID_ARG If
handleoris_readyis NULL
-
static inline esp_gmf_err_t esp_gmf_cache_load(esp_gmf_cache_t *handle, const esp_gmf_payload_t *load_in)
Load new payload data into the cache Call
esp_gmf_cache_ready_for_loadbefore use to check if it is ready to load a new payload.- 参数:
handle – [in] Pointer to the cache handle
load_in – [in] Pointer to the payload data to be loaded
- 返回:
ESP_GMF_ERR_OK Success, payload is loaded
ESP_GMF_ERR_INVALID_ARG If
handleorload_inis NULLESP_GMF_ERR_INVALID_STATE If previous data has not been consumed
-
static inline esp_gmf_err_t esp_gmf_cache_acquire(esp_gmf_cache_t *handle, uint32_t expected_size, esp_gmf_payload_t **load_out)
Acquire a data chunk of the expected size from the cache.
备注
This function must be used in conjunction with
esp_gmf_cache_releaseThe
expected_sizeshould match the internal buffer lengthIf
expected_sizeexceeds the current buffer length, the buffer will be reallocated
- 参数:
handle – [in] Pointer to the cache handle
expected_size – [in] The requested data size in bytes
load_out – [out] Pointer to store the acquired payload data
- 返回:
ESP_GMF_ERR_OK Data successfully acquired
ESP_GMF_ERR_INVALID_ARG If
handleorload_outis NULLESP_GMF_ERR_MEMORY_LACK Insufficient memory to allocate a new buffer
ESP_GMF_ERR_NOT_ENOUGH Cached data size less than expected size
-
static inline esp_gmf_err_t esp_gmf_cache_release(esp_gmf_cache_t *handle, esp_gmf_payload_t *load)
Release the payload data previously acquired with
esp_gmf_cache_acquire备注
The filled portion of the buffer is cleared only if its size equals the buffer length
- 参数:
handle – [in] Pointer to the cache handle
load – [in] Pointer to the payload data to be released
- 返回:
ESP_GMF_ERR_OK Payload data successfully released
ESP_GMF_ERR_INVALID_ARG If either
handleorloadis NULL
-
static inline esp_gmf_err_t esp_gmf_cache_get_cached_size(esp_gmf_cache_t *handle, int *filled)
Get the total amount of cached data.
- 参数:
handle – [in] Pointer to the cache handle
filled – [out] Pointer to store the filled data size
- 返回:
ESP_GMF_ERR_OK Success,
filledis setESP_GMF_ERR_INVALID_ARG If
handleorfilledis NULL
Structures
-
struct esp_gmf_cache_t
This GMF payload cache module is designed to cache a data block of a specified size. Each call to acquire returns this fixed-size block. If the acquired data is smaller than the expected size, new data must be loaded. When sufficient data is available, release clears the cached block. It is commonly used in scenarios where an element can only process a fixed data size, but the input data length is variable. For example, if an element requires exactly 1350 bytes per processing cycle but receives input of varying lengths, you can create a cache with
esp_gmf_cache_new(1350, out_handle), and the element can then fetch data usingesp_gmf_cache_acquireStructure representing a cache for GMF payloads
Public Members
-
uint8_t *buf
Pointer to the cache buffer
-
uint32_t buf_len
Total allocated size of the cache buffer
-
uint32_t buf_filled
Amount of data currently stored in the buffer
-
esp_gmf_payload_t origin_load
Original payload from which data is copied
-
esp_gmf_payload_t load
Current payload structure for acquiring data
-
uint8_t *buf
Header File
Functions
-
esp_gmf_err_t esp_gmf_io_init(esp_gmf_io_handle_t handle, esp_gmf_io_cfg_t *cfg)
Initialize a I/O handle with the given configuration If the stack size is greater than 0, a GMF task is created with the process job registered.
- 参数:
handle – [in] GMF I/O handle to initialize
cfg – [in] Pointer to the configuration structure
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_MEMORY_LACK Insufficient memory to perform the job registration
ESP_GMF_ERR_FAIL Failed to create thread
-
esp_gmf_err_t esp_gmf_io_deinit(esp_gmf_io_handle_t handle)
Deinitialize a GMF I/O handle, freeing associated resources.
- 参数:
handle – [in] GMF I/O handle to deinitialize
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_enable_speed_monitor(esp_gmf_io_handle_t handle, bool enabled)
Enable or disable speed monitor for an I/O.
- 参数:
handle – [in] I/O handle
enabled – [in] true to enable, false to disable
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_speed_stats(esp_gmf_io_handle_t handle, esp_gmf_io_speed_stats_t *stats)
Get speed statistics for an I/O.
- 参数:
handle – [in] I/O handle
stats – [out] Pointer to store speed statistics
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_score(esp_gmf_io_handle_t handle, const char *url, int *score)
Evaluate how well the IO matches a given URL The score ranges from ESP_GMF_IO_SCORE_NONE to ESP_GMF_IO_SCORE_PERFECT:
ESP_GMF_IO_SCORE_NONE: No match (The IO does not support this URL).
ESP_GMF_IO_SCORE_STANDARD: Standard match (e.g., matching the URL scheme like “http://” or “file://”).
ESP_GMF_IO_SCORE_PERFECT: Perfect match (e.g., matching both scheme and file extension, or high priority specialized IO).
- 参数:
handle – [in] GMF I/O handle
url – [in] URL to evaluate
score – [out] Pointer to store the calculated score
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_NOT_SUPPORT The IO does not support scoring
-
esp_gmf_err_t esp_gmf_io_open(esp_gmf_io_handle_t handle)
Open the specific I/O handle, run the thread if it is valid.
- 参数:
handle – [in] GMF I/O handle to open
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_seek(esp_gmf_io_handle_t handle, uint64_t seek_byte_pos)
Seek to a specific byte position in the specific I/O handle If the IO thread is invalid, the IO’s seek function is called directly (synchronous). If the IO thread is valid, the seek is deferred to the IO task: the seek position is stored and the data bus is aborted to wake up the task. The IO task will process the seek at the beginning of its next iteration. The caller will block and wait for the IO task to complete the seek operation.
- 参数:
handle – [in] GMF I/O handle
seek_byte_pos – [in] Byte position to seek to
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_NOT_SUPPORT Not support
Others Indicating failure
-
esp_gmf_err_t esp_gmf_io_close(esp_gmf_io_handle_t handle)
Close a GMF I/O handle If the IO task is active, this function will block and wait up to
task_timeout_msfor the task to finish If both prev_close and thread are valid, they will be called before executing the IO close operation.- 参数:
handle – [in] GMF I/O handle to close
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_io_t esp_gmf_io_acquire_read(esp_gmf_io_handle_t handle, esp_gmf_payload_t *load, uint32_t wanted_size, int wait_ticks)
Acquire read access to the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
load – [out] Pointer to store the acquired payload
wanted_size – [in] Desired payload size
wait_ticks – [in] Timeout value in ticks, in milliseconds
- 返回:
ESP_GMF_IO_OK Operation successful (check load->is_done to see if IO is done)
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_io_release_read(esp_gmf_io_handle_t handle, esp_gmf_payload_t *load, int wait_ticks)
Release read access to the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
load – [in] Pointer to the payload to release
wait_ticks – [in] Timeout value in ticks
- 返回:
ESP_GMF_IO_OK Operation successful (check load->is_done to see if IO is done)
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_io_acquire_write(esp_gmf_io_handle_t handle, esp_gmf_payload_t *load, uint32_t wanted_size, int wait_ticks)
Acquire write access to the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
load – [out] Pointer to store the acquired payload
wanted_size – [in] Desired payload size
wait_ticks – [in] Timeout value in ticks, in milliseconds
- 返回:
ESP_GMF_IO_OK Operation successful (check load->is_done to see if IO is done)
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_io_t esp_gmf_io_release_write(esp_gmf_io_handle_t handle, esp_gmf_payload_t *load, int wait_ticks)
Release write access to the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
load – [in] Pointer to the payload to release
wait_ticks – [in] Timeout value in ticks
- 返回:
ESP_GMF_IO_OK Operation successful (check load->is_done to see if IO is done)
ESP_GMF_IO_FAIL Operation failed or invalid arguments
ESP_GMF_IO_ABORT Operation aborted
ESP_GMF_IO_TIMEOUT Operation timed out
-
esp_gmf_err_t esp_gmf_io_set_info(esp_gmf_io_handle_t handle, esp_gmf_info_file_t *info)
Set information for the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
info – [in] Pointer to the file information structure
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_info(esp_gmf_io_handle_t handle, esp_gmf_info_file_t *info)
Get information from the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
info – [out] Pointer to store the file information
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_set_uri(esp_gmf_io_handle_t handle, const char *uri)
Set the URI for the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
uri – [in] Pointer to the URI string
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_MEMORY_LACK Memory allocation failed
-
esp_gmf_err_t esp_gmf_io_get_uri(esp_gmf_io_handle_t handle, char **uri)
Get the URI from the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
uri – [out] Pointer to store the URI string
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_set_pos(esp_gmf_io_handle_t handle, uint64_t byte_pos)
Set the byte position for the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
byte_pos – [in] Byte position to set
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_update_pos(esp_gmf_io_handle_t handle, uint64_t byte_pos)
Update the byte position for the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
byte_pos – [in] Byte position to update
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_pos(esp_gmf_io_handle_t handle, uint64_t *byte_pos)
Get the current byte position from the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
byte_pos – [out] Pointer to store the current byte position
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_set_size(esp_gmf_io_handle_t handle, uint64_t total_size)
Set the total size for the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
total_size – [in] Total size to set
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_size(esp_gmf_io_handle_t handle, uint64_t *total_size)
Get the total size from the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
total_size – [out] Pointer to store the total size
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_get_db_filled_size(esp_gmf_io_handle_t handle, uint32_t *filled_size)
Get the filled size in the data bus of the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
filled_size – [out] Pointer to store the filled size in the data bus
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_NOT_SUPPORT Operation not supported, the I/O instance does not have a configured data bus
-
esp_gmf_err_t esp_gmf_io_get_type(esp_gmf_io_handle_t handle, esp_gmf_io_type_t *type)
Get the I/O type of the specific I/O handle.
- 参数:
handle – [in] GMF I/O handle
type – [out] Pointer to store the I/O type
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_reset(esp_gmf_io_handle_t handle)
Reset IO will do reset the position and size, call the reset function if it is valid, then reset the task and reload IO process job.
- 参数:
handle – [in] GMF I/O handle
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_reload(esp_gmf_io_handle_t handle, const char *uri)
Reload Io with a URI. Used primarily for seamless transitions (e.g., Hls segments) Reload is specailly designed to optimized download efficiency This function is used to avoid reconnection which is time-consuming when download sequential urls from same host If the IO has an io_process task, this function will reset the done_write (EOF) flag of data_bus and rerun the task.
备注
This function should be called after the current IO operation (e.g., reading the resource) has finished.
- 参数:
handle – [in] GMF I/O handle
uri – [in] Pointer to the URI string
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
ESP_GMF_ERR_NOT_SUPPORT Not support
Others Indicating failure
-
esp_gmf_err_t esp_gmf_io_done(esp_gmf_io_handle_t handle)
Mark an I/O as done, cause the
esp_gmf_io_acquire_*/esp_gmf_io_release_*input parameter(load) to have the is_done flag set If the IO has an io_process task, this function will abort the data_bus and hold the task.备注
Typical usage: when the current data source has been fully consumed (e.g. end of a track in a playlist), call this function to signal downstream consumers that no more data is available. The IO task will be held and stop producing data until
esp_gmf_io_clear_doneis called. Caller MUST callesp_gmf_io_clear_donebefore reusing the IO for the next data source, otherwise the IO task remains permanently held and will not process any further data.- 参数:
handle – [in] GMF I/O handle
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_clear_done(esp_gmf_io_handle_t handle)
Clear the done flag for an I/O, then
esp_gmf_io_acquire_*/esp_gmf_io_release_*will operate normally Reset the position to 0 If the IO has an io_process task, this function will reset the data_bus and release the held task.备注
Typical usage: after
esp_gmf_io_donehas been called and the caller is ready to start a new data source (e.g. switching to the next track), call this function to clear the done state, reset position to 0, and release the held IO task so it can resume producing data. This function should only be called afteresp_gmf_io_done; calling it without a prior done has no effect.- 参数:
handle – [in] GMF I/O handle
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_abort(esp_gmf_io_handle_t handle)
Mark an I/O as aborted, then
esp_gmf_io_acquire_*/esp_gmf_io_release_*will return ESP_GMF_IO_ABORT If the IO has an io_process task, this function will abort the data_bus and hold the task.备注
Typical usage: when the IO operation needs to be interrupted immediately (e.g. user-initiated stop, network error recovery, or switching to a different URI). All ongoing and subsequent acquire/release calls will return ESP_GMF_IO_ABORT until
esp_gmf_io_clear_abortis called. The IO task will be held and stop producing/consuming data. Caller MUST callesp_gmf_io_clear_abortto restore normal operation, otherwise the IO remains permanently in the aborted state and the task stays held.- 参数:
handle – [in] GMF I/O handle
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_clear_abort(esp_gmf_io_handle_t handle)
Clear the abort flag for an I/O, then
esp_gmf_io_acquire_*/esp_gmf_io_release_*will operate normally If the IO has an io_process task, this function will clear abort on the data_bus and release the held task.备注
Typical usage: after
esp_gmf_io_aborthas been called and the error condition has been resolved (e.g. network reconnected, new URI set), call this function to clear the abort state and release the held IO task so it can resume processing data. This function should only be called afteresp_gmf_io_abort; calling it without a prior abort has no effect.- 参数:
handle – [in] GMF I/O handle
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
-
esp_gmf_err_t esp_gmf_io_set_task_timeout(esp_gmf_io_handle_t handle, int32_t timeout_ms)
Set the timeout for internal IO task operation.
- 参数:
handle – [in] GMF I/O handle
timeout_ms – [in] Timeout in milliseconds
- 返回:
ESP_GMF_ERR_OK On success
ESP_GMF_ERR_INVALID_ARG Invalid argument
Structures
-
struct esp_gmf_io_buffer_cfg_t
Buffer configuration for GMF I/O.
Public Members
-
size_t io_size
Each time io size in bytes
-
size_t buffer_size
Buffer size in bytes
-
esp_gmf_err_t (*read_filter)(esp_gmf_io_handle_t obj, void *payload, uint32_t wanted_size, int block_ticks)
Read filter callback function
-
size_t io_size
-
struct esp_gmf_io_speed_stats_t
Structure representing I/O speed statistics.
Public Members
-
uint64_t total_bytes
Total bytes
-
uint64_t last_total_bytes
Last total bytes count for interval calculation
-
uint64_t total_time_ms
Total time in milliseconds
-
uint64_t last_total_time_ms
Last update total time in milliseconds
-
uint32_t current_speed_kbps
Current speed in Kbps
-
uint32_t average_speed_kbps
Average speed in Kbps
-
uint64_t total_bytes
-
struct esp_gmf_io_cfg_t
Configuration structure for a GMF I/O.
Public Members
-
esp_gmf_task_config_t thread
Task configuration
-
esp_gmf_io_buffer_cfg_t buffer_cfg
Buffer configuration
-
bool enable_speed_monitor
Enable speed monitor
-
esp_gmf_task_config_t thread
-
struct esp_gmf_io_
Structure representing a GMF I/O.
Public Members
-
struct esp_gmf_obj_ parent
Parent object
-
esp_gmf_err_t (*get_score)(esp_gmf_io_handle_t obj, const char *url, int *score)
Get score callback function
-
esp_gmf_err_t (*open)(esp_gmf_io_handle_t obj)
Open callback function
-
esp_gmf_err_t (*seek)(esp_gmf_io_handle_t obj, uint64_t data)
Seek callback function
-
esp_gmf_err_t (*close)(esp_gmf_io_handle_t obj)
Close callback function
-
esp_gmf_err_t (*reset)(esp_gmf_io_handle_t obj)
Reset callback function
-
esp_gmf_err_t (*reload)(esp_gmf_io_handle_t obj, const char *uri)
Reload callback function Previous close callback function For some block IO instances, this function can be called before the
closeoperation
-
esp_gmf_err_io_t (*acquire_read)(esp_gmf_io_handle_t handle, void *payload, uint32_t wanted_size, int block_ticks)
Acquire read callback function
-
esp_gmf_err_io_t (*release_read)(esp_gmf_io_handle_t handle, void *payload, int block_ticks)
Release read callback function
-
esp_gmf_err_io_t (*acquire_write)(esp_gmf_io_handle_t handle, void *payload, uint32_t wanted_size, int block_ticks)
Acquire write callback function
-
esp_gmf_err_io_t (*release_write)(esp_gmf_io_handle_t handle, void *payload, int block_ticks)
Release write callback function
-
esp_gmf_task_handle_t task_hd
Task handle
-
esp_gmf_db_handle_t data_bus
Data bus handle for buffer
-
esp_gmf_data_bus_block_t db_block
Data bus block for data bus read/write operation
-
esp_gmf_io_speed_stats_t *speed_stats
Speed statistics for this I/O (dynamically allocated when enabled)
-
void *evt_group
Event group for IO control synchronization
-
esp_gmf_io_dir_t dir
I/O direction
-
esp_gmf_io_type_t type
I/O type
-
esp_gmf_info_file_t attr
File attribute
-
uint64_t seek_pos
Pending seek position (ESP_GMF_IO_SEEK_POS_INVALID = no pending seek)
-
uint8_t _is_done
Flag indicating if a done signal has been received
-
uint8_t _is_abort
Flag indicating if an abort signal has been received
-
uint8_t _is_hold
Flag indicating if the IO task should hold itself
-
int32_t task_timeout_ms
Timeout for internal IO task operation (default 1000 ms)
-
struct esp_gmf_obj_ parent
Macros
-
ESP_GMF_IO_SCORE_NONE
No match (The IO does not support this URL)
-
ESP_GMF_IO_SCORE_STANDARD
Standard match (e.g., matching the URL scheme)
-
ESP_GMF_IO_SCORE_PERFECT
Perfect match (e.g., matching both scheme and extension)
Type Definitions
-
typedef void *esp_gmf_io_handle_t
This GMF I/O abstraction can operate in two modes depending on configuration:
1) Asynchronous I/O (data_bus + task configured) When the I/O is configured with a data bus (buffer configured) and a task (thread stack > 0 in esp_gmf_io_cfg_t), the framework creates an internal `io_process` task. That task is responsible for driving the data flow between the actual underlying I/O (open/seek/close callbacks) and the data bus. In this mode, user-facing APIs such as `esp_gmf_io_acquire_*` / `esp_gmf_io_release_*` interact with the data bus (buffered I/O). The task performs blocking or non-blocking reads/writes against the real I/O device and pushes/pulls data through the data bus. This model is suitable when the element needs fixed-size transfers or when decoupling the producer/consumer timing is desired. 2) Synchronous I/O (no data_bus and no task) If no data bus and no task are configured, the I/O operates in synchronous mode: calls to `esp_gmf_io_acquire_*` / `esp_gmf_io_release_*` invoke the I/O driver's callbacks directly and perform actual I/O in the caller's context. Use this mode when the element can safely perform I/O on the caller thread or when buffering is not needed.Handle to a GMF I/O
-
typedef struct esp_gmf_io_ esp_gmf_io_t
Structure representing a GMF I/O.