/*
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
/*
control IMU heater
*/
#include
#include
#include
#include
#include
#include "AP_BoardConfig.h"
#if HAL_HAVE_IMU_HEATER
extern const AP_HAL::HAL& hal;
#ifndef HAL_HEATER_GPIO_ON
#define HAL_HEATER_GPIO_ON 1
#endif
void AP_BoardConfig::set_imu_temp(float current)
{
int8_t target = heater.imu_target_temperature.get();
// pass to HAL for Linux
hal.util->set_imu_target_temp((int8_t *)&heater.imu_target_temperature);
hal.util->set_imu_temp(current);
if (target == -1) {
// nothing to do, make sure heater is left off
#if defined(HAL_HEATER_GPIO_PIN)
hal.gpio->write(HAL_HEATER_GPIO_PIN, !HAL_HEATER_GPIO_ON);
#endif
#if defined(HAL_HEATER2_GPIO_PIN)
hal.gpio->write(HAL_HEATER2_GPIO_PIN, !HAL_HEATER_GPIO_ON);
#endif
return;
}
// limit to 65 degrees to prevent damage
target = constrain_int16(target, -1, 65);
// average over temperatures to remove noise
heater.count++;
heater.sum += current;
// update at 10Hz
uint32_t now = AP_HAL::millis();
if (now - heater.last_update_ms < 100) {
#if defined(HAL_HEATER_GPIO_PIN)
// output as duty cycle to local pin. Use a random sequence to
// prevent a periodic change to magnetic field
bool heater_on = (get_random16() < uint32_t(heater.output) * 0xFFFFU / 100U);
hal.gpio->write(HAL_HEATER_GPIO_PIN, heater_on?HAL_HEATER_GPIO_ON : !HAL_HEATER_GPIO_ON);
#if defined(HAL_HEATER2_GPIO_PIN)
hal.gpio->write(HAL_HEATER2_GPIO_PIN, heater_on?HAL_HEATER_GPIO_ON : !HAL_HEATER_GPIO_ON);
#endif
#endif
return;
}
float dt = (now - heater.last_update_ms) * 0.001;
dt = constrain_float(dt, 0, 0.5);
heater.last_update_ms = now;
heater.temperature = heater.sum / heater.count;
heater.sum = 0;
heater.count = 0;
if (target < 0) {
heater.output = 0;
} else {
heater.output = heater.pi_controller.update(heater.temperature, target, dt);
heater.output = constrain_float(heater.output, 0, 100);
}
#if HAL_LOGGING_ENABLED
if (now - heater.last_log_ms >= 1000) {
// @LoggerMessage: HEAT
// @Description: IMU Heater data
// @Field: TimeUS: Time since system startup
// @Field: Temp: Current IMU temperature
// @Field: Targ: Target IMU temperature
// @Field: P: Proportional portion of response
// @Field: I: Integral portion of response
// @Field: Out: Controller output to heating element
AP::logger().WriteStreaming("HEAT", "TimeUS,Temp,Targ,P,I,Out", "Qfbfff",
AP_HAL::micros64(),
heater.temperature, target,
heater.pi_controller.get_P(),
heater.pi_controller.get_I(),
heater.output);
heater.last_log_ms = now;
}
#endif // HAL_LOGGING_ENABLED
#if 0
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Heater: Out=%.1f Temp=%.1f",
double(heater.output),
double(avg));
#endif
#if HAL_WITH_IO_MCU
if (io_enabled()) {
AP_IOMCU *iomcu = AP::iomcu();
if (iomcu) {
// tell IOMCU to setup heater
iomcu->set_heater_duty_cycle(heater.output);
}
}
#endif
}
// getter for current temperature, return false if heater disabled
bool AP_BoardConfig::get_board_heater_temperature(float &temperature) const
{
if (heater.imu_target_temperature == -1) {
return false; // heater disabled
}
temperature = heater.temperature;
return true;
}
// getter for min arming temperature, return false if heater disabled or min check disabled
bool AP_BoardConfig::get_board_heater_arming_temperature(int8_t &temperature) const
{
if ((heater.imu_target_temperature == -1) || (heater.imu_arming_temperature_margin_low == 0)) {
return false; // heater or temperature check disabled
}
temperature = heater.imu_target_temperature - heater.imu_arming_temperature_margin_low;
return true;
}
#endif // HAL_HAVE_IMU_HEATER