C中的ADPCM解码

ADPCM decoding in C

当我尝试将 IMA ADPCM 解码为 16 位签名 PCM 时,我得到了半垃圾 PCM

我尝试解码和编码 WAV 文件中的 IMA ADPCM(22050 samples/s,4 bit/sample,36(!)字节(?)对齐(?!),1 通道)。

int16_t decodeImaAdpcmSampleUIS(struct AdpcmState* state, const uint8_t sample){
int diff;
int8_t si=state->stepindex;
int step=i_step_table[si];
int cur=state->current;

diff=step>>3;
if( sample&0x04 ) diff += step;
if( sample&0x02 ) diff += step>>1;
if( sample&0x01 ) diff += step>>2;
if( sample&0x08 ){
cur -= diff;
if(cur<-32768)
    cur=-32768;
}else{
cur += diff;
if(cur>32767)
    cur=32767;
}
//predictor: cur, state.current
//step_index: si, stepindex
si+=ima_index_table[sample&0b111];
if(si < 0)
    si = 0;
if(si > 88)
    si = 88;
state->stepindex=si;

return state->current=cur;
}

void decodeImaAdpcm(uint8_t* src, int16_t* dst, size_t srcLen){
    struct AdpcmState state={0,0};

    for(size_t i=0; i<srcLen; i++){
        *(dst++)=decodeImaAdpcmSampleUIS(&state, (*src)&0xf);
        *(dst++)=decodeImaAdpcmSampleUIS(&state, (*src)>>4);
        src++;
    }
}

带有 WAV 文件的完整项目: https://drive.google.com/open?id=1xuxwXj3Y_QhPDWhrQY7nmz8ycBE1cgyL

A​​DPCM WAV 文件转换为 PCM WAV 包含一些垃圾,但 ffmpeg 可以正常转换。当前未实现 PCM WAV 到 ADPCM WAV 的转换。

发生这种情况是因为 WAV 包含 PLAIN IMA ADPCM。每个块的前 4 个字节都是 ADPCM 状态。

例如:WAV.align=36;//块大小为36字节,块结构为--|current(16 bits)|step index(16 bytes)|data(WAV.align -4(36-4=32) 字节)|--