aboutsummaryrefslogtreecommitdiffstats
path: root/libavcodec/amr.c
diff options
context:
space:
mode:
Diffstat (limited to 'libavcodec/amr.c')
-rw-r--r--libavcodec/amr.c244
1 files changed, 218 insertions, 26 deletions
diff --git a/libavcodec/amr.c b/libavcodec/amr.c
index 50fe003f81..e8da4fd8ac 100644
--- a/libavcodec/amr.c
+++ b/libavcodec/amr.c
@@ -17,31 +17,42 @@
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/*
- This code implements amr-nb audio encoder/decoder through external reference
+ This code implements amr-nb and amr-wb audio encoder/decoder through external reference
code from www.3gpp.org. The licence of the code from 3gpp is unclear so you
have to download the code separately. Two versions exists: One fixed-point
and one with floats. For some reason the float-encoder is significant faster
atleast on a P4 1.5GHz (0.9s instead of 9.9s on a 30s audio clip at MR102).
+ Both float and fixed point is supported for amr-nb, but only float for
+ amr-wb.
+ --AMR-NB--
The fixed-point (TS26.073) can be downloaded from:
- http://www.3gpp.org/ftp/Specs/latest/Rel-5/26_series/26073-510.zip
+ http://www.3gpp.org/ftp/Specs/archive/26_series/26.073/26073-510.zip
Extract the soure into ffmpeg/libavcodec/amr
- To use the float version run "./configure" with "--enable-amr-nb-fixed"
+ To use the fixed version run "./configure" with "--enable-amr-nb-fixed"
The float version (default) can be downloaded from:
- http://www.3gpp.org/ftp/Specs/latest/Rel-5/26_series/26104-510.zip
+ http://www.3gpp.org/ftp/Specs/archive/26_series/26.104/26104-510.zip
Extract the soure into ffmpeg/libavcodec/amr_float
-
+
The specification for amr-nb can be found in TS 26.071
(http://www.3gpp.org/ftp/Specs/html-info/26071.htm) and some other
info at http://www.3gpp.org/ftp/Specs/html-info/26-series.htm
- In the future support for AMR-WB might also be included here.
- Reference code exist in TS26.173 and TS 26.204.
+ --AMR-WB--
+ The reference code can be downloaded from:
+ http://www.3gpp.org/ftp/Specs/archive/26_series/26.204/26204-510.zip
+ It should be extracted to "libavcodec/amrwb_float". Enable it with
+ "--enable-amr-wb".
+
+ The specification for amr-wb can be downloaded from:
+ http://www.3gpp.org/ftp/Specs/archive/26_series/26.171/26171-500.zip
+
+ If someone want to use the fixed point version it can be downloaded
+ from: http://www.3gpp.org/ftp/Specs/archive/26_series/26.173/26173-571.zip
*/
-#define DEBUG
-//#define AMR_NB_FIXED
+
#include "../config.h"
#include "avcodec.h"
@@ -86,7 +97,7 @@ static enum Mode getBitrateMode(int bitrate)
};
int i;
- for(i=0;i<sizeof(rates);i++)
+ for(i=0;i<8;i++)
{
if(rates[i].startrate<=bitrate && rates[i].stoprate>=bitrate)
{
@@ -148,17 +159,19 @@ static int amr_nb_encode_init(AVCodecContext * avctx)
if(avctx->sample_rate!=8000)
{
-#ifdef DEBUG
- printf("Only 8000Hz sample rate supported\n");
-#endif
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only 8000Hz sample rate supported\n");
+ }
return -1;
}
if(avctx->channels!=1)
{
-#ifdef DEBUG
- printf("Only mono supported\n");
-#endif
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only mono supported\n");
+ }
return -1;
}
@@ -167,9 +180,10 @@ static int amr_nb_encode_init(AVCodecContext * avctx)
if(Speech_Encode_Frame_init(&s->enstate, 0, "encoder") || sid_sync_init (&s->sidstate))
{
-#ifdef DEBUG
- printf("Speech_Encode_Frame_init error\n");
-#endif
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Speech_Encode_Frame_init error\n");
+ }
return -1;
}
@@ -343,17 +357,19 @@ static int amr_nb_encode_init(AVCodecContext * avctx)
if(avctx->sample_rate!=8000)
{
-#ifdef DEBUG
- printf("Only 8000Hz sample rate supported\n");
-#endif
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only 8000Hz sample rate supported\n");
+ }
return -1;
}
if(avctx->channels!=1)
{
-#ifdef DEBUG
- printf("Only mono supported\n");
-#endif
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only mono supported\n");
+ }
return -1;
}
@@ -363,7 +379,10 @@ static int amr_nb_encode_init(AVCodecContext * avctx)
s->enstate=Encoder_Interface_init(0);
if(!s->enstate)
{
- printf("Encoder_Interface_init error\n");
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Encoder_Interface_init error\n");
+ }
return -1;
}
@@ -458,3 +477,176 @@ AVCodec amr_nb_encoder =
amr_nb_encode_close,
NULL,
};
+
+/* -----------AMR wideband ------------*/
+#ifdef AMR_WB
+
+#ifdef _TYPEDEF_H
+//To avoid duplicate typedefs from typdef in amr-nb
+#define typedef_h
+#endif
+
+#include "amrwb_float/enc_if.h"
+#include "amrwb_float/dec_if.h"
+
+/* Common code for fixed and float version*/
+typedef struct AMRWB_bitrates
+{
+ int startrate;
+ int stoprate;
+ int mode;
+
+} AMRWB_bitrates;
+
+static int getWBBitrateMode(int bitrate)
+{
+ /* Adjusted so that all bitrates can be used from commandline where
+ only a multiple of 1000 can be specified*/
+ AMRWB_bitrates rates[]={ {0,7999,0}, //6.6kHz
+ {8000,9999,1},//8.85
+ {10000,13000,2},//12.65
+ {13001,14999,3},//14.25
+ {15000,17000,4},//15.85
+ {17001,18000,5},//18.25
+ {18001,22000,6},//19.85
+ {22001,23000,7},//23.05
+ {23001,24000,8},//23.85
+
+ };
+ int i;
+
+ for(i=0;i<9;i++)
+ {
+ if(rates[i].startrate<=bitrate && rates[i].stoprate>=bitrate)
+ {
+ return(rates[i].mode);
+ }
+ }
+ /*Return highest possible*/
+ return(8);
+}
+
+
+typedef struct AMRWBContext {
+ int frameCount;
+ void *state;
+ int mode;
+ Word16 allow_dtx;
+} AMRWBContext;
+
+static int amr_wb_encode_init(AVCodecContext * avctx)
+{
+ AMRWBContext *s = (AMRWBContext*)avctx->priv_data;
+ s->frameCount=0;
+
+ if(avctx->sample_rate!=16000)
+ {
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only 16000Hz sample rate supported\n");
+ }
+ return -1;
+ }
+
+ if(avctx->channels!=1)
+ {
+ if(avctx->debug)
+ {
+ fprintf(stderr, "Only mono supported\n");
+ }
+ return -1;
+ }
+
+ avctx->frame_size=320;
+ avctx->coded_frame= avcodec_alloc_frame();
+
+ s->state = E_IF_init();
+ s->mode=getWBBitrateMode(avctx->bit_rate);
+ s->allow_dtx=0;
+
+ return 0;
+}
+
+static int amr_wb_encode_close(AVCodecContext * avctx)
+{
+ AMRWBContext *s = (AMRWBContext*) avctx->priv_data;
+ E_IF_exit(s->state);
+ av_freep(&avctx->coded_frame);
+ s->frameCount++;
+ return 0;
+}
+
+static int amr_wb_encode_frame(AVCodecContext *avctx,
+ unsigned char *frame/*out*/, int buf_size, void *data/*in*/)
+{
+ AMRWBContext *s = (AMRWBContext*) avctx->priv_data;
+ int size = E_IF_encode(s->state, s->mode, data, frame, s->allow_dtx);
+ return size;
+}
+
+static int amr_wb_decode_init(AVCodecContext * avctx)
+{
+ AMRWBContext *s = (AMRWBContext *)avctx->priv_data;
+ s->frameCount=0;
+ s->state = D_IF_init();
+ return 0;
+}
+
+extern const UWord8 block_size[];
+
+static int amr_wb_decode_frame(AVCodecContext * avctx,
+ void *data, int *data_size,
+ uint8_t * buf, int buf_size)
+{
+ AMRWBContext *s = (AMRWBContext*)avctx->priv_data;
+
+ uint8_t*amrData=buf;
+ int offset=0;
+ int mode;
+ int packet_size;
+ *data_size=0;
+
+ while(offset<buf_size)
+ {
+ s->frameCount++;
+ mode = (Word16)((amrData[offset] >> 3) & 0x0F);
+ packet_size = block_size[mode];
+ D_IF_decode( s->state, &amrData[offset], data+*data_size, _good_frame);
+ *data_size+=320*2;
+ offset+=packet_size;
+ }
+ return buf_size;
+}
+
+static int amr_wb_decode_close(AVCodecContext * avctx)
+{
+ AMRWBContext *s = (AMRWBContext *)avctx->priv_data;
+ D_IF_exit(s->state);
+ return 0;
+}
+
+AVCodec amr_wb_decoder =
+{
+ "amr_wb",
+ CODEC_TYPE_AUDIO,
+ CODEC_ID_AMR_WB,
+ sizeof(AMRWBContext),
+ amr_wb_decode_init,
+ NULL,
+ amr_wb_decode_close,
+ amr_wb_decode_frame,
+};
+
+AVCodec amr_wb_encoder =
+{
+ "amr_wb",
+ CODEC_TYPE_AUDIO,
+ CODEC_ID_AMR_WB,
+ sizeof(AMRWBContext),
+ amr_wb_encode_init,
+ amr_wb_encode_frame,
+ amr_wb_encode_close,
+ NULL,
+};
+
+#endif //AMR_WB