Add time skip option for offline node ()

Introduces a "skip" option which skips first _t_ seconds.
master
Juraj Oršulić 2018-03-05 20:33:57 +01:00 committed by Wally B. Feed
parent 4746d5a734
commit b274743eb7
3 changed files with 28 additions and 13 deletions
cartographer_ros/cartographer_ros

View File

@ -60,6 +60,9 @@ DEFINE_bool(load_frozen_state, true,
DEFINE_bool(keep_running, false, DEFINE_bool(keep_running, false,
"Keep running the offline node after all messages from the bag " "Keep running the offline node after all messages from the bag "
"have been processed."); "have been processed.");
DEFINE_double(skip_seconds, 0,
"Optional amount of seconds to skip from the beginning "
"(i.e. when the earliest bag starts.). ");
namespace cartographer_ros { namespace cartographer_ros {
@ -233,12 +236,21 @@ void RunOfflineNode(const MapBuilderFactory& map_builder_factory) {
// TODO(gaschler): Warn if resolved topics are not in bags. // TODO(gaschler): Warn if resolved topics are not in bags.
std::unordered_map<int, int> bag_index_to_trajectory_id; std::unordered_map<int, int> bag_index_to_trajectory_id;
const ros::Time begin_time = playable_bag_multiplexer.PeekMessageTime();
while (playable_bag_multiplexer.IsMessageAvailable()) { while (playable_bag_multiplexer.IsMessageAvailable()) {
if (!::ros::ok()) {
return;
}
const auto next_msg_tuple = playable_bag_multiplexer.GetNextMessage(); const auto next_msg_tuple = playable_bag_multiplexer.GetNextMessage();
const rosbag::MessageInstance& msg = std::get<0>(next_msg_tuple); const rosbag::MessageInstance& msg = std::get<0>(next_msg_tuple);
const int bag_index = std::get<1>(next_msg_tuple); const int bag_index = std::get<1>(next_msg_tuple);
const bool is_last_message_in_bag = std::get<2>(next_msg_tuple); const bool is_last_message_in_bag = std::get<2>(next_msg_tuple);
if (msg.getTime() < (begin_time + ros::Duration(FLAGS_skip_seconds))) {
continue;
}
int trajectory_id; int trajectory_id;
// Lazily add trajectories only when the first message arrives in order // Lazily add trajectories only when the first message arrives in order
// to avoid blocking the sensor queue. // to avoid blocking the sensor queue.
@ -257,9 +269,6 @@ void RunOfflineNode(const MapBuilderFactory& map_builder_factory) {
trajectory_id = bag_index_to_trajectory_id.at(bag_index); trajectory_id = bag_index_to_trajectory_id.at(bag_index);
} }
if (!::ros::ok()) {
return;
}
const auto bag_topic = std::make_pair( const auto bag_topic = std::make_pair(
bag_index, bag_index,
node.node_handle()->resolveName(msg.getTopic(), false /* resolve */)); node.node_handle()->resolveName(msg.getTopic(), false /* resolve */));

View File

@ -45,12 +45,12 @@ PlayableBag::PlayableBag(
AdvanceUntilMessageAvailable(); AdvanceUntilMessageAvailable();
} }
ros::Time PlayableBag::PeekMessageTime() { ros::Time PlayableBag::PeekMessageTime() const {
CHECK(IsMessageAvailable()); CHECK(IsMessageAvailable());
return buffered_messages_.front().getTime(); return buffered_messages_.front().getTime();
} }
std::tuple<ros::Time, ros::Time> PlayableBag::GetBeginEndTime() { std::tuple<ros::Time, ros::Time> PlayableBag::GetBeginEndTime() const {
return std::make_tuple(view_->getBeginTime(), view_->getEndTime()); return std::make_tuple(view_->getBeginTime(), view_->getEndTime());
} }
@ -67,13 +67,13 @@ rosbag::MessageInstance PlayableBag::GetNextMessage() {
return msg; return msg;
} }
bool PlayableBag::IsMessageAvailable() { bool PlayableBag::IsMessageAvailable() const {
return !buffered_messages_.empty() && return !buffered_messages_.empty() &&
(buffered_messages_.front().getTime() < (buffered_messages_.front().getTime() <
buffered_messages_.back().getTime() - buffer_delay_); buffered_messages_.back().getTime() - buffer_delay_);
} }
int PlayableBag::bag_id() { return bag_id_; } int PlayableBag::bag_id() const { return bag_id_; }
void PlayableBag::AdvanceOneMessage() { void PlayableBag::AdvanceOneMessage() {
CHECK(!finished_); CHECK(!finished_);
@ -106,7 +106,7 @@ void PlayableBagMultiplexer::AddPlayableBag(PlayableBag playable_bag) {
static_cast<int>(playable_bags_.size() - 1)}); static_cast<int>(playable_bags_.size() - 1)});
} }
bool PlayableBagMultiplexer::IsMessageAvailable() { bool PlayableBagMultiplexer::IsMessageAvailable() const {
return !next_message_queue_.empty(); return !next_message_queue_.empty();
} }
@ -126,4 +126,9 @@ PlayableBagMultiplexer::GetNextMessage() {
!current_bag.IsMessageAvailable()); !current_bag.IsMessageAvailable());
} }
ros::Time PlayableBagMultiplexer::PeekMessageTime() const {
CHECK(IsMessageAvailable());
return next_message_queue_.top().message_timestamp;
}
} // namespace cartographer_ros } // namespace cartographer_ros

View File

@ -37,12 +37,12 @@ class PlayableBag {
ros::Time end_time, ros::Duration buffer_delay, ros::Time end_time, ros::Duration buffer_delay,
FilteringEarlyMessageHandler filtering_early_message_handler); FilteringEarlyMessageHandler filtering_early_message_handler);
ros::Time PeekMessageTime(); ros::Time PeekMessageTime() const;
rosbag::MessageInstance GetNextMessage(); rosbag::MessageInstance GetNextMessage();
bool IsMessageAvailable(); bool IsMessageAvailable() const;
std::tuple<ros::Time, ros::Time> GetBeginEndTime(); std::tuple<ros::Time, ros::Time> GetBeginEndTime() const;
int bag_id(); int bag_id() const;
private: private:
void AdvanceOneMessage(); void AdvanceOneMessage();
@ -72,7 +72,8 @@ class PlayableBagMultiplexer {
bool /* is_last_message_in_bag */> bool /* is_last_message_in_bag */>
GetNextMessage(); GetNextMessage();
bool IsMessageAvailable(); bool IsMessageAvailable() const;
ros::Time PeekMessageTime() const;
private: private:
struct BagMessageItem { struct BagMessageItem {