doc/examples/transcoding: use av_packet_alloc() to allocate packets

Signed-off-by: James Almer <jamrial@gmail.com>
This commit is contained in:
James Almer 2021-01-31 14:28:13 -03:00
parent 16dba17329
commit b896832b78
1 changed files with 28 additions and 20 deletions

View File

@ -42,6 +42,7 @@ typedef struct FilteringContext {
AVFilterContext *buffersrc_ctx; AVFilterContext *buffersrc_ctx;
AVFilterGraph *filter_graph; AVFilterGraph *filter_graph;
AVPacket *enc_pkt;
AVFrame *filtered_frame; AVFrame *filtered_frame;
} FilteringContext; } FilteringContext;
static FilteringContext *filter_ctx; static FilteringContext *filter_ctx;
@ -407,6 +408,10 @@ static int init_filters(void)
if (ret) if (ret)
return ret; return ret;
filter_ctx[i].enc_pkt = av_packet_alloc();
if (!filter_ctx[i].enc_pkt)
return AVERROR(ENOMEM);
filter_ctx[i].filtered_frame = av_frame_alloc(); filter_ctx[i].filtered_frame = av_frame_alloc();
if (!filter_ctx[i].filtered_frame) if (!filter_ctx[i].filtered_frame)
return AVERROR(ENOMEM); return AVERROR(ENOMEM);
@ -414,17 +419,17 @@ static int init_filters(void)
return 0; return 0;
} }
static int encode_write_frame(AVFrame *filt_frame, unsigned int stream_index) static int encode_write_frame(unsigned int stream_index, int flush)
{ {
StreamContext *stream = &stream_ctx[stream_index]; StreamContext *stream = &stream_ctx[stream_index];
FilteringContext *filter = &filter_ctx[stream_index];
AVFrame *filt_frame = flush ? NULL : filter->filtered_frame;
AVPacket *enc_pkt = filter->enc_pkt;
int ret; int ret;
AVPacket enc_pkt;
av_log(NULL, AV_LOG_INFO, "Encoding frame\n"); av_log(NULL, AV_LOG_INFO, "Encoding frame\n");
/* encode filtered frame */ /* encode filtered frame */
enc_pkt.data = NULL; av_packet_unref(enc_pkt);
enc_pkt.size = 0;
av_init_packet(&enc_pkt);
ret = avcodec_send_frame(stream->enc_ctx, filt_frame); ret = avcodec_send_frame(stream->enc_ctx, filt_frame);
@ -432,20 +437,20 @@ static int encode_write_frame(AVFrame *filt_frame, unsigned int stream_index)
return ret; return ret;
while (ret >= 0) { while (ret >= 0) {
ret = avcodec_receive_packet(stream->enc_ctx, &enc_pkt); ret = avcodec_receive_packet(stream->enc_ctx, enc_pkt);
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF) if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF)
return 0; return 0;
/* prepare packet for muxing */ /* prepare packet for muxing */
enc_pkt.stream_index = stream_index; enc_pkt->stream_index = stream_index;
av_packet_rescale_ts(&enc_pkt, av_packet_rescale_ts(enc_pkt,
stream->enc_ctx->time_base, stream->enc_ctx->time_base,
ofmt_ctx->streams[stream_index]->time_base); ofmt_ctx->streams[stream_index]->time_base);
av_log(NULL, AV_LOG_DEBUG, "Muxing frame\n"); av_log(NULL, AV_LOG_DEBUG, "Muxing frame\n");
/* mux encoded frame */ /* mux encoded frame */
ret = av_interleaved_write_frame(ofmt_ctx, &enc_pkt); ret = av_interleaved_write_frame(ofmt_ctx, enc_pkt);
} }
return ret; return ret;
@ -481,7 +486,7 @@ static int filter_encode_write_frame(AVFrame *frame, unsigned int stream_index)
} }
filter->filtered_frame->pict_type = AV_PICTURE_TYPE_NONE; filter->filtered_frame->pict_type = AV_PICTURE_TYPE_NONE;
ret = encode_write_frame(filter->filtered_frame, stream_index); ret = encode_write_frame(stream_index, 0);
av_frame_unref(filter->filtered_frame); av_frame_unref(filter->filtered_frame);
if (ret < 0) if (ret < 0)
break; break;
@ -497,13 +502,13 @@ static int flush_encoder(unsigned int stream_index)
return 0; return 0;
av_log(NULL, AV_LOG_INFO, "Flushing stream #%u encoder\n", stream_index); av_log(NULL, AV_LOG_INFO, "Flushing stream #%u encoder\n", stream_index);
return encode_write_frame(NULL, stream_index); return encode_write_frame(stream_index, 1);
} }
int main(int argc, char **argv) int main(int argc, char **argv)
{ {
int ret; int ret;
AVPacket packet = { .data = NULL, .size = 0 }; AVPacket *packet = NULL;
unsigned int stream_index; unsigned int stream_index;
unsigned int i; unsigned int i;
@ -518,12 +523,14 @@ int main(int argc, char **argv)
goto end; goto end;
if ((ret = init_filters()) < 0) if ((ret = init_filters()) < 0)
goto end; goto end;
if (!(packet = av_packet_alloc()))
goto end;
/* read all packets */ /* read all packets */
while (1) { while (1) {
if ((ret = av_read_frame(ifmt_ctx, &packet)) < 0) if ((ret = av_read_frame(ifmt_ctx, packet)) < 0)
break; break;
stream_index = packet.stream_index; stream_index = packet->stream_index;
av_log(NULL, AV_LOG_DEBUG, "Demuxer gave frame of stream_index %u\n", av_log(NULL, AV_LOG_DEBUG, "Demuxer gave frame of stream_index %u\n",
stream_index); stream_index);
@ -532,10 +539,10 @@ int main(int argc, char **argv)
av_log(NULL, AV_LOG_DEBUG, "Going to reencode&filter the frame\n"); av_log(NULL, AV_LOG_DEBUG, "Going to reencode&filter the frame\n");
av_packet_rescale_ts(&packet, av_packet_rescale_ts(packet,
ifmt_ctx->streams[stream_index]->time_base, ifmt_ctx->streams[stream_index]->time_base,
stream->dec_ctx->time_base); stream->dec_ctx->time_base);
ret = avcodec_send_packet(stream->dec_ctx, &packet); ret = avcodec_send_packet(stream->dec_ctx, packet);
if (ret < 0) { if (ret < 0) {
av_log(NULL, AV_LOG_ERROR, "Decoding failed\n"); av_log(NULL, AV_LOG_ERROR, "Decoding failed\n");
break; break;
@ -555,15 +562,15 @@ int main(int argc, char **argv)
} }
} else { } else {
/* remux this frame without reencoding */ /* remux this frame without reencoding */
av_packet_rescale_ts(&packet, av_packet_rescale_ts(packet,
ifmt_ctx->streams[stream_index]->time_base, ifmt_ctx->streams[stream_index]->time_base,
ofmt_ctx->streams[stream_index]->time_base); ofmt_ctx->streams[stream_index]->time_base);
ret = av_interleaved_write_frame(ofmt_ctx, &packet); ret = av_interleaved_write_frame(ofmt_ctx, packet);
if (ret < 0) if (ret < 0)
goto end; goto end;
} }
av_packet_unref(&packet); av_packet_unref(packet);
} }
/* flush filters and encoders */ /* flush filters and encoders */
@ -587,13 +594,14 @@ int main(int argc, char **argv)
av_write_trailer(ofmt_ctx); av_write_trailer(ofmt_ctx);
end: end:
av_packet_unref(&packet); av_packet_free(&packet);
for (i = 0; i < ifmt_ctx->nb_streams; i++) { for (i = 0; i < ifmt_ctx->nb_streams; i++) {
avcodec_free_context(&stream_ctx[i].dec_ctx); avcodec_free_context(&stream_ctx[i].dec_ctx);
if (ofmt_ctx && ofmt_ctx->nb_streams > i && ofmt_ctx->streams[i] && stream_ctx[i].enc_ctx) if (ofmt_ctx && ofmt_ctx->nb_streams > i && ofmt_ctx->streams[i] && stream_ctx[i].enc_ctx)
avcodec_free_context(&stream_ctx[i].enc_ctx); avcodec_free_context(&stream_ctx[i].enc_ctx);
if (filter_ctx && filter_ctx[i].filter_graph) { if (filter_ctx && filter_ctx[i].filter_graph) {
avfilter_graph_free(&filter_ctx[i].filter_graph); avfilter_graph_free(&filter_ctx[i].filter_graph);
av_packet_free(&filter_ctx[i].enc_pkt);
av_frame_free(&filter_ctx[i].filtered_frame); av_frame_free(&filter_ctx[i].filtered_frame);
} }