|  |  |  | @ -4,7 +4,9 @@ AbstractStream *can = nullptr; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | AbstractStream::AbstractStream(QObject *parent, bool is_live_streaming) : is_live_streaming(is_live_streaming), QObject(parent) { | 
			
		
	
		
			
				
					|  |  |  |  |   can = this; | 
			
		
	
		
			
				
					|  |  |  |  |   new_msgs = std::make_unique<QHash<QString, CanData>>(); | 
			
		
	
		
			
				
					|  |  |  |  |   QObject::connect(this, &AbstractStream::received, this, &AbstractStream::process, Qt::QueuedConnection); | 
			
		
	
		
			
				
					|  |  |  |  |   QObject::connect(this, &AbstractStream::seekedTo, this, &AbstractStream::updateLastMsgsTo); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::process(QHash<QString, CanData> *messages) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -18,30 +20,17 @@ void AbstractStream::process(QHash<QString, CanData> *messages) { | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | bool AbstractStream::updateEvent(const Event *event) { | 
			
		
	
		
			
				
					|  |  |  |  |   static std::unique_ptr new_msgs = std::make_unique<QHash<QString, CanData>>(); | 
			
		
	
		
			
				
					|  |  |  |  |   static QHash<QString, ChangeTracker> change_trackers; | 
			
		
	
		
			
				
					|  |  |  |  |   static double prev_update_ts = 0; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   if (event->which == cereal::Event::Which::CAN) { | 
			
		
	
		
			
				
					|  |  |  |  |     double current_sec = currentSec(); | 
			
		
	
		
			
				
					|  |  |  |  |     if (counters_begin_sec == 0 || counters_begin_sec >= current_sec) { | 
			
		
	
		
			
				
					|  |  |  |  |       new_msgs->clear(); | 
			
		
	
		
			
				
					|  |  |  |  |       counters.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |       counters_begin_sec = current_sec; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     auto can_events = event->event.getCan(); | 
			
		
	
		
			
				
					|  |  |  |  |     for (const auto &c : can_events) { | 
			
		
	
		
			
				
					|  |  |  |  |     double current_sec = event->mono_time / 1e9 - routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |     for (const auto &c : event->event.getCan()) { | 
			
		
	
		
			
				
					|  |  |  |  |       QString id = QString("%1:%2").arg(c.getSrc()).arg(c.getAddress(), 1, 16); | 
			
		
	
		
			
				
					|  |  |  |  |       CanData &data = (*new_msgs)[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       data.ts = current_sec; | 
			
		
	
		
			
				
					|  |  |  |  |       data.src = c.getSrc(); | 
			
		
	
		
			
				
					|  |  |  |  |       data.address = c.getAddress(); | 
			
		
	
		
			
				
					|  |  |  |  |       data.dat = QByteArray((char *)c.getDat().begin(), c.getDat().size()); | 
			
		
	
		
			
				
					|  |  |  |  |       data.count = ++counters[id]; | 
			
		
	
		
			
				
					|  |  |  |  |       if (double delta = (current_sec - counters_begin_sec); delta > 0) { | 
			
		
	
		
			
				
					|  |  |  |  |         data.freq = data.count / delta; | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |       data.freq = data.count / std::max(1.0, current_sec); | 
			
		
	
		
			
				
					|  |  |  |  |       change_trackers[id].compute(data.dat, data.ts, data.freq); | 
			
		
	
		
			
				
					|  |  |  |  |       data.colors = change_trackers[id].colors; | 
			
		
	
		
			
				
					|  |  |  |  |       data.last_change_t = change_trackers[id].last_change_t; | 
			
		
	
	
		
			
				
					|  |  |  | @ -60,3 +49,46 @@ bool AbstractStream::updateEvent(const Event *event) { | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   return true; | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | const CanData &AbstractStream::lastMessage(const QString &id) { | 
			
		
	
		
			
				
					|  |  |  |  |   static CanData empty_data; | 
			
		
	
		
			
				
					|  |  |  |  |   auto it = can_msgs.find(id); | 
			
		
	
		
			
				
					|  |  |  |  |   return it != can_msgs.end() ? it.value() : empty_data; | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | void AbstractStream::updateLastMsgsTo(double sec) { | 
			
		
	
		
			
				
					|  |  |  |  |   QHash<std::pair<uint8_t, uint32_t>, CanData> last_msgs;  // Much faster than QHash<String, CanData>
 | 
			
		
	
		
			
				
					|  |  |  |  |   last_msgs.reserve(can_msgs.size()); | 
			
		
	
		
			
				
					|  |  |  |  |   double route_start_time = routeStartTime(); | 
			
		
	
		
			
				
					|  |  |  |  |   uint64_t last_ts = (sec + route_start_time) * 1e9; | 
			
		
	
		
			
				
					|  |  |  |  |   auto last = std::upper_bound(events()->rbegin(), events()->rend(), last_ts, [](uint64_t ts, auto &e) { return e->mono_time < ts; }); | 
			
		
	
		
			
				
					|  |  |  |  |   for (auto it = last; it != events()->rend(); ++it) { | 
			
		
	
		
			
				
					|  |  |  |  |     if ((*it)->which == cereal::Event::Which::CAN) { | 
			
		
	
		
			
				
					|  |  |  |  |       for (const auto &c : (*it)->event.getCan()) { | 
			
		
	
		
			
				
					|  |  |  |  |         auto &m = last_msgs[{c.getSrc(), c.getAddress()}]; | 
			
		
	
		
			
				
					|  |  |  |  |         if (++m.count == 1) { | 
			
		
	
		
			
				
					|  |  |  |  |           m.ts = ((*it)->mono_time / 1e9) - route_start_time; | 
			
		
	
		
			
				
					|  |  |  |  |           m.dat = QByteArray((char *)c.getDat().begin(), c.getDat().size()); | 
			
		
	
		
			
				
					|  |  |  |  |           m.colors = QVector<QColor>(m.dat.size(), QColor(0, 0, 0, 0)); | 
			
		
	
		
			
				
					|  |  |  |  |           m.last_change_t = QVector<double>(m.dat.size(), m.ts); | 
			
		
	
		
			
				
					|  |  |  |  |         } else { | 
			
		
	
		
			
				
					|  |  |  |  |           m.freq = m.count / std::max(1.0, m.ts); | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |       } | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |   // it is thread safe to update data here.
 | 
			
		
	
		
			
				
					|  |  |  |  |   // updateEvent will not be called before replayStream::seekedTo return.
 | 
			
		
	
		
			
				
					|  |  |  |  |   new_msgs->clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   change_trackers.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   counters.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   can_msgs.clear(); | 
			
		
	
		
			
				
					|  |  |  |  |   for (auto it = last_msgs.cbegin(); it != last_msgs.cend(); ++it) { | 
			
		
	
		
			
				
					|  |  |  |  |     QString msg_id = QString("%1:%2").arg(it.key().first).arg(it.key().second, 1, 16); | 
			
		
	
		
			
				
					|  |  |  |  |     can_msgs[msg_id] = it.value(); | 
			
		
	
		
			
				
					|  |  |  |  |     counters[msg_id] = it.value().count; | 
			
		
	
		
			
				
					|  |  |  |  |   } | 
			
		
	
		
			
				
					|  |  |  |  |   emit updated(); | 
			
		
	
		
			
				
					|  |  |  |  |   emit msgsReceived(&can_msgs); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
	
		
			
				
					|  |  |  | 
 |