Go to the documentation of this file.00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018 #include <string>
00019 #include <tf/tf.h>
00020 #include <stdlib.h>
00021
00022 #include "gazebo_plugins/gazebo_ros_p3d.h"
00023
00024 namespace gazebo
00025 {
00026 GZ_REGISTER_MODEL_PLUGIN(GazeboRosP3D);
00027
00029
00030 GazeboRosP3D::GazeboRosP3D()
00031 {
00032 this->seed = 0;
00033 }
00034
00036
00037 GazeboRosP3D::~GazeboRosP3D()
00038 {
00039 event::Events::DisconnectWorldUpdateBegin(this->update_connection_);
00040
00041 this->rosnode_->shutdown();
00042 this->p3d_queue_.clear();
00043 this->p3d_queue_.disable();
00044 this->callback_queue_thread_.join();
00045 delete this->rosnode_;
00046 }
00047
00049
00050 void GazeboRosP3D::Load(physics::ModelPtr _parent, sdf::ElementPtr _sdf)
00051 {
00052
00053 this->world_ = _parent->GetWorld();
00054 this->model_ = _parent;
00055
00056
00057 this->robot_namespace_ = "";
00058 if (_sdf->HasElement("robotNamespace"))
00059 this->robot_namespace_ =
00060 _sdf->GetElement("robotNamespace")->Get<std::string>() + "/";
00061
00062 if (!_sdf->HasElement("bodyName"))
00063 {
00064 ROS_FATAL("p3d plugin missing <bodyName>, cannot proceed");
00065 return;
00066 }
00067 else
00068 this->link_name_ = _sdf->GetElement("bodyName")->Get<std::string>();
00069
00070 this->link_ = _parent->GetLink(this->link_name_);
00071 if (!this->link_)
00072 {
00073 ROS_FATAL("gazebo_ros_p3d plugin error: bodyName: %s does not exist\n",
00074 this->link_name_.c_str());
00075 return;
00076 }
00077
00078 if (!_sdf->HasElement("topicName"))
00079 {
00080 ROS_FATAL("p3d plugin missing <topicName>, cannot proceed");
00081 return;
00082 }
00083 else
00084 this->topic_name_ = _sdf->GetElement("topicName")->Get<std::string>();
00085
00086 if (!_sdf->HasElement("frameName"))
00087 {
00088 ROS_DEBUG("p3d plugin missing <frameName>, defaults to world");
00089 this->frame_name_ = "world";
00090 }
00091 else
00092 this->frame_name_ = _sdf->GetElement("frameName")->Get<std::string>();
00093
00094 if (!_sdf->HasElement("xyzOffset"))
00095 {
00096 ROS_DEBUG("p3d plugin missing <xyzOffset>, defaults to 0s");
00097 this->offset_.pos = math::Vector3(0, 0, 0);
00098 }
00099 else
00100 this->offset_.pos = _sdf->GetElement("xyzOffset")->Get<math::Vector3>();
00101
00102 if (!_sdf->HasElement("rpyOffset"))
00103 {
00104 ROS_DEBUG("p3d plugin missing <rpyOffset>, defaults to 0s");
00105 this->offset_.rot = math::Vector3(0, 0, 0);
00106 }
00107 else
00108 this->offset_.rot = _sdf->GetElement("rpyOffset")->Get<math::Vector3>();
00109
00110 if (!_sdf->HasElement("gaussianNoise"))
00111 {
00112 ROS_DEBUG("p3d plugin missing <gaussianNoise>, defaults to 0.0");
00113 this->gaussian_noise_ = 0;
00114 }
00115 else
00116 this->gaussian_noise_ = _sdf->GetElement("gaussianNoise")->Get<double>();
00117
00118 if (!_sdf->HasElement("updateRate"))
00119 {
00120 ROS_DEBUG("p3d plugin missing <updateRate>, defaults to 0.0"
00121 " (as fast as possible)");
00122 this->update_rate_ = 0;
00123 }
00124 else
00125 this->update_rate_ = _sdf->GetElement("updateRate")->Get<double>();
00126
00127
00128 if (!ros::isInitialized())
00129 {
00130 ROS_FATAL_STREAM("A ROS node for Gazebo has not been initialized, unable to load plugin. "
00131 << "Load the Gazebo system plugin 'libgazebo_ros_api_plugin.so' in the gazebo_ros package)");
00132 return;
00133 }
00134
00135 this->rosnode_ = new ros::NodeHandle(this->robot_namespace_);
00136
00137
00138 this->pmq.startServiceThread();
00139
00140
00141 std::string prefix;
00142 this->rosnode_->getParam(std::string("tf_prefix"), prefix);
00143 this->tf_frame_name_ = tf::resolve(prefix, this->frame_name_);
00144
00145 if (this->topic_name_ != "")
00146 {
00147 this->pub_Queue = this->pmq.addPub<nav_msgs::Odometry>();
00148 this->pub_ =
00149 this->rosnode_->advertise<nav_msgs::Odometry>(this->topic_name_, 1);
00150 }
00151
00152 this->last_time_ = this->world_->GetSimTime();
00153
00154 this->last_vpos_ = this->link_->GetWorldLinearVel();
00155 this->last_veul_ = this->link_->GetWorldAngularVel();
00156 this->apos_ = 0;
00157 this->aeul_ = 0;
00158
00159
00160
00161 if (this->frame_name_ != "world" &&
00162 this->frame_name_ != "/map" &&
00163 this->frame_name_ != "map")
00164 {
00165 this->reference_link_ = this->model_->GetLink(this->frame_name_);
00166 if (!this->reference_link_)
00167 {
00168 ROS_ERROR("gazebo_ros_p3d plugin: frameName: %s does not exist, will"
00169 " not publish pose\n", this->frame_name_.c_str());
00170 return;
00171 }
00172 }
00173
00174
00175 if (this->reference_link_)
00176 {
00177 ROS_DEBUG("got body %s", this->reference_link_->GetName().c_str());
00178 this->frame_apos_ = 0;
00179 this->frame_aeul_ = 0;
00180 this->last_frame_vpos_ = this->reference_link_->GetWorldLinearVel();
00181 this->last_frame_veul_ = this->reference_link_->GetWorldAngularVel();
00182 }
00183
00184
00185
00186 this->callback_queue_thread_ = boost::thread(
00187 boost::bind(&GazeboRosP3D::P3DQueueThread, this));
00188
00189
00190
00191
00192 this->update_connection_ = event::Events::ConnectWorldUpdateBegin(
00193 boost::bind(&GazeboRosP3D::UpdateChild, this));
00194 }
00195
00197
00198 void GazeboRosP3D::UpdateChild()
00199 {
00200 if (!this->link_)
00201 return;
00202
00203 common::Time cur_time = this->world_->GetSimTime();
00204
00205
00206 if (this->update_rate_ > 0 &&
00207 (cur_time-this->last_time_).Double() < (1.0/this->update_rate_))
00208 return;
00209
00210 if (this->pub_.getNumSubscribers() > 0)
00211 {
00212
00213 double tmp_dt = cur_time.Double() - this->last_time_.Double();
00214 if (tmp_dt != 0)
00215 {
00216 this->lock.lock();
00217
00218 if (this->topic_name_ != "")
00219 {
00220
00221 this->pose_msg_.header.frame_id = this->tf_frame_name_;
00222 this->pose_msg_.header.stamp.sec = cur_time.sec;
00223 this->pose_msg_.header.stamp.nsec = cur_time.nsec;
00224
00225
00226 math::Pose pose, frame_pose;
00227 math::Vector3 frame_vpos;
00228 math::Vector3 frame_veul;
00229
00230
00231 math::Vector3 vpos = this->link_->GetWorldLinearVel();
00232 math::Vector3 veul = this->link_->GetWorldAngularVel();
00233
00234
00235 pose = this->link_->GetWorldPose();
00236
00237
00238 if (this->reference_link_)
00239 {
00240
00241 frame_pose = this->reference_link_->GetWorldPose();
00242 pose.pos = pose.pos - frame_pose.pos;
00243 pose.pos = frame_pose.rot.RotateVectorReverse(pose.pos);
00244 pose.rot *= frame_pose.rot.GetInverse();
00245
00246 frame_vpos = this->reference_link_->GetWorldLinearVel();
00247 frame_veul = this->reference_link_->GetWorldAngularVel();
00248 vpos = frame_pose.rot.RotateVector(vpos - frame_vpos);
00249 veul = frame_pose.rot.RotateVector(veul - frame_veul);
00250 }
00251
00252
00253
00254 pose.pos = pose.pos + this->offset_.pos;
00255
00256 pose.rot = this->offset_.rot*pose.rot;
00257 pose.rot.Normalize();
00258
00259
00260 this->apos_ = (this->last_vpos_ - vpos) / tmp_dt;
00261 this->aeul_ = (this->last_veul_ - veul) / tmp_dt;
00262 this->last_vpos_ = vpos;
00263 this->last_veul_ = veul;
00264
00265 this->frame_apos_ = (this->last_frame_vpos_ - frame_vpos) / tmp_dt;
00266 this->frame_aeul_ = (this->last_frame_veul_ - frame_veul) / tmp_dt;
00267 this->last_frame_vpos_ = frame_vpos;
00268 this->last_frame_veul_ = frame_veul;
00269
00270
00271 this->pose_msg_.pose.pose.position.x = pose.pos.x;
00272 this->pose_msg_.pose.pose.position.y = pose.pos.y;
00273 this->pose_msg_.pose.pose.position.z = pose.pos.z;
00274
00275 this->pose_msg_.pose.pose.orientation.x = pose.rot.x;
00276 this->pose_msg_.pose.pose.orientation.y = pose.rot.y;
00277 this->pose_msg_.pose.pose.orientation.z = pose.rot.z;
00278 this->pose_msg_.pose.pose.orientation.w = pose.rot.w;
00279
00280 this->pose_msg_.twist.twist.linear.x = vpos.x +
00281 this->GaussianKernel(0, this->gaussian_noise_);
00282 this->pose_msg_.twist.twist.linear.y = vpos.y +
00283 this->GaussianKernel(0, this->gaussian_noise_);
00284 this->pose_msg_.twist.twist.linear.z = vpos.z +
00285 this->GaussianKernel(0, this->gaussian_noise_);
00286
00287 this->pose_msg_.twist.twist.angular.x = veul.x +
00288 this->GaussianKernel(0, this->gaussian_noise_);
00289 this->pose_msg_.twist.twist.angular.y = veul.y +
00290 this->GaussianKernel(0, this->gaussian_noise_);
00291 this->pose_msg_.twist.twist.angular.z = veul.z +
00292 this->GaussianKernel(0, this->gaussian_noise_);
00293
00294
00296 double gn2 = this->gaussian_noise_*this->gaussian_noise_;
00297 this->pose_msg_.pose.covariance[0] = gn2;
00298 this->pose_msg_.pose.covariance[7] = gn2;
00299 this->pose_msg_.pose.covariance[14] = gn2;
00300 this->pose_msg_.pose.covariance[21] = gn2;
00301 this->pose_msg_.pose.covariance[28] = gn2;
00302 this->pose_msg_.pose.covariance[35] = gn2;
00303
00304 this->pose_msg_.twist.covariance[0] = gn2;
00305 this->pose_msg_.twist.covariance[7] = gn2;
00306 this->pose_msg_.twist.covariance[14] = gn2;
00307 this->pose_msg_.twist.covariance[21] = gn2;
00308 this->pose_msg_.twist.covariance[28] = gn2;
00309 this->pose_msg_.twist.covariance[35] = gn2;
00310
00311
00312 this->pub_Queue->push(this->pose_msg_, this->pub_);
00313 }
00314
00315 this->lock.unlock();
00316
00317
00318 this->last_time_ = cur_time;
00319 }
00320 }
00321 }
00322
00324
00325 double GazeboRosP3D::GaussianKernel(double mu, double sigma)
00326 {
00327
00328
00329
00330
00331 double U = static_cast<double>(rand_r(&this->seed)) /
00332 static_cast<double>(RAND_MAX);
00333
00334
00335 double V = static_cast<double>(rand_r(&this->seed)) /
00336 static_cast<double>(RAND_MAX);
00337
00338 double X = sqrt(-2.0 * ::log(U)) * cos(2.0*M_PI * V);
00339
00340
00341
00342
00343 X = sigma * X + mu;
00344 return X;
00345 }
00346
00348
00349 void GazeboRosP3D::P3DQueueThread()
00350 {
00351 static const double timeout = 0.01;
00352
00353 while (this->rosnode_->ok())
00354 {
00355 this->p3d_queue_.callAvailable(ros::WallDuration(timeout));
00356 }
00357 }
00358 }