ardupilot/libraries/AP_Gripper/AP_Gripper_Servo.cpp
Peter Barker 56a7e520b6 AP_Gripper: correct emitting of grabbed/released messages
these strigs would only be released if release() or grab were called a second time
2024-06-20 10:59:14 +10:00

126 lines
3.6 KiB
C++

#include <AP_Gripper/AP_Gripper_Servo.h>
#if AP_GRIPPER_SERVO_ENABLED
#include <GCS_MAVLink/GCS.h>
#include <AP_Logger/AP_Logger.h>
#include <SRV_Channel/SRV_Channel.h>
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
#include <SITL/SITL.h>
#endif
extern const AP_HAL::HAL& hal;
void AP_Gripper_Servo::init_gripper()
{
// move the servo to the neutral position
SRV_Channels::set_output_pwm(SRV_Channel::k_gripper, config.neutral_pwm);
}
void AP_Gripper_Servo::grab()
{
// check if we are already grabbing
if (config.state == AP_Gripper::STATE_GRABBING) {
// do nothing
return;
}
// check if we are already grabbed
if (config.state == AP_Gripper::STATE_GRABBED) {
// inform user that we are already grabbed
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load grabbed");
return;
}
// flag we are active and grabbing cargo
config.state = AP_Gripper::STATE_GRABBING;
// move the servo to the grab position
SRV_Channels::set_output_pwm(SRV_Channel::k_gripper, config.grab_pwm);
_last_grab_or_release = AP_HAL::millis();
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load grabbing");
LOGGER_WRITE_EVENT(LogEvent::GRIPPER_GRAB);
}
void AP_Gripper_Servo::release()
{
// check if we are already releasing
if (config.state == AP_Gripper::STATE_RELEASING) {
// do nothing
return;
}
// check if we are already released
if (config.state == AP_Gripper::STATE_RELEASED) {
// inform user that we are already released
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load released");
return;
}
// flag we are releasing cargo
config.state = AP_Gripper::STATE_RELEASING;
// move the servo to the release position
SRV_Channels::set_output_pwm(SRV_Channel::k_gripper, config.release_pwm);
_last_grab_or_release = AP_HAL::millis();
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load releasing");
LOGGER_WRITE_EVENT(LogEvent::GRIPPER_RELEASE);
}
bool AP_Gripper_Servo::has_state_pwm(const uint16_t pwm) const
{
// return true if servo is in position represented by pwm
uint16_t current_pwm;
if (!SRV_Channels::get_output_pwm(SRV_Channel::k_gripper, current_pwm)) {
// function not assigned to a channel, perhaps?
return false;
}
if (current_pwm != pwm) {
// last action did not set pwm to the current value
// (e.g. last action was a grab not a release)
return false;
}
if (AP_HAL::millis() - _last_grab_or_release < SERVO_ACTUATION_TIME) {
// servo still moving....
return false;
}
return true;
}
bool AP_Gripper_Servo::released() const
{
return (config.state == AP_Gripper::STATE_RELEASED);
}
bool AP_Gripper_Servo::grabbed() const
{
return (config.state == AP_Gripper::STATE_GRABBED);
}
// type-specific periodic updates:
void AP_Gripper_Servo::update_gripper()
{
// Check for successful grabbed or released
if (config.state == AP_Gripper::STATE_GRABBING && has_state_pwm(config.grab_pwm)) {
config.state = AP_Gripper::STATE_GRABBED;
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load grabbed");
} else if (config.state == AP_Gripper::STATE_RELEASING && has_state_pwm(config.release_pwm)) {
config.state = AP_Gripper::STATE_RELEASED;
GCS_SEND_TEXT(MAV_SEVERITY_INFO, "Gripper load released");
}
}
bool AP_Gripper_Servo::valid() const
{
if (!AP_Gripper_Backend::valid()) {
return false;
}
if (!SRV_Channels::function_assigned(SRV_Channel::k_gripper)) {
return false;
}
return true;
}
#endif // AP_GRIPPER_SERVO_ENABLED