/**************************************************************************** * * Copyright (c) 2020-2022 PX4 Development Team. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * 3. Neither the name PX4 nor the names of its contributors may be * used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. * ****************************************************************************/ #include "GyroFFT.hpp" #include #include #include using namespace matrix; GyroFFT::GyroFFT() : ModuleParams(nullptr), ScheduledWorkItem(MODULE_NAME, px4::wq_configurations::hp_default) { for (int i = 0; i < MAX_NUM_PEAKS; i++) { _sensor_gyro_fft.peak_frequencies_x[i] = NAN; _sensor_gyro_fft.peak_frequencies_y[i] = NAN; _sensor_gyro_fft.peak_frequencies_z[i] = NAN; } _sensor_gyro_fft_pub.advertise(); } GyroFFT::~GyroFFT() { perf_free(_cycle_perf); perf_free(_cycle_interval_perf); perf_free(_fft_perf); perf_free(_gyro_generation_gap_perf); perf_free(_gyro_fifo_generation_gap_perf); delete[] _gyro_data_buffer_x; delete[] _gyro_data_buffer_y; delete[] _gyro_data_buffer_z; delete[] _hanning_window; delete[] _fft_input_buffer; delete[] _fft_outupt_buffer; delete[] _peak_magnitudes_all; } bool GyroFFT::init() { bool buffers_allocated = false; // arm_rfft_init_q15(&_rfft_q15, _imu_gyro_fft_len, 0, 1) manually inlined to save flash _rfft_q15.pTwiddleAReal = (q15_t *) realCoefAQ15; _rfft_q15.pTwiddleBReal = (q15_t *) realCoefBQ15; _rfft_q15.ifftFlagR = 0; _rfft_q15.bitReverseFlagR = 1; switch (_param_imu_gyro_fft_len.get()) { // case 128: // buffers_allocated = AllocateBuffers<128>(); // _rfft_q15.fftLenReal = 128; // _rfft_q15.twidCoefRModifier = 64U; // _rfft_q15.pCfft = &arm_cfft_sR_q15_len64; // break; case 256: buffers_allocated = AllocateBuffers<256>(); _rfft_q15.fftLenReal = 256; _rfft_q15.twidCoefRModifier = 32U; _rfft_q15.pCfft = &arm_cfft_sR_q15_len128; break; case 512: buffers_allocated = AllocateBuffers<512>(); _rfft_q15.fftLenReal = 512; _rfft_q15.twidCoefRModifier = 16U; _rfft_q15.pCfft = &arm_cfft_sR_q15_len256; break; case 1024: buffers_allocated = AllocateBuffers<1024>(); _rfft_q15.fftLenReal = 1024; _rfft_q15.twidCoefRModifier = 8U; _rfft_q15.pCfft = &arm_cfft_sR_q15_len512; break; // case 2048: // buffers_allocated = AllocateBuffers<2048>(); // _rfft_q15.fftLenReal = 2048; // _rfft_q15.twidCoefRModifier = 4U; // _rfft_q15.pCfft = &arm_cfft_sR_q15_len1024; // break; case 4096: buffers_allocated = AllocateBuffers<4096>(); _rfft_q15.fftLenReal = 4096; _rfft_q15.twidCoefRModifier = 2U; _rfft_q15.pCfft = &arm_cfft_sR_q15_len2048; break; // case 8192: // buffers_allocated = AllocateBuffers<8192>(); // _rfft_q15.fftLenReal = 8192; // _rfft_q15.twidCoefRModifier = 1U; // _rfft_q15.pCfft = &arm_cfft_sR_q15_len4096; // break; default: // otherwise default to 256 PX4_ERR("Invalid IMU_GYRO_FFT_LEN=%" PRId32 ", resetting", _param_imu_gyro_fft_len.get()); buffers_allocated = AllocateBuffers<256>(); _param_imu_gyro_fft_len.set(256); _param_imu_gyro_fft_len.commit(); break; } if (buffers_allocated) { _imu_gyro_fft_len = _param_imu_gyro_fft_len.get(); // init Hanning window for (int n = 0; n < _imu_gyro_fft_len; n++) { const float hanning_value = 0.5f * (1.f - cosf(2.f * M_PI_F * n / (_imu_gyro_fft_len - 1))); arm_float_to_q15(&hanning_value, &_hanning_window[n], 1); } if (!SensorSelectionUpdate(true)) { ScheduleDelayed(500_ms); } return true; } PX4_ERR("failed to allocate buffers"); delete[] _gyro_data_buffer_x; delete[] _gyro_data_buffer_y; delete[] _gyro_data_buffer_z; delete[] _hanning_window; delete[] _fft_input_buffer; delete[] _fft_outupt_buffer; return false; } bool GyroFFT::SensorSelectionUpdate(bool force) { if (_sensor_selection_sub.updated() || (_selected_sensor_device_id == 0) || force) { sensor_selection_s sensor_selection{}; _sensor_selection_sub.copy(&sensor_selection); if ((sensor_selection.gyro_device_id != 0) && (_selected_sensor_device_id != sensor_selection.gyro_device_id)) { // prefer sensor_gyro_fifo if available for (uint8_t i = 0; i < MAX_SENSOR_COUNT; i++) { uORB::SubscriptionData sensor_gyro_fifo_sub{ORB_ID(sensor_gyro_fifo), i}; if (sensor_gyro_fifo_sub.get().device_id == sensor_selection.gyro_device_id) { if (_sensor_gyro_fifo_sub.ChangeInstance(i) && _sensor_gyro_fifo_sub.registerCallback()) { _sensor_gyro_sub.unregisterCallback(); _sensor_gyro_fifo_sub.set_required_updates(sensor_gyro_fifo_s::ORB_QUEUE_LENGTH / 2); _selected_sensor_device_id = sensor_selection.gyro_device_id; _gyro_fifo = true; if (_gyro_fifo_generation_gap_perf == nullptr) { _gyro_fifo_generation_gap_perf = perf_alloc(PC_COUNT, MODULE_NAME": gyro FIFO data gap"); } return true; } } } // otherwise use sensor_gyro for (uint8_t i = 0; i < MAX_SENSOR_COUNT; i++) { uORB::SubscriptionData sensor_gyro_sub{ORB_ID(sensor_gyro), i}; if (sensor_gyro_sub.get().device_id == sensor_selection.gyro_device_id) { if (_sensor_gyro_sub.ChangeInstance(i) && _sensor_gyro_sub.registerCallback()) { _sensor_gyro_fifo_sub.unregisterCallback(); _sensor_gyro_sub.set_required_updates(sensor_gyro_s::ORB_QUEUE_LENGTH / 2); _selected_sensor_device_id = sensor_selection.gyro_device_id; _gyro_fifo = false; if (_gyro_generation_gap_perf == nullptr) { _gyro_generation_gap_perf = perf_alloc(PC_COUNT, MODULE_NAME": gyro data gap"); } return true; } } } PX4_ERR("unable to find or subscribe to selected sensor (%" PRIu32 ")", sensor_selection.gyro_device_id); } } return false; } void GyroFFT::VehicleIMUStatusUpdate(bool force) { if (_vehicle_imu_status_sub.updated() || force) { vehicle_imu_status_s vehicle_imu_status; if (_vehicle_imu_status_sub.copy(&vehicle_imu_status)) { // find corresponding vehicle_imu_status instance if the device_id doesn't match if (vehicle_imu_status.gyro_device_id != _selected_sensor_device_id) { for (uint8_t imu_status = 0; imu_status < MAX_SENSOR_COUNT; imu_status++) { uORB::Subscription imu_status_sub{ORB_ID(vehicle_imu_status), imu_status}; if (imu_status_sub.copy(&vehicle_imu_status)) { if (vehicle_imu_status.gyro_device_id == _selected_sensor_device_id) { _vehicle_imu_status_sub.ChangeInstance(imu_status); break; } } } } // update gyro sample rate if ((vehicle_imu_status.gyro_device_id == _selected_sensor_device_id) && (vehicle_imu_status.gyro_rate_hz > 0)) { if (_gyro_fifo) { _gyro_sample_rate_hz = vehicle_imu_status.gyro_raw_rate_hz; } else { _gyro_sample_rate_hz = vehicle_imu_status.gyro_rate_hz; } return; } } } } // helper function used for frequency estimation static inline float tau(float x) { // tau(x) = 1/4 * log(3x^2 + 6x + 1) – sqrt(6)/24 * log((x + 1 – sqrt(2/3)) / (x + 1 + sqrt(2/3))) float p1 = logf(3.f * powf(x, 2.f) + 6.f * x + 1.f); float part1 = x + 1.f - sqrtf(2.f / 3.f); float part2 = x + 1.f + sqrtf(2.f / 3.f); float p2 = logf(part1 / part2); return (0.25f * p1 - sqrtf(6.f) / 24.f * p2); } float GyroFFT::EstimatePeakFrequencyBin(q15_t fft[], int peak_index) { if (peak_index >= 2) { // find peak location using Quinn's Second Estimator (2020-06-14: http://dspguru.com/dsp/howtos/how-to-interpolate-fft-peak/) float real[3] { (float)fft[peak_index - 2], (float)fft[peak_index], (float)fft[peak_index + 2] }; float imag[3] { (float)fft[peak_index - 2 + 1], (float)fft[peak_index + 1], (float)fft[peak_index + 2 + 1] }; static constexpr int k = 1; const float divider = (real[k] * real[k] + imag[k] * imag[k]); // ap = (X[k + 1].r * X[k].r + X[k+1].i * X[k].i) / (X[k].r * X[k].r + X[k].i * X[k].i) float ap = (real[k + 1] * real[k] + imag[k + 1] * imag[k]) / divider; // dp = -ap / (1 – ap) float dp = -ap / (1.f - ap); // am = (X[k - 1].r * X[k].r + X[k – 1].i * X[k].i) / (X[k].r * X[k].r + X[k].i * X[k].i) float am = (real[k - 1] * real[k] + imag[k - 1] * imag[k]) / divider; // dm = am / (1 – am) float dm = am / (1.f - am); // d = (dp + dm) / 2 + tau(dp * dp) – tau(dm * dm) float d = (dp + dm) / 2.f + tau(dp * dp) - tau(dm * dm); // k’ = k + d return peak_index + 2.f * d; } return NAN; } void GyroFFT::Run() { if (should_exit()) { _sensor_gyro_sub.unregisterCallback(); _sensor_gyro_fifo_sub.unregisterCallback(); exit_and_cleanup(); return; } // backup schedule ScheduleDelayed(500_ms); perf_begin(_cycle_perf); perf_count(_cycle_interval_perf); // Check if parameters have changed if (_parameter_update_sub.updated()) { // clear update parameter_update_s param_update; _parameter_update_sub.copy(¶m_update); updateParams(); } const bool selection_updated = SensorSelectionUpdate(); VehicleIMUStatusUpdate(selection_updated); // reset _fft_updated = false; if (_gyro_fifo) { // run on sensor gyro fifo updates sensor_gyro_fifo_s sensor_gyro_fifo; while (_sensor_gyro_fifo_sub.update(&sensor_gyro_fifo)) { if (_sensor_gyro_fifo_sub.get_last_generation() != _gyro_last_generation + 1) { // force reset if we've missed a sample _fft_buffer_index[0] = 0; _fft_buffer_index[1] = 0; _fft_buffer_index[2] = 0; perf_count(_gyro_fifo_generation_gap_perf); } _gyro_last_generation = _sensor_gyro_fifo_sub.get_last_generation(); if (fabsf(sensor_gyro_fifo.scale - _fifo_last_scale) > FLT_EPSILON) { // force reset if scale has changed _fft_buffer_index[0] = 0; _fft_buffer_index[1] = 0; _fft_buffer_index[2] = 0; _fifo_last_scale = sensor_gyro_fifo.scale; } int16_t *input[] {sensor_gyro_fifo.x, sensor_gyro_fifo.y, sensor_gyro_fifo.z}; Update(sensor_gyro_fifo.timestamp_sample, input, sensor_gyro_fifo.samples); } } else { // run on sensor gyro fifo updates sensor_gyro_s sensor_gyro; while (_sensor_gyro_sub.update(&sensor_gyro)) { if (_sensor_gyro_sub.get_last_generation() != _gyro_last_generation + 1) { // force reset if we've missed a sample _fft_buffer_index[0] = 0; _fft_buffer_index[1] = 0; _fft_buffer_index[2] = 0; perf_count(_gyro_generation_gap_perf); } _gyro_last_generation = _sensor_gyro_sub.get_last_generation(); const float gyro_scale = math::radians(1000.f); // arbitrary scaling float32 rad/s -> raw int16 int16_t gyro_x[1] {(int16_t)roundf(sensor_gyro.x * gyro_scale)}; int16_t gyro_y[1] {(int16_t)roundf(sensor_gyro.y * gyro_scale)}; int16_t gyro_z[1] {(int16_t)roundf(sensor_gyro.z * gyro_scale)}; int16_t *input[] {gyro_x, gyro_y, gyro_z}; Update(sensor_gyro.timestamp_sample, input, 1); } } if (_publish) { Publish(); _publish = false; } perf_end(_cycle_perf); } void GyroFFT::Update(const hrt_abstime ×tamp_sample, int16_t *input[], uint8_t N) { q15_t *gyro_data_buffer[] {_gyro_data_buffer_x, _gyro_data_buffer_y, _gyro_data_buffer_z}; for (int axis = 0; axis < 3; axis++) { int &buffer_index = _fft_buffer_index[axis]; for (int n = 0; n < N; n++) { if (buffer_index < _imu_gyro_fft_len) { // convert int16_t -> q15_t (scaling isn't relevant) gyro_data_buffer[axis][buffer_index] = input[axis][n] / 2; buffer_index++; } // if we have enough samples begin processing, but only one FFT per cycle if ((buffer_index >= _imu_gyro_fft_len) && !_fft_updated) { perf_begin(_fft_perf); arm_mult_q15(gyro_data_buffer[axis], _hanning_window, _fft_input_buffer, _imu_gyro_fft_len); arm_rfft_q15(&_rfft_q15, _fft_input_buffer, _fft_outupt_buffer); _fft_updated = true; FindPeaks(timestamp_sample, axis, _fft_outupt_buffer); // reset // shift buffer (3/4 overlap) const int overlap_start = _imu_gyro_fft_len / 4; memmove(&gyro_data_buffer[axis][0], &gyro_data_buffer[axis][overlap_start], sizeof(q15_t) * overlap_start * 3); buffer_index = overlap_start * 3; perf_end(_fft_perf); } } } } void GyroFFT::FindPeaks(const hrt_abstime ×tamp_sample, int axis, q15_t *fft_outupt_buffer) { const float resolution_hz = _gyro_sample_rate_hz / _imu_gyro_fft_len; // sum total energy across all used buckets for SNR float bin_mag_sum = 0; // FFT output buffer is ordered [real[0], imag[0], real[1], imag[1], real[2], imag[2] ... real[(N/2)-1], imag[(N/2)-1] for (uint16_t fft_index = 2; fft_index < _imu_gyro_fft_len; fft_index += 2) { const float real = fft_outupt_buffer[fft_index]; const float imag = fft_outupt_buffer[fft_index + 1]; const float fft_magnitude = sqrtf(real * real + imag * imag); int bin_index = fft_index / 2; _peak_magnitudes_all[bin_index] = fft_magnitude; bin_mag_sum += fft_magnitude; } // find raw peaks uint16_t raw_peak_index[MAX_NUM_PEAKS] {}; float peak_magnitude[MAX_NUM_PEAKS] {}; for (int i = 0; i < MAX_NUM_PEAKS; i++) { float largest_peak = 0; int largest_peak_index = 0; for (int bin_index = 1; bin_index < _imu_gyro_fft_len; bin_index++) { const float freq_hz = bin_index * resolution_hz; if ((_peak_magnitudes_all[bin_index] > largest_peak) && (freq_hz >= _param_imu_gyro_fft_min.get()) && (freq_hz <= _param_imu_gyro_fft_max.get())) { largest_peak = _peak_magnitudes_all[bin_index]; largest_peak_index = bin_index; } } if (largest_peak_index > 1) { raw_peak_index[i] = largest_peak_index; peak_magnitude[i] = _peak_magnitudes_all[largest_peak_index]; // remove peak + sides (included in frequency estimate later) _peak_magnitudes_all[largest_peak_index - 1] = 0; _peak_magnitudes_all[largest_peak_index] = 0; _peak_magnitudes_all[largest_peak_index + 1] = 0; } } // keep if peak has been previously seen and SNR > MIN_SNR // or // peak has SNR > MIN_SNR_INITIAL static constexpr float MIN_SNR = 1.f; // TODO: configurable? int num_peaks_found = 0; float peak_frequencies[MAX_NUM_PEAKS] {}; float peak_snr[MAX_NUM_PEAKS] {}; float *peak_frequencies_publish[] { _sensor_gyro_fft.peak_frequencies_x, _sensor_gyro_fft.peak_frequencies_y, _sensor_gyro_fft.peak_frequencies_z }; float peak_frequencies_prev[MAX_NUM_PEAKS]; for (int i = 0; i < MAX_NUM_PEAKS; i++) { peak_frequencies_prev[i] = peak_frequencies_publish[axis][i]; } for (int peak_new = 0; peak_new < MAX_NUM_PEAKS; peak_new++) { if (raw_peak_index[peak_new] > 0) { const float adjusted_bin = 0.5f * EstimatePeakFrequencyBin(fft_outupt_buffer, 2 * raw_peak_index[peak_new]); if (PX4_ISFINITE(adjusted_bin)) { const float freq_adjusted = resolution_hz * adjusted_bin; const float snr = 10.f * log10f((_imu_gyro_fft_len - 1) * peak_magnitude[peak_new] / (bin_mag_sum - peak_magnitude[peak_new])); if (PX4_ISFINITE(freq_adjusted) && (snr > MIN_SNR) && (freq_adjusted >= _param_imu_gyro_fft_min.get()) && (freq_adjusted <= _param_imu_gyro_fft_max.get())) { // only keep if we're already tracking this frequency or if the SNR is significant for (int peak_prev = 0; peak_prev < MAX_NUM_PEAKS; peak_prev++) { bool snr_acceptable = (snr > _param_imu_gyro_fft_snr.get()); bool peak_close = (fabsf(freq_adjusted - peak_frequencies_prev[peak_prev]) < (resolution_hz * 0.25f)); if (snr_acceptable || peak_close) { // keep peak_frequencies[num_peaks_found] = freq_adjusted; peak_snr[num_peaks_found] = snr; // remove if (peak_close) { peak_frequencies_prev[peak_prev] = NAN; } num_peaks_found++; break; } } } } } } if (num_peaks_found > 0) { UpdateOutput(timestamp_sample, axis, peak_frequencies, peak_snr, num_peaks_found); } } void GyroFFT::UpdateOutput(const hrt_abstime ×tamp_sample, int axis, float peak_frequencies[MAX_NUM_PEAKS], float peak_snr[MAX_NUM_PEAKS], int num_peaks_found) { float *peak_frequencies_publish[] { _sensor_gyro_fft.peak_frequencies_x, _sensor_gyro_fft.peak_frequencies_y, _sensor_gyro_fft.peak_frequencies_z }; float *peak_snr_publish[] { _sensor_gyro_fft.peak_snr_x, _sensor_gyro_fft.peak_snr_y, _sensor_gyro_fft.peak_snr_z }; // new peak: r, old peak: c float peak_frequencies_diff[MAX_NUM_PEAKS][MAX_NUM_PEAKS]; for (int peak_new = 0; peak_new < MAX_NUM_PEAKS; peak_new++) { // compute distance to previous peaks for (int peak_prev = 0; peak_prev < MAX_NUM_PEAKS; peak_prev++) { if ((peak_frequencies[peak_new] > 0) && (peak_frequencies_publish[axis][peak_prev] > 0) && PX4_ISFINITE(peak_frequencies_publish[axis][peak_prev]) ) { peak_frequencies_diff[peak_new][peak_prev] = fabsf(peak_frequencies[peak_new] - peak_frequencies_publish[axis][peak_prev]); } else { peak_frequencies_diff[peak_new][peak_prev] = INFINITY; } } } // go through peak_frequencies_diff and find smallest diff (closest peaks) // - copy new peak to old peak slot // - exclude new peak (row) and old peak (column) in search // - repeat // // - finally copy unmatched peaks to empty slots bool peak_new_copied[MAX_NUM_PEAKS] {}; bool peak_out_filled[MAX_NUM_PEAKS] {}; int peaks_copied = 0; for (int new_peak = 0; new_peak < num_peaks_found; new_peak++) { float smallest_diff = INFINITY; int closest_new_peak = -1; int closest_prev_peak = -1; // find new peak with smallest difference to old peak for (int peak_new = 0; peak_new < num_peaks_found; peak_new++) { for (int peak_prev = 0; peak_prev < MAX_NUM_PEAKS; peak_prev++) { if (!peak_new_copied[peak_new] && !peak_out_filled[peak_prev] && (peak_frequencies_diff[peak_new][peak_prev] < smallest_diff)) { smallest_diff = peak_frequencies_diff[peak_new][peak_prev]; closest_new_peak = peak_new; closest_prev_peak = peak_prev; } } } if (PX4_ISFINITE(smallest_diff) && (smallest_diff > 0)) { // smallest diff found, copy newly found peak into same slot previously published float peak_frequency = _median_filter[axis][closest_prev_peak].apply(peak_frequencies[closest_new_peak]); if (peak_frequency > 0) { peak_frequencies_publish[axis][closest_prev_peak] = peak_frequency; peak_snr_publish[axis][closest_prev_peak] = peak_snr[closest_new_peak]; peaks_copied++; _last_update[axis][closest_prev_peak] = timestamp_sample; _sensor_gyro_fft.timestamp_sample = timestamp_sample; _publish = true; // clear peak_frequencies[closest_new_peak] = NAN; peak_frequencies_diff[closest_new_peak][closest_prev_peak] = NAN; peak_new_copied[closest_new_peak] = true; peak_out_filled[closest_prev_peak] = true; if (peaks_copied == num_peaks_found) { break; } } } } // clear any stale entries for (int peak_out = 0; peak_out < MAX_NUM_PEAKS; peak_out++) { if (timestamp_sample - _last_update[axis][peak_out] > 100_ms) { peak_frequencies_publish[axis][peak_out] = NAN; peak_snr_publish[axis][peak_out] = NAN; _last_update[axis][peak_out] = 0; } } // copy any remaining new (unmatched) peaks to overwrite old or empty slots if (peaks_copied != num_peaks_found) { for (int peak_new = 0; peak_new < num_peaks_found; peak_new++) { if (PX4_ISFINITE(peak_frequencies[peak_new]) && (peak_frequencies[peak_new] > 0)) { int oldest_slot = -1; hrt_abstime oldest = timestamp_sample; // find oldest slot and replace with new peak frequency for (int peak_prev = 0; peak_prev < MAX_NUM_PEAKS; peak_prev++) { if (_last_update[axis][peak_prev] < oldest) { oldest_slot = peak_prev; oldest = _last_update[axis][peak_prev]; } } if (oldest_slot >= 0) { // copy peak to output slot float peak_frequency = _median_filter[axis][oldest_slot].apply(peak_frequencies[peak_new]); if (peak_frequency > 0) { peak_frequencies_publish[axis][oldest_slot] = peak_frequency; peak_snr_publish[axis][oldest_slot] = peak_snr[peak_new]; _last_update[axis][oldest_slot] = timestamp_sample; _sensor_gyro_fft.timestamp_sample = timestamp_sample; _publish = true; } } } } } } void GyroFFT::Publish() { _sensor_gyro_fft.device_id = _selected_sensor_device_id; _sensor_gyro_fft.sensor_sample_rate_hz = _gyro_sample_rate_hz; _sensor_gyro_fft.resolution_hz = _gyro_sample_rate_hz / _imu_gyro_fft_len; _sensor_gyro_fft.timestamp = hrt_absolute_time(); _sensor_gyro_fft_pub.publish(_sensor_gyro_fft); } int GyroFFT::task_spawn(int argc, char *argv[]) { GyroFFT *instance = new GyroFFT(); if (instance) { _object.store(instance); _task_id = task_id_is_work_queue; if (instance->init()) { return PX4_OK; } } else { PX4_ERR("alloc failed"); } delete instance; _object.store(nullptr); _task_id = -1; return PX4_ERROR; } int GyroFFT::print_status() { PX4_INFO("gyro sample rate: %.3f Hz", (double)_gyro_sample_rate_hz); perf_print_counter(_cycle_perf); perf_print_counter(_cycle_interval_perf); perf_print_counter(_fft_perf); perf_print_counter(_gyro_generation_gap_perf); perf_print_counter(_gyro_fifo_generation_gap_perf); return 0; } int GyroFFT::custom_command(int argc, char *argv[]) { return print_usage("unknown command"); } int GyroFFT::print_usage(const char *reason) { if (reason) { PX4_WARN("%s\n", reason); } PRINT_MODULE_DESCRIPTION( R"DESCR_STR( ### Description )DESCR_STR"); PRINT_MODULE_USAGE_NAME("gyro_fft", "system"); PRINT_MODULE_USAGE_COMMAND("start"); PRINT_MODULE_USAGE_DEFAULT_COMMANDS(); return 0; } extern "C" __EXPORT int gyro_fft_main(int argc, char *argv[]) { return GyroFFT::main(argc, argv); }