Cabana: add dynamic/static mode to logs view (#26832)

pull/26846/head
Dean Lee 2 years ago committed by GitHub
parent 981532f0c3
commit e018098571
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 141
      tools/cabana/historylog.cc
  2. 23
      tools/cabana/historylog.h

@ -7,12 +7,6 @@
// HistoryLogModel // HistoryLogModel
HistoryLogModel::HistoryLogModel(QObject *parent) : QAbstractTableModel(parent) {
QObject::connect(can, &CANMessages::seekedTo, [this]() {
if (!msg_id.isEmpty()) setMessage(msg_id);
});
}
QVariant HistoryLogModel::data(const QModelIndex &index, int role) const { QVariant HistoryLogModel::data(const QModelIndex &index, int role) const {
if (role == Qt::DisplayRole) { if (role == Qt::DisplayRole) {
const auto &m = messages[index.row()]; const auto &m = messages[index.row()];
@ -29,17 +23,19 @@ QVariant HistoryLogModel::data(const QModelIndex &index, int role) const {
} }
void HistoryLogModel::setMessage(const QString &message_id) { void HistoryLogModel::setMessage(const QString &message_id) {
beginResetModel(); msg_id = message_id;
sigs.clear(); sigs.clear();
messages.clear(); if (auto dbc_msg = dbc()->msg(msg_id)) {
has_more_data = true;
if (auto dbc_msg = dbc()->msg(message_id)) {
sigs = dbc_msg->getSignals(); sigs = dbc_msg->getSignals();
} }
if (msg_id != message_id || sigs.empty()) { filter_cmp = nullptr;
filter_cmp = nullptr; refresh();
} }
msg_id = message_id;
void HistoryLogModel::refresh() {
beginResetModel();
last_fetch_time = 0;
messages.clear();
updateState(); updateState();
endResetModel(); endResetModel();
} }
@ -60,33 +56,40 @@ QVariant HistoryLogModel::headerData(int section, Qt::Orientation orientation, i
return {}; return {};
} }
void HistoryLogModel::setFilter(int sig_idx, const QString &value, std::function<bool(double, double)> cmp) { void HistoryLogModel::setDynamicMode(int state) {
if (sig_idx < sigs.size()) { dynamic_mode = state != 0;
filter_sig_idx = sig_idx; refresh();
filter_value = value.toDouble(); }
filter_cmp = value.isEmpty() ? nullptr : cmp;
beginResetModel(); void HistoryLogModel::segmentsMerged() {
messages.clear(); if (!dynamic_mode) {
updateState(); has_more_data = true;
endResetModel();
} }
} }
void HistoryLogModel::setFilter(int sig_idx, const QString &value, std::function<bool(double, double)> cmp) {
filter_sig_idx = sig_idx;
filter_value = value.toDouble();
filter_cmp = value.isEmpty() ? nullptr : cmp;
refresh();
}
void HistoryLogModel::updateState() { void HistoryLogModel::updateState() {
if (!msg_id.isEmpty()) { if (!msg_id.isEmpty()) {
uint64_t last_mono_time = messages.empty() ? 0 : messages.front().mono_time; uint64_t current_time = (can->currentSec() + can->routeStartTime()) * 1e9;
auto new_msgs = fetchData(last_mono_time, (can->currentSec() + can->routeStartTime()) * 1e9); auto new_msgs = dynamic_mode ? fetchData(current_time, last_fetch_time) : fetchData(0);
if ((has_more_data = !new_msgs.empty())) { if ((has_more_data = !new_msgs.empty())) {
beginInsertRows({}, 0, new_msgs.size() - 1); beginInsertRows({}, 0, new_msgs.size() - 1);
messages.insert(messages.begin(), std::move_iterator(new_msgs.begin()), std::move_iterator(new_msgs.end())); messages.insert(messages.begin(), std::move_iterator(new_msgs.begin()), std::move_iterator(new_msgs.end()));
endInsertRows(); endInsertRows();
} }
last_fetch_time = current_time;
} }
} }
void HistoryLogModel::fetchMore(const QModelIndex &parent) { void HistoryLogModel::fetchMore(const QModelIndex &parent) {
if (!messages.empty()) { if (!messages.empty()) {
auto new_msgs = fetchData(0, messages.back().mono_time); auto new_msgs = fetchData(messages.back().mono_time);
if ((has_more_data = !new_msgs.empty())) { if ((has_more_data = !new_msgs.empty())) {
beginInsertRows({}, messages.size(), messages.size() + new_msgs.size() - 1); beginInsertRows({}, messages.size(), messages.size() + new_msgs.size() - 1);
messages.insert(messages.end(), std::move_iterator(new_msgs.begin()), std::move_iterator(new_msgs.end())); messages.insert(messages.end(), std::move_iterator(new_msgs.begin()), std::move_iterator(new_msgs.end()));
@ -95,19 +98,12 @@ void HistoryLogModel::fetchMore(const QModelIndex &parent) {
} }
} }
std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t min_mono_time, uint64_t max_mono_time) { template <class InputIt>
auto events = can->events(); std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(InputIt first, InputIt last, uint64_t min_time) {
auto it = std::lower_bound(events->begin(), events->end(), max_mono_time, [=](auto &e, uint64_t ts) {
return e->mono_time < ts;
});
if (it == events->end() || it == events->begin())
return {};
std::deque<HistoryLogModel::Message> msgs; std::deque<HistoryLogModel::Message> msgs;
const auto [src, address] = DBCManager::parseId(msg_id); const auto [src, address] = DBCManager::parseId(msg_id);
uint32_t cnt = 0;
QVector<double> values(sigs.size()); QVector<double> values(sigs.size());
for (--it; it != events->begin() && (*it)->mono_time > min_mono_time; --it) { for (auto it = first; it != last && (*it)->mono_time > min_time; ++it) {
if ((*it)->which == cereal::Event::Which::CAN) { if ((*it)->which == cereal::Event::Which::CAN) {
for (const auto &c : (*it)->event.getCan()) { for (const auto &c : (*it)->event.getCan()) {
if (src == c.getSrc() && address == c.getAddress()) { if (src == c.getSrc() && address == c.getAddress()) {
@ -120,7 +116,7 @@ std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t min_mon
m.mono_time = (*it)->mono_time; m.mono_time = (*it)->mono_time;
m.data = toHex(QByteArray((char *)dat.begin(), dat.size())); m.data = toHex(QByteArray((char *)dat.begin(), dat.size()));
m.sig_values = values; m.sig_values = values;
if (++cnt >= batch_size && min_mono_time == 0) if (msgs.size() >= batch_size && min_time == 0)
return msgs; return msgs;
} }
} }
@ -129,6 +125,25 @@ std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t min_mon
} }
return msgs; return msgs;
} }
template std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData<>(std::vector<const Event*>::iterator first, std::vector<const Event*>::iterator last, uint64_t min_time);
template std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData<>(std::vector<const Event*>::reverse_iterator first, std::vector<const Event*>::reverse_iterator last, uint64_t min_time);
std::deque<HistoryLogModel::Message> HistoryLogModel::fetchData(uint64_t from_time, uint64_t min_time) {
auto events = can->events();
if (dynamic_mode) {
auto it = std::lower_bound(events->rbegin(), events->rend(), from_time, [=](auto &e, uint64_t ts) {
return e->mono_time > ts;
});
if (it != events->rend()) ++it;
return fetchData(it, events->rend(), min_time);
} else {
assert(min_time == 0);
auto it = std::upper_bound(events->begin(), events->end(), from_time, [=](uint64_t ts, auto &e) {
return ts < e->mono_time;
});
return fetchData(it, events->end(), 0);
}
}
// HeaderView // HeaderView
@ -166,9 +181,9 @@ HistoryLog::HistoryLog(QWidget *parent) : QTableView(parent) {
LogsWidget::LogsWidget(QWidget *parent) : QWidget(parent) { LogsWidget::LogsWidget(QWidget *parent) : QWidget(parent) {
QVBoxLayout *main_layout = new QVBoxLayout(this); QVBoxLayout *main_layout = new QVBoxLayout(this);
filter_container = new QWidget(this); QHBoxLayout *h = new QHBoxLayout();
QHBoxLayout *h = new QHBoxLayout(filter_container);
signals_cb = new QComboBox(this); signals_cb = new QComboBox(this);
signals_cb->setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::Preferred);
h->addWidget(signals_cb); h->addWidget(signals_cb);
comp_box = new QComboBox(); comp_box = new QComboBox();
comp_box->addItems({">", "=", "!=", "<"}); comp_box->addItems({">", "=", "!=", "<"});
@ -177,7 +192,9 @@ LogsWidget::LogsWidget(QWidget *parent) : QWidget(parent) {
value_edit->setClearButtonEnabled(true); value_edit->setClearButtonEnabled(true);
value_edit->setValidator(new QDoubleValidator(-500000, 500000, 6, this)); value_edit->setValidator(new QDoubleValidator(-500000, 500000, 6, this));
h->addWidget(value_edit); h->addWidget(value_edit);
main_layout->addWidget(filter_container); dynamic_mode = new QCheckBox(tr("Dynamic"));
h->addWidget(dynamic_mode, 0, Qt::AlignRight);
main_layout->addLayout(h);
model = new HistoryLogModel(this); model = new HistoryLogModel(this);
logs = new HistoryLog(this); logs = new HistoryLog(this);
@ -185,33 +202,38 @@ LogsWidget::LogsWidget(QWidget *parent) : QWidget(parent) {
main_layout->addWidget(logs); main_layout->addWidget(logs);
QObject::connect(logs, &QTableView::doubleClicked, this, &LogsWidget::doubleClicked); QObject::connect(logs, &QTableView::doubleClicked, this, &LogsWidget::doubleClicked);
QObject::connect(signals_cb, SIGNAL(currentIndexChanged(int)), this, SLOT(setFilter())); QObject::connect(signals_cb, SIGNAL(activated(int)), this, SLOT(setFilter()));
QObject::connect(comp_box, SIGNAL(currentIndexChanged(int)), this, SLOT(setFilter())); QObject::connect(comp_box, SIGNAL(activated(int)), this, SLOT(setFilter()));
QObject::connect(value_edit, &QLineEdit::textChanged, this, &LogsWidget::setFilter); QObject::connect(value_edit, &QLineEdit::textChanged, this, &LogsWidget::setFilter);
QObject::connect(dynamic_mode, &QCheckBox::stateChanged, model, &HistoryLogModel::setDynamicMode);
QObject::connect(can, &CANMessages::seekedTo, model, &HistoryLogModel::refresh);
QObject::connect(can, &CANMessages::eventsMerged, model, &HistoryLogModel::segmentsMerged);
} }
void LogsWidget::setMessage(const QString &message_id) { void LogsWidget::setMessage(const QString &message_id) {
blockSignals(true); model->setMessage(message_id);
cur_filter_text = "";
value_edit->setText(""); value_edit->setText("");
signals_cb->clear(); signals_cb->clear();
comp_box->setCurrentIndex(0); comp_box->setCurrentIndex(0);
sigs.clear(); bool has_signals = model->sigs.size() > 0;
if (auto dbc_msg = dbc()->msg(message_id)) { if (has_signals) {
sigs = dbc_msg->getSignals(); for (auto s : model->sigs) {
for (auto s : sigs) {
signals_cb->addItem(s->name.c_str()); signals_cb->addItem(s->name.c_str());
} }
} }
filter_container->setVisible(!sigs.empty()); comp_box->setVisible(has_signals);
model->setMessage(message_id); value_edit->setVisible(has_signals);
blockSignals(false); signals_cb->setVisible(has_signals);
} }
static bool not_equal(double l, double r) { static bool not_equal(double l, double r) { return l != r; }
return l != r;
}
void LogsWidget::setFilter() { void LogsWidget::setFilter() {
if (cur_filter_text.isEmpty() && value_edit->text().isEmpty()) {
return;
}
std::function<bool(double, double)> cmp; std::function<bool(double, double)> cmp;
switch (comp_box->currentIndex()) { switch (comp_box->currentIndex()) {
case 0: cmp = std::greater<double>{}; break; case 0: cmp = std::greater<double>{}; break;
@ -220,6 +242,19 @@ void LogsWidget::setFilter() {
case 3: cmp = std::less<double>{}; break; case 3: cmp = std::less<double>{}; break;
} }
model->setFilter(signals_cb->currentIndex(), value_edit->text(), cmp); model->setFilter(signals_cb->currentIndex(), value_edit->text(), cmp);
cur_filter_text = value_edit->text();
}
void LogsWidget::showEvent(QShowEvent *event) {
if (dynamic_mode->isChecked()) {
model->refresh();
}
}
void LogsWidget::updateState() {
if (dynamic_mode->isChecked()) {
model->updateState();
}
} }
void LogsWidget::doubleClicked(const QModelIndex &index) { void LogsWidget::doubleClicked(const QModelIndex &index) {

@ -1,6 +1,7 @@
#pragma once #pragma once
#include <deque> #include <deque>
#include <QCheckBox>
#include <QComboBox> #include <QComboBox>
#include <QHeaderView> #include <QHeaderView>
#include <QLineEdit> #include <QLineEdit>
@ -17,8 +18,10 @@ public:
}; };
class HistoryLogModel : public QAbstractTableModel { class HistoryLogModel : public QAbstractTableModel {
Q_OBJECT
public: public:
HistoryLogModel(QObject *parent); HistoryLogModel(QObject *parent) : QAbstractTableModel(parent) {}
void setMessage(const QString &message_id); void setMessage(const QString &message_id);
void updateState(); void updateState();
void setFilter(int sig_idx, const QString &value, std::function<bool(double, double)> cmp); void setFilter(int sig_idx, const QString &value, std::function<bool(double, double)> cmp);
@ -28,6 +31,9 @@ public:
inline bool canFetchMore(const QModelIndex &parent) const override { return has_more_data; } inline bool canFetchMore(const QModelIndex &parent) const override { return has_more_data; }
int rowCount(const QModelIndex &parent = QModelIndex()) const override { return messages.size(); } int rowCount(const QModelIndex &parent = QModelIndex()) const override { return messages.size(); }
int columnCount(const QModelIndex &parent = QModelIndex()) const override { return std::max(1ul, sigs.size()) + 1; } int columnCount(const QModelIndex &parent = QModelIndex()) const override { return std::max(1ul, sigs.size()) + 1; }
void setDynamicMode(int state);
void segmentsMerged();
void refresh();
struct Message { struct Message {
uint64_t mono_time = 0; uint64_t mono_time = 0;
@ -35,15 +41,20 @@ public:
QString data; QString data;
}; };
std::deque<Message> fetchData(uint64_t min_mono_time, uint64_t max_mono_time); template <class InputIt>
std::deque<HistoryLogModel::Message> fetchData(InputIt first, InputIt last, uint64_t min_time);
std::deque<Message> fetchData(uint64_t from_time, uint64_t min_time = 0);
QString msg_id; QString msg_id;
bool has_more_data = true; bool has_more_data = true;
const int batch_size = 50; const int batch_size = 50;
int filter_sig_idx = -1; int filter_sig_idx = -1;
double filter_value = 0; double filter_value = 0;
uint64_t last_fetch_time = 0;
std::function<bool(double, double)> filter_cmp = nullptr; std::function<bool(double, double)> filter_cmp = nullptr;
std::deque<Message> messages; std::deque<Message> messages;
std::vector<const Signal*> sigs; std::vector<const Signal*> sigs;
bool dynamic_mode = false;
}; };
class HistoryLog : public QTableView { class HistoryLog : public QTableView {
@ -58,7 +69,7 @@ class LogsWidget : public QWidget {
public: public:
LogsWidget(QWidget *parent); LogsWidget(QWidget *parent);
void setMessage(const QString &message_id); void setMessage(const QString &message_id);
void updateState() { model->updateState(); } void updateState();
signals: signals:
void openChart(const QString &msg_id, const Signal *sig); void openChart(const QString &msg_id, const Signal *sig);
@ -68,12 +79,12 @@ private slots:
private: private:
void doubleClicked(const QModelIndex &index); void doubleClicked(const QModelIndex &index);
void showEvent(QShowEvent *event) override { model->setMessage(model->msg_id); }; void showEvent(QShowEvent *event) override;
HistoryLog *logs; HistoryLog *logs;
HistoryLogModel *model; HistoryLogModel *model;
QWidget *filter_container; QCheckBox *dynamic_mode;
QComboBox *signals_cb, *comp_box; QComboBox *signals_cb, *comp_box;
QLineEdit *value_edit; QLineEdit *value_edit;
std::vector<const Signal*> sigs; QString cur_filter_text;
}; };

Loading…
Cancel
Save