ardupilot/libraries/AP_Generator/AP_Generator_RichenPower.cpp
Peter Barker c8a368d896 AP_Generator: update prearm checks, fill new GENERATOR_STATUS fields
AP_Generator_Richenpower: only update servo channel while disarmed

AP_Generator_Richenpower: move warning for servo output channel to prearm checks

AP_Generator: state is simply off if RPM is zero

AP_Generator: send runtime and seconds-until-maintenance in GENERATOR_STATUS

AP_Generator_RichenPower: correct runtime seconds/minutes position in packets

AP_Generator_RichenPower: correct and expand use of mode packet entry

AP_Generator_RichenPower: fail prearm check if generator not seen

AP_Generator_RichenPower: make prearm check for needing maintenance  warn-only

AP_Generator_RichenPower: rename runstate to pilot_desired_runstate

AP_Generator_RichenPower: tweak decoding of time-to-maintenance

AP_Generator: add hysteresis for warmup/cooldown in IDLE

AP_Generator: stop generator in case of vehicle crash

AP_Generator: generator must go through idle from run to stop

AP_Generator: emit statustext when generator configured but not present

AP_Generator: do not log if LOG_BITMASK is zero

AP_Generator: do not warn user if generator configured but not present

They won't be able to start the generator if we can't see it.

Don't update the runstate if no readings seen.

AP_Generator: do not send generator status if no readings seen
2020-08-05 16:26:19 +09:00

508 lines
16 KiB
C++

