[FFmpeg-devel] [PATCH] new API usage example (adts-aac encoding from raw audio file)
wm4
nfxjfg at googlemail.com
Wed Mar 29 12:58:32 EEST 2017
On Tue, 28 Mar 2017 22:58:18 +0200
Paolo Prete <p4olo_prete-at-yahoo.it at ffmpeg.org> wrote:
> ---
> doc/examples/Makefile | 1 +
> doc/examples/encode_raw_audio_file_to_aac.c | 275 ++++++++++++++++++++++++++++
> 2 files changed, 276 insertions(+)
> create mode 100644 doc/examples/encode_raw_audio_file_to_aac.c
>
> diff --git a/doc/examples/Makefile b/doc/examples/Makefile
> index af38159..81181c7 100644
> --- a/doc/examples/Makefile
> +++ b/doc/examples/Makefile
> @@ -15,6 +15,7 @@ EXAMPLES= avio_dir_cmd \
> avio_reading \
> decoding_encoding \
> demuxing_decoding \
> + encode_raw_audio_file_to_aac \
> extract_mvs \
> filtering_video \
> filtering_audio \
> diff --git a/doc/examples/encode_raw_audio_file_to_aac.c b/doc/examples/encode_raw_audio_file_to_aac.c
> new file mode 100644
> index 0000000..19114e8
> --- /dev/null
> +++ b/doc/examples/encode_raw_audio_file_to_aac.c
> @@ -0,0 +1,275 @@
> +/*
> + * Copyright (c) 2017 Paolo Prete (p4olo_prete at yahoo.it)
> + *
> + * Permission is hereby granted, free of charge, to any person obtaining a copy
> + * of this software and associated documentation files (the "Software"), to deal
> + * in the Software without restriction, including without limitation the rights
> + * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
> + * copies of the Software, and to permit persons to whom the Software is
> + * furnished to do so, subject to the following conditions:
> + *
> + * The above copyright notice and this permission notice shall be included in
> + * all copies or substantial portions of the Software.
> + *
> + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
> + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
> + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
> + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
> + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
> + * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
> + * THE SOFTWARE.
> + */
> +
> +/**
> + * @file
> + * API example for adts-aac encoding raw audio files.
> + * This example reads a raw audio input file, converts it to float-planar format, performs aac encoding and puts the encoded frames into an ADTS container. The encoded stream is written to
> + * a file named "out.aac"
> + * The raw input audio file can be created with: ffmpeg -i some_audio_file -f f32le -acodec pcm_f32le -ac 2 -ar 16000 raw_audio_file.raw
> + *
> + * @example encode_raw_audio_file_to_aac.c
> + */
> +
> +#include <libavcodec/avcodec.h>
> +#include <libavformat/avformat.h>
> +#include <libavutil/timestamp.h>
> +#include <libswresample/swresample.h>
> +
> +#define ENCODER_BITRATE 64000
> +#define SAMPLE_RATE 16000
> +#define INPUT_SAMPLE_FMT AV_SAMPLE_FMT_FLT
> +#define CHANNELS 2
> +
> +static int write_adts_muxed_data (void *opaque, uint8_t *adts_data, int size)
> +{
> + FILE *encoded_audio_file = (FILE *)opaque;
> + fwrite(adts_data, 1, size, encoded_audio_file); //(f)
> + return size;
> +}
> +
> +int main(int argc, char **argv)
> +{
> + FILE *input_audio_file = NULL, *encoded_audio_file = NULL;
> + AVCodec *audio_codec = NULL;
> + AVCodecContext *audio_encoder_ctx = NULL;
> + AVFrame *input_audio_frame = NULL, *converted_audio_frame = NULL;
> + SwrContext *audio_convert_context = NULL;
> + AVOutputFormat *adts_container = NULL;
> + AVFormatContext *adts_container_ctx = NULL;
> + uint8_t *adts_container_buffer = NULL;
> + size_t adts_container_buffer_size = 4096;
> + AVIOContext *adts_avio_ctx = NULL;
> + AVStream *adts_stream = NULL;
> + AVPacket encoded_audio_packet;
There are vague plans of disallowing stack allocation of AVPackets by
removing sizeof(AVPacket) from the ABI (like AVFrame etc.), so it might
be better to allocate the packet with the appropriate functions.
> + int encoded_pkt_counter = 1;
> + int still_pkts_to_flush = 1;
> + int delayed_pkt_counter = 1;
> + int ret_val = 0;
> + int audio_bytes_to_encode;
> + int64_t curr_pts;
This function is obviously getting a bit too long. Why not split it
into multiple functions?
> +
> + if (argc != 2) {
> + printf("Usage: %s <raw audio input file (CHANNELS, INPUT_SAMPLE_FMT, SAMPLE_RATE)>\n", argv[0]);
> + return 1;
> + }
> +
> + input_audio_file = fopen(argv[1], "rb");
> + if (!input_audio_file) {
> + av_log(NULL, AV_LOG_ERROR, "Could not open input audio file\n");
> + return AVERROR_EXIT;
> + }
> +
> + encoded_audio_file = fopen("out.aac", "wb");
> + if (!encoded_audio_file) {
> + av_log(NULL, AV_LOG_ERROR, "Could not open output audio file\n");
> + fclose(input_audio_file);
> + return AVERROR_EXIT;
> + }
> +
> + av_register_all();
> +
> + /**
> + * Allocate the encoder's context and open the encoder
> + */
> + audio_codec = avcodec_find_encoder(AV_CODEC_ID_AAC);
> + if (!audio_codec) {
> + av_log(NULL, AV_LOG_ERROR, "Could not find aac codec\n");
> + ret_val = AVERROR_EXIT;
> + goto end;
> + }
> + audio_encoder_ctx = avcodec_alloc_context3(audio_codec);
> + if (!audio_codec) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate the encoding context\n");
> + ret_val = AVERROR_EXIT;
> + goto end;
> + }
> + audio_encoder_ctx->sample_fmt = AV_SAMPLE_FMT_FLTP;
> + audio_encoder_ctx->bit_rate = ENCODER_BITRATE;
> + audio_encoder_ctx->sample_rate = SAMPLE_RATE; // You can use any other sample rate provided by the input file on condition that it is supported by the codec (use AVCodec::supported_samplerates for listing supported sample rates)
> + audio_encoder_ctx->channels = CHANNELS;
> + audio_encoder_ctx->channel_layout = av_get_default_channel_layout(CHANNELS);
> + audio_encoder_ctx->time_base = (AVRational){1, SAMPLE_RATE};
> + audio_encoder_ctx->codec_type = AVMEDIA_TYPE_AUDIO ;
> + if ((ret_val = avcodec_open2(audio_encoder_ctx, audio_codec, NULL)) < 0) {
> + av_log(NULL, AV_LOG_ERROR, "Could not open input codec (error '%s')\n", av_err2str(ret_val));
> + goto end;
> + }
> +
> + /**
> + * Allocate an AVFrame which will be filled with the input file's data.
> + */
> + if (!(input_audio_frame = av_frame_alloc())) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate input frame\n");
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
> + input_audio_frame->nb_samples = audio_encoder_ctx->frame_size;
> + input_audio_frame->format = INPUT_SAMPLE_FMT;
> + input_audio_frame->channels = CHANNELS;
> + input_audio_frame->sample_rate = SAMPLE_RATE;
> + input_audio_frame->channel_layout = av_get_default_channel_layout(CHANNELS);
> + // Allocate the frame's data buffer
> + if ((ret_val = av_frame_get_buffer(input_audio_frame, 0)) < 0) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate container for input frame samples (error '%s')\n", av_err2str(ret_val));
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
> +
> + /**
> + * Input data must be converted to float-planar format, which is the format required by the AAC encoder. We allocate a SwrContext and an AVFrame (which will contain the converted samples)
> + * for this task. The AVFrame will feed the encoding function (avcodec_send_frame())
> + */
> + audio_convert_context = swr_alloc_set_opts(NULL, av_get_default_channel_layout(CHANNELS), AV_SAMPLE_FMT_FLTP, SAMPLE_RATE, av_get_default_channel_layout(CHANNELS), INPUT_SAMPLE_FMT, SAMPLE_RATE, 0, NULL);
> + if (!audio_convert_context) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate resample context\n");
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
Some encoders (probably not all) signal what rates and formats they
support in the AVCodec struct.
> + if (!(converted_audio_frame = av_frame_alloc())) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate resampled frame\n");
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
> + converted_audio_frame->nb_samples = audio_encoder_ctx->frame_size;
> + converted_audio_frame->format = audio_encoder_ctx->sample_fmt;
> + converted_audio_frame->channels = audio_encoder_ctx->channels;
> + converted_audio_frame->channel_layout = audio_encoder_ctx->channel_layout;
> + converted_audio_frame->sample_rate = SAMPLE_RATE;
> + if ((ret_val = av_frame_get_buffer(converted_audio_frame, 0)) < 0) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate a buffer for resampled frame samples (error '%s')\n", av_err2str(ret_val));
> + goto end;
> + }
> +
> + /**
> + * Create the ADTS container for the encoded frames
> + */
> + adts_container = av_guess_format("adts", NULL, NULL);
> + if (!adts_container) {
> + av_log(NULL, AV_LOG_ERROR, "Could not find adts output format\n");
> + ret_val = AVERROR_EXIT;
> + goto end;
> + }
> + if ((ret_val = avformat_alloc_output_context2(&adts_container_ctx, adts_container, "", NULL)) < 0) {
> + av_log(NULL, AV_LOG_ERROR, "Could not create output context (error '%s')\n", av_err2str(ret_val));
> + goto end;
> + }
> + if (!(adts_container_buffer = av_malloc(adts_container_buffer_size))) {
> + av_log(NULL, AV_LOG_ERROR, "Could not allocate a buffer for the I/O output context\n");
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
> + // Create an I/O context for the adts container with a write callback (write_adts_muxed_data()), so that muxed data will be accessed through this function.
> + if (!(adts_avio_ctx = avio_alloc_context(adts_container_buffer, adts_container_buffer_size, 1, encoded_audio_file, NULL , &write_adts_muxed_data, NULL))) {
> + av_log(NULL, AV_LOG_ERROR, "Could not create I/O output context\n");
> + ret_val = AVERROR_EXIT;
> + goto end;
> + }
The code would be much easier if not using custom I/O...
> + // Link the container's context to the previous I/O context
> + adts_container_ctx->pb = adts_avio_ctx;
> + if (!(adts_stream = avformat_new_stream(adts_container_ctx, NULL))) {
> + av_log(NULL, AV_LOG_ERROR, "Could not create new stream\n");
> + ret_val = AVERROR(ENOMEM);
> + goto end;
> + }
> + adts_stream->id = adts_container_ctx->nb_streams-1;
> + // Copy the encoder's parameters
> + avcodec_parameters_from_context(adts_stream->codecpar, audio_encoder_ctx);
> + // Allocate the stream private data and write the stream header
> + if (avformat_write_header(adts_container_ctx, NULL) < 0) {
> + av_log(NULL, AV_LOG_ERROR, "avformat_write_header() error\n");
> + ret_val = AVERROR_EXIT;
> + goto end;
> + }
Like I said somewhere else, it _might_ be better to wait with that
until the first packet has been encoded (maybe not with the AAC
encoder, but some others potentially). I'll leave this to others to
judge though.
> +
> + /**
> + * Fill the input frame's data buffer with input file data (a),
> + * Convert the input frame to float-planar format (b),
> + * Send the converted frame to the encoder (c),
> + * Get the encoded packet (d),
> + * Send the encoded packet to the adts muxer (e).
> + * Muxed data is caught in write_adts_muxed_data() callback and it is written to the output audio file ( (f) : see above)
> + */
> + av_init_packet(&encoded_audio_packet);
> + while (1) {
> + audio_bytes_to_encode = fread(input_audio_frame->data[0], 1, input_audio_frame->linesize[0], input_audio_file); //(a)
> + swr_convert_frame(audio_convert_context, converted_audio_frame, (const AVFrame *)input_audio_frame); //(b)
What's with the const cast?
You need av_frame_make_writeable() on the output frame.
Also doesn't check for errors.
> + if (audio_bytes_to_encode != input_audio_frame->linesize[0]) {
> + break;
> + }
> + else {
> + ret_val = avcodec_send_frame(audio_encoder_ctx, converted_audio_frame); //(c)
> + if (ret_val == 0)
> + ret_val = avcodec_receive_packet(audio_encoder_ctx, &encoded_audio_packet); //(d)
> + else{
> + av_log(NULL, AV_LOG_ERROR, "Error encoding frame (error '%s')\n", av_err2str(ret_val));
> + goto end;
> + }
> +
> + if (ret_val == 0){
> + curr_pts = converted_audio_frame->nb_samples*(encoded_pkt_counter-1);
> + encoded_audio_packet.pts = encoded_audio_packet.dts = curr_pts;
> + if ((ret_val == av_write_frame(adts_container_ctx, &encoded_audio_packet)) < 0) { //(e)
> + av_log(NULL, AV_LOG_ERROR, "Error calling av_write_frame() (error '%s')\n", av_err2str(ret_val));
> + goto end;
> + }
> + else{
> + av_log(NULL, AV_LOG_INFO, "Encoded AAC packet %d, size=%d, pts_time=%s\n", encoded_pkt_counter, encoded_audio_packet.size, av_ts2timestr(encoded_audio_packet.pts, &audio_encoder_ctx->time_base));
> + ++encoded_pkt_counter;
> + }
> + }
Errors in ret_val are discarded?
Also, if you put the receive call in a loop, it'd actually follow the
proper send/receive data flow.
> + }
> + }
> + // Flush delayed packets
> + while (still_pkts_to_flush) {
> + ret_val = avcodec_send_frame(audio_encoder_ctx, NULL);
This call is needed only once.
> + if (ret_val != 0)
> + still_pkts_to_flush = 0;
> + ret_val = avcodec_receive_packet(audio_encoder_ctx, &encoded_audio_packet);
> + if (ret_val == 0){
> + curr_pts = converted_audio_frame->nb_samples*(encoded_pkt_counter-1);
> + encoded_audio_packet.pts = encoded_audio_packet.dts = curr_pts;
> + av_write_frame(adts_container_ctx, &encoded_audio_packet);
> + av_log(NULL, AV_LOG_INFO, "Flushed encoded AAC delayed packet %d, size=%d, pts_time=%s\n", delayed_pkt_counter, encoded_audio_packet.size, av_ts2timestr(encoded_audio_packet.pts, &audio_encoder_ctx->time_base));
> + ++delayed_pkt_counter;
> + ++encoded_pkt_counter;
Not sure why this logging is done - just seems to inflate the code.
I don't remember how the swr_convert_frame API works, but you might
have to flush out the resampler too.
> + }
> + }
> +
> + av_write_trailer(adts_container_ctx);
> + av_packet_unref(&encoded_audio_packet);
> +
> +end:
> +
> + fclose(input_audio_file);
> + fclose(encoded_audio_file);
> + avcodec_free_context(&audio_encoder_ctx);
> + av_frame_free(&input_audio_frame);
> + swr_free(&audio_convert_context);
> + av_frame_free(&converted_audio_frame);
> + avformat_free_context(adts_container_ctx);
> + av_freep(&adts_avio_ctx);
> + av_freep(&adts_container_buffer);
> +
> + return ret_val;
> +
> +}
More information about the ffmpeg-devel
mailing list