You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							157 lines
						
					
					
						
							5.1 KiB
						
					
					
				
			
		
		
	
	
							157 lines
						
					
					
						
							5.1 KiB
						
					
					
				#pragma once
 | 
						|
 | 
						|
#include <algorithm>
 | 
						|
#include <array>
 | 
						|
#include <condition_variable>
 | 
						|
#include <memory>
 | 
						|
#include <mutex>
 | 
						|
#include <optional>
 | 
						|
#include <set>
 | 
						|
#include <unordered_map>
 | 
						|
#include <utility>
 | 
						|
#include <vector>
 | 
						|
 | 
						|
#include <QColor>
 | 
						|
#include <QDateTime>
 | 
						|
 | 
						|
#include "cereal/messaging/messaging.h"
 | 
						|
#include "tools/cabana/dbc/dbcmanager.h"
 | 
						|
#include "tools/cabana/utils/util.h"
 | 
						|
#include "tools/replay/util.h"
 | 
						|
 | 
						|
struct CanData {
 | 
						|
  void compute(const MessageId &msg_id, const uint8_t *dat, const int size, double current_sec,
 | 
						|
               double playback_speed, const std::vector<uint8_t> &mask, double in_freq = 0);
 | 
						|
 | 
						|
  double ts = 0.;
 | 
						|
  uint32_t count = 0;
 | 
						|
  double freq = 0;
 | 
						|
  std::vector<uint8_t> dat;
 | 
						|
  std::vector<QColor> colors;
 | 
						|
 | 
						|
  struct ByteLastChange {
 | 
						|
    double ts = 0;
 | 
						|
    int delta = 0;
 | 
						|
    int same_delta_counter = 0;
 | 
						|
    bool suppressed = false;
 | 
						|
  };
 | 
						|
  std::vector<ByteLastChange> last_changes;
 | 
						|
  std::vector<std::array<uint32_t, 8>> bit_flip_counts;
 | 
						|
  double last_freq_update_ts = 0;
 | 
						|
};
 | 
						|
 | 
						|
struct CanEvent {
 | 
						|
  uint8_t src;
 | 
						|
  uint32_t address;
 | 
						|
  uint64_t mono_time;
 | 
						|
  uint8_t size;
 | 
						|
  uint8_t dat[];
 | 
						|
};
 | 
						|
 | 
						|
struct CompareCanEvent {
 | 
						|
  constexpr bool operator()(const CanEvent *const e, uint64_t ts) const { return e->mono_time < ts; }
 | 
						|
  constexpr bool operator()(uint64_t ts, const CanEvent *const e) const { return ts < e->mono_time; }
 | 
						|
};
 | 
						|
 | 
						|
typedef std::unordered_map<MessageId, std::vector<const CanEvent *>> MessageEventsMap;
 | 
						|
using CanEventIter = std::vector<const CanEvent *>::const_iterator;
 | 
						|
 | 
						|
