You've already forked FFmpeg
mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-10-06 05:47:18 +02:00
avcodec: add ADPCM IMA Escape audio decoder
(cherry picked from commit 4a663e78c4421da226e7d480d6767de803ee2122)
This commit is contained in:
@@ -987,6 +987,7 @@ OBJS-$(CONFIG_ADPCM_IMA_DK3_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_DK4_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_EA_EACS_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_EA_SEAD_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_ESCAPE_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_HVQM2_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_HVQM4_DECODER) += adpcm.o adpcm_data.o
|
||||
OBJS-$(CONFIG_ADPCM_IMA_ISS_DECODER) += adpcm.o adpcm_data.o
|
||||
|
@@ -420,6 +420,29 @@ static inline int16_t adpcm_agm_expand_nibble(ADPCMChannelStatus *c, int8_t nibb
|
||||
return pred;
|
||||
}
|
||||
|
||||
static inline int16_t adpcm_ima_escape_expand_nibble(ADPCMChannelStatus *c, int8_t nibble)
|
||||
{
|
||||
int step_index;
|
||||
int predictor;
|
||||
int sign, delta, diff, step;
|
||||
|
||||
step = ff_adpcm_step_table[c->step_index];
|
||||
step_index = c->step_index + ff_adpcm_index_table[(unsigned)nibble];
|
||||
step_index = av_clip(step_index, 0, 88);
|
||||
|
||||
sign = nibble & 8;
|
||||
delta = nibble & 7;
|
||||
diff = (delta * step) >> 2;
|
||||
predictor = c->predictor;
|
||||
if (sign) predictor -= diff;
|
||||
else predictor += diff;
|
||||
|
||||
c->predictor = av_clip_int16(predictor);
|
||||
c->step_index = step_index;
|
||||
|
||||
return (int16_t)c->predictor;
|
||||
}
|
||||
|
||||
static inline int16_t adpcm_ima_expand_nibble(ADPCMChannelStatus *c, int8_t nibble, int shift)
|
||||
{
|
||||
int step_index;
|
||||
@@ -1171,6 +1194,7 @@ static int get_nb_samples(AVCodecContext *avctx, GetByteContext *gb,
|
||||
case AV_CODEC_ID_ADPCM_IMA_APC:
|
||||
case AV_CODEC_ID_ADPCM_IMA_CUNNING:
|
||||
case AV_CODEC_ID_ADPCM_IMA_EA_SEAD:
|
||||
case AV_CODEC_ID_ADPCM_IMA_ESCAPE:
|
||||
case AV_CODEC_ID_ADPCM_IMA_OKI:
|
||||
case AV_CODEC_ID_ADPCM_IMA_WS:
|
||||
case AV_CODEC_ID_ADPCM_YAMAHA:
|
||||
@@ -2015,6 +2039,13 @@ static int adpcm_decode_frame(AVCodecContext *avctx, AVFrame *frame,
|
||||
bytestream2_skip(&gb, bytes_remaining);
|
||||
}
|
||||
) /* End of CASE */
|
||||
CASE(ADPCM_IMA_ESCAPE,
|
||||
for (int n = nb_samples >> (1 - st); n > 0; n--) {
|
||||
int byte = bytestream2_get_byteu(&gb);
|
||||
*samples++ = adpcm_ima_escape_expand_nibble(&c->status[0], byte >> 4);
|
||||
*samples++ = adpcm_ima_escape_expand_nibble(&c->status[st], byte & 0xF);
|
||||
}
|
||||
) /* End of CASE */
|
||||
CASE(ADPCM_IMA_EA_EACS,
|
||||
for (int i = 0; i <= st; i++) {
|
||||
c->status[i].step_index = bytestream2_get_le32u(&gb);
|
||||
@@ -2958,6 +2989,7 @@ ADPCM_DECODER(ADPCM_IMA_DK3, sample_fmts_s16, adpcm_ima_dk3, "ADPCM IMA
|
||||
ADPCM_DECODER(ADPCM_IMA_DK4, sample_fmts_s16, adpcm_ima_dk4, "ADPCM IMA Duck DK4")
|
||||
ADPCM_DECODER(ADPCM_IMA_EA_EACS, sample_fmts_s16, adpcm_ima_ea_eacs, "ADPCM IMA Electronic Arts EACS")
|
||||
ADPCM_DECODER(ADPCM_IMA_EA_SEAD, sample_fmts_s16, adpcm_ima_ea_sead, "ADPCM IMA Electronic Arts SEAD")
|
||||
ADPCM_DECODER(ADPCM_IMA_ESCAPE, sample_fmts_s16, adpcm_ima_escape, "ADPCM IMA Acorn Escape")
|
||||
ADPCM_DECODER(ADPCM_IMA_HVQM2, sample_fmts_s16, adpcm_ima_hvqm2, "ADPCM IMA HVQM2")
|
||||
ADPCM_DECODER(ADPCM_IMA_HVQM4, sample_fmts_s16, adpcm_ima_hvqm4, "ADPCM IMA HVQM4")
|
||||
ADPCM_DECODER(ADPCM_IMA_ISS, sample_fmts_s16, adpcm_ima_iss, "ADPCM IMA Funcom ISS")
|
||||
|
@@ -678,6 +678,7 @@ extern const FFCodec ff_adpcm_ima_dk3_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_dk4_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_ea_eacs_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_ea_sead_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_escape_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_hvqm2_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_hvqm4_decoder;
|
||||
extern const FFCodec ff_adpcm_ima_iss_decoder;
|
||||
|
@@ -2683,6 +2683,13 @@ static const AVCodecDescriptor codec_descriptors[] = {
|
||||
.long_name = NULL_IF_CONFIG_SMALL("ADPCM Circus"),
|
||||
.props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSY,
|
||||
},
|
||||
{
|
||||
.id = AV_CODEC_ID_ADPCM_IMA_ESCAPE,
|
||||
.type = AVMEDIA_TYPE_AUDIO,
|
||||
.name = "adpcm_ima_escape",
|
||||
.long_name = NULL_IF_CONFIG_SMALL("ADPCM IMA Acorn Escape"),
|
||||
.props = AV_CODEC_PROP_INTRA_ONLY | AV_CODEC_PROP_LOSSY,
|
||||
},
|
||||
|
||||
/* AMR */
|
||||
{
|
||||
|
@@ -434,6 +434,7 @@ enum AVCodecID {
|
||||
AV_CODEC_ID_ADPCM_IMA_MAGIX,
|
||||
AV_CODEC_ID_ADPCM_PSXC,
|
||||
AV_CODEC_ID_ADPCM_CIRCUS,
|
||||
AV_CODEC_ID_ADPCM_IMA_ESCAPE,
|
||||
|
||||
/* AMR */
|
||||
AV_CODEC_ID_AMR_NB = 0x12000,
|
||||
|
@@ -250,7 +250,7 @@ static int rpl_read_header(AVFormatContext *s)
|
||||
// are all unsigned.
|
||||
ast->codecpar->codec_id = AV_CODEC_ID_PCM_U8;
|
||||
} else if (ast->codecpar->bits_per_coded_sample == 4) {
|
||||
ast->codecpar->codec_id = AV_CODEC_ID_ADPCM_IMA_EA_SEAD;
|
||||
ast->codecpar->codec_id = AV_CODEC_ID_ADPCM_IMA_ESCAPE;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
Reference in New Issue
Block a user