YoC播放器中支持的码流url类型有如下几种:
流类型 URL前缀 URL格式
http流 http(s):// http(s)://ip:port/xx.mp3
文件流 file:// file:///fatfs0/xx.mp3?avformat=%s&avcodec=%u&channel=%u&rate=%u
内存流 mem:// mem://addr=%u&size=%u&avformat=%u&avcodec=%u&channel=%u&rate=%u
fifo流 fifo:// fifo://tts/1?avformat=%s&avcodec=%u&channel=%u&rate=%u
由于fifo是基于生产者-消费者原理来实现的,我们可以基于fifo码流类型来实现实时语音合成码流播放。用户在创建fifo后,通过将fifo传递给播放器播放。与此同时可不断从网络上读取tts流,写入fifo中。
示例代码如下:
//核心代码片段
static player_t* g_player;
static nsfifo_t* g_tts_fifo;
static void _ptask(void *arg)
{
int fd;
int cnt = 0, rc, wlen;
char *val, *pos;
uint8_t reof = 0;
web_session_t *session;
/* 创建一个http会话 */
session = web_session_create();
/* 向指定地址发起http请求 */
rc = web_session_get(session, "http://www.srcbin.net/ai/result.mp3", 3);
if (rc) {
LOGE(TAG, "web_session_get fail. rc = %d, code = %d, phrase = %s", rc, session->code, session->phrase);
goto err;
}
/* 获取内容的长度 */
val = (char*)dict_get_val(&session->hdrs, "Content-Length");
CHECK_RET_TAG_WITH_GOTO(val != NULL, err);
fd = session->fd;
LOGD(TAG, "content len = %d", atoi(val));
for (;;) {
/* 获取fifo的可写指针及长度 */
wlen = nsfifo_get_wpos(g_tts_fifo, &pos, 8*1000);
/* 获取播放器fifo读端是否退出(可能播放出错) */
nsfifo_get_eof(g_tts_fifo, &reof, NULL);
if (wlen <= 0 || reof) {
LOGE(TAG, "get wpos err. wlen = %d, reof = %d", wlen, reof);
break;
}
/* 从网络套接字中读取数据 */
rc = sock_readn(fd, pos, wlen, 6*1000);
if (rc <= 0) {
LOGE(TAG, "readn err. rc = %d", rc);
break;
}
/* 设置写指针 */
nsfifo_set_wpos(g_tts_fifo, rc);
cnt += rc;
}
LOGD(TAG, "rc = %8d, cnt = %8d", rc, cnt);
err:
/* 销毁web会话资源 */
web_session_destroy(session);
return;
}
/ 创建fifo,指定fifo地址和大小 /
g_tts_fifo = nsfifo_open("fifo://tts/1", O_CREAT, 64*1024);
if (g_tts_fifo) {
/* 创建 _ptask任务,用于通过网络获取码流数据 */
aos_task_new("xx_task", _ptask, NULL, 6*1024);
/* 设置播放器取流超时时间 */
player_ioctl(g_player, PLAYER_CMD_RCVTIMEO, AOS_WAIT_FOREVER);
/* 播放之前创建的fifo码流 */
player_play(g_player, "fifo://tts/1", 0);
}