1
0
mirror of https://gitlab.com/obbart/universal_robots_ros_driver.git synced 2026-04-10 10:00:48 +02:00

Action server improvements

This commit is contained in:
Simon Rasmussen
2017-05-19 01:20:30 +02:00
parent c59bfc78cc
commit 8d845c6f38
2 changed files with 110 additions and 21 deletions

View File

@@ -12,6 +12,9 @@ ActionServer::ActionServer(TrajectoryFollower& follower, std::vector<std::string
, joint_names_(joint_names)
, joint_set_(joint_names.begin(), joint_names.end())
, max_velocity_(max_velocity)
, interrupt_traj_(false)
, running_(false)
, has_goal_(false)
, state_(RobotState::Error)
, follower_(follower)
{
@@ -21,22 +24,81 @@ void ActionServer::start()
{
if(running_)
return;
LOG_INFO("Starting ActionServer");
running_ = true;
tj_thread_ = thread(&ActionServer::trajectoryThread, this);
as_.start();
}
void ActionServer::onRobotStateChange(RobotState state)
{
state_ = state;
//don't interrupt if everything is fine
if(state == RobotState::Running)
return;
//don't retry interrupts
if(interrupt_traj_ || !has_goal_)
return;
//on successful lock we're not executing a goal so don't interrupt
if(tj_mutex_.try_lock())
{
tj_mutex_.unlock();
return;
}
interrupt_traj_ = true;
//wait for goal to be interrupted
std::lock_guard<std::mutex> lock(tj_mutex_);
Result res;
res.error_code = -100;
res.error_string = "Received another trajectory";
curr_gh_.setAborted(res, res.error_string);
}
bool ActionServer::updateState(RTShared& data)
{
q_actual_ = data.q_actual;
qd_actual_ = data.qd_actual;
return true;
}
bool ActionServer::consume(RTState_V1_6__7& state)
{
return updateState(state);
}
bool ActionServer::consume(RTState_V1_8& state)
{
return updateState(state);
}
bool ActionServer::consume(RTState_V3_0__1& state)
{
return updateState(state);
}
bool ActionServer::consume(RTState_V3_2__3& state)
{
return updateState(state);
}
void ActionServer::onGoal(GoalHandle gh)
{
Result res;
res.error_code = -100;
LOG_INFO("Received new goal");
if(!validate(gh, res) || !try_execute(gh, res))
{
LOG_WARN("Goal error: %s", res.error_string.c_str());
gh.setRejected(res, res.error_string);
}
}
void ActionServer::onCancel(GoalHandle gh)
@@ -53,7 +115,7 @@ void ActionServer::onCancel(GoalHandle gh)
bool ActionServer::validate(GoalHandle& gh, Result& res)
{
return !validateState(gh, res) || !validateJoints(gh, res) || !validateTrajectory(gh, res);
return validateState(gh, res) && validateJoints(gh, res) && validateTrajectory(gh, res);
}
bool ActionServer::validateState(GoalHandle& gh, Result& res)
@@ -100,6 +162,10 @@ bool ActionServer::validateTrajectory(GoalHandle& gh, Result& res)
auto goal = gh.getGoal();
res.error_code = Result::INVALID_GOAL;
//must at least have one point
if(goal->trajectory.points.size() < 1)
return false;
for(auto const& point : goal->trajectory.points)
{
if(point.velocities.size() != joint_names_.size())
@@ -194,28 +260,41 @@ std::vector<size_t> ActionServer::reorderMap(std::vector<std::string> goal_joint
void ActionServer::trajectoryThread()
{
LOG_INFO("Trajectory thread started");
follower_.start(); //todo check error
//as_.start();
while(running_)
{
std::unique_lock<std::mutex> lk(tj_mutex_);
if(!tj_cv_.wait_for(lk, std::chrono::milliseconds(100), [&]{return running_ && has_goal_;}))
continue;
LOG_DEBUG("Trajectory received and accepted");
LOG_INFO("Trajectory received and accepted");
curr_gh_.setAccepted();
auto goal = curr_gh_.getGoal();
auto mapping = reorderMap(goal->trajectory.joint_names);
std::vector<TrajectoryPoint> trajectory(goal->trajectory.points.size());
//joint names of the goal might have a different ordering compared
//to what URScript expects so need to map between the two
auto mapping = reorderMap(goal->trajectory.joint_names);
LOG_INFO("Translating trajectory");
auto const& fp = goal->trajectory.points[0];
auto fpt = convert(fp.time_from_start)
//make sure we have a proper t0 position
if(fpt > std::chrono::microseconds(0))
{
LOG_INFO("Trajectory without t0 recieved, inserting t0 at currrent position");
trajectory.push_back(TrajectoryPoint(q_actual_, qd_actual_, std::chrono::microseconds(0)));
}
for(auto const& point : goal->trajectory.points)
{
std::array<double, 6> pos, vel;
for(size_t i = 0; i < 6; i++)
{
//joint names of the goal might have a different ordering compared
//to what URScript expects so need to map between the two
size_t idx = mapping[i];
pos[idx] = point.positions[i];
vel[idx] = point.velocities[i];
@@ -223,20 +302,25 @@ void ActionServer::trajectoryThread()
trajectory.push_back(TrajectoryPoint(pos, vel, convert(point.time_from_start)));
}
double t = std::chrono::duration_cast<std::chrono::duration<double>>(trajectory[trajectory.size()-1].time_from_start).count();
LOG_INFO("Trajectory with %d points and duration of %f constructed, executing now", trajectory.size(), t);
Result res;
if(follower_.execute(trajectory, interrupt_traj_))
{
//interrupted goals must be handled by interrupt trigger
if(!interrupt_traj_)
{
LOG_DEBUG("Trajectory executed successfully");
LOG_INFO("Trajectory executed successfully");
res.error_code = Result::SUCCESSFUL;
curr_gh_.setSucceeded(res);
}
else
LOG_INFO("Trajectory interrupted");
}
else
{
LOG_DEBUG("Trajectory failed");
LOG_INFO("Trajectory failed");
res.error_code = -100;
res.error_string = "Connection to robot was lost";
curr_gh_.setAborted(res, res.error_string);