/* * Mathmatical methods. * Copyright (C) 1999 Christopher Edwards * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * */ package org.tritonus.lowlevel.gsm; public class Add { public static short saturate(int x) { return (short) ((x) < Gsm_Def.MIN_WORD ? Gsm_Def.MIN_WORD : (x) > Gsm_Def.MAX_WORD ? Gsm_Def.MAX_WORD: (x)); } public static short saturate(long x) { return (short) ((x) < Gsm_Def.MIN_WORD ? Gsm_Def.MIN_WORD : (x) > Gsm_Def.MAX_WORD ? Gsm_Def.MAX_WORD: (x)); } public static short SASR(int x, int by) { return (short) ((x) >> (by)); } public static short GSM_ADD(short a, short b) { /* * This converts a and b to int implicitely, because * '+' is not defined for short. */ int sum = a + b; return saturate(sum); } public static short GSM_SUB(short a, short b) { /* * This converts a and b to int implicitely, because * '-' is not defined for short. */ int diff = a - b; return saturate(diff); } public static short GSM_MULT(short a, short b) { if (a == Gsm_Def.MIN_WORD && b == Gsm_Def.MIN_WORD) { return Gsm_Def.MAX_WORD; } else { return SASR( ((int)(a)) * ((int)(b)), 15 ); } } public static short GSM_MULT_R(short a, short b) { if (a == Gsm_Def.MIN_WORD && b == Gsm_Def.MIN_WORD) { return Gsm_Def.MAX_WORD; } else { int prod = (int) (((int)(a)) * ((int)(b)) + 16384); prod >>= 15; return (short) (prod & 0xFFFF); } } public static short GSM_ABS(short a) { int b = a < 0 ? (a == Gsm_Def.MIN_WORD ? Gsm_Def.MAX_WORD : -a) : a; return ((short)(b)); } public static int GSM_L_MULT(short a, short b) throws IllegalArgumentException { if( a != Short.MIN_VALUE || b != Short.MIN_VALUE ) { throw new IllegalArgumentException( "One of the aruments must equal " + Short.MIN_VALUE); } return ((int)a * (int)b) << 1; } public static int GSM_L_ADD(int a, int b) { if (a <= 0) { if (b >= 0) { return a + b; } else { long A = (long)-(a + 1) + (long)-(b + 1); return A >= Gsm_Def.MAX_LONGWORD ? Gsm_Def.MIN_LONGWORD :-(int)A-2; } } else if (b <= 0) { return a + b; } else { long A = (long)a + (long) b; return (int)(A > Gsm_Def.MAX_LONGWORD ? Gsm_Def.MAX_LONGWORD : A); } } /* * the number of left shifts needed to normalize the 32 bit * variable L_var1 for positive values on the interval * * with minimum of * minimum of 1073741824 (01000000000000000000000000000000) and * maximum of 2147483647 (01111111111111111111111111111111) * * * and for negative values on the interval with * minimum of -2147483648 (-10000000000000000000000000000000) and * maximum of -1073741824 ( -1000000000000000000000000000000). * * in order to normalize the result, the following * operation must be done: L_norm_var1 = L_var1 << norm( L_var1 ); * * (That's 'ffs', only from the left, not the right..) */ public static short gsm_norm( int a ) throws IllegalArgumentException { if(a == 0) { throw new IllegalArgumentException ("gsm_norm: a cannot = 0."); } if (a < 0) { if (a <= -1073741824) { return 0; } a = ~a; } return (short) ( ((a & 0xffff0000) != 0) ? ( ((a & 0xff000000) != 0) ? -1 + bitoff[ (int)(0xFF & (a >> 24)) ] : 7 + bitoff[ (int)(0xFF & (a >> 16)) ] ) : ( ((a & 0xff00) != 0) ? 15 + bitoff[ (int)(0xFF & (a >> 8)) ] : 23 + bitoff[ (int)(0xFF & a) ] )); } public static short gsm_asl(short a, int n) { if (n >= 16) return ((short) 0); if (n <= -16) if(a<0) return (short) -1; else return (short) -0; if (n < 0) return gsm_asr(a, -n); return ((short)( a << n )); } public static short gsm_asr(short a, int n) { if (n >= 16) if(a<0) return (short) -1; else return (short) -0; if (n <= -16) return ((short) 0); if (n < 0) return ((short)( a << -n )); return ((short)( a >> n )); } public static short gsm_div(short num, short denum) throws IllegalArgumentException { int L_num = num; int L_denum = denum; short div = 0; int k = 15; /* The parameter num sometimes becomes zero. * Although this is explicitly guarded against in 4.2.5, * we assume that the result should then be zero as well. */ /* assert(num != 0); */ if( ! (num >= 0 && denum >= num) ) { throw new IllegalArgumentException ("gsm_div: num >= 0 && denum >= num"); } if (num == 0) return 0; while (k != 0) { k--; div <<= 1; L_num <<= 1; if (L_num >= L_denum) { L_num -= L_denum; div++; } } return div; } private static final short bitoff[] = { 8, 7, 6, 6, 5, 5, 5, 5, 4, 4, 4, 4, 4, 4, 4, 4, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }; }