Merge remote-tracking branch 'qatar/master'
* qatar/master: (24 commits) flvdec: remove incomplete, disabled seeking code mem: add support for _aligned_malloc() as found on Windows lavc: Extend the documentation for avcodec_init_packet flvdec: remove incomplete, disabled seeking code http: replace atoll() with strtoll() mpegts: remove unused/incomplete/broken seeking code af_amix: allow float planar sample format as input af_amix: use AVFloatDSPContext.vector_fmac_scalar() float_dsp: add x86-optimized functions for vector_fmac_scalar() float_dsp: Move vector_fmac_scalar() from libavcodec to libavutil lavr: Add x86-optimized function for flt to s32 conversion lavr: Add x86-optimized function for flt to s16 conversion lavr: Add x86-optimized functions for s32 to flt conversion lavr: Add x86-optimized functions for s32 to s16 conversion lavr: Add x86-optimized functions for s16 to flt conversion lavr: Add x86-optimized function for s16 to s32 conversion rtpenc: Support packetizing iLBC rtpdec: Add a depacketizer for iLBC Implement the iLBC storage file format mov: Support muxing/demuxing iLBC ... Conflicts: Changelog configure libavcodec/avcodec.h libavcodec/dsputil.c libavcodec/version.h libavformat/movenc.c libavformat/mpegts.c libavformat/version.h libavutil/mem.c Merged-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
commit
cabbd271a5
@ -10,6 +10,7 @@ version next:
|
||||
- atempo filter
|
||||
- ffprobe -show_data option
|
||||
- RTMPT protocol support
|
||||
- iLBC encoding/decoding via libilbc
|
||||
|
||||
|
||||
version 0.11:
|
||||
|
10
configure
vendored
10
configure
vendored
@ -180,6 +180,7 @@ External library support:
|
||||
--enable-libfaac enable FAAC support via libfaac [no]
|
||||
--enable-libfreetype enable libfreetype [no]
|
||||
--enable-libgsm enable GSM support via libgsm [no]
|
||||
--enable-libilbc enable iLBC de/encoding via libilbc [no]
|
||||
--enable-libmodplug enable ModPlug via libmodplug [no]
|
||||
--enable-libmp3lame enable MP3 encoding via libmp3lame [no]
|
||||
--enable-libnut enable NUT (de)muxing via libnut,
|
||||
@ -1051,6 +1052,7 @@ CONFIG_LIST="
|
||||
libfaac
|
||||
libfreetype
|
||||
libgsm
|
||||
libilbc
|
||||
libmodplug
|
||||
libmp3lame
|
||||
libnut
|
||||
@ -1168,6 +1170,7 @@ HAVE_LIST="
|
||||
$ARCH_EXT_LIST
|
||||
$HAVE_LIST_PUB
|
||||
$THREADS_LIST
|
||||
aligned_malloc
|
||||
aligned_stack
|
||||
alsa_asoundlib_h
|
||||
altivec_h
|
||||
@ -1588,6 +1591,8 @@ libgsm_decoder_deps="libgsm"
|
||||
libgsm_encoder_deps="libgsm"
|
||||
libgsm_ms_decoder_deps="libgsm"
|
||||
libgsm_ms_encoder_deps="libgsm"
|
||||
libilbc_decoder_deps="libilbc"
|
||||
libilbc_encoder_deps="libilbc"
|
||||
libmodplug_demuxer_deps="libmodplug"
|
||||
libmp3lame_encoder_deps="libmp3lame"
|
||||
libopencore_amrnb_decoder_deps="libopencore_amrnb"
|
||||
@ -3144,6 +3149,7 @@ check_func ${malloc_prefix}memalign && enable memalign
|
||||
check_func mkstemp
|
||||
check_func mmap
|
||||
check_func ${malloc_prefix}posix_memalign && enable posix_memalign
|
||||
check_func_headers malloc.h _aligned_malloc && enable aligned_malloc
|
||||
check_func setrlimit
|
||||
check_func strerror_r
|
||||
check_func strptime
|
||||
@ -3254,6 +3260,7 @@ enabled libcelt && require libcelt celt/celt.h celt_decode -lcelt0 &&
|
||||
enabled libfaac && require2 libfaac "stdint.h faac.h" faacEncGetVersion -lfaac
|
||||
enabled libfreetype && require_pkg_config freetype2 "ft2build.h freetype/freetype.h" FT_Init_FreeType
|
||||
enabled libgsm && require libgsm gsm/gsm.h gsm_create -lgsm
|
||||
enabled libilbc && require libilbc ilbc.h WebRtcIlbcfix_InitDecode -lilbc
|
||||
enabled libmodplug && require libmodplug libmodplug/modplug.h ModPlug_Load -lmodplug
|
||||
enabled libmp3lame && require "libmp3lame >= 3.98.3" lame/lame.h lame_set_VBR_quality -lmp3lame
|
||||
enabled libnut && require libnut libnut.h nut_demuxer_init -lnut
|
||||
@ -3522,7 +3529,7 @@ if test $target_os = "haiku"; then
|
||||
disable posix_memalign
|
||||
fi
|
||||
|
||||
! enabled_any memalign posix_memalign &&
|
||||
! enabled_any memalign posix_memalign aligned_malloc &&
|
||||
enabled_any $need_memalign && enable memalign_hack
|
||||
|
||||
# add_dep lib dep
|
||||
@ -3628,6 +3635,7 @@ echo "libcelt enabled ${libcelt-no}"
|
||||
echo "libdc1394 support ${libdc1394-no}"
|
||||
echo "libfaac enabled ${libfaac-no}"
|
||||
echo "libgsm enabled ${libgsm-no}"
|
||||
echo "libilbc enabled ${libilbc-no}"
|
||||
echo "libmodplug enabled ${libmodplug-no}"
|
||||
echo "libmp3lame enabled ${libmp3lame-no}"
|
||||
echo "libnut enabled ${libnut-no}"
|
||||
|
@ -93,6 +93,17 @@ x264 is under the GNU Public License Version 2 or later
|
||||
details), you must upgrade FFmpeg's license to GPL in order to use it.
|
||||
@end float
|
||||
|
||||
@section libilbc
|
||||
|
||||
iLBC is a narrowband speech codec that has been made freely available
|
||||
by Google as part of the WebRTC project. libilbc is a packaging friendly
|
||||
copy of the iLBC codec. Libav can make use of the libilbc library for
|
||||
iLBC encoding and decoding.
|
||||
|
||||
Go to @url{https://github.com/dekkers/libilbc} and follow the instructions for
|
||||
installing the library. Then pass @code{--enable-libilbc} to configure to
|
||||
enable it.
|
||||
|
||||
|
||||
|
||||
@chapter Supported File Formats, Codecs or Features
|
||||
@ -191,6 +202,7 @@ library:
|
||||
@item IEC61937 encapsulation @tab X @tab X
|
||||
@item IFF @tab @tab X
|
||||
@tab Interchange File Format
|
||||
@item iLBC @tab X @tab X
|
||||
@item Interplay MVE @tab @tab X
|
||||
@tab Format used in various Interplay computer games.
|
||||
@item IV8 @tab @tab X
|
||||
@ -749,6 +761,8 @@ following image formats are supported:
|
||||
@item GSM Microsoft variant @tab E @tab X
|
||||
@tab encoding supported through external library libgsm
|
||||
@item IAC (Indeo Audio Coder) @tab @tab X
|
||||
@item iLBC (Internet Low Bitrate Codec) @tab E @tab E
|
||||
@tab encoding and decoding supported through external library libilbc
|
||||
@item IMC (Intel Music Coder) @tab @tab X
|
||||
@item MACE (Macintosh Audio Compression/Expansion) 3:1 @tab @tab X
|
||||
@item MACE (Macintosh Audio Compression/Expansion) 6:1 @tab @tab X
|
||||
|
@ -247,6 +247,10 @@ times to construct arbitrary AMF sequences.
|
||||
Version of the Flash plugin used to run the SWF player. The default
|
||||
is LNX 9,0,124,2.
|
||||
|
||||
@item rtmp_flush_interval
|
||||
Number of packets flushed in the same request (RTMPT only). The default
|
||||
is 10.
|
||||
|
||||
@item rtmp_live
|
||||
Specify that the media is a live stream. No resuming or seeking in
|
||||
live streams is possible. The default value is @code{any}, which means the
|
||||
|
@ -659,6 +659,8 @@ OBJS-$(CONFIG_LIBGSM_DECODER) += libgsm.o
|
||||
OBJS-$(CONFIG_LIBGSM_ENCODER) += libgsm.o
|
||||
OBJS-$(CONFIG_LIBGSM_MS_DECODER) += libgsm.o
|
||||
OBJS-$(CONFIG_LIBGSM_MS_ENCODER) += libgsm.o
|
||||
OBJS-$(CONFIG_LIBILBC_DECODER) += libilbc.o
|
||||
OBJS-$(CONFIG_LIBILBC_ENCODER) += libilbc.o
|
||||
OBJS-$(CONFIG_LIBMP3LAME_ENCODER) += libmp3lame.o mpegaudiodecheader.o \
|
||||
audio_frame_queue.o
|
||||
OBJS-$(CONFIG_LIBOPENCORE_AMRNB_DECODER) += libopencore-amr.o \
|
||||
|
@ -410,6 +410,7 @@ void avcodec_register_all(void)
|
||||
REGISTER_ENCODER (LIBFAAC, libfaac);
|
||||
REGISTER_ENCDEC (LIBGSM, libgsm);
|
||||
REGISTER_ENCDEC (LIBGSM_MS, libgsm_ms);
|
||||
REGISTER_ENCDEC (LIBILBC, libilbc);
|
||||
REGISTER_ENCODER (LIBMP3LAME, libmp3lame);
|
||||
REGISTER_ENCDEC (LIBOPENCORE_AMRNB, libopencore_amrnb);
|
||||
REGISTER_DECODER (LIBOPENCORE_AMRWB, libopencore_amrwb);
|
||||
|
@ -154,8 +154,6 @@ void ff_vector_fmul_window_neon(float *dst, const float *src0,
|
||||
const float *src1, const float *win, int len);
|
||||
void ff_vector_fmul_scalar_neon(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
void ff_vector_fmac_scalar_neon(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
void ff_butterflies_float_neon(float *v1, float *v2, int len);
|
||||
float ff_scalarproduct_float_neon(const float *v1, const float *v2, int len);
|
||||
void ff_vector_fmul_reverse_neon(float *dst, const float *src0,
|
||||
@ -329,7 +327,6 @@ void ff_dsputil_init_neon(DSPContext *c, AVCodecContext *avctx)
|
||||
|
||||
c->vector_fmul_window = ff_vector_fmul_window_neon;
|
||||
c->vector_fmul_scalar = ff_vector_fmul_scalar_neon;
|
||||
c->vector_fmac_scalar = ff_vector_fmac_scalar_neon;
|
||||
c->butterflies_float = ff_butterflies_float_neon;
|
||||
c->scalarproduct_float = ff_scalarproduct_float_neon;
|
||||
c->vector_fmul_reverse = ff_vector_fmul_reverse_neon;
|
||||
|
@ -682,54 +682,6 @@ NOVFP vdup.32 q8, r2
|
||||
.unreq len
|
||||
endfunc
|
||||
|
||||
function ff_vector_fmac_scalar_neon, export=1
|
||||
VFP len .req r2
|
||||
VFP acc .req r3
|
||||
NOVFP len .req r3
|
||||
NOVFP acc .req r2
|
||||
VFP vdup.32 q15, d0[0]
|
||||
NOVFP vdup.32 q15, r2
|
||||
bics r12, len, #15
|
||||
mov acc, r0
|
||||
beq 3f
|
||||
vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vld1.32 {q1}, [r1,:128]!
|
||||
vld1.32 {q9}, [acc,:128]!
|
||||
1: vmla.f32 q8, q0, q15
|
||||
vld1.32 {q2}, [r1,:128]!
|
||||
vld1.32 {q10}, [acc,:128]!
|
||||
vmla.f32 q9, q1, q15
|
||||
vld1.32 {q3}, [r1,:128]!
|
||||
vld1.32 {q11}, [acc,:128]!
|
||||
vmla.f32 q10, q2, q15
|
||||
vst1.32 {q8}, [r0,:128]!
|
||||
vmla.f32 q11, q3, q15
|
||||
vst1.32 {q9}, [r0,:128]!
|
||||
subs r12, r12, #16
|
||||
beq 2f
|
||||
vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vst1.32 {q10}, [r0,:128]!
|
||||
vld1.32 {q1}, [r1,:128]!
|
||||
vld1.32 {q9}, [acc,:128]!
|
||||
vst1.32 {q11}, [r0,:128]!
|
||||
b 1b
|
||||
2: vst1.32 {q10}, [r0,:128]!
|
||||
vst1.32 {q11}, [r0,:128]!
|
||||
ands len, len, #15
|
||||
it eq
|
||||
bxeq lr
|
||||
3: vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vmla.f32 q8, q0, q15
|
||||
vst1.32 {q8}, [r0,:128]!
|
||||
subs len, len, #4
|
||||
bgt 3b
|
||||
bx lr
|
||||
.unreq len
|
||||
endfunc
|
||||
|
||||
function ff_butterflies_float_neon, export=1
|
||||
1: vld1.32 {q0},[r0,:128]
|
||||
vld1.32 {q1},[r1,:128]
|
||||
|
@ -407,6 +407,7 @@ enum CodecID {
|
||||
CODEC_ID_BMV_AUDIO,
|
||||
CODEC_ID_RALF,
|
||||
CODEC_ID_IAC,
|
||||
CODEC_ID_ILBC,
|
||||
CODEC_ID_FFWAVESYNTH = MKBETAG('F','F','W','S'),
|
||||
CODEC_ID_8SVX_RAW = MKBETAG('8','S','V','X'),
|
||||
CODEC_ID_SONIC = MKBETAG('S','O','N','C'),
|
||||
@ -3470,6 +3471,9 @@ void av_destruct_packet(AVPacket *pkt);
|
||||
/**
|
||||
* Initialize optional fields of a packet with default values.
|
||||
*
|
||||
* Note, this does not touch the data and size members, which have to be
|
||||
* initialized separately.
|
||||
*
|
||||
* @param pkt packet
|
||||
*/
|
||||
void av_init_packet(AVPacket *pkt);
|
||||
|
@ -27,6 +27,7 @@
|
||||
#include <stdio.h>
|
||||
|
||||
#include "libavutil/common.h"
|
||||
#include "libavutil/float_dsp.h"
|
||||
#include "libavutil/intmath.h"
|
||||
#include "libavutil/intreadwrite.h"
|
||||
#include "libavutil/mathematics.h"
|
||||
@ -384,7 +385,7 @@ typedef struct {
|
||||
int profile;
|
||||
|
||||
int debug_flag; ///< used for suppressing repeated error messages output
|
||||
DSPContext dsp;
|
||||
AVFloatDSPContext fdsp;
|
||||
FFTContext imdct;
|
||||
SynthFilterContext synth;
|
||||
DCADSPContext dcadsp;
|
||||
@ -2042,8 +2043,8 @@ static int dca_decode_frame(AVCodecContext *avctx, void *data,
|
||||
float *back_chan = s->samples + s->channel_order_tab[s->xch_base_channel] * 256;
|
||||
float *lt_chan = s->samples + s->channel_order_tab[s->xch_base_channel - 2] * 256;
|
||||
float *rt_chan = s->samples + s->channel_order_tab[s->xch_base_channel - 1] * 256;
|
||||
s->dsp.vector_fmac_scalar(lt_chan, back_chan, -M_SQRT1_2, 256);
|
||||
s->dsp.vector_fmac_scalar(rt_chan, back_chan, -M_SQRT1_2, 256);
|
||||
s->fdsp.vector_fmac_scalar(lt_chan, back_chan, -M_SQRT1_2, 256);
|
||||
s->fdsp.vector_fmac_scalar(rt_chan, back_chan, -M_SQRT1_2, 256);
|
||||
}
|
||||
|
||||
if (avctx->sample_fmt == AV_SAMPLE_FMT_FLT) {
|
||||
@ -2085,7 +2086,7 @@ static av_cold int dca_decode_init(AVCodecContext *avctx)
|
||||
s->avctx = avctx;
|
||||
dca_init_vlcs();
|
||||
|
||||
ff_dsputil_init(&s->dsp, avctx);
|
||||
avpriv_float_dsp_init(&s->fdsp, avctx->flags & CODEC_FLAG_BITEXACT);
|
||||
ff_mdct_init(&s->imdct, 6, 1, 1.0);
|
||||
ff_synth_filter_init(&s->synth);
|
||||
ff_dcadsp_init(&s->dcadsp);
|
||||
|
@ -127,7 +127,7 @@ static const enum PixelFormat dirac_pix_fmt[2][3] = {
|
||||
static int parse_source_parameters(AVCodecContext *avctx, GetBitContext *gb,
|
||||
dirac_source_params *source)
|
||||
{
|
||||
AVRational frame_rate = (AVRational){0,0};
|
||||
AVRational frame_rate = {0,0};
|
||||
unsigned luma_depth = 8, luma_offset = 16;
|
||||
int idx;
|
||||
|
||||
|
@ -2509,14 +2509,6 @@ static void vector_fmul_scalar_c(float *dst, const float *src, float mul,
|
||||
dst[i] = src[i] * mul;
|
||||
}
|
||||
|
||||
static void vector_fmac_scalar_c(float *dst, const float *src, float mul,
|
||||
int len)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < len; i++)
|
||||
dst[i] += src[i] * mul;
|
||||
}
|
||||
|
||||
static void butterflies_float_c(float *av_restrict v1, float *av_restrict v2,
|
||||
int len)
|
||||
{
|
||||
@ -3060,7 +3052,6 @@ av_cold void ff_dsputil_init(DSPContext* c, AVCodecContext *avctx)
|
||||
c->butterflies_float = butterflies_float_c;
|
||||
c->butterflies_float_interleave = butterflies_float_interleave_c;
|
||||
c->vector_fmul_scalar = vector_fmul_scalar_c;
|
||||
c->vector_fmac_scalar = vector_fmac_scalar_c;
|
||||
|
||||
c->shrink[0]= av_image_copy_plane;
|
||||
c->shrink[1]= ff_shrink22;
|
||||
|
@ -420,17 +420,6 @@ typedef struct DSPContext {
|
||||
*/
|
||||
void (*vector_fmul_scalar)(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
/**
|
||||
* Multiply a vector of floats by a scalar float and add to
|
||||
* destination vector. Source and destination vectors must
|
||||
* overlap exactly or not at all.
|
||||
* @param dst result vector, 16-byte aligned
|
||||
* @param src input vector, 16-byte aligned
|
||||
* @param mul scalar value
|
||||
* @param len length of vector, multiple of 4
|
||||
*/
|
||||
void (*vector_fmac_scalar)(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
/**
|
||||
* Calculate the scalar product of two vectors of floats.
|
||||
* @param v1 first vector, 16-byte aligned
|
||||
|
209
libavcodec/libilbc.c
Normal file
209
libavcodec/libilbc.c
Normal file
@ -0,0 +1,209 @@
|
||||
/*
|
||||
* iLBC decoder/encoder stub
|
||||
* Copyright (c) 2012 Martin Storsjo
|
||||
*
|
||||
* 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 <ilbc.h>
|
||||
|
||||
#include "avcodec.h"
|
||||
#include "libavutil/opt.h"
|
||||
#include "internal.h"
|
||||
|
||||
static int get_mode(AVCodecContext *avctx)
|
||||
{
|
||||
if (avctx->block_align == 38)
|
||||
return 20;
|
||||
else if (avctx->block_align == 50)
|
||||
return 30;
|
||||
else if (avctx->bit_rate > 0)
|
||||
return avctx->bit_rate <= 14000 ? 30 : 20;
|
||||
else
|
||||
return -1;
|
||||
}
|
||||
|
||||
typedef struct ILBCDecContext {
|
||||
const AVClass *class;
|
||||
AVFrame frame;
|
||||
iLBC_Dec_Inst_t decoder;
|
||||
int enhance;
|
||||
} ILBCDecContext;
|
||||
|
||||
static const AVOption ilbc_dec_options[] = {
|
||||
{ "enhance", "Enhance the decoded audio (adds delay)", offsetof(ILBCDecContext, enhance), AV_OPT_TYPE_INT, { 0 }, 0, 1, AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_DECODING_PARAM },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
static const AVClass ilbc_dec_class = {
|
||||
"libilbc", av_default_item_name, ilbc_dec_options, LIBAVUTIL_VERSION_INT
|
||||
};
|
||||
|
||||
static av_cold int ilbc_decode_init(AVCodecContext *avctx)
|
||||
{
|
||||
ILBCDecContext *s = avctx->priv_data;
|
||||
int mode;
|
||||
|
||||
if ((mode = get_mode(avctx)) < 0) {
|
||||
av_log(avctx, AV_LOG_ERROR, "iLBC frame mode not indicated\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
|
||||
WebRtcIlbcfix_InitDecode(&s->decoder, mode, s->enhance);
|
||||
avcodec_get_frame_defaults(&s->frame);
|
||||
avctx->coded_frame = &s->frame;
|
||||
|
||||
avctx->channels = 1;
|
||||
avctx->sample_rate = 8000;
|
||||
avctx->sample_fmt = AV_SAMPLE_FMT_S16;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_decode_frame(AVCodecContext *avctx, void *data,
|
||||
int *got_frame_ptr, AVPacket *avpkt)
|
||||
{
|
||||
const uint8_t *buf = avpkt->data;
|
||||
int buf_size = avpkt->size;
|
||||
ILBCDecContext *s = avctx->priv_data;
|
||||
int ret;
|
||||
|
||||
if (s->decoder.no_of_bytes > buf_size) {
|
||||
av_log(avctx, AV_LOG_ERROR, "iLBC frame too short (%u, should be %u)\n",
|
||||
buf_size, s->decoder.no_of_bytes);
|
||||
return AVERROR_INVALIDDATA;
|
||||
}
|
||||
|
||||
s->frame.nb_samples = s->decoder.blockl;
|
||||
if ((ret = avctx->get_buffer(avctx, &s->frame)) < 0) {
|
||||
av_log(avctx, AV_LOG_ERROR, "get_buffer() failed\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
WebRtcIlbcfix_DecodeImpl((WebRtc_Word16*) s->frame.data[0],
|
||||
(const WebRtc_UWord16*) buf, &s->decoder, 1);
|
||||
|
||||
*got_frame_ptr = 1;
|
||||
*(AVFrame *)data = s->frame;
|
||||
|
||||
return s->decoder.no_of_bytes;
|
||||
}
|
||||
|
||||
AVCodec ff_libilbc_decoder = {
|
||||
.name = "libilbc",
|
||||
.type = AVMEDIA_TYPE_AUDIO,
|
||||
.id = CODEC_ID_ILBC,
|
||||
.priv_data_size = sizeof(ILBCDecContext),
|
||||
.init = ilbc_decode_init,
|
||||
.decode = ilbc_decode_frame,
|
||||
.capabilities = CODEC_CAP_DR1,
|
||||
.long_name = NULL_IF_CONFIG_SMALL("Internet Low Bitrate Codec (iLBC)"),
|
||||
.priv_class = &ilbc_dec_class,
|
||||
};
|
||||
|
||||
typedef struct ILBCEncContext {
|
||||
const AVClass *class;
|
||||
iLBC_Enc_Inst_t encoder;
|
||||
int mode;
|
||||
} ILBCEncContext;
|
||||
|
||||
static const AVOption ilbc_enc_options[] = {
|
||||
{ "mode", "iLBC mode (20 or 30 ms frames)", offsetof(ILBCEncContext, mode), AV_OPT_TYPE_INT, { 20 }, 20, 30, AV_OPT_FLAG_AUDIO_PARAM | AV_OPT_FLAG_ENCODING_PARAM },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
static const AVClass ilbc_enc_class = {
|
||||
"libilbc", av_default_item_name, ilbc_enc_options, LIBAVUTIL_VERSION_INT
|
||||
};
|
||||
|
||||
static av_cold int ilbc_encode_init(AVCodecContext *avctx)
|
||||
{
|
||||
ILBCEncContext *s = avctx->priv_data;
|
||||
int mode;
|
||||
|
||||
if (avctx->sample_rate != 8000) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Only 8000Hz sample rate supported\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
|
||||
if (avctx->channels != 1) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Only mono supported\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
|
||||
if ((mode = get_mode(avctx)) > 0)
|
||||
s->mode = mode;
|
||||
else
|
||||
s->mode = s->mode != 30 ? 20 : 30;
|
||||
WebRtcIlbcfix_InitEncode(&s->encoder, s->mode);
|
||||
|
||||
avctx->block_align = s->encoder.no_of_bytes;
|
||||
avctx->frame_size = s->encoder.blockl;
|
||||
#if FF_API_OLD_ENCODE_AUDIO
|
||||
avctx->coded_frame = avcodec_alloc_frame();
|
||||
if (!avctx->coded_frame)
|
||||
return AVERROR(ENOMEM);
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static av_cold int ilbc_encode_close(AVCodecContext *avctx)
|
||||
{
|
||||
#if FF_API_OLD_ENCODE_AUDIO
|
||||
av_freep(&avctx->coded_frame);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_encode_frame(AVCodecContext *avctx, AVPacket *avpkt,
|
||||
const AVFrame *frame, int *got_packet_ptr)
|
||||
{
|
||||
ILBCEncContext *s = avctx->priv_data;
|
||||
int ret;
|
||||
|
||||
if ((ret = ff_alloc_packet(avpkt, 50))) {
|
||||
av_log(avctx, AV_LOG_ERROR, "Error getting output packet\n");
|
||||
return ret;
|
||||
}
|
||||
|
||||
WebRtcIlbcfix_EncodeImpl((WebRtc_UWord16*) avpkt->data, (const WebRtc_Word16*) frame->data[0], &s->encoder);
|
||||
|
||||
avpkt->size = s->encoder.no_of_bytes;
|
||||
*got_packet_ptr = 1;
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const AVCodecDefault ilbc_encode_defaults[] = {
|
||||
{ "b", "0" },
|
||||
{ NULL }
|
||||
};
|
||||
|
||||
AVCodec ff_libilbc_encoder = {
|
||||
.name = "libilbc",
|
||||
.type = AVMEDIA_TYPE_AUDIO,
|
||||
.id = CODEC_ID_ILBC,
|
||||
.priv_data_size = sizeof(ILBCEncContext),
|
||||
.init = ilbc_encode_init,
|
||||
.encode2 = ilbc_encode_frame,
|
||||
.close = ilbc_encode_close,
|
||||
.sample_fmts = (const enum AVSampleFormat[]){ AV_SAMPLE_FMT_S16,
|
||||
AV_SAMPLE_FMT_NONE },
|
||||
.long_name = NULL_IF_CONFIG_SMALL("Internet Low Bitrate Codec (iLBC)"),
|
||||
.defaults = ilbc_encode_defaults,
|
||||
.priv_class = &ilbc_enc_class,
|
||||
};
|
@ -514,14 +514,6 @@ static double predict_size(Predictor *p, double q, double var)
|
||||
return p->coeff*var / (q*p->count);
|
||||
}
|
||||
|
||||
/*
|
||||
static double predict_qp(Predictor *p, double size, double var)
|
||||
{
|
||||
//printf("coeff:%f, count:%f, var:%f, size:%f//\n", p->coeff, p->count, var, size);
|
||||
return p->coeff*var / (size*p->count);
|
||||
}
|
||||
*/
|
||||
|
||||
static void update_predictor(Predictor *p, double q, double var, double size)
|
||||
{
|
||||
double new_coeff= size*q / (var + 1);
|
||||
@ -561,10 +553,6 @@ static void adaptive_quantization(MpegEncContext *s, double q){
|
||||
int mb_y = mb_xy / s->mb_stride;
|
||||
int mb_distance;
|
||||
float mb_factor = 0.0;
|
||||
#if 0
|
||||
if(spat_cplx < q/3) spat_cplx= q/3; //FIXME finetune
|
||||
if(temp_cplx < q/3) temp_cplx= q/3; //FIXME finetune
|
||||
#endif
|
||||
if(spat_cplx < 4) spat_cplx= 4; //FIXME finetune
|
||||
if(temp_cplx < 4) temp_cplx= 4; //FIXME finetune
|
||||
|
||||
|
@ -2138,6 +2138,11 @@ int av_get_audio_frame_duration(AVCodecContext *avctx, int frame_bytes)
|
||||
case 29: return 288;
|
||||
case 37: return 480;
|
||||
}
|
||||
} else if (id == CODEC_ID_ILBC) {
|
||||
switch (ba) {
|
||||
case 38: return 160;
|
||||
case 50: return 240;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -27,7 +27,7 @@
|
||||
*/
|
||||
|
||||
#define LIBAVCODEC_VERSION_MAJOR 54
|
||||
#define LIBAVCODEC_VERSION_MINOR 25
|
||||
#define LIBAVCODEC_VERSION_MINOR 26
|
||||
#define LIBAVCODEC_VERSION_MICRO 100
|
||||
|
||||
#define LIBAVCODEC_VERSION_INT AV_VERSION_INT(LIBAVCODEC_VERSION_MAJOR, \
|
||||
|
@ -32,6 +32,7 @@
|
||||
#include "libavutil/audio_fifo.h"
|
||||
#include "libavutil/avassert.h"
|
||||
#include "libavutil/avstring.h"
|
||||
#include "libavutil/float_dsp.h"
|
||||
#include "libavutil/mathematics.h"
|
||||
#include "libavutil/opt.h"
|
||||
#include "libavutil/samplefmt.h"
|
||||
@ -152,6 +153,7 @@ static int frame_list_add_frame(FrameList *frame_list, int nb_samples, int64_t p
|
||||
|
||||
typedef struct MixContext {
|
||||
const AVClass *class; /**< class for AVOptions */
|
||||
AVFloatDSPContext fdsp;
|
||||
|
||||
int nb_inputs; /**< number of inputs */
|
||||
int active_inputs; /**< number of input currently active */
|
||||
@ -160,6 +162,7 @@ typedef struct MixContext {
|
||||
|
||||
int nb_channels; /**< number of channels */
|
||||
int sample_rate; /**< sample rate */
|
||||
int planar;
|
||||
AVAudioFifo **fifos; /**< audio fifo for each input */
|
||||
uint8_t *input_state; /**< current state of each input */
|
||||
float *input_scale; /**< mixing scale factor for each input */
|
||||
@ -224,6 +227,7 @@ static int config_output(AVFilterLink *outlink)
|
||||
int i;
|
||||
char buf[64];
|
||||
|
||||
s->planar = av_sample_fmt_is_planar(outlink->format);
|
||||
s->sample_rate = outlink->sample_rate;
|
||||
outlink->time_base = (AVRational){ 1, outlink->sample_rate };
|
||||
s->next_pts = AV_NOPTS_VALUE;
|
||||
@ -264,14 +268,6 @@ static int config_output(AVFilterLink *outlink)
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* TODO: move optimized version from DSPContext to libavutil */
|
||||
static void vector_fmac_scalar(float *dst, const float *src, float mul, int len)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < len; i++)
|
||||
dst[i] += src[i] * mul;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read samples from the input FIFOs, mix, and write to the output link.
|
||||
*/
|
||||
@ -294,11 +290,20 @@ static int output_frame(AVFilterLink *outlink, int nb_samples)
|
||||
|
||||
for (i = 0; i < s->nb_inputs; i++) {
|
||||
if (s->input_state[i] == INPUT_ON) {
|
||||
int planes, plane_size, p;
|
||||
|
||||
av_audio_fifo_read(s->fifos[i], (void **)in_buf->extended_data,
|
||||
nb_samples);
|
||||
vector_fmac_scalar((float *)out_buf->extended_data[0],
|
||||
(float *) in_buf->extended_data[0],
|
||||
s->input_scale[i], nb_samples * s->nb_channels);
|
||||
|
||||
planes = s->planar ? s->nb_channels : 1;
|
||||
plane_size = nb_samples * (s->planar ? 1 : s->nb_channels);
|
||||
plane_size = FFALIGN(plane_size, 16);
|
||||
|
||||
for (p = 0; p < planes; p++) {
|
||||
s->fdsp.vector_fmac_scalar((float *)out_buf->extended_data[p],
|
||||
(float *) in_buf->extended_data[p],
|
||||
s->input_scale[i], plane_size);
|
||||
}
|
||||
}
|
||||
}
|
||||
avfilter_unref_buffer(in_buf);
|
||||
@ -501,6 +506,8 @@ static int init(AVFilterContext *ctx, const char *args, void *opaque)
|
||||
ff_insert_inpad(ctx, i, &pad);
|
||||
}
|
||||
|
||||
avpriv_float_dsp_init(&s->fdsp, 0);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
@ -527,6 +534,7 @@ static int query_formats(AVFilterContext *ctx)
|
||||
{
|
||||
AVFilterFormats *formats = NULL;
|
||||
ff_add_format(&formats, AV_SAMPLE_FMT_FLT);
|
||||
ff_add_format(&formats, AV_SAMPLE_FMT_FLTP);
|
||||
ff_set_common_formats(ctx, formats);
|
||||
ff_set_common_channel_layouts(ctx, ff_all_channel_layouts());
|
||||
ff_set_common_samplerates(ctx, ff_all_samplerates());
|
||||
|
@ -123,6 +123,8 @@ OBJS-$(CONFIG_ICO_DEMUXER) += icodec.o
|
||||
OBJS-$(CONFIG_IDCIN_DEMUXER) += idcin.o
|
||||
OBJS-$(CONFIG_IDF_DEMUXER) += bintext.o
|
||||
OBJS-$(CONFIG_IFF_DEMUXER) += iff.o
|
||||
OBJS-$(CONFIG_ILBC_DEMUXER) += ilbc.o
|
||||
OBJS-$(CONFIG_ILBC_MUXER) += ilbc.o
|
||||
OBJS-$(CONFIG_IMAGE2_DEMUXER) += img2dec.o img2.o
|
||||
OBJS-$(CONFIG_IMAGE2_MUXER) += img2enc.o img2.o
|
||||
OBJS-$(CONFIG_IMAGE2PIPE_DEMUXER) += img2dec.o img2.o
|
||||
@ -281,6 +283,7 @@ OBJS-$(CONFIG_RTPDEC) += rdt.o \
|
||||
rtpdec_h263.o \
|
||||
rtpdec_h263_rfc2190.o \
|
||||
rtpdec_h264.o \
|
||||
rtpdec_ilbc.o \
|
||||
rtpdec_latm.o \
|
||||
rtpdec_mpeg4.o \
|
||||
rtpdec_qcelp.o \
|
||||
|
@ -116,6 +116,7 @@ void av_register_all(void)
|
||||
REGISTER_DEMUXER (IDCIN, idcin);
|
||||
REGISTER_DEMUXER (IDF, idf);
|
||||
REGISTER_DEMUXER (IFF, iff);
|
||||
REGISTER_MUXDEMUX (ILBC, ilbc);
|
||||
REGISTER_MUXDEMUX (IMAGE2, image2);
|
||||
REGISTER_MUXDEMUX (IMAGE2PIPE, image2pipe);
|
||||
REGISTER_DEMUXER (INGENIENT, ingenient);
|
||||
|
@ -759,33 +759,6 @@ static int flv_read_seek(AVFormatContext *s, int stream_index,
|
||||
return avio_seek_time(s->pb, stream_index, ts, flags);
|
||||
}
|
||||
|
||||
#if 0 /* don't know enough to implement this */
|
||||
static int flv_read_seek2(AVFormatContext *s, int stream_index,
|
||||
int64_t min_ts, int64_t ts, int64_t max_ts, int flags)
|
||||
{
|
||||
int ret = AVERROR(ENOSYS);
|
||||
|
||||
if (ts - min_ts > (uint64_t)(max_ts - ts)) flags |= AVSEEK_FLAG_BACKWARD;
|
||||
|
||||
if (!s->pb->seekable) {
|
||||
if (stream_index < 0) {
|
||||
stream_index = av_find_default_stream_index(s);
|
||||
if (stream_index < 0)
|
||||
return -1;
|
||||
|
||||
/* timestamp for default must be expressed in AV_TIME_BASE units */
|
||||
ts = av_rescale_rnd(ts, 1000, AV_TIME_BASE,
|
||||
flags & AVSEEK_FLAG_BACKWARD ? AV_ROUND_DOWN : AV_ROUND_UP);
|
||||
}
|
||||
ret = avio_seek_time(s->pb, stream_index, ts, flags);
|
||||
}
|
||||
|
||||
if (ret == AVERROR(ENOSYS))
|
||||
ret = av_seek_frame(s, stream_index, ts, flags);
|
||||
return ret;
|
||||
}
|
||||
#endif
|
||||
|
||||
AVInputFormat ff_flv_demuxer = {
|
||||
.name = "flv",
|
||||
.long_name = NULL_IF_CONFIG_SMALL("FLV format"),
|
||||
@ -794,9 +767,6 @@ AVInputFormat ff_flv_demuxer = {
|
||||
.read_header = flv_read_header,
|
||||
.read_packet = flv_read_packet,
|
||||
.read_seek = flv_read_seek,
|
||||
#if 0
|
||||
.read_seek2 = flv_read_seek2,
|
||||
#endif
|
||||
.read_close = flv_read_close,
|
||||
.extensions = "flv",
|
||||
};
|
||||
|
@ -308,15 +308,15 @@ static int process_line(URLContext *h, char *line, int line_count,
|
||||
strcpy(s->location, p);
|
||||
*new_location = 1;
|
||||
} else if (!av_strcasecmp (tag, "Content-Length") && s->filesize == -1) {
|
||||
s->filesize = atoll(p);
|
||||
s->filesize = strtoll(p, NULL, 10);
|
||||
} else if (!av_strcasecmp (tag, "Content-Range")) {
|
||||
/* "bytes $from-$to/$document_size" */
|
||||
const char *slash;
|
||||
if (!strncmp (p, "bytes ", 6)) {
|
||||
p += 6;
|
||||
s->off = atoll(p);
|
||||
s->off = strtoll(p, NULL, 10);
|
||||
if ((slash = strchr(p, '/')) && strlen(slash) > 0)
|
||||
s->filesize = atoll(slash+1);
|
||||
s->filesize = strtoll(slash+1, NULL, 10);
|
||||
}
|
||||
h->is_streamed = 0; /* we _can_ in fact seek */
|
||||
} else if (!av_strcasecmp(tag, "Accept-Ranges") && !strncmp(p, "bytes", 5)) {
|
||||
|
141
libavformat/ilbc.c
Normal file
141
libavformat/ilbc.c
Normal file
@ -0,0 +1,141 @@
|
||||
/*
|
||||
* iLBC storage file format
|
||||
* Copyright (c) 2012 Martin Storsjo
|
||||
*
|
||||
* 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 "internal.h"
|
||||
|
||||
static const char mode20_header[] = "#!iLBC20\n";
|
||||
static const char mode30_header[] = "#!iLBC30\n";
|
||||
|
||||
static int ilbc_write_header(AVFormatContext *s)
|
||||
{
|
||||
AVIOContext *pb = s->pb;
|
||||
AVCodecContext *enc;
|
||||
|
||||
if (s->nb_streams != 1) {
|
||||
av_log(s, AV_LOG_ERROR, "Unsupported number of streams\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
enc = s->streams[0]->codec;
|
||||
|
||||
if (enc->codec_id != CODEC_ID_ILBC) {
|
||||
av_log(s, AV_LOG_ERROR, "Unsupported codec\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
|
||||
if (enc->block_align == 50) {
|
||||
avio_write(pb, mode30_header, sizeof(mode30_header) - 1);
|
||||
} else if (enc->block_align == 38) {
|
||||
avio_write(pb, mode20_header, sizeof(mode20_header) - 1);
|
||||
} else {
|
||||
av_log(s, AV_LOG_ERROR, "Unsupported mode\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
avio_flush(pb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_write_packet(AVFormatContext *s, AVPacket *pkt)
|
||||
{
|
||||
avio_write(s->pb, pkt->data, pkt->size);
|
||||
avio_flush(s->pb);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_probe(AVProbeData *p)
|
||||
{
|
||||
// Only check for "#!iLBC" which matches both formats
|
||||
if (!memcmp(p->buf, mode20_header, 6))
|
||||
return AVPROBE_SCORE_MAX;
|
||||
else
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_read_header(AVFormatContext *s)
|
||||
{
|
||||
AVIOContext *pb = s->pb;
|
||||
AVStream *st;
|
||||
uint8_t header[9];
|
||||
|
||||
avio_read(pb, header, 9);
|
||||
|
||||
st = avformat_new_stream(s, NULL);
|
||||
if (!st)
|
||||
return AVERROR(ENOMEM);
|
||||
st->codec->codec_id = CODEC_ID_ILBC;
|
||||
st->codec->sample_rate = 8000;
|
||||
st->codec->channels = 1;
|
||||
st->codec->codec_type = AVMEDIA_TYPE_AUDIO;
|
||||
st->start_time = 0;
|
||||
avpriv_set_pts_info(st, 64, 1, st->codec->sample_rate);
|
||||
if (!memcmp(header, mode20_header, sizeof(mode20_header) - 1)) {
|
||||
st->codec->block_align = 38;
|
||||
st->codec->bit_rate = 15200;
|
||||
} else if (!memcmp(header, mode30_header, sizeof(mode30_header) - 1)) {
|
||||
st->codec->block_align = 50;
|
||||
st->codec->bit_rate = 13333;
|
||||
} else {
|
||||
av_log(s, AV_LOG_ERROR, "Unrecognized iLBC file header\n");
|
||||
return AVERROR_INVALIDDATA;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_read_packet(AVFormatContext *s,
|
||||
AVPacket *pkt)
|
||||
{
|
||||
AVCodecContext *enc = s->streams[0]->codec;
|
||||
int ret;
|
||||
|
||||
if ((ret = av_new_packet(pkt, enc->block_align)) < 0)
|
||||
return ret;
|
||||
|
||||
pkt->stream_index = 0;
|
||||
pkt->pos = avio_tell(s->pb);
|
||||
pkt->duration = enc->block_align == 38 ? 160 : 240;
|
||||
if ((ret = avio_read(s->pb, pkt->data, enc->block_align)) != enc->block_align) {
|
||||
av_free_packet(pkt);
|
||||
return ret < 0 ? ret : AVERROR(EIO);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
AVInputFormat ff_ilbc_demuxer = {
|
||||
.name = "ilbc",
|
||||
.long_name = NULL_IF_CONFIG_SMALL("iLBC storage file format"),
|
||||
.read_probe = ilbc_probe,
|
||||
.read_header = ilbc_read_header,
|
||||
.read_packet = ilbc_read_packet,
|
||||
.flags = AVFMT_GENERIC_INDEX,
|
||||
};
|
||||
|
||||
AVOutputFormat ff_ilbc_muxer = {
|
||||
.name = "ilbc",
|
||||
.long_name = NULL_IF_CONFIG_SMALL("iLBC storage file format"),
|
||||
.mime_type = "audio/iLBC",
|
||||
.extensions = "lbc",
|
||||
.audio_codec = CODEC_ID_ILBC,
|
||||
.write_header = ilbc_write_header,
|
||||
.write_packet = ilbc_write_packet,
|
||||
.flags = AVFMT_NOTIMESTAMPS,
|
||||
};
|
@ -259,6 +259,7 @@ const AVCodecTag ff_codec_movaudio_tags[] = {
|
||||
{ CODEC_ID_DVAUDIO, MKTAG('v', 'd', 'v', 'a') },
|
||||
{ CODEC_ID_DVAUDIO, MKTAG('d', 'v', 'c', 'a') },
|
||||
{ CODEC_ID_GSM, MKTAG('a', 'g', 's', 'm') },
|
||||
{ CODEC_ID_ILBC, MKTAG('i', 'l', 'b', 'c') },
|
||||
{ CODEC_ID_MACE3, MKTAG('M', 'A', 'C', '3') },
|
||||
{ CODEC_ID_MACE6, MKTAG('M', 'A', 'C', '6') },
|
||||
{ CODEC_ID_MP1, MKTAG('.', 'm', 'p', '1') },
|
||||
|
@ -1540,6 +1540,7 @@ int ff_mov_read_stsd_entries(MOVContext *c, AVIOContext *pb, int entries)
|
||||
case CODEC_ID_GSM:
|
||||
case CODEC_ID_ADPCM_MS:
|
||||
case CODEC_ID_ADPCM_IMA_WAV:
|
||||
case CODEC_ID_ILBC:
|
||||
st->codec->block_align = sc->bytes_per_frame;
|
||||
break;
|
||||
case CODEC_ID_ALAC:
|
||||
|
@ -3371,7 +3371,8 @@ static int mov_write_header(AVFormatContext *s)
|
||||
av_log(s, AV_LOG_WARNING, "track %d: codec frame size is not set\n", i);
|
||||
track->audio_vbr = 1;
|
||||
}else if(st->codec->codec_id == CODEC_ID_ADPCM_MS ||
|
||||
st->codec->codec_id == CODEC_ID_ADPCM_IMA_WAV){
|
||||
st->codec->codec_id == CODEC_ID_ADPCM_IMA_WAV ||
|
||||
st->codec->codec_id == CODEC_ID_ILBC){
|
||||
if (!st->codec->block_align) {
|
||||
av_log(s, AV_LOG_ERROR, "track %d: codec block align is not set for adpcm\n", i);
|
||||
goto error;
|
||||
|
@ -19,8 +19,6 @@
|
||||
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
||||
*/
|
||||
|
||||
//#define USE_SYNCPOINT_SEARCH
|
||||
|
||||
#include "libavutil/crc.h"
|
||||
#include "libavutil/intreadwrite.h"
|
||||
#include "libavutil/log.h"
|
||||
@ -2161,92 +2159,6 @@ static int64_t mpegts_get_dts(AVFormatContext *s, int stream_index,
|
||||
return AV_NOPTS_VALUE;
|
||||
}
|
||||
|
||||
#ifdef USE_SYNCPOINT_SEARCH
|
||||
|
||||
static int read_seek2(AVFormatContext *s,
|
||||
int stream_index,
|
||||
int64_t min_ts,
|
||||
int64_t target_ts,
|
||||
int64_t max_ts,
|
||||
int flags)
|
||||
{
|
||||
int64_t pos;
|
||||
|
||||
int64_t ts_ret, ts_adj;
|
||||
int stream_index_gen_search;
|
||||
AVStream *st;
|
||||
AVParserState *backup;
|
||||
|
||||
backup = ff_store_parser_state(s);
|
||||
|
||||
// detect direction of seeking for search purposes
|
||||
flags |= (target_ts - min_ts > (uint64_t)(max_ts - target_ts)) ?
|
||||
AVSEEK_FLAG_BACKWARD : 0;
|
||||
|
||||
if (flags & AVSEEK_FLAG_BYTE) {
|
||||
// use position directly, we will search starting from it
|
||||
pos = target_ts;
|
||||
} else {
|
||||
// search for some position with good timestamp match
|
||||
if (stream_index < 0) {
|
||||
stream_index_gen_search = av_find_default_stream_index(s);
|
||||
if (stream_index_gen_search < 0) {
|
||||
ff_restore_parser_state(s, backup);
|
||||
return -1;
|
||||
}
|
||||
|
||||
st = s->streams[stream_index_gen_search];
|
||||
// timestamp for default must be expressed in AV_TIME_BASE units
|
||||
ts_adj = av_rescale(target_ts,
|
||||
st->time_base.den,
|
||||
AV_TIME_BASE * (int64_t)st->time_base.num);
|
||||
} else {
|
||||
ts_adj = target_ts;
|
||||
stream_index_gen_search = stream_index;
|
||||
}
|
||||
pos = ff_gen_search(s, stream_index_gen_search, ts_adj,
|
||||
0, INT64_MAX, -1,
|
||||
AV_NOPTS_VALUE,
|
||||
AV_NOPTS_VALUE,
|
||||
flags, &ts_ret, mpegts_get_pcr);
|
||||
if (pos < 0) {
|
||||
ff_restore_parser_state(s, backup);
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
|
||||
// search for actual matching keyframe/starting position for all streams
|
||||
if (ff_gen_syncpoint_search(s, stream_index, pos,
|
||||
min_ts, target_ts, max_ts,
|
||||
flags) < 0) {
|
||||
ff_restore_parser_state(s, backup);
|
||||
return -1;
|
||||
}
|
||||
|
||||
ff_free_parser_state(s, backup);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int read_seek(AVFormatContext *s, int stream_index, int64_t target_ts, int flags)
|
||||
{
|
||||
int ret;
|
||||
if (flags & AVSEEK_FLAG_BACKWARD) {
|
||||
flags &= ~AVSEEK_FLAG_BACKWARD;
|
||||
ret = read_seek2(s, stream_index, INT64_MIN, target_ts, target_ts, flags);
|
||||
if (ret < 0)
|
||||
// for compatibility reasons, seek to the best-fitting timestamp
|
||||
ret = read_seek2(s, stream_index, INT64_MIN, target_ts, INT64_MAX, flags);
|
||||
} else {
|
||||
ret = read_seek2(s, stream_index, target_ts, target_ts, INT64_MAX, flags);
|
||||
if (ret < 0)
|
||||
// for compatibility reasons, seek to the best-fitting timestamp
|
||||
ret = read_seek2(s, stream_index, INT64_MIN, target_ts, INT64_MAX, flags);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
/**************************************************************/
|
||||
/* parsing functions - called from other demuxers such as RTP */
|
||||
|
||||
@ -2313,9 +2225,6 @@ AVInputFormat ff_mpegts_demuxer = {
|
||||
.read_close = mpegts_read_close,
|
||||
.read_timestamp = mpegts_get_dts,
|
||||
.flags = AVFMT_SHOW_IDS | AVFMT_TS_DISCONT,
|
||||
#ifdef USE_SYNCPOINT_SEARCH
|
||||
.read_seek2 = read_seek2,
|
||||
#endif
|
||||
};
|
||||
|
||||
AVInputFormat ff_mpegtsraw_demuxer = {
|
||||
@ -2327,8 +2236,5 @@ AVInputFormat ff_mpegtsraw_demuxer = {
|
||||
.read_close = mpegts_read_close,
|
||||
.read_timestamp = mpegts_get_dts,
|
||||
.flags = AVFMT_SHOW_IDS | AVFMT_TS_DISCONT,
|
||||
#ifdef USE_SYNCPOINT_SEARCH
|
||||
.read_seek2 = read_seek2,
|
||||
#endif
|
||||
.priv_class = &mpegtsraw_class,
|
||||
};
|
||||
|
@ -76,6 +76,7 @@ typedef struct RTMPContext {
|
||||
uint8_t* flv_data; ///< buffer with data for demuxer
|
||||
int flv_size; ///< current buffer size
|
||||
int flv_off; ///< number of bytes read from current buffer
|
||||
int flv_nb_packets; ///< number of flv packets published
|
||||
RTMPPacket out_pkt; ///< rtmp packet, created from flv a/v or metadata (for output)
|
||||
uint32_t client_report_size; ///< number of bytes after which client should report to server
|
||||
uint32_t bytes_read; ///< number of bytes read from server
|
||||
@ -90,6 +91,7 @@ typedef struct RTMPContext {
|
||||
char* swfurl; ///< url of the swf player
|
||||
int server_bw; ///< server bandwidth
|
||||
int client_buffer_time; ///< client buffer time in ms
|
||||
int flush_interval; ///< number of packets flushed in the same request (RTMPT only)
|
||||
} RTMPContext;
|
||||
|
||||
#define PLAYER_KEY_OPEN_PART_LEN 30 ///< length of partial key used for first client digest signing
|
||||
@ -1361,9 +1363,14 @@ static int rtmp_write(URLContext *s, const uint8_t *buf, int size)
|
||||
rt->flv_size = 0;
|
||||
rt->flv_off = 0;
|
||||
rt->flv_header_bytes = 0;
|
||||
rt->flv_nb_packets++;
|
||||
}
|
||||
} while (buf_temp - buf < size);
|
||||
|
||||
if (rt->flv_nb_packets < rt->flush_interval)
|
||||
return size;
|
||||
rt->flv_nb_packets = 0;
|
||||
|
||||
/* set stream into nonblocking mode */
|
||||
rt->stream->flags |= AVIO_FLAG_NONBLOCK;
|
||||
|
||||
@ -1404,6 +1411,7 @@ static const AVOption rtmp_options[] = {
|
||||
{"rtmp_buffer", "Set buffer time in milliseconds. The default is 3000.", OFFSET(client_buffer_time), AV_OPT_TYPE_INT, {3000}, 0, INT_MAX, DEC|ENC},
|
||||
{"rtmp_conn", "Append arbitrary AMF data to the Connect message", OFFSET(conn), AV_OPT_TYPE_STRING, {.str = NULL }, 0, 0, DEC|ENC},
|
||||
{"rtmp_flashver", "Version of the Flash plugin used to run the SWF player.", OFFSET(flashver), AV_OPT_TYPE_STRING, {.str = NULL }, 0, 0, DEC|ENC},
|
||||
{"rtmp_flush_interval", "Number of packets flushed in the same request (RTMPT only).", OFFSET(flush_interval), AV_OPT_TYPE_INT, {10}, 0, INT_MAX, ENC},
|
||||
{"rtmp_live", "Specify that the media is a live stream.", OFFSET(live), AV_OPT_TYPE_INT, {-2}, INT_MIN, INT_MAX, DEC, "rtmp_live"},
|
||||
{"any", "both", 0, AV_OPT_TYPE_CONST, {-2}, 0, 0, DEC, "rtmp_live"},
|
||||
{"live", "live stream", 0, AV_OPT_TYPE_CONST, {-1}, 0, 0, DEC, "rtmp_live"},
|
||||
|
@ -68,6 +68,7 @@ void av_register_rtp_dynamic_payload_handlers(void)
|
||||
ff_register_dynamic_payload_handler(&ff_h263_2000_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_h263_rfc2190_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_h264_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_ilbc_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_vorbis_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_theora_dynamic_handler);
|
||||
ff_register_dynamic_payload_handler(&ff_qdm2_dynamic_handler);
|
||||
|
@ -45,6 +45,7 @@ extern RTPDynamicProtocolHandler ff_h263_1998_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_h263_2000_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_h263_rfc2190_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_h264_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_ilbc_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_mp4a_latm_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_mp4v_es_dynamic_handler;
|
||||
extern RTPDynamicProtocolHandler ff_mpeg4_generic_dynamic_handler;
|
||||
|
73
libavformat/rtpdec_ilbc.c
Normal file
73
libavformat/rtpdec_ilbc.c
Normal file
@ -0,0 +1,73 @@
|
||||
/*
|
||||
* RTP iLBC Depacketizer, RFC 3952
|
||||
* Copyright (c) 2012 Martin Storsjo
|
||||
*
|
||||
* 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 "rtpdec_formats.h"
|
||||
#include "libavutil/avstring.h"
|
||||
|
||||
static int ilbc_parse_fmtp(AVStream *stream, PayloadContext *data,
|
||||
char *attr, char *value)
|
||||
{
|
||||
if (!strcmp(attr, "mode")) {
|
||||
int mode = atoi(value);
|
||||
switch (mode) {
|
||||
case 20:
|
||||
stream->codec->block_align = 38;
|
||||
break;
|
||||
case 30:
|
||||
stream->codec->block_align = 50;
|
||||
break;
|
||||
default:
|
||||
av_log(NULL, AV_LOG_ERROR, "Unsupported iLBC mode %d\n", mode);
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int ilbc_parse_sdp_line(AVFormatContext *s, int st_index,
|
||||
PayloadContext *data, const char *line)
|
||||
{
|
||||
const char *p;
|
||||
AVStream *st;
|
||||
|
||||
if (st_index < 0)
|
||||
return 0;
|
||||
st = s->streams[st_index];
|
||||
|
||||
if (av_strstart(line, "fmtp:", &p)) {
|
||||
int ret = ff_parse_fmtp(st, data, p, ilbc_parse_fmtp);
|
||||
if (ret < 0)
|
||||
return ret;
|
||||
if (!st->codec->block_align) {
|
||||
av_log(s, AV_LOG_ERROR, "No iLBC mode set\n");
|
||||
return AVERROR(EINVAL);
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
RTPDynamicProtocolHandler ff_ilbc_dynamic_handler = {
|
||||
.enc_name = "iLBC",
|
||||
.codec_type = AVMEDIA_TYPE_AUDIO,
|
||||
.codec_id = CODEC_ID_ILBC,
|
||||
.parse_sdp_a_line = ilbc_parse_sdp_line,
|
||||
};
|
@ -74,6 +74,7 @@ static int is_supported(enum CodecID id)
|
||||
case CODEC_ID_VP8:
|
||||
case CODEC_ID_ADPCM_G722:
|
||||
case CODEC_ID_ADPCM_G726:
|
||||
case CODEC_ID_ILBC:
|
||||
return 1;
|
||||
default:
|
||||
return 0;
|
||||
@ -187,6 +188,16 @@ static int rtp_write_header(AVFormatContext *s1)
|
||||
* 8000, even if the sample rate is 16000. See RFC 3551. */
|
||||
avpriv_set_pts_info(st, 32, 1, 8000);
|
||||
break;
|
||||
case CODEC_ID_ILBC:
|
||||
if (st->codec->block_align != 38 && st->codec->block_align != 50) {
|
||||
av_log(s1, AV_LOG_ERROR, "Incorrect iLBC block size specified\n");
|
||||
goto fail;
|
||||
}
|
||||
if (!s->max_frames_per_packet)
|
||||
s->max_frames_per_packet = 1;
|
||||
s->max_frames_per_packet = FFMIN(s->max_frames_per_packet,
|
||||
s->max_payload_size / st->codec->block_align);
|
||||
goto defaultcase;
|
||||
case CODEC_ID_AMR_NB:
|
||||
case CODEC_ID_AMR_WB:
|
||||
if (!s->max_frames_per_packet)
|
||||
@ -395,6 +406,36 @@ static void rtp_send_mpegts_raw(AVFormatContext *s1,
|
||||
}
|
||||
}
|
||||
|
||||
static int rtp_send_ilbc(AVFormatContext *s1, const uint8_t *buf, int size)
|
||||
{
|
||||
RTPMuxContext *s = s1->priv_data;
|
||||
AVStream *st = s1->streams[0];
|
||||
int frame_duration = av_get_audio_frame_duration(st->codec, 0);
|
||||
int frame_size = st->codec->block_align;
|
||||
int frames = size / frame_size;
|
||||
|
||||
while (frames > 0) {
|
||||
int n = FFMIN(s->max_frames_per_packet - s->num_frames, frames);
|
||||
|
||||
if (!s->num_frames) {
|
||||
s->buf_ptr = s->buf;
|
||||
s->timestamp = s->cur_timestamp;
|
||||
}
|
||||
memcpy(s->buf_ptr, buf, n * frame_size);
|
||||
frames -= n;
|
||||
s->num_frames += n;
|
||||
s->buf_ptr += n * frame_size;
|
||||
buf += n * frame_size;
|
||||
s->cur_timestamp += n * frame_duration;
|
||||
|
||||
if (s->num_frames == s->max_frames_per_packet) {
|
||||
ff_rtp_send_data(s1, s->buf, s->buf_ptr - s->buf, 1);
|
||||
s->num_frames = 0;
|
||||
}
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int rtp_write_packet(AVFormatContext *s1, AVPacket *pkt)
|
||||
{
|
||||
RTPMuxContext *s = s1->priv_data;
|
||||
@ -483,6 +524,9 @@ static int rtp_write_packet(AVFormatContext *s1, AVPacket *pkt)
|
||||
case CODEC_ID_VP8:
|
||||
ff_rtp_send_vp8(s1, pkt->data, size);
|
||||
break;
|
||||
case CODEC_ID_ILBC:
|
||||
rtp_send_ilbc(s1, pkt->data, size);
|
||||
break;
|
||||
default:
|
||||
/* better than nothing : send the codec raw data */
|
||||
rtp_send_raw(s1, pkt->data, size);
|
||||
|
@ -1279,7 +1279,7 @@ int ff_rtsp_make_setup_request(AVFormatContext *s, const char *host, int port,
|
||||
"%s/UDP;multicast", trans_pref);
|
||||
}
|
||||
if (s->oformat) {
|
||||
av_strlcat(transport, ";mode=receive", sizeof(transport));
|
||||
av_strlcat(transport, ";mode=record", sizeof(transport));
|
||||
} else if (rt->server_type == RTSP_SERVER_REAL ||
|
||||
rt->server_type == RTSP_SERVER_WMS)
|
||||
av_strlcat(transport, ";mode=play", sizeof(transport));
|
||||
|
@ -549,6 +549,12 @@ static char *sdp_write_media_attributes(char *buff, int size, AVCodecContext *c,
|
||||
c->sample_rate);
|
||||
break;
|
||||
}
|
||||
case CODEC_ID_ILBC:
|
||||
av_strlcatf(buff, size, "a=rtpmap:%d iLBC/%d\r\n"
|
||||
"a=fmtp:%d mode=%d\r\n",
|
||||
payload_type, c->sample_rate,
|
||||
payload_type, c->block_align == 38 ? 20 : 30);
|
||||
break;
|
||||
default:
|
||||
/* Nothing special to do here... */
|
||||
break;
|
||||
|
@ -30,7 +30,7 @@
|
||||
#include "libavutil/avutil.h"
|
||||
|
||||
#define LIBAVFORMAT_VERSION_MAJOR 54
|
||||
#define LIBAVFORMAT_VERSION_MINOR 8
|
||||
#define LIBAVFORMAT_VERSION_MINOR 9
|
||||
#define LIBAVFORMAT_VERSION_MICRO 100
|
||||
|
||||
#define LIBAVFORMAT_VERSION_INT AV_VERSION_INT(LIBAVFORMAT_VERSION_MAJOR, \
|
||||
|
@ -1,6 +1,7 @@
|
||||
;******************************************************************************
|
||||
;* x86 optimized Format Conversion Utils
|
||||
;* Copyright (c) 2008 Loren Merritt
|
||||
;* Copyright (c) 2012 Justin Ruggles <justin.ruggles@gmail.com>
|
||||
;*
|
||||
;* This file is part of Libav.
|
||||
;*
|
||||
@ -21,9 +22,217 @@
|
||||
|
||||
%include "x86inc.asm"
|
||||
%include "x86util.asm"
|
||||
%include "util.asm"
|
||||
|
||||
SECTION_RODATA 32
|
||||
|
||||
pf_s32_inv_scale: times 8 dd 0x30000000
|
||||
pf_s32_scale: times 8 dd 0x4f000000
|
||||
pf_s16_inv_scale: times 4 dd 0x38000000
|
||||
pf_s16_scale: times 4 dd 0x47000000
|
||||
|
||||
SECTION_TEXT
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_s16_to_s32(int32_t *dst, const int16_t *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
INIT_XMM sse2
|
||||
cglobal conv_s16_to_s32, 3,3,3, dst, src, len
|
||||
lea lenq, [2*lend]
|
||||
lea dstq, [dstq+2*lenq]
|
||||
add srcq, lenq
|
||||
neg lenq
|
||||
.loop:
|
||||
mova m2, [srcq+lenq]
|
||||
pxor m0, m0
|
||||
pxor m1, m1
|
||||
punpcklwd m0, m2
|
||||
punpckhwd m1, m2
|
||||
mova [dstq+2*lenq ], m0
|
||||
mova [dstq+2*lenq+mmsize], m1
|
||||
add lenq, mmsize
|
||||
jl .loop
|
||||
REP_RET
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_s16_to_flt(float *dst, const int16_t *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
%macro CONV_S16_TO_FLT 0
|
||||
cglobal conv_s16_to_flt, 3,3,3, dst, src, len
|
||||
lea lenq, [2*lend]
|
||||
add srcq, lenq
|
||||
lea dstq, [dstq + 2*lenq]
|
||||
neg lenq
|
||||
mova m2, [pf_s16_inv_scale]
|
||||
ALIGN 16
|
||||
.loop:
|
||||
mova m0, [srcq+lenq]
|
||||
S16_TO_S32_SX 0, 1
|
||||
cvtdq2ps m0, m0
|
||||
cvtdq2ps m1, m1
|
||||
mulps m0, m2
|
||||
mulps m1, m2
|
||||
mova [dstq+2*lenq ], m0
|
||||
mova [dstq+2*lenq+mmsize], m1
|
||||
add lenq, mmsize
|
||||
jl .loop
|
||||
REP_RET
|
||||
%endmacro
|
||||
|
||||
INIT_XMM sse2
|
||||
CONV_S16_TO_FLT
|
||||
INIT_XMM sse4
|
||||
CONV_S16_TO_FLT
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_s32_to_s16(int16_t *dst, const int32_t *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
%macro CONV_S32_TO_S16 0
|
||||
cglobal conv_s32_to_s16, 3,3,4, dst, src, len
|
||||
lea lenq, [2*lend]
|
||||
lea srcq, [srcq+2*lenq]
|
||||
add dstq, lenq
|
||||
neg lenq
|
||||
.loop:
|
||||
mova m0, [srcq+2*lenq ]
|
||||
mova m1, [srcq+2*lenq+ mmsize]
|
||||
mova m2, [srcq+2*lenq+2*mmsize]
|
||||
mova m3, [srcq+2*lenq+3*mmsize]
|
||||
psrad m0, 16
|
||||
psrad m1, 16
|
||||
psrad m2, 16
|
||||
psrad m3, 16
|
||||
packssdw m0, m1
|
||||
packssdw m2, m3
|
||||
mova [dstq+lenq ], m0
|
||||
mova [dstq+lenq+mmsize], m2
|
||||
add lenq, mmsize*2
|
||||
jl .loop
|
||||
%if mmsize == 8
|
||||
emms
|
||||
RET
|
||||
%else
|
||||
REP_RET
|
||||
%endif
|
||||
%endmacro
|
||||
|
||||
INIT_MMX mmx
|
||||
CONV_S32_TO_S16
|
||||
INIT_XMM sse2
|
||||
CONV_S32_TO_S16
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_s32_to_flt(float *dst, const int32_t *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
%macro CONV_S32_TO_FLT 0
|
||||
cglobal conv_s32_to_flt, 3,3,3, dst, src, len
|
||||
lea lenq, [4*lend]
|
||||
add srcq, lenq
|
||||
add dstq, lenq
|
||||
neg lenq
|
||||
mova m0, [pf_s32_inv_scale]
|
||||
ALIGN 16
|
||||
.loop:
|
||||
cvtdq2ps m1, [srcq+lenq ]
|
||||
cvtdq2ps m2, [srcq+lenq+mmsize]
|
||||
mulps m1, m1, m0
|
||||
mulps m2, m2, m0
|
||||
mova [dstq+lenq ], m1
|
||||
mova [dstq+lenq+mmsize], m2
|
||||
add lenq, mmsize*2
|
||||
jl .loop
|
||||
%if mmsize == 32
|
||||
vzeroupper
|
||||
RET
|
||||
%else
|
||||
REP_RET
|
||||
%endif
|
||||
%endmacro
|
||||
|
||||
INIT_XMM sse2
|
||||
CONV_S32_TO_FLT
|
||||
%if HAVE_AVX
|
||||
INIT_YMM avx
|
||||
CONV_S32_TO_FLT
|
||||
%endif
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_flt_to_s16(int16_t *dst, const float *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
INIT_XMM sse2
|
||||
cglobal conv_flt_to_s16, 3,3,5, dst, src, len
|
||||
lea lenq, [2*lend]
|
||||
lea srcq, [srcq+2*lenq]
|
||||
add dstq, lenq
|
||||
neg lenq
|
||||
mova m4, [pf_s16_scale]
|
||||
.loop:
|
||||
mova m0, [srcq+2*lenq ]
|
||||
mova m1, [srcq+2*lenq+1*mmsize]
|
||||
mova m2, [srcq+2*lenq+2*mmsize]
|
||||
mova m3, [srcq+2*lenq+3*mmsize]
|
||||
mulps m0, m4
|
||||
mulps m1, m4
|
||||
mulps m2, m4
|
||||
mulps m3, m4
|
||||
cvtps2dq m0, m0
|
||||
cvtps2dq m1, m1
|
||||
cvtps2dq m2, m2
|
||||
cvtps2dq m3, m3
|
||||
packssdw m0, m1
|
||||
packssdw m2, m3
|
||||
mova [dstq+lenq ], m0
|
||||
mova [dstq+lenq+mmsize], m2
|
||||
add lenq, mmsize*2
|
||||
jl .loop
|
||||
REP_RET
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_conv_flt_to_s32(int32_t *dst, const float *src, int len);
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
%macro CONV_FLT_TO_S32 0
|
||||
cglobal conv_flt_to_s32, 3,3,5, dst, src, len
|
||||
lea lenq, [lend*4]
|
||||
add srcq, lenq
|
||||
add dstq, lenq
|
||||
neg lenq
|
||||
mova m4, [pf_s32_scale]
|
||||
.loop:
|
||||
mulps m0, m4, [srcq+lenq ]
|
||||
mulps m1, m4, [srcq+lenq+1*mmsize]
|
||||
mulps m2, m4, [srcq+lenq+2*mmsize]
|
||||
mulps m3, m4, [srcq+lenq+3*mmsize]
|
||||
cvtps2dq m0, m0
|
||||
cvtps2dq m1, m1
|
||||
cvtps2dq m2, m2
|
||||
cvtps2dq m3, m3
|
||||
mova [dstq+lenq ], m0
|
||||
mova [dstq+lenq+1*mmsize], m1
|
||||
mova [dstq+lenq+2*mmsize], m2
|
||||
mova [dstq+lenq+3*mmsize], m3
|
||||
add lenq, mmsize*4
|
||||
jl .loop
|
||||
%if mmsize == 32
|
||||
vzeroupper
|
||||
RET
|
||||
%else
|
||||
REP_RET
|
||||
%endif
|
||||
%endmacro
|
||||
|
||||
INIT_XMM sse2
|
||||
CONV_FLT_TO_S32
|
||||
%if HAVE_AVX
|
||||
INIT_YMM avx
|
||||
CONV_FLT_TO_S32
|
||||
%endif
|
||||
|
||||
;-----------------------------------------------------------------------------
|
||||
; void ff_conv_fltp_to_flt_6ch(float *dst, float *const *src, int len,
|
||||
; int channels);
|
||||
|
@ -22,6 +22,22 @@
|
||||
#include "libavutil/cpu.h"
|
||||
#include "libavresample/audio_convert.h"
|
||||
|
||||
extern void ff_conv_s16_to_s32_sse2(int16_t *dst, const int32_t *src, int len);
|
||||
|
||||
extern void ff_conv_s16_to_flt_sse2(float *dst, const int16_t *src, int len);
|
||||
extern void ff_conv_s16_to_flt_sse4(float *dst, const int16_t *src, int len);
|
||||
|
||||
extern void ff_conv_s32_to_s16_mmx (int16_t *dst, const int32_t *src, int len);
|
||||
extern void ff_conv_s32_to_s16_sse2(int16_t *dst, const int32_t *src, int len);
|
||||
|
||||
extern void ff_conv_s32_to_flt_sse2(float *dst, const int32_t *src, int len);
|
||||
extern void ff_conv_s32_to_flt_avx (float *dst, const int32_t *src, int len);
|
||||
|
||||
extern void ff_conv_flt_to_s16_sse2(int16_t *dst, const float *src, int len);
|
||||
|
||||
extern void ff_conv_flt_to_s32_sse2(int32_t *dst, const float *src, int len);
|
||||
extern void ff_conv_flt_to_s32_avx (int32_t *dst, const float *src, int len);
|
||||
|
||||
extern void ff_conv_fltp_to_flt_6ch_mmx (float *dst, float *const *src, int len);
|
||||
extern void ff_conv_fltp_to_flt_6ch_sse4(float *dst, float *const *src, int len);
|
||||
extern void ff_conv_fltp_to_flt_6ch_avx (float *dst, float *const *src, int len);
|
||||
@ -32,6 +48,8 @@ av_cold void ff_audio_convert_init_x86(AudioConvert *ac)
|
||||
int mm_flags = av_get_cpu_flags();
|
||||
|
||||
if (mm_flags & AV_CPU_FLAG_MMX && HAVE_MMX) {
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S16, AV_SAMPLE_FMT_S32,
|
||||
0, 1, 8, "MMX", ff_conv_s32_to_s16_mmx);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_FLTP,
|
||||
6, 1, 4, "MMX", ff_conv_fltp_to_flt_6ch_mmx);
|
||||
}
|
||||
@ -43,5 +61,31 @@ av_cold void ff_audio_convert_init_x86(AudioConvert *ac)
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_FLTP,
|
||||
6, 16, 4, "AVX", ff_conv_fltp_to_flt_6ch_avx);
|
||||
}
|
||||
if (mm_flags & AV_CPU_FLAG_SSE2 && HAVE_SSE) {
|
||||
if (!(mm_flags & AV_CPU_FLAG_SSE2SLOW)) {
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S16, AV_SAMPLE_FMT_S32,
|
||||
0, 16, 16, "SSE2", ff_conv_s32_to_s16_sse2);
|
||||
}
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S32, AV_SAMPLE_FMT_S16,
|
||||
0, 16, 8, "SSE2", ff_conv_s16_to_s32_sse2);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_S16,
|
||||
0, 16, 8, "SSE2", ff_conv_s16_to_flt_sse2);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_S32,
|
||||
0, 16, 8, "SSE2", ff_conv_s32_to_flt_sse2);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S16, AV_SAMPLE_FMT_FLT,
|
||||
0, 16, 16, "SSE2", ff_conv_flt_to_s16_sse2);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S32, AV_SAMPLE_FMT_FLT,
|
||||
0, 16, 16, "SSE2", ff_conv_flt_to_s32_sse2);
|
||||
}
|
||||
if (mm_flags & AV_CPU_FLAG_SSE4 && HAVE_SSE) {
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_S16,
|
||||
0, 16, 8, "SSE4", ff_conv_s16_to_flt_sse4);
|
||||
}
|
||||
if (mm_flags & AV_CPU_FLAG_AVX && HAVE_AVX) {
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_FLT, AV_SAMPLE_FMT_S32,
|
||||
0, 32, 16, "AVX", ff_conv_s32_to_flt_avx);
|
||||
ff_audio_convert_set_func(ac, AV_SAMPLE_FMT_S32, AV_SAMPLE_FMT_FLT,
|
||||
0, 32, 32, "AVX", ff_conv_flt_to_s32_avx);
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
@ -26,7 +26,11 @@
|
||||
|
||||
void ff_vector_fmul_neon(float *dst, const float *src0, const float *src1, int len);
|
||||
|
||||
void ff_vector_fmac_scalar_neon(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
|
||||
void ff_float_dsp_init_neon(AVFloatDSPContext *fdsp)
|
||||
{
|
||||
fdsp->vector_fmul = ff_vector_fmul_neon;
|
||||
fdsp->vector_fmac_scalar = ff_vector_fmac_scalar_neon;
|
||||
}
|
||||
|
@ -62,3 +62,51 @@ function ff_vector_fmul_neon, export=1
|
||||
3: vst1.32 {d16-d19},[r0,:128]!
|
||||
bx lr
|
||||
endfunc
|
||||
|
||||
function ff_vector_fmac_scalar_neon, export=1
|
||||
VFP len .req r2
|
||||
VFP acc .req r3
|
||||
NOVFP len .req r3
|
||||
NOVFP acc .req r2
|
||||
VFP vdup.32 q15, d0[0]
|
||||
NOVFP vdup.32 q15, r2
|
||||
bics r12, len, #15
|
||||
mov acc, r0
|
||||
beq 3f
|
||||
vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vld1.32 {q1}, [r1,:128]!
|
||||
vld1.32 {q9}, [acc,:128]!
|
||||
1: vmla.f32 q8, q0, q15
|
||||
vld1.32 {q2}, [r1,:128]!
|
||||
vld1.32 {q10}, [acc,:128]!
|
||||
vmla.f32 q9, q1, q15
|
||||
vld1.32 {q3}, [r1,:128]!
|
||||
vld1.32 {q11}, [acc,:128]!
|
||||
vmla.f32 q10, q2, q15
|
||||
vst1.32 {q8}, [r0,:128]!
|
||||
vmla.f32 q11, q3, q15
|
||||
vst1.32 {q9}, [r0,:128]!
|
||||
subs r12, r12, #16
|
||||
beq 2f
|
||||
vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vst1.32 {q10}, [r0,:128]!
|
||||
vld1.32 {q1}, [r1,:128]!
|
||||
vld1.32 {q9}, [acc,:128]!
|
||||
vst1.32 {q11}, [r0,:128]!
|
||||
b 1b
|
||||
2: vst1.32 {q10}, [r0,:128]!
|
||||
vst1.32 {q11}, [r0,:128]!
|
||||
ands len, len, #15
|
||||
it eq
|
||||
bxeq lr
|
||||
3: vld1.32 {q0}, [r1,:128]!
|
||||
vld1.32 {q8}, [acc,:128]!
|
||||
vmla.f32 q8, q0, q15
|
||||
vst1.32 {q8}, [r0,:128]!
|
||||
subs len, len, #4
|
||||
bgt 3b
|
||||
bx lr
|
||||
.unreq len
|
||||
endfunc
|
||||
|
@ -31,9 +31,18 @@ static void vector_fmul_c(float *dst, const float *src0, const float *src1,
|
||||
dst[i] = src0[i] * src1[i];
|
||||
}
|
||||
|
||||
static void vector_fmac_scalar_c(float *dst, const float *src, float mul,
|
||||
int len)
|
||||
{
|
||||
int i;
|
||||
for (i = 0; i < len; i++)
|
||||
dst[i] += src[i] * mul;
|
||||
}
|
||||
|
||||
void avpriv_float_dsp_init(AVFloatDSPContext *fdsp, int bit_exact)
|
||||
{
|
||||
fdsp->vector_fmul = vector_fmul_c;
|
||||
fdsp->vector_fmac_scalar = vector_fmac_scalar_c;
|
||||
|
||||
#if ARCH_ARM
|
||||
ff_float_dsp_init_arm(fdsp);
|
||||
|
@ -35,6 +35,22 @@ typedef struct AVFloatDSPContext {
|
||||
*/
|
||||
void (*vector_fmul)(float *dst, const float *src0, const float *src1,
|
||||
int len);
|
||||
|
||||
/**
|
||||
* Multiply a vector of floats by a scalar float and add to
|
||||
* destination vector. Source and destination vectors must
|
||||
* overlap exactly or not at all.
|
||||
*
|
||||
* @param dst result vector
|
||||
* constraints: 32-byte aligned
|
||||
* @param src input vector
|
||||
* constraints: 32-byte aligned
|
||||
* @param mul scalar value
|
||||
* @param len length of vector
|
||||
* constraints: multiple of 16
|
||||
*/
|
||||
void (*vector_fmac_scalar)(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
} AVFloatDSPContext;
|
||||
|
||||
/**
|
||||
|
@ -94,6 +94,8 @@ void *av_malloc(size_t size)
|
||||
if (size) //OS X on SDK 10.6 has a broken posix_memalign implementation
|
||||
if (posix_memalign(&ptr,ALIGN,size))
|
||||
ptr = NULL;
|
||||
#elif HAVE_ALIGNED_MALLOC
|
||||
ptr = _aligned_malloc(size, ALIGN);
|
||||
#elif HAVE_MEMALIGN
|
||||
ptr = memalign(ALIGN,size);
|
||||
/* Why 64?
|
||||
@ -145,6 +147,8 @@ void *av_realloc(void *ptr, size_t size)
|
||||
ptr= realloc((char*)ptr - diff, size + diff);
|
||||
if(ptr) ptr = (char*)ptr + diff;
|
||||
return ptr;
|
||||
#elif HAVE_ALIGNED_MALLOC
|
||||
return _aligned_realloc(ptr, size + !size, ALIGN);
|
||||
#else
|
||||
return realloc(ptr, size + !size);
|
||||
#endif
|
||||
@ -170,6 +174,8 @@ void av_free(void *ptr)
|
||||
#if CONFIG_MEMALIGN_HACK
|
||||
if (ptr)
|
||||
free((char*)ptr - ((char*)ptr)[-1]);
|
||||
#elif HAVE_ALIGNED_MALLOC
|
||||
_aligned_free(ptr);
|
||||
#else
|
||||
free(ptr);
|
||||
#endif
|
||||
|
@ -21,6 +21,7 @@
|
||||
;******************************************************************************
|
||||
|
||||
%include "x86inc.asm"
|
||||
%include "x86util.asm"
|
||||
|
||||
SECTION .text
|
||||
|
||||
@ -55,3 +56,49 @@ VECTOR_FMUL
|
||||
INIT_YMM avx
|
||||
VECTOR_FMUL
|
||||
%endif
|
||||
|
||||
;------------------------------------------------------------------------------
|
||||
; void ff_vector_fmac_scalar(float *dst, const float *src, float mul, int len)
|
||||
;------------------------------------------------------------------------------
|
||||
|
||||
%macro VECTOR_FMAC_SCALAR 0
|
||||
%if UNIX64
|
||||
cglobal vector_fmac_scalar, 3,3,3, dst, src, len
|
||||
%else
|
||||
cglobal vector_fmac_scalar, 4,4,3, dst, src, mul, len
|
||||
%endif
|
||||
%if WIN64
|
||||
SWAP 0, 2
|
||||
%endif
|
||||
%if ARCH_X86_32
|
||||
VBROADCASTSS m0, mulm
|
||||
%else
|
||||
shufps xmm0, xmm0, 0
|
||||
%if cpuflag(avx)
|
||||
vinsertf128 m0, m0, xmm0, 1
|
||||
%endif
|
||||
%endif
|
||||
lea lenq, [lend*4-2*mmsize]
|
||||
.loop
|
||||
mulps m1, m0, [srcq+lenq ]
|
||||
mulps m2, m0, [srcq+lenq+mmsize]
|
||||
addps m1, m1, [dstq+lenq ]
|
||||
addps m2, m2, [dstq+lenq+mmsize]
|
||||
mova [dstq+lenq ], m1
|
||||
mova [dstq+lenq+mmsize], m2
|
||||
sub lenq, 2*mmsize
|
||||
jge .loop
|
||||
%if mmsize == 32
|
||||
vzeroupper
|
||||
RET
|
||||
%else
|
||||
REP_RET
|
||||
%endif
|
||||
%endmacro
|
||||
|
||||
INIT_XMM sse
|
||||
VECTOR_FMAC_SCALAR
|
||||
%if HAVE_AVX
|
||||
INIT_YMM avx
|
||||
VECTOR_FMAC_SCALAR
|
||||
%endif
|
||||
|
@ -26,6 +26,11 @@ extern void ff_vector_fmul_sse(float *dst, const float *src0, const float *src1,
|
||||
extern void ff_vector_fmul_avx(float *dst, const float *src0, const float *src1,
|
||||
int len);
|
||||
|
||||
extern void ff_vector_fmac_scalar_sse(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
extern void ff_vector_fmac_scalar_avx(float *dst, const float *src, float mul,
|
||||
int len);
|
||||
|
||||
void ff_float_dsp_init_x86(AVFloatDSPContext *fdsp)
|
||||
{
|
||||
#if HAVE_YASM
|
||||
@ -33,9 +38,11 @@ void ff_float_dsp_init_x86(AVFloatDSPContext *fdsp)
|
||||
|
||||
if (mm_flags & AV_CPU_FLAG_SSE && HAVE_SSE) {
|
||||
fdsp->vector_fmul = ff_vector_fmul_sse;
|
||||
fdsp->vector_fmac_scalar = ff_vector_fmac_scalar_sse;
|
||||
}
|
||||
if (mm_flags & AV_CPU_FLAG_AVX && HAVE_AVX) {
|
||||
fdsp->vector_fmul = ff_vector_fmul_avx;
|
||||
fdsp->vector_fmac_scalar = ff_vector_fmac_scalar_avx;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user