/**************************************************************************** * * Copyright (c) 2020-2021 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 "FakeImu.hpp" using namespace time_literals; FakeImu::FakeImu() : ModuleParams(nullptr), ScheduledWorkItem(MODULE_NAME, px4::wq_configurations::hp_default), _px4_accel(1310988), // 1310988: DRV_IMU_DEVTYPE_SIM, BUS: 1, ADDR: 1, TYPE: SIMULATION _px4_gyro(1310988) // 1310988: DRV_IMU_DEVTYPE_SIM, BUS: 1, ADDR: 1, TYPE: SIMULATION { _sensor_interval_us = roundf(1.e6f / _px4_gyro.get_max_rate_hz()); PX4_INFO("Rate %.3f, Interval: %d us", (double)_px4_gyro.get_max_rate_hz(), _sensor_interval_us); _px4_accel.set_range(2000.f); // don't care _px4_gyro.set_scale(math::radians(2000.f) / static_cast(INT16_MAX - 1)); // 2000 degrees/second max } bool FakeImu::init() { ScheduleOnInterval(_sensor_interval_us); return true; } void FakeImu::Run() { if (should_exit()) { ScheduleClear(); exit_and_cleanup(); return; } sensor_gyro_fifo_s gyro{}; gyro.timestamp_sample = hrt_absolute_time(); gyro.samples = roundf(IMU_RATE_HZ / (1e6 / _sensor_interval_us)); gyro.dt = 1e6 / IMU_RATE_HZ; const double dt_s = 1 / IMU_RATE_HZ; const double x_f0 = 0.0; // 0 Hz X frequency start const double x_f1 = 10.0; // 10 Hz X frequency stop const double y_f0 = 0.0; // 10 Hz Y frequency start const double y_f1 = 100.0; // 1000 Hz Y frequency stop const double z_f0 = 0.0; // 100 Hz Z frequency start const double z_f1 = 1000.0; // 1000 Hz Z frequency stop // amplitude static constexpr double A = (INT16_MAX - 1); if (_time_start_us == 0) { _time_start_us = gyro.timestamp_sample; } // 10 second sweep const double T = 10.0; const double timestamp_sample_s = static_cast(gyro.timestamp_sample - _time_start_us) / 1e6; float x_freq = 0; float y_freq = 0; float z_freq = 0; for (int n = 0; n < gyro.samples; n++) { // timestamp_sample corresponds to last sample const double t = timestamp_sample_s - (gyro.samples - n - 1) * dt_s; // linear-frequency chirp, see https://en.wikipedia.org/wiki/Chirp const double x_F = x_f0 + (x_f1 - x_f0) * t / (2 * T); const double y_F = y_f0 + (y_f1 - y_f0) * t / (2 * T); const double z_F = z_f0 + (z_f1 - z_f0) * t / (2 * T); gyro.x[n] = roundf(A * sin(2 * M_PI * x_F * t)); gyro.y[n] = roundf(A * sin(2 * M_PI * y_F * t)); gyro.z[n] = roundf(A * sin(2 * M_PI * z_F * t)); if (n == 0) { x_freq = (x_f1 - x_f0) * (t / T) + x_f0; y_freq = (y_f1 - y_f0) * (t / T) + y_f0; z_freq = (z_f1 - z_f0) * (t / T) + z_f0; _px4_accel.update(gyro.timestamp_sample, x_freq, y_freq, z_freq); } } _px4_gyro.updateFIFO(gyro); #if defined(FAKE_IMU_FAKE_ESC_STATUS) // publish fake esc status at ~10 Hz if (hrt_elapsed_time(&_esc_status_pub.get().timestamp) > 100_ms) { auto &esc_status = _esc_status_pub.get(); esc_status.esc_count = 3; // ESC 0 follow X axis RPM if (!(timestamp_sample_s > 1.5 && timestamp_sample_s < 2.0)) { // simulate drop out at 1.5 to 2 seconds esc_status.esc[0].timestamp = hrt_absolute_time(); esc_status.esc[0].esc_rpm = x_freq * 60; } // ESC 1 follow Y axis RPM if (!(timestamp_sample_s > 2.5 && timestamp_sample_s < 3.0)) { // simulate drop out at 2.5 to 3 seconds esc_status.esc[1].timestamp = hrt_absolute_time(); esc_status.esc[1].esc_rpm = y_freq * 60; } // ESC 2 follow Z axis RPM if (!(timestamp_sample_s > 3.5 && timestamp_sample_s < 4.0)) { // simulate drop out at 3.5 to 4 seconds esc_status.esc[2].timestamp = hrt_absolute_time(); esc_status.esc[2].esc_rpm = z_freq * 60; } // simulate brief dropout of all data if (timestamp_sample_s > 5.5 && timestamp_sample_s < 5.6) { esc_status.esc[0].esc_rpm = 0; esc_status.esc[1].esc_rpm = 0; esc_status.esc[2].esc_rpm = 0; } esc_status.timestamp = hrt_absolute_time(); _esc_status_pub.update(); } #endif // FAKE_IMU_FAKE_ESC_STATUS } int FakeImu::task_spawn(int argc, char *argv[]) { FakeImu *instance = new FakeImu(); 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 FakeImu::custom_command(int argc, char *argv[]) { return print_usage("unknown command"); } int FakeImu::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("fake_imu", "driver"); PRINT_MODULE_USAGE_COMMAND("start"); PRINT_MODULE_USAGE_DEFAULT_COMMANDS(); return 0; } extern "C" __EXPORT int fake_imu_main(int argc, char *argv[]) { return FakeImu::main(argc, argv); }