$search
00001 /* 00002 * map_saver 00003 * Copyright (c) 2008, Willow Garage, Inc. 00004 * All rights reserved. 00005 * 00006 * Redistribution and use in source and binary forms, with or without 00007 * modification, are permitted provided that the following conditions are met: 00008 * 00009 * * Redistributions of source code must retain the above copyright 00010 * notice, this list of conditions and the following disclaimer. 00011 * * Redistributions in binary form must reproduce the above copyright 00012 * notice, this list of conditions and the following disclaimer in the 00013 * documentation and/or other materials provided with the distribution. 00014 * * Neither the name of the <ORGANIZATION> nor the names of its 00015 * contributors may be used to endorse or promote products derived from 00016 * this software without specific prior written permission. 00017 * 00018 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 00019 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 00020 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 00021 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 00022 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 00023 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 00024 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 00025 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 00026 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 00027 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 00028 * POSSIBILITY OF SUCH DAMAGE. 00029 */ 00030 00031 #include <cstdio> 00032 #include "ros/ros.h" 00033 #include "ros/console.h" 00034 #include "nav_msgs/GetMap.h" 00035 #include "LinearMath/btMatrix3x3.h" 00036 #include "geometry_msgs/Quaternion.h" 00037 00038 using namespace std; 00039 00043 class MapGenerator 00044 { 00045 00046 public: 00047 MapGenerator(const std::string& mapname) : mapname_(mapname), saved_map_(false) 00048 { 00049 ros::NodeHandle n; 00050 ROS_INFO("Waiting for the map"); 00051 map_sub_ = n.subscribe("map", 1, &MapGenerator::mapCallback, this); 00052 } 00053 00054 void mapCallback(const nav_msgs::OccupancyGridConstPtr& map) 00055 { 00056 ROS_INFO("Received a %d X %d map @ %.3f m/pix", 00057 map->info.width, 00058 map->info.height, 00059 map->info.resolution); 00060 00061 00062 std::string mapdatafile = mapname_ + ".pgm"; 00063 ROS_INFO("Writing map occupancy data to %s", mapdatafile.c_str()); 00064 FILE* out = fopen(mapdatafile.c_str(), "w"); 00065 if (!out) 00066 { 00067 ROS_ERROR("Couldn't save map file to %s", mapdatafile.c_str()); 00068 return; 00069 } 00070 00071 fprintf(out, "P5\n# CREATOR: Map_generator.cpp %.3f m/pix\n%d %d\n255\n", 00072 map->info.resolution, map->info.width, map->info.height); 00073 for(unsigned int y = 0; y < map->info.height; y++) { 00074 for(unsigned int x = 0; x < map->info.width; x++) { 00075 unsigned int i = x + (map->info.height - y - 1) * map->info.width; 00076 if (map->data[i] == 0) { //occ [0,0.1) 00077 fputc(254, out); 00078 } else if (map->data[i] == +100) { //occ (0.65,1] 00079 fputc(000, out); 00080 } else { //occ [0.1,0.65] 00081 fputc(205, out); 00082 } 00083 } 00084 } 00085 00086 fclose(out); 00087 00088 00089 std::string mapmetadatafile = mapname_ + ".yaml"; 00090 ROS_INFO("Writing map occupancy data to %s", mapmetadatafile.c_str()); 00091 FILE* yaml = fopen(mapmetadatafile.c_str(), "w"); 00092 00093 00094 /* 00095 resolution: 0.100000 00096 origin: [0.000000, 0.000000, 0.000000] 00097 # 00098 negate: 0 00099 occupied_thresh: 0.65 00100 free_thresh: 0.196 00101 00102 */ 00103 00104 geometry_msgs::Quaternion orientation = map->info.origin.orientation; 00105 btMatrix3x3 mat(btQuaternion(orientation.x, orientation.y, orientation.z, orientation.w)); 00106 double yaw, pitch, roll; 00107 mat.getEulerYPR(yaw, pitch, roll); 00108 00109 fprintf(yaml, "image: %s\nresolution: %f\norigin: [%f, %f, %f]\nnegate: 0\noccupied_thresh: 0.65\nfree_thresh: 0.196\n\n", 00110 mapdatafile.c_str(), map->info.resolution, map->info.origin.position.x, map->info.origin.position.y, yaw); 00111 00112 fclose(yaml); 00113 00114 ROS_INFO("Done\n"); 00115 saved_map_ = true; 00116 } 00117 00118 std::string mapname_; 00119 ros::Subscriber map_sub_; 00120 bool saved_map_; 00121 00122 }; 00123 00124 #define USAGE "Usage: \n" \ 00125 " map_saver -h\n"\ 00126 " map_saver [-f <mapname>] [ROS remapping args]" 00127 00128 int main(int argc, char** argv) 00129 { 00130 ros::init(argc, argv, "map_saver"); 00131 std::string mapname = "map"; 00132 00133 for(int i=1; i<argc; i++) 00134 { 00135 if(!strcmp(argv[i], "-h")) 00136 { 00137 puts(USAGE); 00138 return 0; 00139 } 00140 else if(!strcmp(argv[i], "-f")) 00141 { 00142 if(++i < argc) 00143 mapname = argv[i]; 00144 else 00145 { 00146 puts(USAGE); 00147 return 1; 00148 } 00149 } 00150 else 00151 { 00152 puts(USAGE); 00153 return 1; 00154 } 00155 } 00156 00157 MapGenerator mg(mapname); 00158 00159 while(!mg.saved_map_) 00160 ros::spinOnce(); 00161 00162 return 0; 00163 } 00164 00165