2012-03-04 23:31:47 -04:00
|
|
|
/// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
//
|
|
|
|
// Unit tests for the AP_Math euler code
|
|
|
|
//
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
#include <stdlib.h>
|
2012-03-04 23:31:47 -04:00
|
|
|
#include <AP_Common.h>
|
2012-10-27 00:55:17 -03:00
|
|
|
#include <AP_Progmem.h>
|
2012-08-20 20:22:44 -03:00
|
|
|
#include <AP_Param.h>
|
2012-03-04 23:31:47 -04:00
|
|
|
#include <AP_Math.h>
|
2012-09-18 15:08:18 -03:00
|
|
|
#include <AP_HAL.h>
|
2012-03-04 23:31:47 -04:00
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
#include <AP_HAL_AVR.h>
|
2012-12-18 21:26:58 -04:00
|
|
|
const AP_HAL::HAL& hal = AP_HAL_BOARD_DRIVER;
|
2012-03-12 08:49:36 -03:00
|
|
|
|
2012-03-04 23:31:47 -04:00
|
|
|
static float rad_diff(float rad1, float rad2)
|
|
|
|
{
|
|
|
|
float diff = rad1 - rad2;
|
|
|
|
if (diff > PI) {
|
|
|
|
diff -= 2*PI;
|
|
|
|
}
|
|
|
|
if (diff < -PI) {
|
|
|
|
diff += 2*PI;
|
|
|
|
}
|
2013-01-10 14:42:24 -04:00
|
|
|
return fabsf(diff);
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
|
2012-03-05 07:42:29 -04:00
|
|
|
static void check_result(float roll, float pitch, float yaw,
|
|
|
|
float roll2, float pitch2, float yaw2)
|
2012-03-04 23:31:47 -04:00
|
|
|
{
|
|
|
|
if (isnan(roll2) ||
|
|
|
|
isnan(pitch2) ||
|
|
|
|
isnan(yaw2)) {
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf("NAN eulers roll=%f pitch=%f yaw=%f\n",
|
2012-03-04 23:31:47 -04:00
|
|
|
roll, pitch, yaw);
|
|
|
|
}
|
2012-03-05 07:42:29 -04:00
|
|
|
|
|
|
|
if (rad_diff(roll2,roll) > ToRad(179)) {
|
|
|
|
// reverse all 3
|
|
|
|
roll2 += fmod(roll2+PI, 2*PI);
|
|
|
|
pitch2 += fmod(pitch2+PI, 2*PI);
|
|
|
|
yaw2 += fmod(yaw2+PI, 2*PI);
|
|
|
|
}
|
|
|
|
|
2012-03-04 23:31:47 -04:00
|
|
|
if (rad_diff(roll2,roll) > 0.01 ||
|
|
|
|
rad_diff(pitch2, pitch) > 0.01 ||
|
|
|
|
rad_diff(yaw2, yaw) > 0.01) {
|
2012-03-10 02:29:32 -04:00
|
|
|
if (pitch >= PI/2 ||
|
|
|
|
pitch <= -PI/2 ||
|
|
|
|
ToDeg(rad_diff(pitch, PI/2)) < 1 ||
|
2012-03-05 07:42:29 -04:00
|
|
|
ToDeg(rad_diff(pitch, -PI/2)) < 1) {
|
|
|
|
// we expect breakdown at these poles
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf_P(
|
|
|
|
PSTR("breakdown eulers roll=%f/%f pitch=%f/%f yaw=%f/%f\n"),
|
|
|
|
ToDeg(roll), ToDeg(roll2),
|
|
|
|
ToDeg(pitch), ToDeg(pitch2),
|
|
|
|
ToDeg(yaw), ToDeg(yaw2));
|
2012-03-05 07:42:29 -04:00
|
|
|
} else {
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf_P(
|
|
|
|
PSTR("incorrect eulers roll=%f/%f pitch=%f/%f yaw=%f/%f\n"),
|
|
|
|
ToDeg(roll), ToDeg(roll2),
|
|
|
|
ToDeg(pitch), ToDeg(pitch2),
|
|
|
|
ToDeg(yaw), ToDeg(yaw2));
|
2012-03-05 07:42:29 -04:00
|
|
|
}
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-05 07:42:29 -04:00
|
|
|
static void test_euler(float roll, float pitch, float yaw)
|
|
|
|
{
|
|
|
|
Matrix3f m;
|
|
|
|
float roll2, pitch2, yaw2;
|
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
m.from_euler(roll, pitch, yaw);
|
|
|
|
m.to_euler(&roll2, &pitch2, &yaw2);
|
2012-03-05 07:42:29 -04:00
|
|
|
check_result(roll, pitch, yaw, roll2, pitch2, yaw2);
|
|
|
|
}
|
|
|
|
|
2012-03-04 23:31:47 -04:00
|
|
|
#define ARRAY_LENGTH(x) (sizeof((x))/sizeof((x)[0]))
|
|
|
|
|
|
|
|
static const float angles[] = { 0, PI/8, PI/4, PI/2, PI,
|
|
|
|
-PI/8, -PI/4, -PI/2, -PI};
|
|
|
|
|
|
|
|
void test_matrix_eulers(void)
|
|
|
|
{
|
|
|
|
uint8_t i, j, k;
|
|
|
|
uint8_t N = ARRAY_LENGTH(angles);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("rotation matrix unit tests\n");
|
2012-03-04 23:31:47 -04:00
|
|
|
|
|
|
|
for (i=0; i<N; i++)
|
|
|
|
for (j=0; j<N; j++)
|
|
|
|
for (k=0; k<N; k++)
|
|
|
|
test_euler(angles[i], angles[j], angles[k]);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("tests done\n");
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void test_quaternion(float roll, float pitch, float yaw)
|
|
|
|
{
|
|
|
|
Quaternion q;
|
|
|
|
float roll2, pitch2, yaw2;
|
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
q.from_euler(roll, pitch, yaw);
|
|
|
|
q.to_euler(&roll2, &pitch2, &yaw2);
|
2012-03-05 07:42:29 -04:00
|
|
|
check_result(roll, pitch, yaw, roll2, pitch2, yaw2);
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_quaternion_eulers(void)
|
|
|
|
{
|
|
|
|
uint8_t i, j, k;
|
|
|
|
uint8_t N = ARRAY_LENGTH(angles);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("quaternion unit tests\n");
|
2012-03-04 23:31:47 -04:00
|
|
|
|
|
|
|
test_quaternion(PI/4, 0, 0);
|
|
|
|
test_quaternion(0, PI/4, 0);
|
|
|
|
test_quaternion(0, 0, PI/4);
|
|
|
|
test_quaternion(-PI/4, 0, 0);
|
|
|
|
test_quaternion(0, -PI/4, 0);
|
|
|
|
test_quaternion(0, 0, -PI/4);
|
|
|
|
test_quaternion(-PI/4, 1, 1);
|
|
|
|
test_quaternion(1, -PI/4, 1);
|
|
|
|
test_quaternion(1, 1, -PI/4);
|
|
|
|
|
|
|
|
test_quaternion(ToRad(89), 0, 0.1);
|
|
|
|
test_quaternion(0, ToRad(89), 0.1);
|
|
|
|
test_quaternion(0.1, 0, ToRad(89));
|
|
|
|
|
|
|
|
test_quaternion(ToRad(91), 0, 0.1);
|
|
|
|
test_quaternion(0, ToRad(91), 0.1);
|
|
|
|
test_quaternion(0.1, 0, ToRad(91));
|
|
|
|
|
|
|
|
for (i=0; i<N; i++)
|
|
|
|
for (j=0; j<N; j++)
|
|
|
|
for (k=0; k<N; k++)
|
|
|
|
test_quaternion(angles[i], angles[j], angles[k]);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("tests done\n");
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
|
2012-03-05 07:42:29 -04:00
|
|
|
|
|
|
|
static void test_conversion(float roll, float pitch, float yaw)
|
|
|
|
{
|
|
|
|
Quaternion q;
|
|
|
|
Matrix3f m, m2;
|
|
|
|
|
|
|
|
float roll2, pitch2, yaw2;
|
2012-03-10 02:06:46 -04:00
|
|
|
float roll3, pitch3, yaw3;
|
2012-03-05 07:42:29 -04:00
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
q.from_euler(roll, pitch, yaw);
|
2012-03-10 02:29:32 -04:00
|
|
|
q.to_euler(&roll2, &pitch2, &yaw2);
|
|
|
|
check_result(roll, pitch, yaw, roll2, pitch2, yaw2);
|
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
q.rotation_matrix(m);
|
|
|
|
m.to_euler(&roll2, &pitch2, &yaw2);
|
2012-03-10 02:29:32 -04:00
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
m2.from_euler(roll, pitch, yaw);
|
|
|
|
m2.to_euler(&roll3, &pitch3, &yaw3);
|
2012-03-05 07:42:29 -04:00
|
|
|
if (m.is_nan()) {
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf("NAN matrix roll=%f pitch=%f yaw=%f\n",
|
2012-03-05 07:42:29 -04:00
|
|
|
roll, pitch, yaw);
|
|
|
|
}
|
2012-03-10 02:29:32 -04:00
|
|
|
|
2012-03-05 07:42:29 -04:00
|
|
|
check_result(roll, pitch, yaw, roll2, pitch2, yaw2);
|
2012-03-10 02:06:46 -04:00
|
|
|
check_result(roll, pitch, yaw, roll3, pitch3, yaw3);
|
2012-03-05 07:42:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_conversions(void)
|
|
|
|
{
|
|
|
|
uint8_t i, j, k;
|
|
|
|
uint8_t N = ARRAY_LENGTH(angles);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("matrix/quaternion tests\n");
|
2012-03-05 07:42:29 -04:00
|
|
|
|
|
|
|
test_conversion(1, 1.1, 1.2);
|
|
|
|
test_conversion(1, -1.1, 1.2);
|
|
|
|
test_conversion(1, -1.1, -1.2);
|
|
|
|
test_conversion(-1, 1.1, -1.2);
|
|
|
|
test_conversion(-1, 1.1, 1.2);
|
|
|
|
|
|
|
|
for (i=0; i<N; i++)
|
|
|
|
for (j=0; j<N; j++)
|
|
|
|
for (k=0; k<N; k++)
|
|
|
|
test_conversion(angles[i], angles[j], angles[k]);
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("tests done\n");
|
2012-03-05 07:42:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_frame_transforms(void)
|
|
|
|
{
|
|
|
|
Vector3f v, v2;
|
|
|
|
Quaternion q;
|
|
|
|
Matrix3f m;
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("frame transform tests\n");
|
2012-03-05 07:42:29 -04:00
|
|
|
|
2012-03-10 02:06:46 -04:00
|
|
|
q.from_euler(ToRad(90), 0, 0);
|
2012-03-05 07:42:29 -04:00
|
|
|
v2 = v = Vector3f(0, 0, 1);
|
2012-03-10 02:06:46 -04:00
|
|
|
q.earth_to_body(v2);
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf("%f %f %f\n", v2.x, v2.y, v2.z);
|
2012-03-05 07:42:29 -04:00
|
|
|
}
|
|
|
|
|
2012-03-19 03:23:22 -03:00
|
|
|
// generate a random float between -1 and 1
|
|
|
|
static float rand_num(void)
|
|
|
|
{
|
2012-08-17 03:20:14 -03:00
|
|
|
float ret = ((unsigned)random()) % 2000000;
|
|
|
|
return (ret - 1.0e6) / 1.0e6;
|
2012-03-19 03:23:22 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void test_matrix_rotate(void)
|
|
|
|
{
|
|
|
|
Matrix3f m1, m2, diff;
|
|
|
|
Vector3f r;
|
|
|
|
|
|
|
|
m1.identity();
|
|
|
|
m2.identity();
|
|
|
|
r.x = rand_num();
|
|
|
|
r.y = rand_num();
|
|
|
|
r.z = rand_num();
|
|
|
|
|
|
|
|
for (uint16_t i = 0; i<1000; i++) {
|
|
|
|
// old method
|
|
|
|
Matrix3f temp_matrix;
|
|
|
|
temp_matrix.a.x = 0;
|
|
|
|
temp_matrix.a.y = -r.z;
|
|
|
|
temp_matrix.a.z = r.y;
|
|
|
|
temp_matrix.b.x = r.z;
|
|
|
|
temp_matrix.b.y = 0;
|
|
|
|
temp_matrix.b.z = -r.x;
|
|
|
|
temp_matrix.c.x = -r.y;
|
|
|
|
temp_matrix.c.y = r.x;
|
|
|
|
temp_matrix.c.z = 0;
|
|
|
|
temp_matrix = m1 * temp_matrix;
|
|
|
|
m1 += temp_matrix;
|
|
|
|
|
|
|
|
// new method
|
|
|
|
m2.rotate(r);
|
|
|
|
|
|
|
|
// check they behave in the same way
|
|
|
|
diff = m1 - m2;
|
|
|
|
float err = diff.a.length() + diff.b.length() + diff.c.length();
|
|
|
|
|
|
|
|
if (err > 0) {
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->printf("ERROR: i=%u err=%f\n", (unsigned)i, err);
|
2012-03-19 03:23:22 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-04 23:31:47 -04:00
|
|
|
/*
|
2012-08-17 03:20:14 -03:00
|
|
|
* euler angle tests
|
2012-03-04 23:31:47 -04:00
|
|
|
*/
|
|
|
|
void setup(void)
|
|
|
|
{
|
2012-09-18 15:08:18 -03:00
|
|
|
hal.console->println("euler unit tests\n");
|
2012-03-10 02:29:32 -04:00
|
|
|
|
|
|
|
test_conversion(0, PI, 0);
|
|
|
|
|
2012-03-05 07:42:29 -04:00
|
|
|
test_frame_transforms();
|
|
|
|
test_conversions();
|
2012-03-04 23:31:47 -04:00
|
|
|
test_quaternion_eulers();
|
2012-03-05 07:42:29 -04:00
|
|
|
test_matrix_eulers();
|
2012-03-19 03:23:22 -03:00
|
|
|
test_matrix_rotate();
|
2012-03-04 23:31:47 -04:00
|
|
|
}
|
|
|
|
|
2012-09-18 15:08:18 -03:00
|
|
|
void loop(void){}
|
|
|
|
|
|
|
|
AP_HAL_MAIN();
|