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
ADPCM 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) 字节)|--
当我尝试将 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
ADPCM 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) 字节)|--