I have been working on an propeller based autopilot for a while now, and things have been coming along great. Although I was nervous about the lack of interrupts, the faster speed and multiple cogs have made timing the program efficient. I purchased the Pololu brand IMU v2 for $50, and my only complaint might be that the board is TOO small. I am implementing a complementary filter to filter all vibration noise, and it seems to be working just fine.
I am able to set the attitude of the plane using a simple PID loop, and I am receiving data back from the airplane to my laptop via xbee. The only problem is: I cannot tilt-compensate the compass module.
The LSM303DLHC contains both an accelerometer and magnetometer (I assume specifically for this purpose!) I am able to read the whole sensor using the I2C bus.
I have tried both the algorithms found at these websites:
Before implementing the tilt-compensation, I normalize the magnetometer values by find the total magnitude:
r = sqrt( magX ^2 + magY^2 + magZ^2)
and then divinding each component by magnitude.
However, after using my atan2 function, I do not get a full 360 degrees!!
I don't know if my IMU is orientated correctly:
X-axis -> towards nose of plane
Y-axis -> towards right wingtip
Z-axis -> downwards
While this reference frame is used in the first link above ^^ it is not what I am used to from math class lol
My yaw value stays center at about 170 degrees and -170. It actually changes very little as I spin the plane horizontally, however, the value scales up and down as I change the pitch and roll of the plane.
If i don't normalize the magnetometer vectors, I get a value between 60-120 for my yaw while spinning it level, but again is not tilt-compensated.
What is more confusing, if i use atan2(mz,mx) and add 360 if its less than 0, i get a full 360 degrees while spinning the plane horizontally. The only problem is that its not tilt compensated...
Any ideas on this one guys? Has anyone succesfully tilt compensated this magnetometer before? I do not understand why this algorithm is not working, we only care about the planes direction in the xy plane, so using atan2 makes sense enough.
Here is a clip of the code I am using (I apologize ahead of time, float math is more troublesome then i would like on the propeller chip)::
mx := fm.ffloat(IMU.getMX)
my := fm.ffloat(IMU.getMY)
mz := fm.ffloat(IMU.getMZ)
mv := fm.fsqr(fm.fadd(fm.fadd(fm.fmul(mx,mx),fm.fmul(my,my)),fm.fmul(mz,mz)))
pitchRad := fm.radians(pitch)
rollRad := fm.radians(roll)
mx := fm.fdiv(mx,mv)
my := fm.fdiv(my,mv)
mz := fm.fdiv(mz,mv)
magY := fm.FSub(fm.fadd(fm.fmul(fm.fmul(fm.cos(pitchRad),mx),fm.sin(pitchRad)),fm.fmul(fm.cos(rollRad),my)),fm.fmul(fm.fmul(fm.sin(rollRad),mz),fm.cos(pitchRad)))
//sorry for how ugly this term is ^^^
magX := fm.fadd(fm.fmul(fm.cos(pitchRad),mx),fm.fmul(fm.sin(pitchRad),mz))
yaw := fm.degrees(atan2(magY, magX))
PRI atan2(_y, _x) | _z, atan //this is my atan2 function
if _x == 0.0
if _y > 0.0
if _y == 0.0
_z := fm.fdiv(_y, _x)
if fm.fabs(_z) < 1.0
atan := fm.fdiv(_z, fm.fadd(1.0,fm.fmul(fm.fmul(0.28, _z), _z)))
if _x < 0.0
if _y < 0.0
atan := fm.fsub(PIBY2_FLOAT, fm.fdiv(_z , fm.fadd(fm.fmul(_z, _z), 0.28)))
if _y < 0.0
THANKS FOR ANY HELP!!!
Remind me -- did you calibrate the compass? At least for hard iron offset?
If not, nothing will work.
A little more info here :