avformat/aea: add aea muxer
Signed-off-by: asivery <asivery@protonmail.com>
This commit is contained in:
committed by
Stefano Sabatini
parent
78803a4b8a
commit
9124d807dc
@@ -32,6 +32,7 @@ version <next>:
|
|||||||
- DVD-Video demuxer, powered by libdvdnav and libdvdread
|
- DVD-Video demuxer, powered by libdvdnav and libdvdread
|
||||||
- ffprobe -show_stream_groups option
|
- ffprobe -show_stream_groups option
|
||||||
- ffprobe (with -export_side_data film_grain) now prints film grain metadata
|
- ffprobe (with -export_side_data film_grain) now prints film grain metadata
|
||||||
|
- AEA muxer
|
||||||
|
|
||||||
|
|
||||||
version 6.1:
|
version 6.1:
|
||||||
|
@@ -684,6 +684,16 @@ Enable to set MPEG version bit in the ADTS frame header to 1 which
|
|||||||
indicates MPEG-2. Default is 0, which indicates MPEG-4.
|
indicates MPEG-2. Default is 0, which indicates MPEG-4.
|
||||||
@end table
|
@end table
|
||||||
|
|
||||||
|
@anchor{aea}
|
||||||
|
@section aea
|
||||||
|
MD STUDIO audio muxer.
|
||||||
|
|
||||||
|
This muxer accepts a single ATRAC1 audio stream with either one or two channels
|
||||||
|
and a sample rate of 44100Hz.
|
||||||
|
|
||||||
|
As AEA supports storing the track title, this muxer will also write
|
||||||
|
the title from stream's metadata to the container.
|
||||||
|
|
||||||
@anchor{aiff}
|
@anchor{aiff}
|
||||||
@section aiff
|
@section aiff
|
||||||
Audio Interchange File Format muxer.
|
Audio Interchange File Format muxer.
|
||||||
|
@@ -91,7 +91,8 @@ OBJS-$(CONFIG_ADTS_MUXER) += adtsenc.o apetag.o img2.o \
|
|||||||
id3v2enc.o
|
id3v2enc.o
|
||||||
OBJS-$(CONFIG_ADX_DEMUXER) += adxdec.o
|
OBJS-$(CONFIG_ADX_DEMUXER) += adxdec.o
|
||||||
OBJS-$(CONFIG_ADX_MUXER) += rawenc.o
|
OBJS-$(CONFIG_ADX_MUXER) += rawenc.o
|
||||||
OBJS-$(CONFIG_AEA_DEMUXER) += aea.o pcm.o
|
OBJS-$(CONFIG_AEA_DEMUXER) += aeadec.o pcm.o
|
||||||
|
OBJS-$(CONFIG_AEA_MUXER) += aeaenc.o rawenc.o
|
||||||
OBJS-$(CONFIG_AFC_DEMUXER) += afc.o
|
OBJS-$(CONFIG_AFC_DEMUXER) += afc.o
|
||||||
OBJS-$(CONFIG_AIFF_DEMUXER) += aiffdec.o aiff.o pcm.o \
|
OBJS-$(CONFIG_AIFF_DEMUXER) += aiffdec.o aiff.o pcm.o \
|
||||||
mov_chan.o replaygain.o
|
mov_chan.o replaygain.o
|
||||||
|
115
libavformat/aeaenc.c
Normal file
115
libavformat/aeaenc.c
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
/*
|
||||||
|
* MD STUDIO audio muxer
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 asivery
|
||||||
|
*
|
||||||
|
* This file is part of FFmpeg.
|
||||||
|
*
|
||||||
|
* FFmpeg is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU Lesser General Public
|
||||||
|
* License as published by the Free Software Foundation; either
|
||||||
|
* version 2.1 of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* FFmpeg is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with FFmpeg; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "avformat.h"
|
||||||
|
#include "avio_internal.h"
|
||||||
|
#include "rawenc.h"
|
||||||
|
#include "mux.h"
|
||||||
|
|
||||||
|
static int aea_write_header(AVFormatContext *s)
|
||||||
|
{
|
||||||
|
const AVDictionaryEntry *title_entry;
|
||||||
|
size_t title_length = 0;
|
||||||
|
AVStream *st;
|
||||||
|
|
||||||
|
if (s->nb_streams > 1) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "Got more than one stream to encode. This is not supported.\n");
|
||||||
|
return AVERROR(EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
st = s->streams[0];
|
||||||
|
if (st->codecpar->ch_layout.nb_channels != 1 && st->codecpar->ch_layout.nb_channels != 2) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "Only maximum 2 channels are supported in the audio"
|
||||||
|
" stream, %d channels were found.\n", st->codecpar->ch_layout.nb_channels);
|
||||||
|
return AVERROR(EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (st->codecpar->codec_id != AV_CODEC_ID_ATRAC1) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "AEA can only store ATRAC1 streams, %s was found.\n", avcodec_get_name(st->codecpar->codec_id));
|
||||||
|
return AVERROR(EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (st->codecpar->sample_rate != 44100) {
|
||||||
|
av_log(s, AV_LOG_ERROR, "Invalid sample rate (%d) AEA only supports 44.1kHz.\n", st->codecpar->sample_rate);
|
||||||
|
return AVERROR(EINVAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Write magic */
|
||||||
|
avio_wl32(s->pb, 2048);
|
||||||
|
|
||||||
|
/* Write AEA title */
|
||||||
|
title_entry = av_dict_get(st->metadata, "title", NULL, 0);
|
||||||
|
if (title_entry) {
|
||||||
|
const char *title_contents = title_entry->value;
|
||||||
|
title_length = strlen(title_contents);
|
||||||
|
if (title_length > 256) {
|
||||||
|
av_log(s, AV_LOG_WARNING, "Title too long, truncated to 256 bytes.\n");
|
||||||
|
title_length = 256;
|
||||||
|
}
|
||||||
|
avio_write(s->pb, title_contents, title_length);
|
||||||
|
}
|
||||||
|
|
||||||
|
ffio_fill(s->pb, 0, 256 - title_length);
|
||||||
|
|
||||||
|
/* Write number of frames (zero at header-writing time, will seek later), number of channels */
|
||||||
|
avio_wl32(s->pb, 0);
|
||||||
|
avio_w8(s->pb, st->codecpar->ch_layout.nb_channels);
|
||||||
|
avio_w8(s->pb, 0);
|
||||||
|
|
||||||
|
/* Pad the header to 2048 bytes */
|
||||||
|
ffio_fill(s->pb, 0, 1782);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int aea_write_trailer(struct AVFormatContext *s)
|
||||||
|
{
|
||||||
|
int64_t total_blocks;
|
||||||
|
AVIOContext *pb = s->pb;
|
||||||
|
AVStream *st = s->streams[0];
|
||||||
|
if (pb->seekable & AVIO_SEEKABLE_NORMAL) {
|
||||||
|
/* Seek to rewrite the block count. */
|
||||||
|
avio_seek(pb, 260, SEEK_SET);
|
||||||
|
total_blocks = st->nb_frames * st->codecpar->ch_layout.nb_channels;
|
||||||
|
if (total_blocks > UINT32_MAX) {
|
||||||
|
av_log(s, AV_LOG_WARNING, "Too many frames in the file to properly encode the header (%ld)."
|
||||||
|
" Block count in the header will be truncated.\n", total_blocks);
|
||||||
|
total_blocks = UINT32_MAX;
|
||||||
|
}
|
||||||
|
avio_wl32(pb, total_blocks);
|
||||||
|
} else {
|
||||||
|
av_log(s, AV_LOG_WARNING, "Unable to rewrite AEA header.\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FFOutputFormat ff_aea_muxer = {
|
||||||
|
.p.name = "aea",
|
||||||
|
.p.long_name = NULL_IF_CONFIG_SMALL("MD STUDIO audio"),
|
||||||
|
.p.extensions = "aea",
|
||||||
|
.p.audio_codec = AV_CODEC_ID_ATRAC1,
|
||||||
|
|
||||||
|
.write_header = aea_write_header,
|
||||||
|
.write_packet = ff_raw_write_packet,
|
||||||
|
.write_trailer = aea_write_trailer,
|
||||||
|
};
|
@@ -47,6 +47,7 @@ extern const FFOutputFormat ff_adts_muxer;
|
|||||||
extern const FFInputFormat ff_adx_demuxer;
|
extern const FFInputFormat ff_adx_demuxer;
|
||||||
extern const FFOutputFormat ff_adx_muxer;
|
extern const FFOutputFormat ff_adx_muxer;
|
||||||
extern const FFInputFormat ff_aea_demuxer;
|
extern const FFInputFormat ff_aea_demuxer;
|
||||||
|
extern const FFOutputFormat ff_aea_muxer;
|
||||||
extern const FFInputFormat ff_afc_demuxer;
|
extern const FFInputFormat ff_afc_demuxer;
|
||||||
extern const FFInputFormat ff_aiff_demuxer;
|
extern const FFInputFormat ff_aiff_demuxer;
|
||||||
extern const FFOutputFormat ff_aiff_muxer;
|
extern const FFOutputFormat ff_aiff_muxer;
|
||||||
|
Reference in New Issue
Block a user