class AbstractStream : public QObject {
 | 
						|
  Q_OBJECT
 | 
						|
 | 
						|
public:
 | 
						|
  AbstractStream(QObject *parent);
 | 
						|
  virtual ~AbstractStream() {}
 | 
						|
  virtual void start() = 0;
 | 
						|
  virtual bool liveStreaming() const { return true; }
 | 
						|
  virtual void seekTo(double ts) {}
 | 
						|
  virtual QString routeName() const = 0;
 | 
						|
  virtual QString carFingerprint() const { return ""; }
 | 
						|
  virtual QDateTime beginDateTime() const { return {}; }
 | 
						|
  virtual uint64_t beginMonoTime() const { return 0; }
 | 
						|
  virtual double minSeconds() const { return 0; }
 | 
						|
  virtual double maxSeconds() const { return 0; }
 | 
						|
  virtual void setSpeed(float speed) {}
 | 
						|
  virtual double getSpeed() { return 1; }
 | 
						|
  virtual bool isPaused() const { return false; }
 | 
						|
  virtual void pause(bool pause) {}
 | 
						|
  void setTimeRange(const std::optional<std::pair<double, double>> &range);
 | 
						|
  const std::optional<std::pair<double, double>> &timeRange() const { return time_range_; }
 | 
						|
 | 
						|
  inline double currentSec() const { return current_sec_; }
 | 
						|
  inline uint64_t toMonoTime(double sec) const { return beginMonoTime() + std::max(sec, 0.0) * 1e9; }
 | 
						|
  inline double toSeconds(uint64_t mono_time) const { return std::max(0.0, (mono_time - beginMonoTime()) / 1e9); }
 | 
						|
 | 
						|
  inline const std::unordered_map<MessageId, CanData> &lastMessages() const { return last_msgs; }
 | 
						|
  bool isMessageActive(const MessageId &id) const;
 | 
						|
  inline const MessageEventsMap &eventsMap() const { return events_; }
 | 
						|
  inline const std::vector<const CanEvent *> &allEvents() const { return all_events_; }
 | 
						|
  const CanData &lastMessage(const MessageId &id) const;
 | 
						|
  const std::vector<const CanEvent *> &events(const MessageId &id) const;
 | 
						|
  std::pair<CanEventIter, CanEventIter> eventsInRange(const MessageId &id, std::optional<std::pair<double, double>> time_range) const;
 | 
						|
 | 
						|
  size_t suppressHighlighted();
 | 
						|
  void clearSuppressed();
 | 
						|
  void suppressDefinedSignals(bool suppress);
 | 
						|
 | 
						|
signals:
 | 
						|
  void paused();
 | 
						|
  void resume();
 | 
						|
  void seeking(double sec);
 | 
						|
  void seekedTo(double sec);
 | 
						|
  void timeRangeChanged(const std::optional<std::pair<double, double>> &range);
 | 
						|
  void eventsMerged(const MessageEventsMap &events_map);
 | 
						|
  void msgsReceived(const std::set<MessageId> *new_msgs, bool has_new_ids);
 | 
						|
  void sourcesUpdated(const SourceSet &s);
 | 
						|
  void privateUpdateLastMsgsSignal();
 | 
						|
 | 
						|
public:
 | 
						|
  SourceSet sources;
 | 
						|
 | 
						|
protected:
 | 
						|
  void mergeEvents(const std::vector<const CanEvent *> &events);
 | 
						|
  const CanEvent *newEvent(uint64_t mono_time, const cereal::CanData::Reader &c);
 | 
						|
  void updateEvent(const MessageId &id, double sec, const uint8_t *data, uint8_t size);
 | 
						|
  void waitForSeekFinshed();
 | 
						|
  std::vector<const CanEvent *> all_events_;
 | 
						|
  double current_sec_ = 0;
 | 
						|
  std::optional<std::pair<double, double>> time_range_;
 | 
						|
 | 
						|
private:
 | 
						|
  void updateLastMessages();
 | 
						|
  void updateLastMsgsTo(double sec);
 | 
						|
  void updateMasks();
 | 
						|
 | 
						|
  MessageEventsMap events_;
 | 
						|
  std::unordered_map<MessageId, CanData> last_msgs;
 | 
						|
  std::unique_ptr<MonotonicBuffer> event_buffer_;
 | 
						|
 | 
						|
  // Members accessed in multiple threads. (mutex protected)
 | 
						|
  std::mutex mutex_;
 | 
						|
  std::condition_variable seek_finished_cv_;
 | 
						|
  bool seek_finished_ = false;
 | 
						|
  std::set<MessageId> new_msgs_;
 | 
						|
  std::unordered_map<MessageId, CanData> messages_;
 | 
						|
  std::unordered_map<MessageId, std::vector<uint8_t>> masks_;
 | 
						|
};
 | 
						|
 | 
						|
class AbstractOpenStreamWidget : public QWidget {
 | 
						|
  Q_OBJECT
 | 
						|
public:
 | 
						|
  AbstractOpenStreamWidget(QWidget *parent = nullptr) : QWidget(parent) {}
 | 
						|
  virtual AbstractStream *open() = 0;
 | 
						|
 | 
						|
signals:
 | 
						|
  void enableOpenButton(bool);
 | 
						|
};
 | 
						|
 | 
						|
class DummyStream : public AbstractStream {
 | 
						|
  Q_OBJECT
 | 
						|
public:
 | 
						|
  DummyStream(QObject *parent) : AbstractStream(parent) {}
 | 
						|
  QString routeName() const override { return tr("No Stream"); }
 | 
						|
  void start() override {}
 | 
						|
};
 | 
						|
 | 
						|
// A global pointer referring to the unique AbstractStream object
 | 
						|
extern AbstractStream *can;
 | 
						|
 |