00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00036
00037
00038 #include <ros/ros.h>
00039
00040 #include <tf/tf.h>
00041 #include <tf/transform_listener.h>
00042 #include <tf/transform_broadcaster.h>
00043
00044 #include <boost/bind.hpp>
00045
00046 #include "pcl/io/pcd_io.h"
00047 #include "pcl/point_types.h"
00048
00049 #include <sensor_msgs/point_cloud_conversion.h>
00050
00051 #include <visualization_msgs/Marker.h>
00052 #include <visualization_msgs/MarkerArray.h>
00053
00054 #include <object_manipulation_msgs/GraspPlanning.h>
00055 #include <object_manipulation_msgs/FindClusterBoundingBox.h>
00056 #include <object_manipulator/tools/service_action_wrappers.h>
00057
00058 #include <dynamic_reconfigure/server.h>
00059 #include "pr2_grasp_adjust/EstimateConfig.h"
00060 #include "pr2_grasp_adjust/DebugConfig.h"
00061 #include "pr2_grasp_adjust/GraspAdjust.h"
00062
00063
00064 #include <iostream>
00065 #include <fstream>
00066 #include <queue>
00067
00068 #include "gripper_model.h"
00069 #include "helpers.h"
00070 #include "grasp_adjust.h"
00071
00072 #include <pcl/ModelCoefficients.h>
00073 #include <pcl/features/normal_3d.h>
00074 #include <pcl/filters/extract_indices.h>
00075 #include <pcl/filters/passthrough.h>
00076 #include <pcl/sample_consensus/method_types.h>
00077 #include <pcl/sample_consensus/model_types.h>
00078 #include <pcl/segmentation/sac_segmentation.h>
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090 typedef pcl::PointXYZRGBNormal PointT;
00091
00092 const int GLOBAL_SEARCH = pr2_grasp_adjust::Estimate_global_search;
00093 const int LOCAL_SEARCH = pr2_grasp_adjust::Estimate_local_search;
00094 const int SINGLE_POSE = pr2_grasp_adjust::Estimate_single_pose;
00095
00096
00097 std::ofstream outputFile;
00098
00099 class GraspAdjustServer{
00100
00101 pcl::PointCloud<PointT> last_cloud_;
00102
00103 ros::NodeHandle nh_, nh_pvt_;
00104 ros::Subscriber sub_cloud_;
00105 ros::Publisher pub_cloud_, pub_marker_, pub_marker_array_;
00106 ros::Publisher pub_cloud_roi_, pub_cloud_debug_2;
00107
00108 ros::ServiceServer adjust_srv_, evaluate_srv_, plan_srv_;
00109 ros::Timer update_timer_;
00110
00111 std::string input_topic_;
00112 std::string output_topic_;
00113
00114 tf::TransformListener listener_;
00115 tf::TransformBroadcaster broadcaster_;
00116
00117 ros::Time now_;
00118
00119 bool got_first_cloud_;
00120
00121 GraspAdjust grasp_adjust_;
00122
00123
00124 typedef pr2_grasp_adjust::EstimateConfig Config;
00125 dynamic_reconfigure::Server<Config> dyn_srv;
00126 dynamic_reconfigure::Server<Config>::CallbackType dyn_cb;
00127
00128 typedef pr2_grasp_adjust::DebugConfig Debug;
00129 dynamic_reconfigure::Server<Debug> debug_srv;
00130 dynamic_reconfigure::Server<Debug>::CallbackType debug_cb;
00131
00132
00133 object_manipulator::ServiceWrapper<object_manipulation_msgs::FindClusterBoundingBox> bounding_box_client_;
00134
00135
00136 bool training_;
00137 std::ofstream train_file_;
00138
00139 public:
00143 GraspAdjustServer(char* argv[]):
00144 nh_("/"),
00145 nh_pvt_("~"),
00146 input_topic_("/cloud_in"),
00147 output_topic_("/cloud_out"),
00148 dyn_srv(ros::NodeHandle("~config")),
00149 debug_srv(ros::NodeHandle("~debug")),
00150 bounding_box_client_("/find_cluster_bounding_box"),
00151 training_(0)
00152 {
00153 got_first_cloud_ = false;
00154 now_ = ros::Time::now() - ros::Duration(1.0);
00155
00156
00157 sub_cloud_ = nh_.subscribe<sensor_msgs::PointCloud2>(input_topic_, 1, boost::bind(&GraspAdjustServer::cloud_callback, this, _1));
00158 pub_cloud_ = nh_.advertise<sensor_msgs::PointCloud2>(output_topic_, 1);
00159 pub_cloud_roi_ = nh_.advertise<sensor_msgs::PointCloud2>("/roi_snapshot", 1);
00160
00161
00162 pub_marker_ = nh_.advertise<visualization_msgs::Marker>("/markers", 20);
00163 pub_marker_array_ = nh_.advertise<visualization_msgs::MarkerArray>("/markers_array", 20);
00164
00165 grasp_adjust_.pub_cloud_ = &pub_cloud_;
00166 grasp_adjust_.pub_cloud_roi_ = &pub_cloud_roi_;
00167 grasp_adjust_.pub_marker_ = &pub_marker_;
00168 grasp_adjust_.pub_marker_array_ = &pub_marker_array_;
00169 grasp_adjust_.listener_ = &listener_;
00170 grasp_adjust_.broadcaster_ = &broadcaster_;
00171
00172
00173 update_timer_ = nh_.createTimer(ros::Duration(0.5), boost::bind(&GraspAdjustServer::timerCallback, this));
00174
00175 bool load_debug_server = false;
00176 nh_pvt_.param("load_debug_server", load_debug_server, load_debug_server);
00177
00178
00179 dyn_cb = boost::bind( &GraspAdjustServer::dynamicCallback, this, _1, _2 );
00180 dyn_srv.setCallback(dyn_cb);
00181
00182
00183 if(load_debug_server){
00184 ROS_INFO("Loading the debug dynamic reconfigure server");
00185 debug_cb = boost::bind( &GraspAdjustServer::debugCallback, this, _1, _2 );
00186 debug_srv.setCallback(debug_cb);
00187 }
00188
00189 adjust_srv_ = nh_.advertiseService("pr2_grasp_adjust", &GraspAdjustServer::graspAdjustCallback, this);
00190 evaluate_srv_ = nh_.advertiseService("pr2_gripper_grasp_evaluate", &GraspAdjustServer::graspEvaluateCallback, this);
00191 plan_srv_ = nh_.advertiseService("pr2_gripper_grasp_plan", &GraspAdjustServer::graspPlanCallback, this);
00192
00193
00194 if(training_){
00195 ROS_INFO("in training mode, opening training_output.txt");
00196 train_file_.open("/u/hsiao/gripper_stereo_data/training_data/training_output.txt");
00197 }
00198
00199 ROS_INFO("Finished constructor!");
00200 }
00201
00202
00203 private:
00204
00208 void timerCallback()
00209 {
00210 now_= last_cloud_.header.stamp;
00211 geometry_msgs::PoseStamped pose_msg;
00212 tf::Pose pose;
00213
00214 if(got_first_cloud_ && grasp_adjust_.debug_.run_continuous)
00215 {
00216 if(grasp_adjust_.debug_.custom_pose)
00217 {
00218 if(grasp_adjust_.debug_.custom_frame.empty())
00219 pose_msg.header.frame_id = "torso_lift_link";
00220 else
00221 pose_msg.header.frame_id = grasp_adjust_.debug_.custom_frame;
00222 pose_msg.header.stamp = now_;
00223
00224 float x = grasp_adjust_.debug_.custom_x;
00225 float y = grasp_adjust_.debug_.custom_y;
00226 float z = grasp_adjust_.debug_.custom_z;
00227 float roll = grasp_adjust_.debug_.custom_roll * M_PI/180.0;
00228 float pitch = grasp_adjust_.debug_.custom_pitch * M_PI/180.0;
00229 float yaw = grasp_adjust_.debug_.custom_yaw * M_PI/180.0;
00230
00231 tf::Transform T_rot;
00232 tf::Transform T_offset;
00233 tf::Transform T_trans = tf::Transform(tf::Quaternion(0,0,0,1),
00234 tf::Vector3(x, y, z));
00235 T_rot = tf::Transform(tf::createQuaternionFromRPY(roll, pitch, yaw),
00236 tf::Vector3( 0, 0, 0) );
00237 T_offset = T_trans * grasp_adjust_.DEFAULT_GRIPPER.tool_frame_ * T_rot * grasp_adjust_.DEFAULT_GRIPPER.tool_frame_.inverse();
00238 pose = T_offset;
00239 }
00240 else
00241 {
00242 pose_msg.header.frame_id = "r_wrist_roll_link";
00243 pose_msg.header.stamp = now_;;
00244 pose = tf::Pose(tf::Quaternion(0,0,0,1), tf::Vector3(0.06, 0, 0));
00245 }
00246
00247 tf::poseTFToMsg(pose, pose_msg.pose);
00248 std::vector<geometry_msgs::PoseStamped> adjusted_poses;
00249 std::vector<double> pose_scores;
00250
00251 int value = grasp_adjust_.findGrasps(pose_msg, &adjusted_poses, &pose_scores, grasp_adjust_.config_.search_mode);
00252 ROS_INFO("pr2_grasp_adjust::grasp_adjust returned %d", value);
00253 }
00254 }
00255
00261 bool graspAdjustCallback( pr2_grasp_adjust::GraspAdjust::Request &req,
00262 pr2_grasp_adjust::GraspAdjust::Response &res )
00263 {
00264
00265 int mode = req.search_mode;
00266 res.result.value = grasp_adjust_.findGrasps(req.grasp_pose, &(res.grasp_poses), &(res.pose_scores), mode);
00267
00268 ROS_INFO("pr2_grasp_adjust::grasp_adjust returned %d", res.result.value);
00269
00270 return true;
00271 }
00272
00278 void setupGraspPlanning(object_manipulation_msgs::GraspPlanning::Request &req, pcl::PointCloud<PointT> &cloud)
00279 {
00280 pcl::PointCloud<pcl::PointXYZ> cloud_in;
00281
00282 sensor_msgs::PointCloud2 cloud_msg;
00283
00284 sensor_msgs::convertPointCloudToPointCloud2(req.target.cluster, cloud_msg);
00285 ROS_INFO("Cloud_in in frame %s, reference frame %s!", cloud_msg.header.frame_id.c_str(), req.target.reference_frame_id.c_str());
00286
00287 pcl::fromROSMsg(cloud_msg, cloud_in);
00288
00289
00290
00291 pcl::KdTreeFLANN<pcl::PointXYZ>::Ptr tree (new pcl::KdTreeFLANN<pcl::PointXYZ> ());
00292 pcl::NormalEstimation<pcl::PointXYZ, PointT> ne;
00293 ne.setSearchMethod (tree);
00294 ne.setInputCloud (boost::make_shared<pcl::PointCloud<pcl::PointXYZ> >(cloud_in));
00295 ne.setKSearch (15);
00296 ne.compute (cloud);
00297 for(unsigned int i = 0; i < cloud_in.size(); i++){
00298 cloud.points[i].x = cloud_in.points[i].x;
00299 cloud.points[i].y = cloud_in.points[i].y;
00300 cloud.points[i].z = cloud_in.points[i].z;
00301 }
00302 ROS_INFO("Cloud_out in frame %s has %d points, stamp %f!", cloud.header.frame_id.c_str(), (int)cloud.size(), cloud.header.stamp.toSec());
00303
00304 sensor_msgs::PointCloud2 cloud_out;
00305 pcl::toROSMsg(cloud, cloud_out);
00306 pub_cloud_.publish(cloud_out);
00307
00308 grasp_adjust_.setInputCloud(cloud);
00309 }
00310
00311
00315 geometry_msgs::PoseStamped generateRelativePose(geometry_msgs::PoseStamped original_pose, double roll, double pitch,
00316 double yaw, double xdist, double ydist, double zdist)
00317 {
00318 tf::Pose pose_tf;
00319 tf::poseMsgToTF(original_pose.pose, pose_tf);
00320 tf::Transform T_rot = tf::Transform(tf::createQuaternionFromRPY(roll, pitch, yaw),
00321 tf::Vector3( xdist, ydist, zdist) );
00322 tf::Transform output_rot = pose_tf * T_rot;
00323 geometry_msgs::PoseStamped output_pose;
00324 output_pose.header.frame_id = original_pose.header.frame_id;
00325 tf::poseTFToMsg(output_rot, output_pose.pose);
00326 return output_pose;
00327 }
00328
00334 bool graspPlanCallback( object_manipulation_msgs::GraspPlanning::Request &req,
00335 object_manipulation_msgs::GraspPlanning::Response &res )
00336 {
00337
00338 unsigned int MAX_GRASPS = 20;
00339 GripperModel clear_model;
00340 clear_model.setStamp(ros::Time::now());
00341 clear_model.setFrameId(req.target.reference_frame_id);
00342 if(pub_marker_array_.getNumSubscribers() > 0)
00343 for(unsigned int i = 0; i < MAX_GRASPS; i++)
00344 {
00345 clear_model.publishModel(pub_marker_array_, "grasp_planning", i, ros::Duration(), 1, visualization_msgs::Marker::DELETE);
00346 }
00347
00348
00349 ROS_INFO("starting graspPlanCallback");
00350 pcl::PointCloud<PointT> cloud;
00351 setupGraspPlanning(req, cloud);
00352
00353
00354 std::string arm_name = "right_arm";
00355 if(req.arm_name.length() != 0) arm_name = req.arm_name;
00356 else ROS_ERROR("point cluster grasp planner: missing arm_name in request! Using right_arm");
00357 std::vector<std::string> hand_joints;
00358
00359 XmlRpc::XmlRpcValue my_list;
00360 nh_.getParam(std::string("/hand_description/")+arm_name+std::string("/hand_joints"), my_list);
00361 ROS_ASSERT(my_list.getType() == XmlRpc::XmlRpcValue::TypeArray);
00362 for (int32_t i = 0; i < my_list.size(); ++i)
00363 {
00364 hand_joints.push_back(static_cast<std::string>(my_list[i]));
00365 }
00366
00367
00368 object_manipulation_msgs::FindClusterBoundingBox bounding_box_srv;
00369 ros::ServiceClient client = bounding_box_client_.client(ros::Duration(5.0));
00370 bounding_box_srv.request.cluster = req.target.cluster;
00371 bool result = client.call(bounding_box_srv);
00372 if(!result)
00373 {
00374 ROS_ERROR("Failed to call find_cluster_bounding_box service!");
00375 res.error_code.value = object_manipulation_msgs::GraspPlanningErrorCode::OTHER_ERROR;
00376 return true;
00377 }
00378
00379
00380 std::vector<geometry_msgs::PoseStamped> start_poses;
00381 double start_dist = .01;
00382
00383 start_poses.push_back(generateRelativePose(bounding_box_srv.response.pose, 0.,0.,0.,
00384 -(bounding_box_srv.response.box_dims.x/2+.185+start_dist),0.,0.));
00385
00386
00387
00388
00389 start_poses.push_back(generateRelativePose(bounding_box_srv.response.pose, 0.,0.,-M_PI/2,
00390 0.,bounding_box_srv.response.box_dims.y/2+.185+start_dist,0));
00391
00392 start_poses.push_back(generateRelativePose(bounding_box_srv.response.pose, 0.,0.,M_PI,
00393 bounding_box_srv.response.box_dims.x/2+.185+start_dist,0.,0.));
00394
00395 start_poses.push_back(generateRelativePose(bounding_box_srv.response.pose, 0.,M_PI/2,0.,
00396 0.,0.,bounding_box_srv.response.box_dims.z/2+.185+start_dist));
00397
00398 start_poses.push_back(generateRelativePose(bounding_box_srv.response.pose, M_PI/2,M_PI/2.0,0.,
00399 0.,0.,bounding_box_srv.response.box_dims.z/2+.185+start_dist));
00400
00401
00402 std::priority_queue<GripperModel, std::vector<GripperModel>, GripperModel::compareModels> ranked_poses;
00403
00404
00405 for(size_t start_pose_ind=0; start_pose_ind < start_poses.size(); start_pose_ind++)
00406 {
00407 int mode = GLOBAL_SEARCH;
00408 std::vector<geometry_msgs::PoseStamped> adjusted_poses;
00409 std::vector<double> pose_scores;
00410
00411 int value = grasp_adjust_.findGrasps(start_poses[start_pose_ind], &ranked_poses, mode, req.target.reference_frame_id);
00412 }
00413
00414 ROS_INFO("Globally, found %d valid poses!", (int)ranked_poses.size());
00415 while(!ranked_poses.empty() && res.grasps.size() < MAX_GRASPS)
00416 {
00417 GripperModel model = ranked_poses.top();
00418 ranked_poses.pop();
00419
00420 geometry_msgs::PoseStamped ps;
00421 model.toPoseStamped(ps);
00422
00423
00424 object_manipulation_msgs::Grasp new_grasp;
00425 for(size_t i=0; i<hand_joints.size(); i++)
00426 {
00427 new_grasp.pre_grasp_posture.name.push_back(hand_joints[i]);
00428 new_grasp.pre_grasp_posture.position.push_back(0.5);
00429 new_grasp.grasp_posture.name.push_back(hand_joints[i]);
00430 new_grasp.grasp_posture.position.push_back(0);
00431 }
00432 new_grasp.grasp_pose = ps.pose;
00433 new_grasp.success_probability = model.score_;
00434
00435
00436 res.grasps.push_back(new_grasp);
00437
00438 ROS_INFO("Added pose % 2d with score %.3f", (int)res.grasps.size(), model.score_);
00439 model.publishModel(pub_marker_array_, "grasp_planning", res.grasps.size(), ros::Duration(), model.score_*model.score_);
00440
00441 }
00442
00443
00444 res.error_code.value = object_manipulation_msgs::GraspPlanningErrorCode::SUCCESS;
00445 ROS_INFO("pr2_grasp_adjust::grasp_evaluate returned %d", res.error_code.value);
00446 return true;
00447 }
00448
00449
00455 bool graspEvaluateCallback( object_manipulation_msgs::GraspPlanning::Request &req,
00456 object_manipulation_msgs::GraspPlanning::Response &res )
00457 {
00458
00459 pcl::PointCloud<PointT> cloud;
00460 setupGraspPlanning(req, cloud);
00461
00462
00463 tf::Vector3 centroid = tf::Vector3(0,0,0);
00464 for(unsigned int i = 0; i < cloud.points.size(); i++ )
00465 {
00466 PointT point = cloud.points[i];
00467 centroid += tf::Vector3(point.x, point.y, point.z);
00468 }
00469 centroid /= cloud.points.size() + 0.001;
00470
00471 ROS_INFO("There are %d grasps to evaluate", (int)req.grasps_to_evaluate.size());
00472 ROS_INFO("object grasp centroid symmetry points pnt_bonus normals collision");
00473 bool CONTINUE = false;
00474
00475 for(unsigned j = 0; j < req.grasps_to_evaluate.size(); j ++)
00476 {
00477
00478 this->broadcaster_.sendTransform(tf::StampedTransform(tf::Transform(tf::Quaternion(0,0,0,1), centroid),
00479 cloud.header.stamp, cloud.header.frame_id.c_str(), "/centroid"));
00480
00481 geometry_msgs::Pose pose_msg = req.grasps_to_evaluate[j].grasp_pose;
00482 tf::Pose pose_tf;
00483 tf::poseMsgToTF(pose_msg, pose_tf);
00484
00485 FeatureWeights features;
00486 GripperModel gripper;
00487 gripper.setHeader(cloud.header);
00488 gripper.transform(pose_tf, cloud.header.frame_id.c_str());
00489
00490 gripper.publishModel(pub_marker_array_, "gripper_model", 0, ros::Duration(20.0), 2.0);
00491 gripper.evaluateGrasp(cloud, features, centroid, false, true);
00492
00493 char feature_str[200];
00494
00495 if(gripper.state_.no_object) {
00496 sprintf(feature_str, "% 6d % 6d No points in contact with gripper!", req.target.potential_models[0].model_id, j);
00497 }
00498 else {
00499 sprintf(feature_str, "% 6d % 6d % 10.5f % 10.5f % 10.5f % 10.5f % 10.5f % 10.5f",
00500 req.target.potential_models[0].model_id, j,
00501 features.centroid, features.symmetry, features.point,
00502 features.Iyz, features.normal, features.collision
00503 );
00504 }
00505
00506 outputFile << feature_str << std::endl;
00507 ROS_INFO_STREAM(feature_str);
00508
00509 if(!CONTINUE && grasp_adjust_.config_.debug_stepping) {
00510 char key[256]; std::cin.getline(key, 256);
00511 if(key[0] == 'c') CONTINUE = true;
00512 }
00513
00514 usleep(grasp_adjust_.config_.pause_time*1000000);
00515 }
00516
00517 res.error_code.value = object_manipulation_msgs::GraspPlanningErrorCode::SUCCESS;
00518 ROS_INFO("pr2_grasp_adjust::grasp_evaluate returned %d", res.error_code.value);
00519 return true;
00520 }
00521
00522
00526 void cloud_callback(const sensor_msgs::PointCloud2ConstPtr &input)
00527 {
00528 ROS_DEBUG("Got a cloud with %d points!", input->width*input->height);
00529
00530 if( !strcmp(input->fields[4].name.c_str(), "normal_x") )
00531 {
00532 pcl::fromROSMsg(*input, last_cloud_);
00533 grasp_adjust_.setInputCloud(last_cloud_);
00534
00535 if(!got_first_cloud_){
00536 got_first_cloud_ = true;
00537 ROS_INFO("Got first cloud!");
00538 }
00539 }
00540 }
00541
00542
00546 void dynamicCallback(Config &new_config, uint32_t id)
00547 {
00548
00549 char status[256] = "\0";
00550
00551 switch(id)
00552 {
00553 case(-1):
00554 break;
00555 case(0):
00556 printf("Reconfigure GUI connected to me!\n");
00557 new_config = grasp_adjust_.config_;
00558 break;
00559 case(91):
00560 if(new_config.write_file)
00561 {
00562 ROS_INFO("In training mode, opening %s", new_config.filename.c_str());
00563 outputFile.open(new_config.filename.c_str());
00564 outputFile << "object grasp centroid symmetry points pnt_bonus normals collision" << std::endl;
00565 }
00566 else
00567 {
00568 ROS_INFO("Closing file %s", new_config.filename.c_str());
00569 outputFile.close();
00570
00571 }
00572 break;
00573 default:
00574 ROS_INFO("Dynamic reconfigure did something, variable %d.", id);
00575 }
00576 grasp_adjust_.config_ = new_config;
00577 new_config.status = status;
00578 }
00579
00580
00584 void debugCallback(Debug &new_config, uint32_t id)
00585 {
00586
00587 char status[256] = "\0";
00588
00589 switch(id){
00590 case(-1):
00591 break;
00592 case(0):
00593 printf("Reconfigure GUI connected to me!\n");
00594 new_config = grasp_adjust_.debug_;
00595 break;
00596 default:
00597 ROS_INFO("Dynamic reconfigure did something, variable %d.", id);
00598 }
00599 grasp_adjust_.debug_ = new_config;
00600 new_config.status = status;
00601 }
00602 };
00603
00604
00605
00606
00607 int main (int argc, char* argv[])
00608 {
00609 ros::init(argc, argv, "filter");
00610
00611 GraspAdjustServer server(argv);
00612
00613 ros::spin();
00614
00615 return (0);
00616 }
00617