mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 06:38:29 -04:00
AP_AccelCal: use union instead of reference
The current approach to access the same memory location by using a reference is giving this warning: ../libraries/AP_AccelCal/AccelCalibrator.cpp: In constructor ‘AccelCalibrator::AccelCalibrator()’: ../libraries/AP_AccelCal/AccelCalibrator.cpp:34:64: warning: dereferencing type-punned pointer will break strict-aliasing rules [-Wstrict-aliasing] _param_struct(*reinterpret_cast<struct param_t *>(&_param_array)) ^ ../libraries/AP_AccelCal/AccelCalibrator.cpp: In member function ‘void AccelCalibrator::run_fit(uint8_t, float&)’: ../libraries/AP_AccelCal/AccelCalibrator.cpp:336:79: warning: dereferencing type-punned pointer will break strict-aliasing rules [-Wstrict-aliasing] struct param_t &fit_param(*reinterpret_cast<struct param_t *>(¶m_array)); ^ Using a union allows us to get rid of the warning, make sure the sizes of the different structs match and have a more elegant solution.
This commit is contained in:
parent
841f34effa
commit
f3ee7a9a85
@ -30,8 +30,7 @@ const extern AP_HAL::HAL& hal;
|
|||||||
|
|
||||||
AccelCalibrator::AccelCalibrator() :
|
AccelCalibrator::AccelCalibrator() :
|
||||||
_conf_tolerance(ACCEL_CAL_TOLERANCE),
|
_conf_tolerance(ACCEL_CAL_TOLERANCE),
|
||||||
_sample_buffer(NULL),
|
_sample_buffer(NULL)
|
||||||
_param_struct(*reinterpret_cast<struct param_t *>(&_param_array))
|
|
||||||
{
|
{
|
||||||
clear();
|
clear();
|
||||||
}
|
}
|
||||||
@ -74,9 +73,9 @@ void AccelCalibrator::start(enum accel_cal_fit_type_t fit_type, uint8_t num_samp
|
|||||||
const float theta = 0.5f * acosf(cosf(a) / (1.0f - cosf(a)));
|
const float theta = 0.5f * acosf(cosf(a) / (1.0f - cosf(a)));
|
||||||
_min_sample_dist = GRAVITY_MSS * 2*sinf(theta/2);
|
_min_sample_dist = GRAVITY_MSS * 2*sinf(theta/2);
|
||||||
|
|
||||||
_param_struct.offset = offset;
|
_param.s.offset = offset;
|
||||||
_param_struct.diag = diag;
|
_param.s.diag = diag;
|
||||||
_param_struct.offdiag = offdiag;
|
_param.s.offdiag = offdiag;
|
||||||
|
|
||||||
switch (_conf_fit_type) {
|
switch (_conf_fit_type) {
|
||||||
case ACCEL_CAL_AXIS_ALIGNED_ELLIPSOID:
|
case ACCEL_CAL_AXIS_ALIGNED_ELLIPSOID:
|
||||||
@ -153,12 +152,12 @@ void AccelCalibrator::new_sample(const Vector3f& delta_velocity, float dt) {
|
|||||||
|
|
||||||
// determines if the result is acceptable
|
// determines if the result is acceptable
|
||||||
bool AccelCalibrator::accept_result() const {
|
bool AccelCalibrator::accept_result() const {
|
||||||
if (fabsf(_param_struct.offset.x) > GRAVITY_MSS ||
|
if (fabsf(_param.s.offset.x) > GRAVITY_MSS ||
|
||||||
fabsf(_param_struct.offset.y) > GRAVITY_MSS ||
|
fabsf(_param.s.offset.y) > GRAVITY_MSS ||
|
||||||
fabsf(_param_struct.offset.z) > GRAVITY_MSS ||
|
fabsf(_param.s.offset.z) > GRAVITY_MSS ||
|
||||||
_param_struct.diag.x < 0.8f || _param_struct.diag.x > 1.2f ||
|
_param.s.diag.x < 0.8f || _param.s.diag.x > 1.2f ||
|
||||||
_param_struct.diag.y < 0.8f || _param_struct.diag.y > 1.2f ||
|
_param.s.diag.y < 0.8f || _param.s.diag.y > 1.2f ||
|
||||||
_param_struct.diag.z < 0.8f || _param_struct.diag.z > 1.2f) {
|
_param.s.diag.z < 0.8f || _param.s.diag.z > 1.2f) {
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
return true;
|
return true;
|
||||||
@ -183,12 +182,12 @@ bool AccelCalibrator::get_sample_corrected(uint8_t i, Vector3f& s) const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Matrix3f M (
|
Matrix3f M (
|
||||||
_param_struct.diag.x , _param_struct.offdiag.x , _param_struct.offdiag.y,
|
_param.s.diag.x , _param.s.offdiag.x , _param.s.offdiag.y,
|
||||||
_param_struct.offdiag.x , _param_struct.diag.y , _param_struct.offdiag.z,
|
_param.s.offdiag.x , _param.s.diag.y , _param.s.offdiag.z,
|
||||||
_param_struct.offdiag.y , _param_struct.offdiag.z , _param_struct.diag.z
|
_param.s.offdiag.y , _param.s.offdiag.z , _param.s.diag.z
|
||||||
);
|
);
|
||||||
|
|
||||||
s = M*(s+_param_struct.offset);
|
s = M*(s+_param.s.offset);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -203,20 +202,20 @@ void AccelCalibrator::check_for_timeout() {
|
|||||||
|
|
||||||
// returns spherical fit paramteters
|
// returns spherical fit paramteters
|
||||||
void AccelCalibrator::get_calibration(Vector3f& offset) const {
|
void AccelCalibrator::get_calibration(Vector3f& offset) const {
|
||||||
offset = -_param_struct.offset;
|
offset = -_param.s.offset;
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns axis aligned ellipsoidal fit parameters
|
// returns axis aligned ellipsoidal fit parameters
|
||||||
void AccelCalibrator::get_calibration(Vector3f& offset, Vector3f& diag) const {
|
void AccelCalibrator::get_calibration(Vector3f& offset, Vector3f& diag) const {
|
||||||
offset = -_param_struct.offset;
|
offset = -_param.s.offset;
|
||||||
diag = _param_struct.diag;
|
diag = _param.s.diag;
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns generic ellipsoidal fit parameters
|
// returns generic ellipsoidal fit parameters
|
||||||
void AccelCalibrator::get_calibration(Vector3f& offset, Vector3f& diag, Vector3f& offdiag) const {
|
void AccelCalibrator::get_calibration(Vector3f& offset, Vector3f& diag, Vector3f& offdiag) const {
|
||||||
offset = -_param_struct.offset;
|
offset = -_param.s.offset;
|
||||||
diag = _param_struct.diag;
|
diag = _param.s.diag;
|
||||||
offdiag = _param_struct.offdiag;
|
offdiag = _param.s.offdiag;
|
||||||
}
|
}
|
||||||
|
|
||||||
/////////////////////////////////////////////////////////////
|
/////////////////////////////////////////////////////////////
|
||||||
@ -330,10 +329,9 @@ void AccelCalibrator::run_fit(uint8_t max_iterations, float& fitness)
|
|||||||
if (_sample_buffer == NULL) {
|
if (_sample_buffer == NULL) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
fitness = calc_mean_squared_residuals(_param_struct);
|
fitness = calc_mean_squared_residuals(_param.s);
|
||||||
float min_fitness = fitness;
|
float min_fitness = fitness;
|
||||||
VectorP param_array = _param_array;
|
union param_u fit_param = _param;
|
||||||
struct param_t &fit_param(*reinterpret_cast<struct param_t *>(¶m_array));
|
|
||||||
uint8_t num_iterations = 0;
|
uint8_t num_iterations = 0;
|
||||||
|
|
||||||
while(num_iterations < max_iterations) {
|
while(num_iterations < max_iterations) {
|
||||||
@ -348,7 +346,7 @@ void AccelCalibrator::run_fit(uint8_t max_iterations, float& fitness)
|
|||||||
|
|
||||||
VectorN<float,ACCEL_CAL_MAX_NUM_PARAMS> jacob;
|
VectorN<float,ACCEL_CAL_MAX_NUM_PARAMS> jacob;
|
||||||
|
|
||||||
calc_jacob(sample, fit_param, jacob);
|
calc_jacob(sample, fit_param.s, jacob);
|
||||||
|
|
||||||
for(uint8_t i = 0; i < get_num_params(); i++) {
|
for(uint8_t i = 0; i < get_num_params(); i++) {
|
||||||
// compute JTJ
|
// compute JTJ
|
||||||
@ -356,7 +354,7 @@ void AccelCalibrator::run_fit(uint8_t max_iterations, float& fitness)
|
|||||||
JTJ[i*get_num_params()+j] += jacob[i] * jacob[j];
|
JTJ[i*get_num_params()+j] += jacob[i] * jacob[j];
|
||||||
}
|
}
|
||||||
// compute JTFI
|
// compute JTFI
|
||||||
JTFI[i] += jacob[i] * calc_residual(sample, fit_param);
|
JTFI[i] += jacob[i] * calc_residual(sample, fit_param.s);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -366,11 +364,11 @@ void AccelCalibrator::run_fit(uint8_t max_iterations, float& fitness)
|
|||||||
|
|
||||||
for(uint8_t row=0; row < get_num_params(); row++) {
|
for(uint8_t row=0; row < get_num_params(); row++) {
|
||||||
for(uint8_t col=0; col < get_num_params(); col++) {
|
for(uint8_t col=0; col < get_num_params(); col++) {
|
||||||
param_array[row] -= JTFI[col] * JTJ[row*get_num_params()+col];
|
fit_param.a[row] -= JTFI[col] * JTJ[row*get_num_params()+col];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fitness = calc_mean_squared_residuals(fit_param);
|
fitness = calc_mean_squared_residuals(fit_param.s);
|
||||||
|
|
||||||
if (isnan(fitness) || isinf(fitness)) {
|
if (isnan(fitness) || isinf(fitness)) {
|
||||||
return;
|
return;
|
||||||
@ -378,7 +376,7 @@ void AccelCalibrator::run_fit(uint8_t max_iterations, float& fitness)
|
|||||||
|
|
||||||
if (fitness < min_fitness) {
|
if (fitness < min_fitness) {
|
||||||
min_fitness = fitness;
|
min_fitness = fitness;
|
||||||
_param_struct = fit_param;
|
_param = fit_param;
|
||||||
}
|
}
|
||||||
|
|
||||||
num_iterations++;
|
num_iterations++;
|
||||||
@ -403,7 +401,7 @@ float AccelCalibrator::calc_residual(const Vector3f& sample, const struct param_
|
|||||||
// converged to LSq estimator so far
|
// converged to LSq estimator so far
|
||||||
float AccelCalibrator::calc_mean_squared_residuals() const
|
float AccelCalibrator::calc_mean_squared_residuals() const
|
||||||
{
|
{
|
||||||
return calc_mean_squared_residuals(_param_struct);
|
return calc_mean_squared_residuals(_param.s);
|
||||||
}
|
}
|
||||||
|
|
||||||
// calculated the total mean squared fitness of all the collected samples using parameters
|
// calculated the total mean squared fitness of all the collected samples using parameters
|
||||||
|
@ -95,6 +95,17 @@ private:
|
|||||||
};
|
};
|
||||||
typedef VectorN<float, ACCEL_CAL_MAX_NUM_PARAMS> VectorP;
|
typedef VectorN<float, ACCEL_CAL_MAX_NUM_PARAMS> VectorP;
|
||||||
|
|
||||||
|
union param_u {
|
||||||
|
struct param_t s;
|
||||||
|
VectorN<float, ACCEL_CAL_MAX_NUM_PARAMS> a;
|
||||||
|
|
||||||
|
param_u() : a{}
|
||||||
|
{
|
||||||
|
static_assert(sizeof(*this) == sizeof(struct param_t),
|
||||||
|
"Invalid union members: sizes do not match");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
//configuration
|
//configuration
|
||||||
uint8_t _conf_num_samples;
|
uint8_t _conf_num_samples;
|
||||||
float _conf_sample_time;
|
float _conf_sample_time;
|
||||||
@ -105,8 +116,7 @@ private:
|
|||||||
accel_cal_status_t _status;
|
accel_cal_status_t _status;
|
||||||
struct AccelSample* _sample_buffer;
|
struct AccelSample* _sample_buffer;
|
||||||
uint8_t _samples_collected;
|
uint8_t _samples_collected;
|
||||||
struct param_t &_param_struct;
|
union param_u _param;
|
||||||
VectorP _param_array;
|
|
||||||
float _fitness;
|
float _fitness;
|
||||||
uint32_t _last_samp_frag_collected_ms;
|
uint32_t _last_samp_frag_collected_ms;
|
||||||
float _min_sample_dist;
|
float _min_sample_dist;
|
||||||
|
Loading…
Reference in New Issue
Block a user