gtsam/slam/dataset.cpp

225 lines
7.1 KiB
C++
Raw Normal View History

/* ----------------------------------------------------------------------------
* GTSAM Copyright 2010, Georgia Tech Research Corporation,
* Atlanta, Georgia 30332-0415
* All Rights Reserved
* Authors: Frank Dellaert, et al. (see THANKS for the full author list)
* See LICENSE for the license information
* -------------------------------------------------------------------------- */
2010-01-23 04:18:40 +08:00
/*
* dataset.cpp
*
* Created on: Jan 22, 2010
* Author: nikai
* Description: utility functions for loading datasets
*/
#include <fstream>
#include <sstream>
#include <cstdlib>
#include <gtsam/inference/graph-inl.h>
2010-01-23 04:18:40 +08:00
#include <gtsam/slam/dataset.h>
2010-01-23 04:18:40 +08:00
using namespace std;
using namespace gtsam;
#define LINESIZE 81920
typedef boost::shared_ptr<Pose2Graph> sharedPose2Graph;
typedef boost::shared_ptr<Pose2Values> sharedPose2Values;
2010-01-23 04:18:40 +08:00
namespace gtsam {
/* ************************************************************************* */
pair<string, boost::optional<SharedDiagonal> > dataset(const string& dataset, const string& user_path) {
string path = user_path, set = dataset;
boost::optional<SharedDiagonal> null_model;
boost::optional<SharedDiagonal> identity(noiseModel::Unit::Create(3));
boost::optional<SharedDiagonal> small(noiseModel::Diagonal::Variances(
gtsam::Vector_(3, 0.0001, 0.0001, 0.0003)));
2010-01-23 04:18:40 +08:00
if (path.empty()) path = string(getenv("HOME")) + "/";
if (set.empty()) set = string(getenv("DATASET"));
if (set == "intel") return make_pair(path + "borg/CitySLAM/data/Intel/intel.graph", null_model);
if (set == "intel-gfs") return make_pair(path + "borg/CitySLAM/data/Intel/intel.gfs.graph", null_model);
if (set == "Killian-gfs") return make_pair(path + "borg/CitySLAM/data/Killian/Killian.gfs.graph", null_model);
if (set == "Killian") return make_pair(path + "borg/CitySLAM/data/Killian/Killian.graph", small);
if (set == "Killian-noised") return make_pair(path + "borg/CitySLAM/data/Killian/Killian-noised.graph", null_model);
if (set == "3") return make_pair(path + "borg/CitySLAM/data/TORO/w3-odom.graph", identity);
2010-03-05 13:41:21 +08:00
if (set == "100") return make_pair(path + "borg/CitySLAM/data/TORO/w100-odom.graph", identity);
if (set == "10K") return make_pair(path + "borg/CitySLAM/data/TORO/w10000-odom.graph", identity);
2010-07-15 12:41:17 +08:00
if (set == "10K2") return make_pair(path + "borg/hogman/data/2D/w10000.graph",
noiseModel::Diagonal::Variances(gtsam::Vector_(3, 0.1, 0.1, 0.05)));
2010-03-13 05:56:14 +08:00
if (set == "Eiffel100") return make_pair(path + "borg/CitySLAM/data/TORO/w100-Eiffel.graph", identity);
if (set == "Eiffel10K") return make_pair(path + "borg/CitySLAM/data/TORO/w10000-Eiffel.graph", identity);
if (set == "olson") return make_pair(path + "borg/CitySLAM/data/Olson/olson06icra.graph", null_model);
2010-03-09 04:35:35 +08:00
if (set == "victoria") return make_pair(path + "borg/CitySLAM/data/VictoriaPark/victoria_park.graph", null_model);
if (set == "beijing") return make_pair(path + "borg/CitySLAM/data/Beijing/beijingData_trips.graph", null_model);
return make_pair("unknown", null_model);
2010-01-23 04:18:40 +08:00
}
/* ************************************************************************* */
pair<sharedPose2Graph, sharedPose2Values> load2D(
pair<string, boost::optional<SharedDiagonal> > dataset,
int maxID, bool addNoise, bool smart) {
return load2D(dataset.first, dataset.second, maxID, addNoise, smart);
}
pair<sharedPose2Graph, sharedPose2Values> load2D(const string& filename,
boost::optional<SharedDiagonal> model, int maxID, bool addNoise, bool smart) {
2010-01-23 04:18:40 +08:00
cout << "Will try to read " << filename << endl;
ifstream is(filename.c_str());
if (!is) {
cout << "load2D: can not find the file!";
exit(-1);
}
sharedPose2Values poses(new Pose2Values);
2010-01-23 04:18:40 +08:00
sharedPose2Graph graph(new Pose2Graph);
string tag;
2010-01-23 04:18:40 +08:00
// load the poses
while (is) {
is >> tag;
2010-01-23 04:18:40 +08:00
if ((tag == "VERTEX2") || (tag == "VERTEX")) {
int id;
double x, y, yaw;
is >> id >> x >> y >> yaw;
2010-01-23 04:18:40 +08:00
// optional filter
if (maxID && id >= maxID) continue;
poses->insert(id, Pose2(x, y, yaw));
}
is.ignore(LINESIZE, '\n');
2010-01-23 04:18:40 +08:00
}
is.clear(); /* clears the end-of-file and error flags */
is.seekg(0, ios::beg);
// load the factors
while (is) {
is >> tag;
2010-01-23 04:18:40 +08:00
if ((tag == "EDGE2") || (tag == "EDGE") || (tag == "ODOMETRY")) {
int id1, id2;
double x, y, yaw;
2010-03-09 04:35:35 +08:00
is >> id1 >> id2 >> x >> y >> yaw;
2010-01-23 04:18:40 +08:00
Matrix m = eye(3);
is >> m(0, 0) >> m(0, 1) >> m(1, 1) >> m(2, 2) >> m(0, 2) >> m(1, 2);
2010-01-23 04:18:40 +08:00
m(2, 0) = m(0, 2);
m(2, 1) = m(1, 2);
m(1, 0) = m(0, 1);
// optional filter
if (maxID && (id1 >= maxID || id2 >= maxID)) continue;
2010-01-27 11:41:23 +08:00
Pose2 l1Xl2(x, y, yaw);
2010-01-23 04:18:40 +08:00
// SharedGaussian noise = noiseModel::Gaussian::Covariance(m, smart);
2010-01-23 04:18:40 +08:00
if (!model) {
2010-03-09 02:45:22 +08:00
Vector variances = Vector_(3,m(0,0),m(1,1),m(2,2));
2010-01-23 04:18:40 +08:00
model = noiseModel::Diagonal::Variances(variances, smart);
}
if (addNoise)
l1Xl2 = l1Xl2.expmap((*model)->sample());
2010-01-23 04:18:40 +08:00
// Insert vertices if pure odometry file
if (!poses->exists(id1)) poses->insert(id1, Pose2());
2010-01-27 11:41:23 +08:00
if (!poses->exists(id2)) poses->insert(id2, poses->at(id1) * l1Xl2);
2010-01-23 04:18:40 +08:00
2010-01-27 11:41:23 +08:00
Pose2Graph::sharedFactor factor(new Pose2Factor(id1, id2, l1Xl2, *model));
2010-01-23 04:18:40 +08:00
graph->push_back(factor);
}
is.ignore(LINESIZE, '\n');
2010-01-23 04:18:40 +08:00
}
cout << "load2D read a graph file with " << poses->size() << " vertices and "
<< graph->nrFactors() << " factors" << endl;
return make_pair(graph, poses);
}
/* ************************************************************************* */
void save2D(const Pose2Graph& graph, const Pose2Values& config, const SharedDiagonal model,
2010-01-23 04:18:40 +08:00
const string& filename) {
typedef Pose2Values::Key Key;
2010-01-23 04:18:40 +08:00
fstream stream(filename.c_str(), fstream::out);
// save poses
Pose2Values::Key key;
2010-01-23 04:18:40 +08:00
Pose2 pose;
BOOST_FOREACH(boost::tie(key, pose), config)
stream << "VERTEX2 " << key.index() << " " << pose.x() << " " << pose.y() << " " << pose.theta() << endl;
// save edges
Matrix R = model->R();
Matrix RR = prod(trans(R),R);
BOOST_FOREACH(boost::shared_ptr<NonlinearFactor<Pose2Values> > factor_, graph) {
2010-01-23 04:18:40 +08:00
boost::shared_ptr<Pose2Factor> factor = boost::dynamic_pointer_cast<Pose2Factor>(factor_);
if (!factor) continue;
pose = factor->measured().inverse();
2010-01-23 04:18:40 +08:00
stream << "EDGE2 " << factor->key2().index() << " " << factor->key1().index()
<< " " << pose.x() << " " << pose.y() << " " << pose.theta()
<< " " << RR(0, 0) << " " << RR(0, 1) << " " << RR(1, 1) << " " << RR(2, 2)
<< " " << RR(0, 2) << " " << RR(1, 2) << endl;
2010-01-23 04:18:40 +08:00
}
stream.close();
}
/* ************************************************************************* */
bool load3D(const string& filename) {
ifstream is(filename.c_str());
if (!is) return false;
while (is) {
char buf[LINESIZE];
is.getline(buf, LINESIZE);
istringstream ls(buf);
string tag;
ls >> tag;
if (tag == "VERTEX3") {
int id;
double x, y, z, roll, pitch, yaw;
ls >> id >> x >> y >> z >> roll >> pitch >> yaw;
}
}
is.clear(); /* clears the end-of-file and error flags */
is.seekg(0, ios::beg);
while (is) {
char buf[LINESIZE];
is.getline(buf, LINESIZE);
istringstream ls(buf);
string tag;
ls >> tag;
if (tag == "EDGE3") {
int id1, id2;
double x, y, z, roll, pitch, yaw;
ls >> id1 >> id2 >> x >> y >> z >> roll >> pitch >> yaw;
Matrix m = eye(6);
for (int i = 0; i < 6; i++)
for (int j = i; j < 6; j++)
ls >> m(i, j);
}
}
return true;
}
}