mirror of https://github.com/FFmpeg/FFmpeg.git
parent
db1a642cd2
commit
d5d328059e
6 changed files with 389 additions and 2 deletions
@ -0,0 +1,334 @@ |
||||
/*
|
||||
* Audio Toolbox system codecs |
||||
* |
||||
* copyright (c) 2016 Rodger Combs |
||||
* |
||||
* 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 <AudioToolbox/AudioToolbox.h> |
||||
|
||||
#include "config.h" |
||||
#include "avcodec.h" |
||||
#include "bytestream.h" |
||||
#include "internal.h" |
||||
#include "libavutil/avassert.h" |
||||
#include "libavutil/opt.h" |
||||
#include "libavutil/log.h" |
||||
|
||||
typedef struct ATDecodeContext { |
||||
AVClass *av_class; |
||||
|
||||
AudioConverterRef converter; |
||||
AudioStreamPacketDescription pkt_desc; |
||||
AVPacket in_pkt; |
||||
AVPacket new_in_pkt; |
||||
|
||||
unsigned pkt_size; |
||||
int64_t last_pts; |
||||
int eof; |
||||
} ATDecodeContext; |
||||
|
||||
static UInt32 ffat_get_format_id(enum AVCodecID codec, int profile) |
||||
{ |
||||
switch (codec) { |
||||
case AV_CODEC_ID_AAC: |
||||
return kAudioFormatMPEG4AAC; |
||||
case AV_CODEC_ID_AC3: |
||||
return kAudioFormatAC3; |
||||
case AV_CODEC_ID_ADPCM_IMA_QT: |
||||
return kAudioFormatAppleIMA4; |
||||
case AV_CODEC_ID_ALAC: |
||||
return kAudioFormatAppleLossless; |
||||
case AV_CODEC_ID_AMR_NB: |
||||
return kAudioFormatAMR; |
||||
case AV_CODEC_ID_GSM_MS: |
||||
return kAudioFormatMicrosoftGSM; |
||||
case AV_CODEC_ID_ILBC: |
||||
return kAudioFormatiLBC; |
||||
case AV_CODEC_ID_MP1: |
||||
return kAudioFormatMPEGLayer1; |
||||
case AV_CODEC_ID_MP2: |
||||
return kAudioFormatMPEGLayer2; |
||||
case AV_CODEC_ID_MP3: |
||||
return kAudioFormatMPEGLayer3; |
||||
case AV_CODEC_ID_PCM_ALAW: |
||||
return kAudioFormatALaw; |
||||
case AV_CODEC_ID_PCM_MULAW: |
||||
return kAudioFormatULaw; |
||||
case AV_CODEC_ID_QDMC: |
||||
return kAudioFormatQDesign; |
||||
case AV_CODEC_ID_QDM2: |
||||
return kAudioFormatQDesign2; |
||||
default: |
||||
av_assert0(!"Invalid codec ID!"); |
||||
return 0; |
||||
} |
||||
} |
||||
|
||||
static void ffat_update_ctx(AVCodecContext *avctx) |
||||
{ |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
AudioStreamBasicDescription in_format; |
||||
UInt32 size = sizeof(in_format); |
||||
if (!AudioConverterGetProperty(at->converter, |
||||
kAudioConverterCurrentInputStreamDescription, |
||||
&size, &in_format)) { |
||||
avctx->channels = in_format.mChannelsPerFrame; |
||||
at->pkt_size = in_format.mFramesPerPacket; |
||||
} |
||||
|
||||
if (!at->pkt_size) |
||||
at->pkt_size = 2048; |
||||
} |
||||
|
||||
static void put_descr(PutByteContext *pb, int tag, unsigned int size) |
||||
{ |
||||
int i = 3; |
||||
bytestream2_put_byte(pb, tag); |
||||
for (; i > 0; i--) |
||||
bytestream2_put_byte(pb, (size >> (7 * i)) | 0x80); |
||||
bytestream2_put_byte(pb, size & 0x7F); |
||||
} |
||||
|
||||
static av_cold int ffat_init_decoder(AVCodecContext *avctx) |
||||
{ |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
OSStatus status; |
||||
|
||||
enum AVSampleFormat sample_fmt = (avctx->bits_per_raw_sample == 32) ? |
||||
AV_SAMPLE_FMT_S32 : AV_SAMPLE_FMT_S16; |
||||
|
||||
AudioStreamBasicDescription in_format = { |
||||
.mSampleRate = avctx->sample_rate ? avctx->sample_rate : 44100, |
||||
.mFormatID = ffat_get_format_id(avctx->codec_id, avctx->profile), |
||||
.mBytesPerPacket = avctx->block_align, |
||||
.mChannelsPerFrame = avctx->channels ? avctx->channels : 1, |
||||
}; |
||||
AudioStreamBasicDescription out_format = { |
||||
.mSampleRate = in_format.mSampleRate, |
||||
.mFormatID = kAudioFormatLinearPCM, |
||||
.mFormatFlags = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagIsPacked, |
||||
.mFramesPerPacket = 1, |
||||
.mChannelsPerFrame = in_format.mChannelsPerFrame, |
||||
.mBitsPerChannel = av_get_bytes_per_sample(sample_fmt) * 8, |
||||
}; |
||||
|
||||
avctx->sample_fmt = sample_fmt; |
||||
|
||||
if (avctx->codec_id == AV_CODEC_ID_ADPCM_IMA_QT) |
||||
in_format.mFramesPerPacket = 64; |
||||
|
||||
status = AudioConverterNew(&in_format, &out_format, &at->converter); |
||||
|
||||
if (status != 0) { |
||||
av_log(avctx, AV_LOG_ERROR, "AudioToolbox init error: %i\n", (int)status); |
||||
return AVERROR_UNKNOWN; |
||||
} |
||||
|
||||
if (avctx->extradata_size) { |
||||
char *extradata = avctx->extradata; |
||||
int extradata_size = avctx->extradata_size; |
||||
if (avctx->codec_id == AV_CODEC_ID_AAC) { |
||||
PutByteContext pb; |
||||
extradata_size = 5 + 3 + 5+13 + 5+avctx->extradata_size; |
||||
if (!(extradata = av_malloc(extradata_size))) |
||||
return AVERROR(ENOMEM); |
||||
|
||||
bytestream2_init_writer(&pb, extradata, extradata_size); |
||||
|
||||
// ES descriptor
|
||||
put_descr(&pb, 0x03, 3 + 5+13 + 5+avctx->extradata_size); |
||||
bytestream2_put_be16(&pb, 0); |
||||
bytestream2_put_byte(&pb, 0x00); // flags (= no flags)
|
||||
|
||||
// DecoderConfig descriptor
|
||||
put_descr(&pb, 0x04, 13 + 5+avctx->extradata_size); |
||||
|
||||
// Object type indication
|
||||
bytestream2_put_byte(&pb, 0x40); |
||||
|
||||
bytestream2_put_byte(&pb, 0x15); // flags (= Audiostream)
|
||||
|
||||
bytestream2_put_be24(&pb, 0); // Buffersize DB
|
||||
|
||||
bytestream2_put_be32(&pb, 0); // maxbitrate
|
||||
bytestream2_put_be32(&pb, 0); // avgbitrate
|
||||
|
||||
// DecoderSpecific info descriptor
|
||||
put_descr(&pb, 0x05, avctx->extradata_size); |
||||
bytestream2_put_buffer(&pb, avctx->extradata, avctx->extradata_size); |
||||
} |
||||
|
||||
status = AudioConverterSetProperty(at->converter, |
||||
kAudioConverterDecompressionMagicCookie, |
||||
extradata_size, extradata); |
||||
if (status != 0) |
||||
av_log(avctx, AV_LOG_WARNING, "AudioToolbox cookie error: %i\n", (int)status); |
||||
} |
||||
|
||||
ffat_update_ctx(avctx); |
||||
|
||||
at->last_pts = AV_NOPTS_VALUE; |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static OSStatus ffat_decode_callback(AudioConverterRef converter, UInt32 *nb_packets, |
||||
AudioBufferList *data, |
||||
AudioStreamPacketDescription **packets, |
||||
void *inctx) |
||||
{ |
||||
AVCodecContext *avctx = inctx; |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
|
||||
if (at->eof) { |
||||
*nb_packets = 0; |
||||
if (packets) { |
||||
*packets = &at->pkt_desc; |
||||
at->pkt_desc.mDataByteSize = 0; |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
av_packet_move_ref(&at->in_pkt, &at->new_in_pkt); |
||||
at->new_in_pkt.data = 0; |
||||
at->new_in_pkt.size = 0; |
||||
|
||||
if (!at->in_pkt.data) { |
||||
*nb_packets = 0; |
||||
return 1; |
||||
} |
||||
|
||||
data->mNumberBuffers = 1; |
||||
data->mBuffers[0].mNumberChannels = 0; |
||||
data->mBuffers[0].mDataByteSize = at->in_pkt.size; |
||||
data->mBuffers[0].mData = at->in_pkt.data; |
||||
*nb_packets = 1; |
||||
|
||||
if (packets) { |
||||
*packets = &at->pkt_desc; |
||||
at->pkt_desc.mDataByteSize = at->in_pkt.size; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int ffat_decode(AVCodecContext *avctx, void *data, |
||||
int *got_frame_ptr, AVPacket *avpkt) |
||||
{ |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
AVFrame *frame = data; |
||||
OSStatus ret; |
||||
|
||||
AudioBufferList out_buffers = { |
||||
.mNumberBuffers = 1, |
||||
.mBuffers = { |
||||
{ |
||||
.mNumberChannels = avctx->channels, |
||||
.mDataByteSize = av_get_bytes_per_sample(avctx->sample_fmt) * at->pkt_size * avctx->channels, |
||||
} |
||||
} |
||||
}; |
||||
|
||||
av_packet_unref(&at->new_in_pkt); |
||||
|
||||
if (avpkt->size) { |
||||
if ((ret = av_packet_ref(&at->new_in_pkt, avpkt)) < 0) |
||||
return ret; |
||||
} else { |
||||
at->eof = 1; |
||||
} |
||||
|
||||
frame->sample_rate = avctx->sample_rate; |
||||
|
||||
frame->nb_samples = at->pkt_size; |
||||
ff_get_buffer(avctx, frame, 0); |
||||
|
||||
out_buffers.mBuffers[0].mData = frame->data[0]; |
||||
|
||||
ret = AudioConverterFillComplexBuffer(at->converter, ffat_decode_callback, avctx, |
||||
&frame->nb_samples, &out_buffers, NULL); |
||||
if ((!ret || ret == 1) && frame->nb_samples) { |
||||
*got_frame_ptr = 1; |
||||
if (at->last_pts != AV_NOPTS_VALUE) { |
||||
frame->pts = at->last_pts; |
||||
at->last_pts = avpkt->pts; |
||||
} |
||||
} else if (ret && ret != 1) { |
||||
av_log(avctx, AV_LOG_WARNING, "Decode error: %i\n", ret); |
||||
} else { |
||||
at->last_pts = avpkt->pts; |
||||
} |
||||
|
||||
return avpkt->size; |
||||
} |
||||
|
||||
static av_cold void ffat_decode_flush(AVCodecContext *avctx) |
||||
{ |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
AudioConverterReset(at->converter); |
||||
av_packet_unref(&at->new_in_pkt); |
||||
av_packet_unref(&at->in_pkt); |
||||
} |
||||
|
||||
static av_cold int ffat_close_decoder(AVCodecContext *avctx) |
||||
{ |
||||
ATDecodeContext *at = avctx->priv_data; |
||||
AudioConverterDispose(at->converter); |
||||
av_packet_unref(&at->new_in_pkt); |
||||
av_packet_unref(&at->in_pkt); |
||||
return 0; |
||||
} |
||||
|
||||
#define FFAT_DEC_CLASS(NAME) \ |
||||
static const AVClass ffat_##NAME##_dec_class = { \
|
||||
.class_name = "at_" #NAME "_dec", \
|
||||
.version = LIBAVUTIL_VERSION_INT, \
|
||||
}; |
||||
|
||||
#define FFAT_DEC(NAME, ID) \ |
||||
FFAT_DEC_CLASS(NAME) \
|
||||
AVCodec ff_##NAME##_at_decoder = { \
|
||||
.name = #NAME "_at", \
|
||||
.long_name = NULL_IF_CONFIG_SMALL(#NAME " (AudioToolbox)"), \
|
||||
.type = AVMEDIA_TYPE_AUDIO, \
|
||||
.id = ID, \
|
||||
.priv_data_size = sizeof(ATDecodeContext), \
|
||||
.init = ffat_init_decoder, \
|
||||
.close = ffat_close_decoder, \
|
||||
.decode = ffat_decode, \
|
||||
.flush = ffat_decode_flush, \
|
||||
.priv_class = &ffat_##NAME##_dec_class, \
|
||||
.capabilities = AV_CODEC_CAP_DR1 | AV_CODEC_CAP_DELAY, \
|
||||
.caps_internal = FF_CODEC_CAP_INIT_THREADSAFE, \
|
||||
}; |
||||
|
||||
FFAT_DEC(aac, AV_CODEC_ID_AAC) |
||||
FFAT_DEC(ac3, AV_CODEC_ID_AC3) |
||||
FFAT_DEC(adpcm_ima_qt, AV_CODEC_ID_ADPCM_IMA_QT) |
||||
FFAT_DEC(alac, AV_CODEC_ID_ALAC) |
||||
FFAT_DEC(amr_nb, AV_CODEC_ID_AMR_NB) |
||||
FFAT_DEC(gsm_ms, AV_CODEC_ID_GSM_MS) |
||||
FFAT_DEC(ilbc, AV_CODEC_ID_ILBC) |
||||
FFAT_DEC(mp1, AV_CODEC_ID_MP1) |
||||
FFAT_DEC(mp2, AV_CODEC_ID_MP2) |
||||
FFAT_DEC(mp3, AV_CODEC_ID_MP3) |
||||
FFAT_DEC(pcm_alaw, AV_CODEC_ID_PCM_ALAW) |
||||
FFAT_DEC(pcm_mulaw, AV_CODEC_ID_PCM_MULAW) |
||||
FFAT_DEC(qdmc, AV_CODEC_ID_QDMC) |
||||
FFAT_DEC(qdm2, AV_CODEC_ID_QDM2) |
Loading…
Reference in new issue