2011-05-31 11:22:16 +02:00
/*
* Copyright ( c ) 2011 Broadcom Corporation
*
* Permission to use , copy , modify , and / or distribute this software for any
* purpose with or without fee is hereby granted , provided that the above
* copyright notice and this permission notice appear in all copies .
*
* THE SOFTWARE IS PROVIDED " AS IS " AND THE AUTHOR DISCLAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS . IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
* SPECIAL , DIRECT , INDIRECT , OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE , DATA OR PROFITS , WHETHER IN AN ACTION
* OF CONTRACT , NEGLIGENCE OR OTHER TORTIOUS ACTION , ARISING OUT OF OR IN
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE .
*/
# include <linux/module.h>
# include <linux/cordic.h>
# define CORDIC_ANGLE_GEN 39797
# define CORDIC_PRECISION_SHIFT 16
# define CORDIC_NUM_ITER (CORDIC_PRECISION_SHIFT + 2)
# define FIXED(X) ((s32)((X) << CORDIC_PRECISION_SHIFT))
# define FLOAT(X) (((X) >= 0) \
? ( ( ( ( X ) > > ( CORDIC_PRECISION_SHIFT - 1 ) ) + 1 ) > > 1 ) \
: - ( ( ( ( - ( X ) ) > > ( CORDIC_PRECISION_SHIFT - 1 ) ) + 1 ) > > 1 ) )
static const s32 arctan_table [ ] = {
2949120 ,
1740967 ,
919879 ,
466945 ,
234379 ,
117304 ,
58666 ,
29335 ,
14668 ,
7334 ,
3667 ,
1833 ,
917 ,
458 ,
229 ,
115 ,
57 ,
29
} ;
/*
* cordic_calc_iq ( ) - calculates the i / q coordinate for given angle
*
* theta : angle in degrees for which i / q coordinate is to be calculated
* coord : function output parameter holding the i / q coordinate
*/
struct cordic_iq cordic_calc_iq ( s32 theta )
{
struct cordic_iq coord ;
s32 angle , valtmp ;
unsigned iter ;
int signx = 1 ;
int signtheta ;
coord . i = CORDIC_ANGLE_GEN ;
coord . q = 0 ;
angle = 0 ;
theta = FIXED ( theta ) ;
signtheta = ( theta < 0 ) ? - 1 : 1 ;
theta = ( ( theta + FIXED ( 180 ) * signtheta ) % FIXED ( 360 ) ) -
FIXED ( 180 ) * signtheta ;
if ( FLOAT ( theta ) > 90 ) {
theta - = FIXED ( 180 ) ;
signx = - 1 ;
} else if ( FLOAT ( theta ) < - 90 ) {
theta + = FIXED ( 180 ) ;
signx = - 1 ;
}
for ( iter = 0 ; iter < CORDIC_NUM_ITER ; iter + + ) {
if ( theta > angle ) {
valtmp = coord . i - ( coord . q > > iter ) ;
coord . q + = ( coord . i > > iter ) ;
angle + = arctan_table [ iter ] ;
} else {
valtmp = coord . i + ( coord . q > > iter ) ;
coord . q - = ( coord . i > > iter ) ;
angle - = arctan_table [ iter ] ;
}
coord . i = valtmp ;
}
coord . i * = signx ;
coord . q * = signx ;
return coord ;
}
EXPORT_SYMBOL ( cordic_calc_iq ) ;
2011-07-29 12:59:51 +00:00
MODULE_DESCRIPTION ( " CORDIC algorithm " ) ;
2011-05-31 11:22:16 +02:00
MODULE_AUTHOR ( " Broadcom Corporation " ) ;
MODULE_LICENSE ( " Dual BSD/GPL " ) ;