add Creative 8 bits ADPCM schemes support
Originally committed as revision 5024 to svn://svn.ffmpeg.org/ffmpeg/trunk
This commit is contained in:
		@@ -833,6 +833,7 @@ other implementations.
 | 
				
			|||||||
@item Electronic Arts ADPCM  @tab      @tab X
 | 
					@item Electronic Arts ADPCM  @tab      @tab X
 | 
				
			||||||
@tab Used in various EA titles.
 | 
					@tab Used in various EA titles.
 | 
				
			||||||
@item Creative ADPCM         @tab      @tab X
 | 
					@item Creative ADPCM         @tab      @tab X
 | 
				
			||||||
 | 
					@tab 16 -> 4, 8 -> 4, 8 -> 3, 8 -> 2
 | 
				
			||||||
@item RA144                  @tab      @tab X
 | 
					@item RA144                  @tab      @tab X
 | 
				
			||||||
@tab Real 14400 bit/s codec
 | 
					@tab Real 14400 bit/s codec
 | 
				
			||||||
@item RA288                  @tab      @tab X
 | 
					@item RA288                  @tab      @tab X
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -514,6 +514,34 @@ static inline short adpcm_ct_expand_nibble(ADPCMChannelStatus *c, char nibble)
 | 
				
			|||||||
    return (short)predictor;
 | 
					    return (short)predictor;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static inline short adpcm_sbpro_expand_nibble(ADPCMChannelStatus *c, char nibble, int size, int shift)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    int sign, delta, diff;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    sign = nibble & (1<<(size-1));
 | 
				
			||||||
 | 
					    delta = nibble & ((1<<(size-1))-1);
 | 
				
			||||||
 | 
					    diff = delta << (7 + c->step + shift);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (sign)
 | 
				
			||||||
 | 
					        c->predictor -= diff;
 | 
				
			||||||
 | 
					    else
 | 
				
			||||||
 | 
					        c->predictor += diff;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* clamp result */
 | 
				
			||||||
 | 
					    if (c->predictor > 16256)
 | 
				
			||||||
 | 
					        c->predictor = 16256;
 | 
				
			||||||
 | 
					    else if (c->predictor < -16384)
 | 
				
			||||||
 | 
					        c->predictor = -16384;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /* calculate new step */
 | 
				
			||||||
 | 
					    if (delta >= (2*size - 3) && c->step < 3)
 | 
				
			||||||
 | 
					        c->step++;
 | 
				
			||||||
 | 
					    else if (delta == 0 && c->step > 0)
 | 
				
			||||||
 | 
					        c->step--;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return (short) c->predictor;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static inline short adpcm_yamaha_expand_nibble(ADPCMChannelStatus *c, unsigned char nibble)
 | 
					static inline short adpcm_yamaha_expand_nibble(ADPCMChannelStatus *c, unsigned char nibble)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
    if(!c->step) {
 | 
					    if(!c->step) {
 | 
				
			||||||
@@ -644,7 +672,7 @@ static int adpcm_decode_frame(AVCodecContext *avctx,
 | 
				
			|||||||
    samples = data;
 | 
					    samples = data;
 | 
				
			||||||
    src = buf;
 | 
					    src = buf;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    st = avctx->channels == 2;
 | 
					    st = avctx->channels == 2 ? 1 : 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    switch(avctx->codec->id) {
 | 
					    switch(avctx->codec->id) {
 | 
				
			||||||
    case CODEC_ID_ADPCM_IMA_QT:
 | 
					    case CODEC_ID_ADPCM_IMA_QT:
 | 
				
			||||||
@@ -973,6 +1001,48 @@ static int adpcm_decode_frame(AVCodecContext *avctx,
 | 
				
			|||||||
            src++;
 | 
					            src++;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        break;
 | 
					        break;
 | 
				
			||||||
 | 
					    case CODEC_ID_ADPCM_SBPRO_4:
 | 
				
			||||||
 | 
					    case CODEC_ID_ADPCM_SBPRO_3:
 | 
				
			||||||
 | 
					    case CODEC_ID_ADPCM_SBPRO_2:
 | 
				
			||||||
 | 
					        if (!c->status[0].step_index) {
 | 
				
			||||||
 | 
					            /* the first byte is a raw sample */
 | 
				
			||||||
 | 
					            *samples++ = 128 * (*src++ - 0x80);
 | 
				
			||||||
 | 
					            if (st)
 | 
				
			||||||
 | 
					              *samples++ = 128 * (*src++ - 0x80);
 | 
				
			||||||
 | 
					            c->status[0].step_index = 1;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if (avctx->codec->id == CODEC_ID_ADPCM_SBPRO_4) {
 | 
				
			||||||
 | 
					            while (src < buf + buf_size) {
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    (src[0] >> 4) & 0x0F, 4, 0);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[st],
 | 
				
			||||||
 | 
					                    src[0] & 0x0F, 4, 0);
 | 
				
			||||||
 | 
					                src++;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        } else if (avctx->codec->id == CODEC_ID_ADPCM_SBPRO_3) {
 | 
				
			||||||
 | 
					            while (src < buf + buf_size) {
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    (src[0] >> 5) & 0x07, 3, 0);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    (src[0] >> 2) & 0x07, 3, 0);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    src[0] & 0x03, 2, 0);
 | 
				
			||||||
 | 
					                src++;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        } else {
 | 
				
			||||||
 | 
					            while (src < buf + buf_size) {
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    (src[0] >> 6) & 0x03, 2, 2);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[st],
 | 
				
			||||||
 | 
					                    (src[0] >> 4) & 0x03, 2, 2);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[0],
 | 
				
			||||||
 | 
					                    (src[0] >> 2) & 0x03, 2, 2);
 | 
				
			||||||
 | 
					                *samples++ = adpcm_sbpro_expand_nibble(&c->status[st],
 | 
				
			||||||
 | 
					                    src[0] & 0x03, 2, 2);
 | 
				
			||||||
 | 
					                src++;
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        break;
 | 
				
			||||||
    case CODEC_ID_ADPCM_SWF:
 | 
					    case CODEC_ID_ADPCM_SWF:
 | 
				
			||||||
    {
 | 
					    {
 | 
				
			||||||
        GetBitContext gb;
 | 
					        GetBitContext gb;
 | 
				
			||||||
@@ -1117,5 +1187,8 @@ ADPCM_CODEC(CODEC_ID_ADPCM_EA, adpcm_ea);
 | 
				
			|||||||
ADPCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
				
			||||||
ADPCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
				
			||||||
ADPCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
				
			||||||
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_SBPRO_4, adpcm_sbpro_4);
 | 
				
			||||||
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_SBPRO_3, adpcm_sbpro_3);
 | 
				
			||||||
 | 
					ADPCM_CODEC(CODEC_ID_ADPCM_SBPRO_2, adpcm_sbpro_2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#undef ADPCM_CODEC
 | 
					#undef ADPCM_CODEC
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -593,6 +593,9 @@ PCM_CODEC(CODEC_ID_ADPCM_G726, adpcm_g726);
 | 
				
			|||||||
PCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
				
			||||||
PCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
				
			||||||
PCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_4, adpcm_sbpro_4);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_3, adpcm_sbpro_3);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_2, adpcm_sbpro_2);
 | 
				
			||||||
#undef PCM_CODEC
 | 
					#undef PCM_CODEC
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* subtitles */
 | 
					    /* subtitles */
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -21,8 +21,8 @@ extern "C" {
 | 
				
			|||||||
#define AV_STRINGIFY(s)         AV_TOSTRING(s)
 | 
					#define AV_STRINGIFY(s)         AV_TOSTRING(s)
 | 
				
			||||||
#define AV_TOSTRING(s) #s
 | 
					#define AV_TOSTRING(s) #s
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define LIBAVCODEC_VERSION_INT  ((51<<16)+(5<<8)+0)
 | 
					#define LIBAVCODEC_VERSION_INT  ((51<<16)+(6<<8)+0)
 | 
				
			||||||
#define LIBAVCODEC_VERSION      51.5.0
 | 
					#define LIBAVCODEC_VERSION      51.6.0
 | 
				
			||||||
#define LIBAVCODEC_BUILD        LIBAVCODEC_VERSION_INT
 | 
					#define LIBAVCODEC_BUILD        LIBAVCODEC_VERSION_INT
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#define LIBAVCODEC_IDENT        "Lavc" AV_STRINGIFY(LIBAVCODEC_VERSION)
 | 
					#define LIBAVCODEC_IDENT        "Lavc" AV_STRINGIFY(LIBAVCODEC_VERSION)
 | 
				
			||||||
@@ -153,6 +153,9 @@ enum CodecID {
 | 
				
			|||||||
    CODEC_ID_ADPCM_CT,
 | 
					    CODEC_ID_ADPCM_CT,
 | 
				
			||||||
    CODEC_ID_ADPCM_SWF,
 | 
					    CODEC_ID_ADPCM_SWF,
 | 
				
			||||||
    CODEC_ID_ADPCM_YAMAHA,
 | 
					    CODEC_ID_ADPCM_YAMAHA,
 | 
				
			||||||
 | 
					    CODEC_ID_ADPCM_SBPRO_4,
 | 
				
			||||||
 | 
					    CODEC_ID_ADPCM_SBPRO_3,
 | 
				
			||||||
 | 
					    CODEC_ID_ADPCM_SBPRO_2,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /* AMR */
 | 
					    /* AMR */
 | 
				
			||||||
    CODEC_ID_AMR_NB= 0x12000,
 | 
					    CODEC_ID_AMR_NB= 0x12000,
 | 
				
			||||||
@@ -2271,6 +2274,9 @@ PCM_CODEC(CODEC_ID_ADPCM_G726, adpcm_g726);
 | 
				
			|||||||
PCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_CT, adpcm_ct);
 | 
				
			||||||
PCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SWF, adpcm_swf);
 | 
				
			||||||
PCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
					PCM_CODEC(CODEC_ID_ADPCM_YAMAHA, adpcm_yamaha);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_4, adpcm_sbpro_4);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_3, adpcm_sbpro_3);
 | 
				
			||||||
 | 
					PCM_CODEC(CODEC_ID_ADPCM_SBPRO_2, adpcm_sbpro_2);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#undef PCM_CODEC
 | 
					#undef PCM_CODEC
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -41,6 +41,9 @@ static const unsigned char voc_magic[] = "Creative Voice File\x1A";
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
static const CodecTag voc_codec_tags[] = {
 | 
					static const CodecTag voc_codec_tags[] = {
 | 
				
			||||||
    {CODEC_ID_PCM_U8,        0x00},
 | 
					    {CODEC_ID_PCM_U8,        0x00},
 | 
				
			||||||
 | 
					    {CODEC_ID_ADPCM_SBPRO_4, 0x01},
 | 
				
			||||||
 | 
					    {CODEC_ID_ADPCM_SBPRO_3, 0x02},
 | 
				
			||||||
 | 
					    {CODEC_ID_ADPCM_SBPRO_2, 0x03},
 | 
				
			||||||
    {CODEC_ID_PCM_S16LE,     0x04},
 | 
					    {CODEC_ID_PCM_S16LE,     0x04},
 | 
				
			||||||
    {CODEC_ID_PCM_ALAW,      0x06},
 | 
					    {CODEC_ID_PCM_ALAW,      0x06},
 | 
				
			||||||
    {CODEC_ID_PCM_MULAW,     0x07},
 | 
					    {CODEC_ID_PCM_MULAW,     0x07},
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user