Overte C++ Documentation
PacketReceiver.h
1 //
2 // PacketReceiver.h
3 // libraries/networking/src
4 //
5 // Created by Stephen Birarda on 1/23/2014.
6 // Update by Ryan Huffman on 7/8/2015.
7 // Copyright 2014 High Fidelity, Inc.
8 // Copyright 2021 Vircadia contributors.
9 //
10 // Distributed under the Apache License, Version 2.0.
11 // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
12 //
13 
14 #ifndef hifi_PacketReceiver_h
15 #define hifi_PacketReceiver_h
16 
17 #include <vector>
18 #include <unordered_map>
19 
20 #include <QtCore/QMutex>
21 #include <QtCore/QObject>
22 #include <QtCore/QPointer>
23 #include <QtCore/QSet>
24 #include <QtCore/QSharedPointer>
25 #include <QtCore/QEnableSharedFromThis>
26 
27 #include "NLPacket.h"
28 #include "NLPacketList.h"
29 #include "ReceivedMessage.h"
30 #include "udt/PacketHeaders.h"
31 
33 class Node;
35 
36 namespace std {
37  template <>
38  struct hash<std::pair<SockAddr, udt::Packet::MessageNumber>> {
39  size_t operator()(const std::pair<SockAddr, udt::Packet::MessageNumber>& pair) const {
40  return hash<SockAddr>()(pair.first) ^ hash<udt::Packet::MessageNumber>()(pair.second);
41  }
42  };
43 }
44 
45 class PacketReceiver : public QObject {
46  Q_OBJECT
47 public:
48  class ListenerReference : public QEnableSharedFromThis<ListenerReference> {
49  public:
50  virtual bool invokeDirectly(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>& sourceNode) = 0;
51  bool invokeWithQt(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>& sourceNode);
52  virtual bool isSourced() const = 0;
53  virtual QObject* getObject() const = 0;
54  };
55  typedef QSharedPointer<ListenerReference> ListenerReferencePointer;
56 
57  template<class T>
58  static ListenerReferencePointer makeUnsourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>));
59 
60  template <class T>
61  static ListenerReferencePointer makeSourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>, QSharedPointer<Node>));
62 
63 public:
64  using PacketTypeList = std::vector<PacketType>;
65 
66  PacketReceiver(QObject* parent = 0);
67  PacketReceiver(const PacketReceiver&) = delete;
68 
69  PacketReceiver& operator=(const PacketReceiver&) = delete;
70 
71  void setShouldDropPackets(bool shouldDropPackets) { _shouldDropPackets = shouldDropPackets; }
72 
73  // If deliverPending is false, ReceivedMessage will only be delivered once all packets for the message have
74  // been received. If deliverPending is true, ReceivedMessage will be delivered as soon as the first packet
75  // for the message is received.
76  bool registerListener(PacketType type, const ListenerReferencePointer& listener, bool deliverPending = false);
77  bool registerListenerForTypes(PacketTypeList types, const ListenerReferencePointer& listener);
78  void unregisterListener(QObject* listener);
79 
80  void handleVerifiedPacket(std::unique_ptr<udt::Packet> packet);
81  void handleVerifiedMessagePacket(std::unique_ptr<udt::Packet> message);
82  void handleMessageFailure(SockAddr from, udt::Packet::MessageNumber messageNumber);
83 
84 private:
85  template <class T>
86  class UnsourcedListenerReference : public ListenerReference {
87  public:
88  inline UnsourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>));
89  virtual bool invokeDirectly(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>& sourceNode) override;
90  virtual bool isSourced() const override { return false; }
91  virtual QObject* getObject() const override { return _target; }
92 
93  private:
94  QPointer<T> _target;
95  void (T::*_slot)(QSharedPointer<ReceivedMessage>);
96  };
97 
98  template <class T>
99  class SourcedListenerReference : public ListenerReference {
100  public:
101  inline SourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>, QSharedPointer<Node>));
102  virtual bool invokeDirectly(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>& sourceNode) override;
103  virtual bool isSourced() const override { return true; }
104  virtual QObject* getObject() const override { return _target; }
105 
106  private:
107  QPointer<T> _target;
108  void (T::*_slot)(QSharedPointer<ReceivedMessage>, QSharedPointer<Node>);
109  };
110 
111  struct Listener {
112  ListenerReferencePointer listener;
113  bool deliverPending;
114  };
115 
116  void handleVerifiedMessage(QSharedPointer<ReceivedMessage> message, bool justReceived);
117 
118  // these are brutal hacks for now - ideally GenericThread / ReceivedPacketProcessor
119  // should be changed to have a true event loop and be able to handle our QMetaMethod::invoke
120  void registerDirectListenerForTypes(PacketTypeList types, const ListenerReferencePointer& listener);
121  void registerDirectListener(PacketType type, const ListenerReferencePointer& listener);
122 
123  bool matchingMethodForListener(PacketType type, const ListenerReferencePointer& listener) const;
124  void registerVerifiedListener(PacketType type, const ListenerReferencePointer& listener, bool deliverPending = false);
125 
126  QMutex _packetListenerLock;
127  QHash<PacketType, Listener> _messageListenerMap;
128 
129  bool _shouldDropPackets = false;
130  QMutex _directConnectSetMutex;
131  QSet<QObject*> _directlyConnectedObjects;
132 
133  std::unordered_map<std::pair<SockAddr, udt::Packet::MessageNumber>, QSharedPointer<ReceivedMessage>> _pendingMessages;
134 
135  friend class EntityEditPacketSender;
136  friend class OctreePacketProcessor;
137 };
138 
139 template <class T>
140 PacketReceiver::ListenerReferencePointer PacketReceiver::makeUnsourcedListenerReference(T* target, void (T::* slot)(QSharedPointer<ReceivedMessage>)) {
141  return QSharedPointer<UnsourcedListenerReference<T>>::create(target, slot);
142 }
143 
144 template <class T>
145 PacketReceiver::ListenerReferencePointer PacketReceiver::makeSourcedListenerReference(T* target, void (T::* slot)(QSharedPointer<ReceivedMessage>, QSharedPointer<Node>)) {
146  return QSharedPointer<SourcedListenerReference<T>>::create(target, slot);
147 }
148 
149 template <class T>
150 PacketReceiver::UnsourcedListenerReference<T>::UnsourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>)) :
151  _target(target),_slot(slot) {
152 }
153 
154 template <class T>
155 bool PacketReceiver::UnsourcedListenerReference<T>::invokeDirectly(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>&) {
156  if (_target.isNull()) {
157  return false;
158  }
159  (_target->*_slot)(receivedMessagePointer);
160  return true;
161 }
162 
163 template <class T>
164 PacketReceiver::SourcedListenerReference<T>::SourcedListenerReference(T* target, void (T::*slot)(QSharedPointer<ReceivedMessage>, QSharedPointer<Node>)) :
165  _target(target),_slot(slot) {
166 }
167 
168 template <class T>
169 bool PacketReceiver::SourcedListenerReference<T>::invokeDirectly(const QSharedPointer<ReceivedMessage>& receivedMessagePointer, const QSharedPointer<Node>& sourceNode) {
170  if (_target.isNull()) {
171  return false;
172  }
173  (_target->*_slot)(receivedMessagePointer, sourceNode);
174  return true;
175 }
176 
177 #endif // hifi_PacketReceiver_h
Utility for processing, packing, queueing and sending of outbound edit voxel messages.
Definition: EntityEditPacketSender.h:25
Definition: OctreePacketProcessor.h:24