180 lines
5.6 KiB
C++
180 lines
5.6 KiB
C++
/*
|
|
* Copyright 2016 The Cartographer Authors
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "cartographer/sensor/ordered_multi_queue.h"
|
|
|
|
#include <algorithm>
|
|
#include <sstream>
|
|
#include <vector>
|
|
|
|
#include "cartographer/common/make_unique.h"
|
|
#include "glog/logging.h"
|
|
|
|
namespace cartographer {
|
|
namespace sensor {
|
|
|
|
namespace {
|
|
|
|
// Number of items that can be queued up before we log which queues are waiting
|
|
// for data.
|
|
const int kMaxQueueSize = 500;
|
|
|
|
} // namespace
|
|
|
|
inline std::ostream& operator<<(std::ostream& out, const QueueKey& key) {
|
|
return out << '(' << key.trajectory_id << ", " << key.sensor_id << ')';
|
|
}
|
|
|
|
OrderedMultiQueue::OrderedMultiQueue() {}
|
|
|
|
OrderedMultiQueue::~OrderedMultiQueue() {
|
|
for (auto& entry : queues_) {
|
|
CHECK(entry.second.finished);
|
|
}
|
|
}
|
|
|
|
void OrderedMultiQueue::AddQueue(const QueueKey& queue_key, Callback callback) {
|
|
CHECK_EQ(queues_.count(queue_key), 0);
|
|
queues_[queue_key].callback = std::move(callback);
|
|
}
|
|
|
|
void OrderedMultiQueue::MarkQueueAsFinished(const QueueKey& queue_key) {
|
|
auto it = queues_.find(queue_key);
|
|
CHECK(it != queues_.end()) << "Did not find '" << queue_key << "'.";
|
|
auto& queue = it->second;
|
|
CHECK(!queue.finished);
|
|
queue.finished = true;
|
|
Dispatch();
|
|
}
|
|
|
|
void OrderedMultiQueue::Add(const QueueKey& queue_key,
|
|
std::unique_ptr<Data> data) {
|
|
auto it = queues_.find(queue_key);
|
|
if (it == queues_.end()) {
|
|
LOG_EVERY_N(WARNING, 1000)
|
|
<< "Ignored data for queue: '" << queue_key << "'";
|
|
return;
|
|
}
|
|
it->second.queue.Push(std::move(data));
|
|
Dispatch();
|
|
}
|
|
|
|
void OrderedMultiQueue::Flush() {
|
|
std::vector<QueueKey> unfinished_queues;
|
|
for (auto& entry : queues_) {
|
|
if (!entry.second.finished) {
|
|
unfinished_queues.push_back(entry.first);
|
|
}
|
|
}
|
|
for (auto& unfinished_queue : unfinished_queues) {
|
|
MarkQueueAsFinished(unfinished_queue);
|
|
}
|
|
}
|
|
|
|
QueueKey OrderedMultiQueue::GetBlocker() const {
|
|
CHECK(!queues_.empty());
|
|
return blocker_;
|
|
}
|
|
|
|
void OrderedMultiQueue::Dispatch() {
|
|
while (true) {
|
|
const Data* next_data = nullptr;
|
|
Queue* next_queue = nullptr;
|
|
QueueKey next_queue_key;
|
|
for (auto it = queues_.begin(); it != queues_.end();) {
|
|
const auto* data = it->second.queue.Peek<Data>();
|
|
if (data == nullptr) {
|
|
if (it->second.finished) {
|
|
queues_.erase(it++);
|
|
continue;
|
|
}
|
|
CannotMakeProgress(it->first);
|
|
return;
|
|
}
|
|
if (next_data == nullptr || data->GetTime() < next_data->GetTime()) {
|
|
next_data = data;
|
|
next_queue = &it->second;
|
|
next_queue_key = it->first;
|
|
}
|
|
CHECK_LE(last_dispatched_time_, next_data->GetTime())
|
|
<< "Non-sorted data added to queue: '" << it->first << "'";
|
|
++it;
|
|
}
|
|
if (next_data == nullptr) {
|
|
CHECK(queues_.empty());
|
|
return;
|
|
}
|
|
|
|
// If we haven't dispatched any data for this trajectory yet, fast forward
|
|
// all queues of this trajectory until a common start time has been reached.
|
|
const common::Time common_start_time =
|
|
GetCommonStartTime(next_queue_key.trajectory_id);
|
|
|
|
if (next_data->GetTime() >= common_start_time) {
|
|
// Happy case, we are beyond the 'common_start_time' already.
|
|
last_dispatched_time_ = next_data->GetTime();
|
|
next_queue->callback(next_queue->queue.Pop());
|
|
} else if (next_queue->queue.Size() < 2) {
|
|
if (!next_queue->finished) {
|
|
// We cannot decide whether to drop or dispatch this yet.
|
|
CannotMakeProgress(next_queue_key);
|
|
return;
|
|
}
|
|
last_dispatched_time_ = next_data->GetTime();
|
|
next_queue->callback(next_queue->queue.Pop());
|
|
} else {
|
|
// We take a peek at the time after next data. If it also is not beyond
|
|
// 'common_start_time' we drop 'next_data', otherwise we just found the
|
|
// first packet to dispatch from this queue.
|
|
std::unique_ptr<Data> next_data_owner = next_queue->queue.Pop();
|
|
if (next_queue->queue.Peek<Data>()->GetTime() > common_start_time) {
|
|
last_dispatched_time_ = next_data->GetTime();
|
|
next_queue->callback(std::move(next_data_owner));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void OrderedMultiQueue::CannotMakeProgress(const QueueKey& queue_key) {
|
|
blocker_ = queue_key;
|
|
for (auto& entry : queues_) {
|
|
if (entry.second.queue.Size() > kMaxQueueSize) {
|
|
LOG_EVERY_N(WARNING, 60) << "Queue waiting for data: " << queue_key;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
common::Time OrderedMultiQueue::GetCommonStartTime(const int trajectory_id) {
|
|
auto emplace_result = common_start_time_per_trajectory_.emplace(
|
|
trajectory_id, common::Time::min());
|
|
common::Time& common_start_time = emplace_result.first->second;
|
|
if (emplace_result.second) {
|
|
for (auto& entry : queues_) {
|
|
if (entry.first.trajectory_id == trajectory_id) {
|
|
common_start_time = std::max(
|
|
common_start_time, entry.second.queue.Peek<Data>()->GetTime());
|
|
}
|
|
}
|
|
LOG(INFO) << "All sensor data for trajectory " << trajectory_id
|
|
<< " is available starting at '" << common_start_time << "'.";
|
|
}
|
|
return common_start_time;
|
|
}
|
|
|
|
} // namespace sensor
|
|
} // namespace cartographer
|