| /* |
| * Siren Encoder/Decoder library |
| * |
| * @author: Youness Alaoui <kakaroto@kakaroto.homelinux.net> |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public |
| * License along with this library; if not, write to the |
| * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| */ |
| |
| |
| #include "siren7.h" |
| |
| |
| static int rmlt_initialized = 0; |
| static float rmlt_window_640[640]; |
| static float rmlt_window_320[320]; |
| |
| #define PI_2 1.57079632679489661923 |
| |
| void |
| siren_rmlt_init (void) |
| { |
| int i = 0; |
| float angle; |
| |
| for (i = 0; i < 640; i++) { |
| angle = (float) (((i + 0.5) * PI_2) / 640); |
| rmlt_window_640[i] = (float) sin (angle); |
| } |
| for (i = 0; i < 320; i++) { |
| angle = (float) (((i + 0.5) * PI_2) / 320); |
| rmlt_window_320[i] = (float) sin (angle); |
| } |
| |
| rmlt_initialized = 1; |
| } |
| |
| int |
| siren_rmlt_encode_samples (float *samples, float *old_samples, int dct_length, |
| float *rmlt_coefs) |
| { |
| int half_dct_length = dct_length / 2; |
| float *old_ptr = old_samples + half_dct_length; |
| float *coef_high = rmlt_coefs + half_dct_length; |
| float *coef_low = rmlt_coefs + half_dct_length; |
| float *samples_low = samples; |
| float *samples_high = samples + dct_length; |
| float *window_low = NULL; |
| float *window_high = NULL; |
| int i = 0; |
| |
| if (rmlt_initialized == 0) |
| siren_rmlt_init (); |
| |
| if (dct_length == 320) |
| window_low = rmlt_window_320; |
| else if (dct_length == 640) |
| window_low = rmlt_window_640; |
| else |
| return 4; |
| |
| window_high = window_low + dct_length; |
| |
| |
| for (i = 0; i < half_dct_length; i++) { |
| *--coef_low = *--old_ptr; |
| *coef_high++ = |
| (*samples_low * *--window_high) - (*--samples_high * *window_low); |
| *old_ptr = |
| (*samples_high * *window_high) + (*samples_low++ * *window_low++); |
| } |
| siren_dct4 (rmlt_coefs, rmlt_coefs, dct_length); |
| |
| return 0; |
| } |
| |
| |
| |
| int |
| siren_rmlt_decode_samples (float *coefs, float *old_coefs, int dct_length, |
| float *samples) |
| { |
| int half_dct_length = dct_length / 2; |
| float *old_low = old_coefs; |
| float *old_high = old_coefs + half_dct_length; |
| float *samples_low = samples; |
| float *samples_high = samples + dct_length; |
| float *samples_middle_low = samples + half_dct_length; |
| float *samples_middle_high = samples + half_dct_length; |
| float *window_low = NULL; |
| float *window_high = NULL; |
| float *window_middle_low = NULL; |
| float *window_middle_high = NULL; |
| float sample_low_val; |
| float sample_high_val; |
| float sample_middle_low_val; |
| float sample_middle_high_val; |
| int i = 0; |
| |
| if (rmlt_initialized == 0) |
| siren_rmlt_init (); |
| |
| if (dct_length == 320) |
| window_low = rmlt_window_320; |
| else if (dct_length == 640) |
| window_low = rmlt_window_640; |
| else |
| return 4; |
| |
| |
| window_high = window_low + dct_length; |
| window_middle_low = window_low + half_dct_length; |
| window_middle_high = window_low + half_dct_length; |
| |
| siren_dct4 (coefs, samples, dct_length); |
| |
| for (i = 0; i < half_dct_length; i += 2) { |
| sample_low_val = *samples_low; |
| sample_high_val = *--samples_high; |
| sample_middle_low_val = *--samples_middle_low; |
| sample_middle_high_val = *samples_middle_high; |
| *samples_low++ = |
| (*old_low * *--window_high) + (sample_middle_low_val * *window_low); |
| *samples_high = |
| (sample_middle_low_val * *window_high) - (*old_low * *window_low++); |
| *samples_middle_high++ = |
| (sample_low_val * *window_middle_high) - |
| (*--old_high * *--window_middle_low); |
| *samples_middle_low = |
| (*old_high * *window_middle_high++) + |
| (sample_low_val * *window_middle_low); |
| *old_low++ = sample_middle_high_val; |
| *old_high = sample_high_val; |
| } |
| |
| return 0; |
| } |