• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #ifndef CHROME_BROWSER_EXTENSIONS_API_DIAL_DIAL_SERVICE_H_
6 #define CHROME_BROWSER_EXTENSIONS_API_DIAL_DIAL_SERVICE_H_
7 
8 #include <string>
9 
10 #include "base/gtest_prod_util.h"
11 #include "base/memory/scoped_ptr.h"
12 #include "base/memory/scoped_vector.h"
13 #include "base/memory/weak_ptr.h"
14 #include "base/observer_list.h"
15 #include "base/threading/thread_checker.h"
16 #include "base/timer/timer.h"
17 #include "net/base/net_log.h"
18 #include "net/udp/udp_socket.h"
19 
20 namespace net {
21 class IPEndPoint;
22 class IPAddress;
23 class IOBuffer;
24 class StringIOBuffer;
25 struct NetworkInterface;
26 }
27 
28 namespace extensions {
29 
30 class DialDeviceData;
31 
32 // DialService accepts requests to discover devices, sends multiple M-SEARCH
33 // requests via UDP multicast, and notifies observers when a DIAL-compliant
34 // device responds.
35 //
36 // Each time Discover() is called, kDialNumRequests M-SEARCH requests are sent
37 // (with a delay of kDialRequestIntervalMillis in between):
38 //
39 // Time    Action
40 // ----    ------
41 // T1      Request 1 sent, OnDiscoveryReqest() called
42 // ...
43 // Tk      Request kDialNumRequests sent, OnDiscoveryReqest() called
44 // Tf      OnDiscoveryFinished() called
45 //
46 // Any time a valid response is received between T1 and Tf, it is parsed and
47 // OnDeviceDiscovered() is called with the result.  Tf is set to Tk +
48 // kDialResponseTimeoutSecs (the response timeout passed in each request).
49 //
50 // Calling Discover() again between T1 and Tf has no effect.
51 //
52 // All relevant constants are defined in dial_service.cc.
53 //
54 // TODO(mfoltz): Port this into net/.
55 // See https://code.google.com/p/chromium/issues/detail?id=164473
56 class DialService {
57  public:
58   enum DialServiceErrorCode {
59     DIAL_SERVICE_NO_INTERFACES = 0,
60     DIAL_SERVICE_SOCKET_ERROR
61   };
62 
63   class Observer {
64    public:
65     // Called when a single discovery request was sent.
66     virtual void OnDiscoveryRequest(DialService* service) = 0;
67 
68     // Called when a device responds to a request.
69     virtual void OnDeviceDiscovered(DialService* service,
70                                     const DialDeviceData& device) = 0;
71 
72     // Called when we have all responses from the last discovery request.
73     virtual void OnDiscoveryFinished(DialService* service) = 0;
74 
75     // Called when an error occurs.
76     virtual void OnError(DialService* service,
77                          const DialServiceErrorCode& code) = 0;
78 
79    protected:
~Observer()80     virtual ~Observer() {}
81   };
82 
~DialService()83   virtual ~DialService() {}
84 
85   // Starts a new round of discovery.  Returns |true| if discovery was started
86   // successfully or there is already one active. Returns |false| on error.
87   virtual bool Discover() = 0;
88 
89   // Called by listeners to this service to add/remove themselves as observers.
90   virtual void AddObserver(Observer* observer) = 0;
91   virtual void RemoveObserver(Observer* observer) = 0;
92   virtual bool HasObserver(Observer* observer) = 0;
93 };
94 
95 // Implements DialService.
96 //
97 // NOTE(mfoltz): It would make this class cleaner to refactor most of the state
98 // associated with a single discovery cycle into its own |DiscoveryOperation|
99 // object.  This would also simplify lifetime of the object w.r.t. DialRegistry;
100 // the Registry would not need to create/destroy the Service on demand.
101 class DialServiceImpl : public DialService,
102                         public base::SupportsWeakPtr<DialServiceImpl> {
103  public:
104   explicit DialServiceImpl(net::NetLog* net_log);
105   virtual ~DialServiceImpl();
106 
107   // DialService implementation
108   virtual bool Discover() OVERRIDE;
109   virtual void AddObserver(Observer* observer) OVERRIDE;
110   virtual void RemoveObserver(Observer* observer) OVERRIDE;
111   virtual bool HasObserver(Observer* observer) OVERRIDE;
112 
113  private:
114   // Represents a socket binding to a single network interface.
115   class DialSocket {
116    public:
117     // TODO(imcheng): Consider writing a DialSocket::Delegate interface that
118     // declares methods for these callbacks, and taking a ptr to the delegate
119     // here.
120     DialSocket(
121         const base::Closure& discovery_request_cb,
122         const base::Callback<void(const DialDeviceData&)>& device_discovered_cb,
123         const base::Closure& on_error_cb);
124     ~DialSocket();
125 
126     // Creates a socket using |net_log| and |net_log_source| and binds it to
127     // |bind_ip_address|.
128     bool CreateAndBindSocket(const net::IPAddressNumber& bind_ip_address,
129                              net::NetLog* net_log,
130                              net::NetLog::Source net_log_source);
131 
132     // Sends a single discovery request |send_buffer| to |send_address|
133     // over the socket.
134     void SendOneRequest(const net::IPEndPoint& send_address,
135                         const scoped_refptr<net::StringIOBuffer>& send_buffer);
136 
137     // Returns true if the socket is closed.
138     bool IsClosed();
139 
140    private:
141     // Checks the result of a socket operation.  The name of the socket
142     // operation is given by |operation| and the result of the operation is
143     // given by |result|. If the result is an error, closes the socket,
144     // calls |on_error_cb_|, and returns |false|.  Returns
145     // |true| otherwise. |operation| and |result| are logged.
146     bool CheckResult(const char* operation, int result);
147 
148     // Closes the socket.
149     void Close();
150 
151     // Callback invoked for socket writes.
152     void OnSocketWrite(int buffer_size, int result);
153 
154     // Establishes the callback to read from the socket.  Returns true if
155     // successful.
156     bool ReadSocket();
157 
158     // Callback invoked for socket reads.
159     void OnSocketRead(int result);
160 
161     // Callback invoked for socket reads.
162     void HandleResponse(int bytes_read);
163 
164     // Parses a response into a DialDeviceData object. If the DIAL response is
165     // invalid or does not contain enough information, then the return
166     // value will be false and |device| is not changed.
167     static bool ParseResponse(const std::string& response,
168                               const base::Time& response_time,
169                               DialDeviceData* device);
170 
171     // The UDP socket.
172     scoped_ptr<net::UDPSocket> socket_;
173 
174     // Buffer for socket reads.
175     scoped_refptr<net::IOBufferWithSize> recv_buffer_;
176 
177     // The source of of the last socket read.
178     net::IPEndPoint recv_address_;
179 
180     // Thread checker.
181     base::ThreadChecker thread_checker_;
182 
183     // The callback to be invoked when a discovery request was made.
184     base::Closure discovery_request_cb_;
185 
186     // The callback to be invoked when a device has been discovered.
187     base::Callback<void(const DialDeviceData&)> device_discovered_cb_;
188 
189     // The callback to be invoked when there is an error with socket operations.
190     base::Closure on_error_cb_;
191 
192     // Marks whether there is an active write callback.
193     bool is_writing_;
194 
195     // Marks whether there is an active read callback.
196     bool is_reading_;
197 
198     FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestNotifyOnError);
199     FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestOnDeviceDiscovered);
200     FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestOnDiscoveryRequest);
201     FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestResponseParsing);
202     DISALLOW_COPY_AND_ASSIGN(DialSocket);
203   };
204 
205   // Starts the control flow for one discovery cycle.
206   void StartDiscovery();
207 
208   // For each network interface in |list|, finds all unqiue IPv4 network
209   // interfaces and call |DiscoverOnAddresses()| with their IP addresses.
210   void SendNetworkList(const net::NetworkInterfaceList& list);
211 
212   // Calls |BindAndAddSocket()| for each address in |ip_addresses|, calls
213   // |SendOneRequest()|, and start the timer to finish discovery if needed.
214   // The (Address family, interface index) of each address in |ip_addresses|
215   // must be unique. If |ip_address| is empty, calls |FinishDiscovery()|.
216   void DiscoverOnAddresses(
217       const std::vector<net::IPAddressNumber>& ip_addresses);
218 
219   // Creates a DialSocket, binds it to |bind_ip_address| and if
220   // successful, add the DialSocket to |dial_sockets_|.
221   void BindAndAddSocket(const net::IPAddressNumber& bind_ip_address);
222 
223   // Creates a DialSocket with callbacks to this object.
224   scoped_ptr<DialSocket> CreateDialSocket();
225 
226   // Sends a single discovery request to every socket that are currently open.
227   void SendOneRequest();
228 
229   // Notify observers that a discovery request was made.
230   void NotifyOnDiscoveryRequest();
231 
232   // Notify observers a device has been discovered.
233   void NotifyOnDeviceDiscovered(const DialDeviceData& device_data);
234 
235   // Notify observers that there has been an error with one of the DialSockets.
236   void NotifyOnError();
237 
238   // Called from finish_timer_ when we are done with the current round of
239   // discovery.
240   void FinishDiscovery();
241 
242   // Returns |true| if there are open sockets.
243   bool HasOpenSockets();
244 
245   // DialSockets for each network interface whose ip address was
246   // successfully bound.
247   ScopedVector<DialSocket> dial_sockets_;
248 
249   // The NetLog for this service.
250   net::NetLog* net_log_;
251 
252   // The NetLog source for this service.
253   net::NetLog::Source net_log_source_;
254 
255   // The multicast address:port for search requests.
256   net::IPEndPoint send_address_;
257 
258   // Buffer for socket writes.
259   scoped_refptr<net::StringIOBuffer> send_buffer_;
260 
261   // True when we are currently doing discovery.
262   bool discovery_active_;
263 
264   // The number of requests that have been sent in the current discovery.
265   int num_requests_sent_;
266 
267   // The maximum number of requests to send per discovery cycle.
268   int max_requests_;
269 
270   // Timer for finishing discovery.
271   base::OneShotTimer<DialServiceImpl> finish_timer_;
272 
273   // The delay for |finish_timer_|; how long to wait for discovery to finish.
274   // Setting this to zero disables the timer.
275   base::TimeDelta finish_delay_;
276 
277   // Timer for sending multiple requests at fixed intervals.
278   base::RepeatingTimer<DialServiceImpl> request_timer_;
279 
280   // The delay for |request_timer_|; how long to wait between successive
281   // requests.
282   base::TimeDelta request_interval_;
283 
284   // List of observers.
285   ObserverList<Observer> observer_list_;
286 
287   // Thread checker.
288   base::ThreadChecker thread_checker_;
289 
290   friend class DialServiceTest;
291   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestSendMultipleRequests);
292   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestMultipleNetworkInterfaces);
293   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestNotifyOnError);
294   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestOnDeviceDiscovered);
295   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestOnDiscoveryFinished);
296   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestOnDiscoveryRequest);
297   FRIEND_TEST_ALL_PREFIXES(DialServiceTest, TestResponseParsing);
298   DISALLOW_COPY_AND_ASSIGN(DialServiceImpl);
299 };
300 
301 }  // namespace extensions
302 
303 #endif  // CHROME_BROWSER_EXTENSIONS_API_DIAL_DIAL_SERVICE_H_
304