gpt4 book ai didi

c++ - 带字符串输入的 ffmpeg avformat_open_input 模拟

转载 作者:搜寻专家 更新时间:2023-10-31 00:31:01 28 4
gpt4 key购买 nike

ffmpeg avformat_open_input() 需要流的文件名才能打开。

但是我的程序生成了自己的 SDP 文件,其中包含所有需要的信息到一个字符串中。

是否有任何类似的函数可以用于字符串而不是文件?因为我不想为此创建文件。

或者我必须为它创建自己的解析器?

最佳答案

来自俄罗斯 StackOverflow 的示例(非常感谢 Monah Tuk):

#include <iostream>
#include <vector>
#include <cassert>

extern "C" {#include <libavformat/avformat.h>}

using namespace std;

static const char* SDP_DATA = R"(
v=0
o=- 1376063087593 1 IN IP4 127.0.0.1
s=-
t=0 0
m=audio 50008 RTP/AVP 0
c=IN IP4 192.168.2.196
a=rtcp:50009 IN IP4 192.168.2.196
a=rtpmap:0 PCMU/8000
a=sendrecv
m=video 50010 RTP/AVP 120
c=IN IP4 192.168.2.196
a=rtcp:50011 IN IP4 192.168.2.196
a=rtpmap:120 VP8/90000
a=sendrecv
a=rtcp-fb:* nack
a=rtcp-fb:* ccm fir
)";

struct SdpOpaque
{
using Vector = std::vector<uint8_t>;
Vector data;
Vector::iterator pos;
};

int sdp_read(void *opaque, uint8_t *buf, int size)
{
assert(opaque);
assert(buf);
auto octx = static_cast<SdpOpaque*>(opaque);

if (octx->pos == octx->data.end()) {
return 0;
}

auto dist = static_cast<int>(std::distance(octx->pos, octx->data.end()));
auto count = std::min(size, dist);

std::copy(octx->pos, octx->pos + count, buf);
octx->pos += count;

return count;
}

int sdp_open(AVFormatContext **pctx, const char *data, AVDictionary **options)
{
assert(pctx);
*pctx = avformat_alloc_context();
assert(*pctx);

const size_t avioBufferSize = 4096;
auto avioBuffer = static_cast<uint8_t*>(av_malloc(avioBufferSize));
auto opaque = new SdpOpaque();

opaque->data = SdpOpaque::Vector(data, data + strlen(data));
opaque->pos = opaque->data.begin();

auto pbctx = avio_alloc_context(avioBuffer, avioBufferSize, 0, opaque, sdp_read, nullptr, nullptr);
assert(pbctx);

(*pctx)->pb = pbctx;

auto infmt = av_find_input_format("sdp");

return avformat_open_input(pctx, "memory.sdp", infmt, options);
}

void sdp_close(AVFormatContext **fctx)
{
assert(fctx);
auto ctx = *fctx;

// Opaque can be non-POD type, free it before and assign to null
auto opaque = static_cast<SdpOpaque*>(ctx->pb->opaque);
delete opaque;
ctx->pb->opaque = nullptr;

avio_close(ctx->pb);
avformat_close_input(fctx);
}

int main()
{
av_register_all();
avformat_network_init();

AVFormatContext *sdpctx = nullptr;
sdp_open(&sdpctx, SDP_DATA, nullptr);

av_dump_format(sdpctx, 0, "memory.sdp", 0);
// Copy settings to target context from SDP context:
/*
for (size_t i = 0; i < sdpctx->nb_streams; ++i) {
AVStream *st = avformat_new_stream(otherctx, nullptr);
st->id = i;
avcodec_copy_context(st->codec, sdpctx->streams[i]->codec);
st->time_base = sdpctx->streams[i]->time_base;
}
*/

sdp_close(&sdpctx);
return 0;
}

除了 std::copy 之外,它在 VS2013 中完全有效。

关于c++ - 带字符串输入的 ffmpeg avformat_open_input 模拟,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35132647/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com