/*
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 <http://www.gnu.org/licenses/>.
*/
#include "AP_Generator_RichenPower.h"
#if GENERATOR_ENABLED
#include <AP_HAL/AP_HAL.h>
#include <AP_Logger/AP_Logger.h>
#include <AP_SerialManager/AP_SerialManager.h>
#include <GCS_MAVLink/GCS.h>
#include <AP_Vehicle/AP_Vehicle.h>
#include <AP_HAL/utility/sparse-endian.h>
extern const AP_HAL::HAL& hal;
// Generator constructor; only one generator is curently permitted
AP_Generator_RichenPower::AP_Generator_RichenPower()
{
if (_singleton) {
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
AP_HAL::panic("Too many richenpower generators");
#endif
return;
}
_singleton = this;
}
// init method; configure communications with the generator
void AP_Generator_RichenPower::init()
{
const AP_SerialManager &serial_manager = AP::serialmanager();
uart = serial_manager.find_serial(AP_SerialManager::SerialProtocol_Generator, 0);
if (uart != nullptr) {
const uint32_t baud = serial_manager.find_baudrate(AP_SerialManager::SerialProtocol_Generator, 0);
uart->begin(baud, 256, 256);
}
}
// find a RichenPower message in the buffer, starting at
// initial_offset. If dound, that message (or partial message) will
// be moved to the start of the buffer.
void AP_Generator_RichenPower::move_header_in_buffer(uint8_t initial_offset)
{
uint8_t header_offset;
for (header_offset=initial_offset; header_offset<body_length; header_offset++) {
if (u.parse_buffer[header_offset] == HEADER_MAGIC1) {
break;
}
}
if (header_offset != 0) {
// header was found, but not at index 0; move it back to start of array
memmove(u.parse_buffer, &u.parse_buffer[header_offset], body_length - header_offset);
body_length -= header_offset;
}
}
// read - read serial port, return true if a new reading has been found
bool AP_Generator_RichenPower::get_reading()
{
// Example of a packet from a H2 controller:
//AA 55 00 0A 00 00 00 00 00 04 1E B0 00 10 00 00 23 7A 23
//7A 11 1D 00 00 00 00 01 00 00 00 00 00 00 00 00 00 00 00
//00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
//00 00 00 00 00 00 00 00 00 1E BE 55 AA
//16 bit one data Hight + Low
// fill our buffer some more:
uint32_t nbytes = uart->read(&u.parse_buffer[body_length],
ARRAY_SIZE(u.parse_buffer)-body_length);
if (nbytes == 0) {
return false;
}
body_length += nbytes;
move_header_in_buffer(0);
// header byte 1 is correct.
if (body_length < ARRAY_SIZE(u.parse_buffer)) {
// need a full buffer to have a valid message...
return false;
}
if (u.packet.headermagic2 != HEADER_MAGIC2) {
move_header_in_buffer(1);
return false;
}
// check for the footer signature:
if (u.packet.footermagic1 != FOOTER_MAGIC1) {
move_header_in_buffer(1);
return false;
}
if (u.packet.footermagic2 != FOOTER_MAGIC2) {
move_header_in_buffer(1);
return false;
}
// calculate checksum....
uint16_t checksum = 0;
const uint8_t *checksum_buffer = &u.parse_buffer[2];
for (uint8_t i=0; i<5; i++) {
checksum += be16toh_ptr(&checksum_buffer[2*i]);
}
if (checksum != be16toh(u.packet.checksum)) {
move_header_in_buffer(1);
return false;
}
// check the version:
const uint16_t version = be16toh(u.packet.version);
const uint8_t major = version / 100;
const uint8_t minor = (version % 100) / 10;
const uint8_t point = version % 10;
if (!protocol_information_anounced) {
gcs().send_text(MAV_SEVERITY_INFO, "RichenPower: protocol %u.%u.%u", major, minor, point);
protocol_information_anounced = true;
}
last_reading.runtime = be16toh(u.packet.runtime_hours) * 3600 +
u.packet.runtime_minutes * 60 +
u.packet.runtime_seconds;
last_reading.seconds_until_maintenance = be16toh(u.packet.seconds_until_maintenance_high) * 65536 + be16toh(u.packet.seconds_until_maintenance_low);
last_reading.errors = be16toh(u.packet.errors);
last_reading.rpm = be16toh(u.packet.rpm);
last_reading.output_voltage = be16toh(u.packet.output_voltage) / 100.0f;
last_reading.output_current = be16toh(u.packet.output_current) / 100.0f;
last_reading.mode = (Mode)u.packet.mode;
last_reading_ms = AP_HAL::millis();
body_length = 0;
// update the time we started idling at:
if (last_reading.mode == Mode::IDLE) {
if (idle_state_start_ms == 0) {
idle_state_start_ms = last_reading_ms;
}
} else {
idle_state_start_ms = 0;
}
return true;
}
// update the synthetic heat measurement we are keeping for the
// generator. We keep a synthetic heat measurement as the telemetry
// from the unit does not include the motor temperature, so we
// approximate one by assuming it produces heat in proportion to its
// current RPM.
void AP_Generator_RichenPower::update_heat()
{
// assume heat increase is directly proportional to RPM.
const uint32_t now = AP_HAL::millis();
uint16_t rpm = last_reading.rpm;
if (now - last_reading_ms > 2000) {
// if we're not getting updates, assume we're getting colder
rpm = 0;
// ... and resend the version information when we get something again
protocol_information_anounced = false;
}
const uint32_t time_delta_ms = now - last_heat_update_ms;
last_heat_update_ms = now;
heat += rpm * time_delta_ms * (1/1000.0f);
// cap the heat of the motor:
heat = MIN(heat, 60 * RUN_RPM); // so cap heat at 60 seconds at run-speed
// now lose some heat to the environment
heat -= (heat * heat_environment_loss_factor * (time_delta_ms * (1/1000.0f))); // lose some % of heat per second
}
// returns true if the generator should be allowed to move into
// the "run" (high-RPM) state:
bool AP_Generator_RichenPower::generator_ok_to_run() const
{
return heat > heat_required_for_run();
}
// returns an amount of synthetic heat required for the generator
// to move into the "run" state:
constexpr float AP_Generator_RichenPower::heat_required_for_run()
{
// assume that heat is proportional to RPM. Return a number
// proportial to RPM. Reduce it to account for the cooling some%/s
// cooling
return (45 * IDLE_RPM) * heat_environment_loss_30s;
}
/*
update the state of the sensor
*/
void AP_Generator_RichenPower::update(void)
{
if (uart == nullptr) {
return;
}
if (last_reading_ms != 0) {
update_runstate();
}
(void)get_reading();
update_heat();
Log_Write();
}
// update_runstate updates the servo output we use to control the
// generator. Which state we request the generator move to depends on
// the RC inputcontrol and the temperature the generator is at.
void AP_Generator_RichenPower::update_runstate()
{
// don't run the generator while the safety is on:
// if (hal.util->safety_switch_state() == AP_HAL::Util::SAFETY_DISARMED) {
// _servo_channel->set_output_pwm(SERVO_PWM_STOP);
// return;
// }
static const uint16_t SERVO_PWM_STOP = 1200;
static const uint16_t SERVO_PWM_IDLE = 1500;
static const uint16_t SERVO_PWM_RUN = 1900;
// if the vehicle crashes then we assume the pilot wants to stop
// the motor. This is done as a once-off when the crash is
// detected to allow the operator to rearm the vehicle, or we end
// up in a catch-22 situation where we force the stop state on the
// generator so they can't arm and can't start the generator
// because the vehicle is crashed.
if (AP::vehicle()->is_crashed()) {
if (!vehicle_was_crashed) {
gcs().send_text(MAV_SEVERITY_INFO, "Crash; stopping generator");
pilot_desired_runstate = RunState::STOP;
vehicle_was_crashed = true;
}
} else {
vehicle_was_crashed = false;
}
if (commanded_runstate != pilot_desired_runstate &&
!hal.util->get_soft_armed()) {
// consider changing the commanded runstate to the pilot
// desired runstate:
commanded_runstate = RunState::IDLE;
switch (pilot_desired_runstate) {
case RunState::STOP:
// The H2 will keep the motor running for ~20 seconds for cool-down
// we must go via the idle state or the H2 disallows moving to stop!
if (time_in_idle_state_ms() > 1000) {
commanded_runstate = pilot_desired_runstate;
}
break;
case RunState::IDLE:
// can always switch to idle
commanded_runstate = pilot_desired_runstate;
break;
case RunState::RUN:
// must have idled for a while before moving to run:
if (generator_ok_to_run()) {
commanded_runstate = pilot_desired_runstate;
}
break;
}
}
switch (commanded_runstate) {
case RunState::STOP:
SRV_Channels::set_output_pwm(SRV_Channel::k_generator_control, SERVO_PWM_STOP);
break;
case RunState::IDLE:
SRV_Channels::set_output_pwm(SRV_Channel::k_generator_control, SERVO_PWM_IDLE);
break;
case RunState::RUN:
SRV_Channels::set_output_pwm(SRV_Channel::k_generator_control, SERVO_PWM_RUN);
break;
}
}
// log generator status to the onboard log
void AP_Generator_RichenPower::Log_Write()
{
#define MASK_LOG_ANY 0xFFFF
if (!AP::logger().should_log(MASK_LOG_ANY)) {
return;
}
if (last_logged_reading_ms == last_reading_ms) {
return;
}
last_logged_reading_ms = last_reading_ms;
AP::logger().Write(
"GEN",
"TimeUS,runTime,maintTime,errors,rpm,ovolt,ocurr,mode",
"s-------",
"F-------",
"QIIHHffB",
AP_HAL::micros64(),
last_reading.runtime,
last_reading.seconds_until_maintenance,
last_reading.errors,
last_reading.rpm,
last_reading.output_voltage,
last_reading.output_current,
last_reading.mode
);
}
// generator prearm checks; notably, if we never see a generator we do
// not run the checks. Generators are attached/detached at will, and
// reconfiguring is painful.
bool AP_Generator_RichenPower::pre_arm_check(char *failmsg, uint8_t failmsg_len) const
{
if (uart == nullptr) {
// not configured in serial manager
return true;
}
if (last_reading_ms == 0) {
// allow optional use of generator
return true;
}
const uint32_t now = AP_HAL::millis();
if (now - last_reading_ms > 2000) { // we expect @1Hz
snprintf(failmsg, failmsg_len, "no messages in %ums", unsigned(now - last_reading_ms));
return false;
}
if (last_reading.seconds_until_maintenance == 0) {
snprintf(failmsg, failmsg_len, "requires maintenance");
}
if (SRV_Channels::get_channel_for(SRV_Channel::k_generator_control) == nullptr) {
snprintf(failmsg, failmsg_len, "need a servo output channel");
return false;
}
if (last_reading.errors) {
for (uint16_t i=0; i<16; i++) {
if (last_reading.errors & (1U << (uint16_t)i)) {
if (i < (uint16_t)Errors::LAST) {
snprintf(failmsg, failmsg_len, "error: %s", error_strings[i]);
} else {
snprintf(failmsg, failmsg_len, "unknown error: 1U<<%u", i);
}
}
}
return false;
}
if (pilot_desired_runstate != RunState::RUN) {
snprintf(failmsg, failmsg_len, "requested state is not RUN");
return false;
}
if (commanded_runstate != RunState::RUN) {
snprintf(failmsg, failmsg_len, "Generator warming up (%.0f%%)", (heat *100 / heat_required_for_run()));
return false;
}
if (last_reading.mode != Mode::RUN &&
last_reading.mode != Mode::CHARGE &&
last_reading.mode != Mode::BALANCE) {
snprintf(failmsg, failmsg_len, "not running");
return false;
}
return true;
}
// voltage returns the last voltage read from the generator telemetry
bool AP_Generator_RichenPower::voltage(float &v) const
{
if (last_reading_ms == 0) {
return false;
}
v = last_reading.output_voltage;
return true;
}
// current returns the last current read from the generator telemetry
bool AP_Generator_RichenPower::current(float &curr) const
{
if (last_reading_ms == 0) {
return false;
}
curr = last_reading.output_current;
return true;
}
// healthy returns true if the generator is not present, or it is
// present, providing telemetry and not indicating an errors.
bool AP_Generator_RichenPower::healthy() const
{
const uint32_t now = AP_HAL::millis();
if (last_reading_ms == 0 || now - last_reading_ms > 2000) {
return false;
}
if (last_reading.errors) {
return false;
}
return true;
}
//send mavlink generator status
void AP_Generator_RichenPower::send_generator_status(const GCS_MAVLINK &channel)
{
if (last_reading_ms == 0) {
// nothing to report
return;
}
uint64_t status = 0;
if (last_reading.rpm == 0) {
status |= MAV_GENERATOR_STATUS_FLAG_OFF;
} else {
switch (last_reading.mode) {
case Mode::OFF:
status |= MAV_GENERATOR_STATUS_FLAG_OFF;
break;
case Mode::IDLE:
if (pilot_desired_runstate == RunState::RUN) {
status |= MAV_GENERATOR_STATUS_FLAG_WARMING_UP;
} else {
status |= MAV_GENERATOR_STATUS_FLAG_IDLE;
}
break;
case Mode::RUN:
status |= MAV_GENERATOR_STATUS_FLAG_GENERATING;
break;
case Mode::CHARGE:
status |= MAV_GENERATOR_STATUS_FLAG_GENERATING;
status |= MAV_GENERATOR_STATUS_FLAG_CHARGING;
break;
case Mode::BALANCE:
status |= MAV_GENERATOR_STATUS_FLAG_GENERATING;
status |= MAV_GENERATOR_STATUS_FLAG_CHARGING;
break;
}
}
if (last_reading.errors & (uint8_t)Errors::Overload) {
status |= MAV_GENERATOR_STATUS_FLAG_OVERCURRENT_FAULT;
}
if (last_reading.errors & (uint8_t)Errors::LowVoltageOutput) {
status |= MAV_GENERATOR_STATUS_FLAG_REDUCED_POWER;
}
if (last_reading.errors & (uint8_t)Errors::MaintenanceRequired) {
status |= MAV_GENERATOR_STATUS_FLAG_MAINTENANCE_REQUIRED;
}
if (last_reading.errors & (uint8_t)Errors::StartDisabled) {
status |= MAV_GENERATOR_STATUS_FLAG_START_INHIBITED;
}
if (last_reading.errors & (uint8_t)Errors::LowBatteryVoltage) {
status |= MAV_GENERATOR_STATUS_FLAG_BATTERY_UNDERVOLT_FAULT;
}
mavlink_msg_generator_status_send(
channel.get_chan(),
status,
last_reading.rpm, // generator_speed
std::numeric_limits<double>::quiet_NaN(), // battery_current; current into/out of battery
last_reading.output_current, // load_current; Current going to UAV
std::numeric_limits<double>::quiet_NaN(), // power_generated; the power being generated
last_reading.output_voltage, // bus_voltage; Voltage of the bus seen at the generator
INT16_MAX, // rectifier_temperature
std::numeric_limits<double>::quiet_NaN(), // bat_current_setpoint; The target battery current
INT16_MAX, // generator temperature
last_reading.runtime,
(int32_t)last_reading.seconds_until_maintenance
);
}
/*
* Get the AP_Generator singleton
*/
AP_Generator_RichenPower *AP_Generator_RichenPower::_singleton;
AP_Generator_RichenPower *AP_Generator_RichenPower::get_singleton()
{
return _singleton;
}
namespace AP {
AP_Generator_RichenPower *generator()
{
return AP_Generator_RichenPower::get_singleton();
}
};
#endif