FreeCalypso > hg > gsm-codec-lib
comparison libtwamr/dec_main.c @ 427:357d1faad55d
libtwamr: implement decoder top level
| author | Mychaela Falconia <falcon@freecalypso.org> |
|---|---|
| date | Tue, 07 May 2024 21:45:47 +0000 |
| parents | libtwamr/enc_main.c@2a094af3d384 |
| children |
comparison
equal
deleted
inserted
replaced
| 426:7bef001cd8b8 | 427:357d1faad55d |
|---|---|
| 1 /* | |
| 2 * This C module is the top level entity for our stateful decoder engine. | |
| 3 */ | |
| 4 | |
| 5 #include <stdint.h> | |
| 6 #include <stdlib.h> | |
| 7 #include <string.h> | |
| 8 #include "tw_amr.h" | |
| 9 #include "namespace.h" | |
| 10 #include "typedef.h" | |
| 11 #include "cnst.h" | |
| 12 #include "dec_amr.h" | |
| 13 #include "pstfilt.h" | |
| 14 #include "post_pro.h" | |
| 15 #include "bitno.h" | |
| 16 | |
| 17 struct amr_decoder_state { | |
| 18 Decoder_amrState dec; | |
| 19 Post_FilterState pstfilt; | |
| 20 Post_ProcessState posthp; | |
| 21 enum Mode prev_mode; | |
| 22 Flag is_homed; | |
| 23 }; | |
| 24 | |
| 25 struct amr_decoder_state *amr_decoder_create(void) | |
| 26 { | |
| 27 struct amr_decoder_state *st; | |
| 28 | |
| 29 st = malloc(sizeof(struct amr_decoder_state)); | |
| 30 if (st) | |
| 31 amr_decoder_reset(st); | |
| 32 return st; | |
| 33 } | |
| 34 | |
| 35 void amr_decoder_reset(struct amr_decoder_state *st) | |
| 36 { | |
| 37 Decoder_amr_reset(&st->dec, 0); | |
| 38 Post_Filter_reset(&st->pstfilt); | |
| 39 Post_Process_reset(&st->posthp); | |
| 40 st->prev_mode = (enum Mode) 0; | |
| 41 st->is_homed = 1; | |
| 42 } | |
| 43 | |
| 44 void amr_decode_frame(struct amr_decoder_state *st, | |
| 45 const struct amr_param_frame *frame, int16_t *pcm) | |
| 46 { | |
| 47 enum Mode mode; | |
| 48 Word16 parm[MAX_PRM_SIZE]; | |
| 49 Word16 Az_dec[AZ_SIZE]; | |
| 50 Word16 i; | |
| 51 | |
| 52 /* fast home state handling needs to be first */ | |
| 53 if (st->is_homed && amr_check_dhf(frame, 1)) { | |
| 54 for (i = 0; i < L_FRAME; i++) | |
| 55 pcm[i] = EHF_MASK; | |
| 56 return; | |
| 57 } | |
| 58 /* "unpack" into internal form */ | |
| 59 if (frame->type == RX_NO_DATA) | |
| 60 mode = st->prev_mode; | |
| 61 else { | |
| 62 mode = frame->mode & 7; | |
| 63 st->prev_mode = mode; | |
| 64 } | |
| 65 memcpy(parm, frame->param, prmno[mode] * sizeof(int16_t)); | |
| 66 /* now we can call the guts of the decoder */ | |
| 67 Decoder_amr(&st->dec, mode, parm, frame->type, pcm, Az_dec); | |
| 68 Post_Filter(&st->pstfilt, mode, pcm, Az_dec); | |
| 69 Post_Process(&st->posthp, pcm, L_FRAME); | |
| 70 /* | |
| 71 * The 3 lsbs of each speech sample typically won't be all 0 | |
| 72 * out of Post_Process(), hence we have to clear them explicitly | |
| 73 * to maintain overall bit-exact operation. | |
| 74 */ | |
| 75 for (i = 0; i < L_FRAME; i++) | |
| 76 pcm[i] &= 0xFFF8; | |
| 77 /* final check for full DHF */ | |
| 78 if (amr_check_dhf(frame, 0)) | |
| 79 amr_decoder_reset(st); | |
| 80 else | |
| 81 st->is_homed = 0; | |
| 82 } |
