150 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			150 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * 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;
 | |
| }
 |