cabana: improve frequency accuracy (#29631)

* improve frequency accuracy

* apply review

* 2 less lines!

* another!

---------

Co-authored-by: Shane Smiskol <shane@smiskol.com>
pull/29636/head
Dean Lee 2 years ago committed by GitHub
parent 4d6bb32df2
commit 2484548fdd
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 4
      tools/cabana/historylog.cc
  2. 2
      tools/cabana/messageswidget.cc
  3. 55
      tools/cabana/streams/abstractstream.cc
  4. 5
      tools/cabana/streams/abstractstream.h

@ -155,7 +155,7 @@ std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t from_ti
auto msgs = fetchData(first, events.rend(), min_time);
if (update_colors && (min_time > 0 || messages.empty())) {
for (auto it = msgs.rbegin(); it != msgs.rend(); ++it) {
hex_colors.compute(it->data.data(), it->data.size(), it->mono_time / (double)1e9, speed, nullptr, freq);
hex_colors.compute(msg_id, it->data.data(), it->data.size(), it->mono_time / (double)1e9, speed, nullptr, freq);
it->colors = hex_colors.colors;
}
}
@ -168,7 +168,7 @@ std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t from_ti
auto msgs = fetchData(first, events.cend(), 0);
if (update_colors) {
for (auto it = msgs.begin(); it != msgs.end(); ++it) {
hex_colors.compute(it->data.data(), it->data.size(), it->mono_time / (double)1e9, speed, nullptr, freq);
hex_colors.compute(msg_id, it->data.data(), it->data.size(), it->mono_time / (double)1e9, speed, nullptr, freq);
it->colors = hex_colors.colors;
}
}

@ -174,7 +174,7 @@ QVariant MessageListModel::data(const QModelIndex &index, int role) const {
auto getFreq = [](const CanData &d) -> QString {
if (d.freq > 0 && (can->currentSec() - d.ts - 1.0 / settings.fps) < (5.0 / d.freq)) {
return d.freq >= 1 ? QString::number(std::nearbyint(d.freq)) : QString::number(d.freq, 'f', 2);
return d.freq >= 0.95 ? QString::number(std::nearbyint(d.freq)) : QString::number(d.freq, 'f', 2);
} else {
return "--";
}

@ -62,7 +62,7 @@ void AbstractStream::updateEvent(const MessageId &id, double sec, const uint8_t
std::lock_guard lk(mutex);
auto mask_it = masks.find(id);
std::vector<uint8_t> *mask = mask_it == masks.end() ? nullptr : &mask_it->second;
all_msgs[id].compute((const char *)data, size, sec, getSpeed(), mask);
all_msgs[id].compute(id, (const char *)data, size, sec, getSpeed(), mask);
if (!new_msgs->contains(id)) {
new_msgs->insert(id, {});
}
@ -113,9 +113,8 @@ void AbstractStream::updateLastMsgsTo(double sec) {
if (it != ev.crend()) {
double ts = (*it)->mono_time / 1e9 - routeStartTime();
auto &m = all_msgs[id];
m.compute((const char *)(*it)->dat, (*it)->size, ts, getSpeed(), mask);
m.compute(id, (const char *)(*it)->dat, (*it)->size, ts, getSpeed(), mask);
m.count = std::distance(it, ev.crend());
m.freq = m.count / std::max(1.0, ts);
}
}
@ -165,18 +164,20 @@ void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std
}
}
auto compare = [](const CanEvent *l, const CanEvent *r) {
return l->mono_time < r->mono_time;
};
bool append = new_events.front()->mono_time > lastest_event_ts;
for (auto &[id, new_e] : new_events_map) {
auto &e = events_[id];
auto pos = append ? e.end() : std::upper_bound(e.cbegin(), e.cend(), new_e.front(), [](const CanEvent *l, const CanEvent *r) {
return l->mono_time < r->mono_time;
});
auto pos = append ? e.end()
: std::upper_bound(e.cbegin(), e.cend(), new_e.front(), compare);
e.insert(pos, new_e.cbegin(), new_e.cend());
}
auto pos = append ? all_events_.end() : std::upper_bound(all_events_.begin(), all_events_.end(), new_events.front(), [](auto l, auto r) {
return l->mono_time < r->mono_time;
});
auto pos = append ? all_events_.end()
: std::upper_bound(all_events_.begin(), all_events_.end(), new_events.front(), compare);
all_events_.insert(pos, new_events.cbegin(), new_events.cend());
lastest_event_ts = all_events_.back()->mono_time;
@ -185,6 +186,8 @@ void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std
// CanData
namespace {
constexpr int periodic_threshold = 10;
constexpr int start_alpha = 128;
constexpr float fade_time = 2.0;
@ -195,15 +198,41 @@ const QColor CYAN_LIGHTER = QColor(0, 187, 255, start_alpha).lighter(135);
const QColor RED_LIGHTER = QColor(255, 0, 0, start_alpha).lighter(135);
const QColor GREYISH_BLUE_LIGHTER = QColor(102, 86, 169, start_alpha / 2).lighter(135);
static inline QColor blend(const QColor &a, const QColor &b) {
inline QColor blend(const QColor &a, const QColor &b) {
return QColor((a.red() + b.red()) / 2, (a.green() + b.green()) / 2, (a.blue() + b.blue()) / 2, (a.alpha() + b.alpha()) / 2);
}
void CanData::compute(const char *can_data, const int size, double current_sec, double playback_speed, const std::vector<uint8_t> *mask, uint32_t in_freq) {
// Calculate the frequency of the past minute.
double calc_freq(const MessageId &msg_id, double current_sec) {
auto compare = [](const CanEvent *e, uint64_t mono_time) {
return e->mono_time < mono_time;
};
const auto &events = can->events(msg_id);
uint64_t cur_mono_time = (can->routeStartTime() + current_sec) * 1e9;
uint64_t first_mono_time = std::max<int64_t>(0, cur_mono_time - 59 * 1e9);
auto first = std::lower_bound(events.begin(), events.end(), first_mono_time, compare);
auto second = std::lower_bound(first, events.end(), cur_mono_time, compare);
if (first != events.end() && second != events.end()) {
double duration = ((*second)->mono_time - (*first)->mono_time) / 1e9;
uint32_t count = std::distance(first, second);
return count / std::max(1.0, duration);
}
return 0;
}
} // namespace
void CanData::compute(const MessageId &msg_id, const char *can_data, const int size, double current_sec,
double playback_speed, const std::vector<uint8_t> *mask, double in_freq) {
ts = current_sec;
++count;
const double sec_to_first_event = current_sec - (can->allEvents().front()->mono_time / 1e9 - can->routeStartTime());
freq = in_freq == 0 ? count / std::max(1.0, sec_to_first_event) : in_freq;
if (auto sec = seconds_since_boot(); (sec - last_freq_update_ts) >= 1) {
last_freq_update_ts = sec;
freq = !in_freq ? calc_freq(msg_id, ts) : in_freq;
}
if (dat.size() != size) {
dat.resize(size);
bit_change_counts.resize(size);

@ -11,13 +11,15 @@
#include <QColor>
#include <QHash>
#include "common/timing.h"
#include "tools/cabana/dbc/dbcmanager.h"
#include "tools/cabana/settings.h"
#include "tools/cabana/util.h"
#include "tools/replay/replay.h"
struct CanData {
void compute(const char *dat, const int size, double current_sec, double playback_speed, const std::vector<uint8_t> *mask, uint32_t in_freq = 0);
void compute(const MessageId &msg_id, const char *dat, const int size, double current_sec,
double playback_speed, const std::vector<uint8_t> *mask = nullptr, double in_freq = 0);
double ts = 0.;
uint32_t count = 0;
@ -28,6 +30,7 @@ struct CanData {
std::vector<std::array<uint32_t, 8>> bit_change_counts;
std::vector<int> last_delta;
std::vector<int> same_delta_counter;
double last_freq_update_ts = seconds_since_boot();
};
struct CanEvent {

Loading…
Cancel
Save