-
Notifications
You must be signed in to change notification settings - Fork 80
/
lwr_hw_fri_node.cpp
176 lines (142 loc) · 4.44 KB
/
lwr_hw_fri_node.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
// SYS
#include <sys/mman.h>
#include <cmath>
#include <time.h>
#include <signal.h>
#include <stdexcept>
// ROS headers
#include <ros/ros.h>
#include <controller_manager/controller_manager.h>
#include <std_msgs/Bool.h>
// the lwr hw fri interface
#include "lwr_hw/lwr_hw_fri.hpp"
bool g_quit = false;
void quitRequested(int sig)
{
g_quit = true;
}
bool isStopPressed = false;
bool wasStopHandled = true;
void eStopCB(const std_msgs::BoolConstPtr& e_stop_msg)
{
isStopPressed = e_stop_msg->data;
}
// Get the URDF XML from the parameter server
std::string getURDF(ros::NodeHandle &model_nh_, std::string param_name)
{
std::string urdf_string;
std::string robot_description = "/robot_description";
// search and wait for robot_description on param server
while (urdf_string.empty())
{
std::string search_param_name;
if (model_nh_.searchParam(param_name, search_param_name))
{
ROS_INFO_ONCE_NAMED("LWRHWFRI", "LWRHWFRI node is waiting for model"
" URDF in parameter [%s] on the ROS param server.", search_param_name.c_str());
model_nh_.getParam(search_param_name, urdf_string);
}
else
{
ROS_INFO_ONCE_NAMED("LWRHWFRI", "LWRHWFRI node is waiting for model"
" URDF in parameter [%s] on the ROS param server.", robot_description.c_str());
model_nh_.getParam(param_name, urdf_string);
}
usleep(100000);
}
ROS_DEBUG_STREAM_NAMED("LWRHWFRI", "Received URDF from param server, parsing...");
return urdf_string;
}
int main( int argc, char** argv )
{
// initialize ROS
ros::init(argc, argv, "lwr_hw_interface", ros::init_options::NoSigintHandler);
// ros spinner
ros::AsyncSpinner spinner(1);
spinner.start();
// custom signal handlers
signal(SIGTERM, quitRequested);
signal(SIGINT, quitRequested);
signal(SIGHUP, quitRequested);
// create a node
ros::NodeHandle lwr_nh;
// get params or give default values
int port;
std::string hintToRemoteHost;
std::string name;
lwr_nh.param("port", port, 49939);
lwr_nh.param("ip", hintToRemoteHost, std::string("192.168.0.10") );
lwr_nh.param("name", name, std::string("lwr"));
// advertise the e-stop topic
ros::Subscriber estop_sub = lwr_nh.subscribe(lwr_nh.resolveName("emergency_stop"), 1, eStopCB);
// get the general robot description, the lwr class will take care of parsing what's useful to itself
std::string urdf_string = getURDF(lwr_nh, "/robot_description");
// construct and start the real lwr
lwr_hw::LWRHWFRI lwr_robot;
lwr_robot.create(name, urdf_string);
lwr_robot.setPort(port);
lwr_robot.setIP(hintToRemoteHost);
if(!lwr_robot.init())
{
ROS_FATAL_NAMED("lwr_hw","Could not initialize robot real interface");
return -1;
}
// timer variables
struct timespec ts = {0, 0};
ros::Time last(ts.tv_sec, ts.tv_nsec), now(ts.tv_sec, ts.tv_nsec);
ros::Duration period(1.0);
float sampling_time = lwr_robot.getSampleTime();
ROS_INFO("Sampling time on robot: %f", sampling_time);
//the controller manager
controller_manager::ControllerManager manager(&lwr_robot, lwr_nh);
// run as fast as the robot interface, or as fast as possible
ros::Rate rate(1.0/sampling_time);
while( !g_quit )
{
// get the time / period
if (!clock_gettime(CLOCK_MONOTONIC, &ts))
{
now.sec = ts.tv_sec;
now.nsec = ts.tv_nsec;
period = now - last;
last = now;
}
else
{
ROS_FATAL("Failed to poll realtime clock!");
break;
}
// read the state from the lwr
lwr_robot.read(now, period);
// Compute the controller commands
bool resetControllers;
if(!wasStopHandled && !resetControllers)
{
ROS_WARN("E-STOP HAS BEEN PRESSED: Controllers will be restarted, but the robot won't move until you release the E-Stop");
ROS_WARN("HOW TO RELEASE E-STOP: rostopic pub -r 10 /NAMESPACE/emergency_stop std_msgs/Bool 'data: false'");
resetControllers = true;
wasStopHandled = true;
}
if( isStopPressed )
{
wasStopHandled = false;
}
else
{
resetControllers = false;
wasStopHandled = true;
}
// update the controllers
manager.update(now, period, resetControllers);
// write the command to the lwr
lwr_robot.write(now, period);
// if there is time left, sleep
rate.sleep();
}
std::cerr<<"Stopping spinner..."<<std::endl;
spinner.stop();
//std::cerr<<"Stopping LWR..."<<std::endl;
//lwr_robot.stopFRI();
std::cerr<<"Bye!"<<std::endl;
return 0;
}