mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-04-14 10:07:39 +08:00
- Reporter class that keeps track of results (list of events + whether arming is possible + health). And only reports when one or more checks change.
244 lines
8.3 KiB
C++
244 lines
8.3 KiB
C++
/****************************************************************************
|
|
*
|
|
* Copyright (c) 2020 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 "Common.hpp"
|
|
|
|
void Report::healthFailure(NavModes required_modes, HealthComponentIndex component, uint32_t event_id,
|
|
const events::LogLevels &log_levels, const char *message)
|
|
{
|
|
healthFailure(required_modes, component, log_levels.external);
|
|
addEvent(event_id, log_levels, message, (uint32_t)reportedModes(required_modes), component.index);
|
|
}
|
|
|
|
void Report::armingCheckFailure(NavModes required_modes, HealthComponentIndex component, uint32_t event_id,
|
|
const events::LogLevels &log_levels, const char *message)
|
|
{
|
|
armingCheckFailure(required_modes, component, log_levels.external);
|
|
addEvent(event_id, log_levels, message, (uint32_t)reportedModes(required_modes), component.index);
|
|
}
|
|
|
|
NavModes Report::reportedModes(NavModes required_modes)
|
|
{
|
|
// Make sure optional checks are still shown in the UI
|
|
if (required_modes == NavModes::None) {
|
|
return NavModes::All;
|
|
}
|
|
|
|
return required_modes;
|
|
}
|
|
|
|
void Report::setIsPresent(health_component_t component)
|
|
{
|
|
HealthResults &health = _results[_current_result].health;
|
|
health.is_present = health.is_present | component;
|
|
}
|
|
|
|
void Report::setHealth(health_component_t component, bool is_present, bool warning, bool error)
|
|
{
|
|
HealthResults &health = _results[_current_result].health;
|
|
|
|
if (is_present) {
|
|
health.is_present = health.is_present | component;
|
|
}
|
|
|
|
if (warning) {
|
|
health.warning = health.warning | component;
|
|
}
|
|
|
|
if (error) {
|
|
health.error = health.error | component;
|
|
}
|
|
}
|
|
|
|
void Report::healthFailure(NavModes required_modes, HealthComponentIndex component, events::Log log_level)
|
|
{
|
|
// update current health results
|
|
HealthResults &health = _results[_current_result].health;
|
|
|
|
if (log_level <= events::Log::Error) {
|
|
health.error = health.error | (events::px4::enums::health_component_t)(1 << component.index);
|
|
|
|
} else if (log_level <= events::Log::Warning) {
|
|
health.warning = health.warning | (events::px4::enums::health_component_t)(1 << component.index);
|
|
}
|
|
|
|
// clear relevant arming bits
|
|
clearArmingBits(required_modes);
|
|
}
|
|
|
|
void Report::armingCheckFailure(NavModes required_modes, HealthComponentIndex component, events::Log log_level)
|
|
{
|
|
// update current arming check results
|
|
ArmingCheckResults &arming_checks = _results[_current_result].arming_checks;
|
|
|
|
if (log_level <= events::Log::Error) {
|
|
arming_checks.error = arming_checks.error | (events::px4::enums::health_component_t)(1 << component.index);
|
|
|
|
} else if (log_level <= events::Log::Warning) {
|
|
arming_checks.warning = arming_checks.warning | (events::px4::enums::health_component_t)(1 << component.index);
|
|
}
|
|
|
|
// clear relevant arming bits
|
|
clearArmingBits(required_modes);
|
|
}
|
|
|
|
void Report::clearArmingBits(NavModes modes)
|
|
{
|
|
ArmingCheckResults &arming_checks = _results[_current_result].arming_checks;
|
|
NavModes &can_arm = arming_checks.can_arm;
|
|
can_arm = can_arm & ~modes;
|
|
}
|
|
|
|
void Report::clearCanRunBits(NavModes modes)
|
|
{
|
|
ArmingCheckResults &arming_checks = _results[_current_result].arming_checks;
|
|
NavModes &can_run = arming_checks.can_run;
|
|
can_run = can_run & ~modes;
|
|
}
|
|
|
|
void Report::reset()
|
|
{
|
|
_current_result = (_current_result + 1) % 2;
|
|
_results[_current_result].reset();
|
|
_next_buffer_idx = 0;
|
|
_buffer_overflowed = false;
|
|
_failing_checks_prevent_mode_switch = 0;
|
|
}
|
|
|
|
NavModes Report::getModeGroup(uint8_t nav_state) const
|
|
{
|
|
// Note: this needs to match with the json metadata definition "navigation_mode_groups"
|
|
return (NavModes)(1u << nav_state);
|
|
}
|
|
|
|
void Report::finalize()
|
|
{
|
|
_results[_current_result].arming_checks.valid = true;
|
|
_already_reported = false;
|
|
}
|
|
|
|
bool Report::report(bool is_armed, bool force)
|
|
{
|
|
const hrt_abstime now = hrt_absolute_time();
|
|
const bool has_difference = _had_unreported_difference || _results[0] != _results[1];
|
|
|
|
if (now - _last_report < min_reporting_interval && !force) {
|
|
if (has_difference) {
|
|
_had_unreported_difference = true;
|
|
}
|
|
|
|
PX4_DEBUG("not reporting yet (%i)", has_difference);
|
|
return false;
|
|
}
|
|
|
|
const bool need_to_report = (has_difference && !_already_reported) || force;
|
|
|
|
if (!need_to_report) {
|
|
PX4_DEBUG("not reporting, no update (%i)", has_difference);
|
|
return false;
|
|
}
|
|
|
|
const Results ¤t_results = _results[_current_result];
|
|
|
|
// If we have too many events, the result is still correct, we just don't report everything
|
|
const int max_num_events = event_s::ORB_QUEUE_LENGTH - 2;
|
|
|
|
if (_buffer_overflowed || current_results.num_events > max_num_events) {
|
|
PX4_WARN("Too many arming check events (%i, %i > %i). Not reporting all", _buffer_overflowed,
|
|
current_results.num_events, max_num_events);
|
|
}
|
|
|
|
if (!current_results.arming_checks.valid) {
|
|
PX4_ERR("BUG: arming checks not valid");
|
|
}
|
|
|
|
_last_report = now;
|
|
_already_reported = true;
|
|
_had_unreported_difference = false;
|
|
PX4_DEBUG("Sending Arming/Health report (num events: %i, can_arm: 0x%x)", _results[_current_result].num_events + 2,
|
|
(int)current_results.arming_checks.can_arm);
|
|
|
|
// send arming summary
|
|
navigation_mode_group_t can_arm_and_run;
|
|
|
|
if (is_armed) {
|
|
can_arm_and_run = (navigation_mode_group_t)current_results.arming_checks.can_run;
|
|
|
|
} else {
|
|
can_arm_and_run = (navigation_mode_group_t)current_results.arming_checks.can_arm;
|
|
}
|
|
|
|
/* EVENT
|
|
* @arg1 chunk_idx
|
|
* @arg2 error
|
|
* @arg3 warning
|
|
* @arg4 can_arm_and_run
|
|
* @type summary
|
|
* @group arming_check
|
|
*/
|
|
events::send<uint8_t, events::px4::enums::health_component_t, events::px4::enums::health_component_t,
|
|
events::px4::enums::navigation_mode_group_t>(events::ID("commander_arming_check_summary"), events::Log::Protocol,
|
|
"Arming check summary event", 0, current_results.arming_checks.error, current_results.arming_checks.warning,
|
|
can_arm_and_run);
|
|
|
|
// send all events
|
|
int offset = 0;
|
|
events::EventType event;
|
|
|
|
for (int i = 0; i < max_num_events && offset < _next_buffer_idx; ++i) {
|
|
EventBufferHeader *header = (EventBufferHeader *)(_event_buffer + offset);
|
|
memcpy(&event.id, &header->id, sizeof(event.id));
|
|
event.log_levels = header->log_levels;
|
|
memcpy(event.arguments, _event_buffer + offset + sizeof(EventBufferHeader), header->size);
|
|
memset(event.arguments + header->size, 0, sizeof(event.arguments) - header->size);
|
|
events::send(event);
|
|
offset += sizeof(EventBufferHeader) + header->size;
|
|
}
|
|
|
|
// send health summary
|
|
/* EVENT
|
|
* @arg1 chunk_idx
|
|
* @arg2 is_present
|
|
* @arg3 error
|
|
* @arg4 warning
|
|
* @type summary
|
|
* @group health
|
|
*/
|
|
events::send<uint8_t, events::px4::enums::health_component_t, events::px4::enums::health_component_t,
|
|
events::px4::enums::health_component_t>(events::ID("commander_health_summary"),
|
|
events::Log::Protocol,
|
|
"Health report summary event", 0, current_results.health.is_present,
|
|
current_results.health.error, current_results.health.warning);
|
|
return true;
|
|
}
|