|  |  |  | @ -11,8 +11,14 @@ AbstractStream::AbstractStream(QObject *parent, bool is_live_streaming) : is_liv | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::process(QHash<MessageId, CanData> *messages) { | 
			
		
	
		
			
				
					|  |  |  |  |   auto prev_src_size = sources.size(); | 
			
		
	
		
			
				
					|  |  |  |  |   for (auto it = messages->begin(); it != messages->end(); ++it) { | 
			
		
	
		
			
				
					|  |  |  |  |     last_msgs[it.key()] = it.value(); | 
			
		
	
		
			
				
					|  |  |  |  |     const auto &id = it.key(); | 
			
		
	
		
			
				
					|  |  |  |  |     last_msgs[id] = it.value(); | 
			
		
	
		
			
				
					|  |  |  |  |     sources.insert(id.source); | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   if (sources.size() != prev_src_size) { | 
			
		
	
		
			
				
					|  |  |  |  |     emit sourcesUpdated(sources); | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   emit updated(); | 
			
		
	
		
			
				
					|  |  |  |  |   emit msgsReceived(messages); | 
			
		
	
	
		
			
				
					|  |  |  | @ -22,34 +28,24 @@ void AbstractStream::process(QHash<MessageId, CanData> *messages) { | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | bool AbstractStream::updateEvent(const Event *event) { | 
			
		
	
		
			
				
					|  |  |  |  |   static double prev_update_ts = 0; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   if (event->which == cereal::Event::Which::CAN) { | 
			
		
	
		
			
				
					|  |  |  |  |     double current_sec = event->mono_time / 1e9 - routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |     for (const auto &c : event->event.getCan()) { | 
			
		
	
		
			
				
					|  |  |  |  |       MessageId id = {.source = c.getSrc(), .address = c.getAddress()}; | 
			
		
	
		
			
				
					|  |  |  |  |       CanData &data = (*new_msgs)[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       data.ts = current_sec; | 
			
		
	
		
			
				
					|  |  |  |  |       data.dat = QByteArray((char *)c.getDat().begin(), c.getDat().size()); | 
			
		
	
		
			
				
					|  |  |  |  |       data.count = ++counters[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       data.freq = data.count / std::max(1.0, current_sec); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |       auto &tracker = change_trackers[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       tracker.compute(data.dat, data.ts, data.freq); | 
			
		
	
		
			
				
					|  |  |  |  |       data.colors = tracker.colors; | 
			
		
	
		
			
				
					|  |  |  |  |       data.last_change_t = tracker.last_change_t; | 
			
		
	
		
			
				
					|  |  |  |  |       data.bit_change_counts = tracker.bit_change_counts; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |       if (!sources.contains(id.source)) { | 
			
		
	
		
			
				
					|  |  |  |  |         sources.insert(id.source); | 
			
		
	
		
			
				
					|  |  |  |  |         emit sourcesUpdated(sources); | 
			
		
	
		
			
				
					|  |  |  |  |       const auto dat = c.getDat(); | 
			
		
	
		
			
				
					|  |  |  |  |       all_msgs[id].compute((const char *)dat.begin(), dat.size(), current_sec); | 
			
		
	
		
			
				
					|  |  |  |  |       if (!new_msgs->contains(id)) { | 
			
		
	
		
			
				
					|  |  |  |  |         new_msgs->insert(id, {}); | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     double ts = millis_since_boot(); | 
			
		
	
		
			
				
					|  |  |  |  |     // delay posting CAN message if UI thread is busy
 | 
			
		
	
		
			
				
					|  |  |  |  |     if ((ts - prev_update_ts) > (1000.0 / settings.fps) && !processing && !new_msgs->isEmpty()) { | 
			
		
	
		
			
				
					|  |  |  |  |       // delay posting CAN message if UI thread is busy
 | 
			
		
	
		
			
				
					|  |  |  |  |       processing = true; | 
			
		
	
		
			
				
					|  |  |  |  |       prev_update_ts = ts; | 
			
		
	
		
			
				
					|  |  |  |  |       for (auto it = new_msgs->begin(); it != new_msgs->end(); ++it) { | 
			
		
	
		
			
				
					|  |  |  |  |         it.value() = all_msgs[it.key()]; | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |       // use pointer to avoid data copy in queued connection.
 | 
			
		
	
		
			
				
					|  |  |  |  |       emit received(new_msgs.release()); | 
			
		
	
		
			
				
					|  |  |  |  |       new_msgs.reset(new QHash<MessageId, CanData>); | 
			
		
	
	
		
			
				
					|  |  |  | @ -69,25 +65,22 @@ const CanData &AbstractStream::lastMessage(const MessageId &id) { | 
			
		
	
		
			
				
					|  |  |  |  | // updateEvent will not be called before replayStream::seekedTo return.
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::updateLastMsgsTo(double sec) { | 
			
		
	
		
			
				
					|  |  |  |  |   new_msgs->clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   change_trackers.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   all_msgs.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   last_msgs.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   counters.clear(); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   CanEvent last_event = {.mono_time = uint64_t((sec + routeStartTime()) * 1e9)}; | 
			
		
	
		
			
				
					|  |  |  |  |   for (auto &[id, e] : events_) { | 
			
		
	
		
			
				
					|  |  |  |  |     auto it = std::lower_bound(e.crbegin(), e.crend(), last_event, std::greater<CanEvent>()); | 
			
		
	
		
			
				
					|  |  |  |  |     if (it != e.crend()) { | 
			
		
	
		
			
				
					|  |  |  |  |       auto &m = last_msgs[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       m.dat = QByteArray((const char *)it->dat, it->size); | 
			
		
	
		
			
				
					|  |  |  |  |       m.ts = it->mono_time / 1e9 - routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |       double ts = it->mono_time / 1e9 - routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |       auto &m = all_msgs[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       m.compute((const char *)it->dat, it->size, ts); | 
			
		
	
		
			
				
					|  |  |  |  |       m.count = std::distance(it, e.crend()); | 
			
		
	
		
			
				
					|  |  |  |  |       m.freq = m.count / std::max(1.0, m.ts); | 
			
		
	
		
			
				
					|  |  |  |  |       m.last_change_t = QVector<double>(m.dat.size(), m.ts); | 
			
		
	
		
			
				
					|  |  |  |  |       m.colors = QVector<QColor>(m.dat.size(), QColor(0, 0, 0, 0)); | 
			
		
	
		
			
				
					|  |  |  |  |       m.bit_change_counts = QVector<std::array<uint32_t, 8>>(m.dat.size()); | 
			
		
	
		
			
				
					|  |  |  |  |       counters[id] = m.count; | 
			
		
	
		
			
				
					|  |  |  |  |       m.freq = m.count / std::max(1.0, ts); | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   last_msgs = all_msgs; | 
			
		
	
		
			
				
					|  |  |  |  |   // use a timer to prevent recursive calls
 | 
			
		
	
		
			
				
					|  |  |  |  |   QTimer::singleShot(0, [this]() { | 
			
		
	
		
			
				
					|  |  |  |  |     emit updated(); | 
			
		
	
		
			
				
					|  |  |  |  |     emit msgsReceived(&last_msgs); | 
			
		
	
	
		
			
				
					|  |  |  | @ -96,18 +89,20 @@ void AbstractStream::updateLastMsgsTo(double sec) { | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::parseEvents(std::unordered_map<MessageId, std::deque<CanEvent>> &msgs, | 
			
		
	
		
			
				
					|  |  |  |  |                                  std::vector<Event *>::const_iterator first, std::vector<Event *>::const_iterator last) { | 
			
		
	
		
			
				
					|  |  |  |  |   uint64_t ts = 0; | 
			
		
	
		
			
				
					|  |  |  |  |   for (; first != last; ++first) { | 
			
		
	
		
			
				
					|  |  |  |  |     if ((*first)->which == cereal::Event::Which::CAN) { | 
			
		
	
		
			
				
					|  |  |  |  |       ts = (*first)->mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |       for (const auto &c : (*first)->event.getCan()) { | 
			
		
	
		
			
				
					|  |  |  |  |         auto dat = c.getDat(); | 
			
		
	
		
			
				
					|  |  |  |  |         auto &m = msgs[{.source = c.getSrc(), .address = c.getAddress()}].emplace_back(); | 
			
		
	
		
			
				
					|  |  |  |  |         m.size = std::min(dat.size(), std::size(m.dat)); | 
			
		
	
		
			
				
					|  |  |  |  |         memcpy(m.dat, (uint8_t *)dat.begin(), m.size); | 
			
		
	
		
			
				
					|  |  |  |  |         m.mono_time = (*first)->mono_time; | 
			
		
	
		
			
				
					|  |  |  |  |         m.mono_time = ts; | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |       last_event_ts = std::max(last_event_ts, (*first)->mono_time); | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   last_event_ts = std::max(last_event_ts, ts); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std::vector<Event *>::const_iterator last, bool append) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -126,3 +121,67 @@ void AbstractStream::mergeEvents(std::vector<Event *>::const_iterator first, std | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   emit eventsMerged(); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | // CanData
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | constexpr int periodic_threshold = 10; | 
			
		
	
		
			
				
					|  |  |  |  | constexpr int start_alpha = 128; | 
			
		
	
		
			
				
					|  |  |  |  | constexpr float fade_time = 2.0; | 
			
		
	
		
			
				
					|  |  |  |  | const QColor CYAN = QColor(0, 187, 255, start_alpha); | 
			
		
	
		
			
				
					|  |  |  |  | const QColor RED = QColor(255, 0, 0, start_alpha); | 
			
		
	
		
			
				
					|  |  |  |  | const QColor GREYISH_BLUE = QColor(102, 86, 169, start_alpha / 2); | 
			
		
	
		
			
				
					|  |  |  |  | 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) { | 
			
		
	
		
			
				
					|  |  |  |  |   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, uint32_t in_freq) { | 
			
		
	
		
			
				
					|  |  |  |  |   ts = current_sec; | 
			
		
	
		
			
				
					|  |  |  |  |   ++count; | 
			
		
	
		
			
				
					|  |  |  |  |   freq = in_freq == 0 ? count / std::max(1.0, current_sec) : in_freq; | 
			
		
	
		
			
				
					|  |  |  |  |   if (dat.size() != size) { | 
			
		
	
		
			
				
					|  |  |  |  |     dat.resize(size); | 
			
		
	
		
			
				
					|  |  |  |  |     bit_change_counts.resize(size); | 
			
		
	
		
			
				
					|  |  |  |  |     colors = QVector(size, QColor(0, 0, 0, 0)); | 
			
		
	
		
			
				
					|  |  |  |  |     last_change_t = QVector(size, ts); | 
			
		
	
		
			
				
					|  |  |  |  |   } else { | 
			
		
	
		
			
				
					|  |  |  |  |     bool lighter = settings.theme == DARK_THEME; | 
			
		
	
		
			
				
					|  |  |  |  |     const QColor &cyan = !lighter ? CYAN : CYAN_LIGHTER; | 
			
		
	
		
			
				
					|  |  |  |  |     const QColor &red = !lighter ? RED : RED_LIGHTER; | 
			
		
	
		
			
				
					|  |  |  |  |     const QColor &greyish_blue = !lighter ? GREYISH_BLUE : GREYISH_BLUE_LIGHTER; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     for (int i = 0; i < size; ++i) { | 
			
		
	
		
			
				
					|  |  |  |  |       const uint8_t last = dat[i]; | 
			
		
	
		
			
				
					|  |  |  |  |       const uint8_t cur = can_data[i]; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |       if (last != cur) { | 
			
		
	
		
			
				
					|  |  |  |  |         double delta_t = ts - last_change_t[i]; | 
			
		
	
		
			
				
					|  |  |  |  |         if (delta_t * freq > periodic_threshold) { | 
			
		
	
		
			
				
					|  |  |  |  |           // Last change was while ago, choose color based on delta up or down
 | 
			
		
	
		
			
				
					|  |  |  |  |           colors[i] = (cur > last) ? cyan : red; | 
			
		
	
		
			
				
					|  |  |  |  |         } else { | 
			
		
	
		
			
				
					|  |  |  |  |           // Periodic changes
 | 
			
		
	
		
			
				
					|  |  |  |  |           colors[i] = blend(colors[i], greyish_blue); | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // Track bit level changes
 | 
			
		
	
		
			
				
					|  |  |  |  |         const uint8_t tmp = (cur ^ last); | 
			
		
	
		
			
				
					|  |  |  |  |         for (int bit = 0; bit < 8; bit++) { | 
			
		
	
		
			
				
					|  |  |  |  |           if (tmp & (1 << bit)) { | 
			
		
	
		
			
				
					|  |  |  |  |             bit_change_counts[i][bit] += 1; | 
			
		
	
		
			
				
					|  |  |  |  |           } | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         last_change_t[i] = ts; | 
			
		
	
		
			
				
					|  |  |  |  |       } else { | 
			
		
	
		
			
				
					|  |  |  |  |         // Fade out
 | 
			
		
	
		
			
				
					|  |  |  |  |         float alpha_delta = 1.0 / (freq + 1) / fade_time; | 
			
		
	
		
			
				
					|  |  |  |  |         colors[i].setAlphaF(std::max(0.0, colors[i].alphaF() - alpha_delta)); | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   memcpy(dat.data(), can_data, size); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
	
		
			
				
					|  |  |  | 
 |