Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(132)

Side by Side Diff: components/proximity_auth/ble/bluetooth_low_energy_weave_client_connection.h

Issue 2075313002: Substituting legacy protocol with uWeave protocol (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: renamed to weave client connection Created 4 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2015 The Chromium Authors. All rights reserved. 1 // Copyright 2015 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #ifndef COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_CONNECTION_H_ 5 #ifndef COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_WEAVE_CLIENT_CONNECTI ON_H_
6 #define COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_CONNECTION_H_ 6 #define COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_WEAVE_CLIENT_CONNECTI ON_H_
7 7
8 #include <stddef.h> 8 #include <stddef.h>
9 #include <stdint.h> 9 #include <stdint.h>
10 10
11 #include <memory> 11 #include <memory>
12 #include <queue> 12 #include <queue>
13 #include <string> 13 #include <string>
14 14
15 #include "base/macros.h" 15 #include "base/macros.h"
16 #include "base/memory/ref_counted.h" 16 #include "base/memory/ref_counted.h"
17 #include "base/memory/weak_ptr.h" 17 #include "base/memory/weak_ptr.h"
18 #include "base/time/time.h" 18 #include "base/time/time.h"
19 #include "components/proximity_auth/ble/bluetooth_low_energy_characteristics_fin der.h" 19 #include "components/proximity_auth/ble/bluetooth_low_energy_characteristics_fin der.h"
20 #include "components/proximity_auth/ble/bluetooth_low_energy_weave_packet_genera tor.h"
21 #include "components/proximity_auth/ble/bluetooth_low_energy_weave_packet_receiv er.h"
20 #include "components/proximity_auth/ble/fake_wire_message.h" 22 #include "components/proximity_auth/ble/fake_wire_message.h"
21 #include "components/proximity_auth/ble/remote_attribute.h" 23 #include "components/proximity_auth/ble/remote_attribute.h"
22 #include "components/proximity_auth/connection.h" 24 #include "components/proximity_auth/connection.h"
23 #include "device/bluetooth/bluetooth_adapter.h" 25 #include "device/bluetooth/bluetooth_adapter.h"
24 #include "device/bluetooth/bluetooth_device.h" 26 #include "device/bluetooth/bluetooth_device.h"
25 #include "device/bluetooth/bluetooth_gatt_notify_session.h" 27 #include "device/bluetooth/bluetooth_gatt_notify_session.h"
26 #include "device/bluetooth/bluetooth_remote_gatt_characteristic.h" 28 #include "device/bluetooth/bluetooth_remote_gatt_characteristic.h"
27 #include "device/bluetooth/bluetooth_uuid.h" 29 #include "device/bluetooth/bluetooth_uuid.h"
28 30
29 namespace base { 31 namespace base {
30 class TaskRunner; 32 class TaskRunner;
31 } 33 }
32 34
33 namespace proximity_auth { 35 namespace proximity_auth {
34 36
35 class BluetoothThrottler; 37 class BluetoothThrottler;
36 38
37 // Represents a connection with a remote device over Bluetooth low energy. The 39 // Creates GATT connection on top of the BLE connection and act as a Client.
38 // connection is a persistent bidirectional channel for sending and receiving 40 // uWeave communication follows the flow:
39 // wire messages. The remote device is the peripheral mode and the service 41 // Client | Server
40 // contains two characteristics: one to send data and another to receive it. 42 // ---------------------------------|--------------------------------
41 // 43 // send connection request |
42 // The connection flow is described below. 44 // | receive connection request
43 // 45 // | send connection response
44 // Discover Reader and Writer Characteristics 46 // receive connection response |
45 // | 47 // opt: send data | opt: send data
46 // | 48 // receive data | receive data
47 // | 49 // opt: close connection | opt: close connection
48 // Start Notify Session 50 class BluetoothLowEnergyWeaveClientConnection
Kyle Horimoto 2016/06/27 18:06:03 This class will also require a Factory for when we
jingxuy 2016/06/30 00:27:21 Done.
49 // | 51 : public Connection,
50 // | 52 public device::BluetoothAdapter::Observer {
51 // |
52 // Write kInviteToConnectSignal to Writer Characteristic
53 // |
54 // |
55 // |
56 // Read kInvitationResponseSignal from Reader Characteristic
57 // |
58 // |
59 // |
60 // Proximity Auth Connection Established
61 class BluetoothLowEnergyConnection : public Connection,
62 public device::BluetoothAdapter::Observer {
63 public: 53 public:
64 // Signals sent to the remote device to indicate connection related events. 54 // Signals sent to the remote device to indicate connection related events.
65 enum class ControlSignal : uint32_t { 55 enum class ControlSignal : uint32_t {
Kyle Horimoto 2016/06/27 18:06:02 This class is not needed anymore.
sacomoto 2016/06/29 14:26:58 +1
jingxuy 2016/06/30 00:27:21 Done.
66 kInviteToConnectSignal = 0, 56 kInviteToConnectSignal = 0,
67 kInvitationResponseSignal = 1, 57 kInvitationResponseSignal = 1,
68 kSendSignal = 2, 58 kSendSignal = 2,
69 kDisconnectSignal = 3, 59 kDisconnectSignal = 3,
70 }; 60 };
71 61
72 // The sub-state of a proximity_auth::BluetoothLowEnergyConnection class 62 // The sub-state of a proximity_auth::BluetoothLowEnergyWeaveClientConnection
63 // class
sacomoto 2016/06/29 14:26:58 nit: kill the empty line.
jingxuy 2016/06/30 00:27:21 Done.
73 // extends the IN_PROGRESS state of proximity_auth::Connection::Status. 64 // extends the IN_PROGRESS state of proximity_auth::Connection::Status.
74 enum class SubStatus { 65 enum SubStatus {
75 DISCONNECTED, 66 DISCONNECTED,
76 WAITING_GATT_CONNECTION, 67 WAITING_GATT_CONNECTION,
77 WAITING_CHARACTERISTICS, 68 WAITING_CHARACTERISTICS,
78 CHARACTERISTICS_FOUND, 69 CHARACTERISTICS_FOUND,
79 WAITING_NOTIFY_SESSION, 70 WAITING_NOTIFY_SESSION,
80 NOTIFY_SESSION_READY, 71 NOTIFY_SESSION_READY,
81 WAITING_RESPONSE_SIGNAL, 72 WAITING_RESPONSE_SIGNAL,
Kyle Horimoto 2016/06/27 18:06:03 It's not called a response signal in uWeave.
sacomoto 2016/06/29 14:26:58 +1
jingxuy 2016/06/30 00:27:20 Done.
82 CONNECTED, 73 CONNECTED,
83 }; 74 };
84 75
85 // Constructs a Bluetooth low energy connection to the service with 76 // Constructs a Bluetooth low energy connection to the service with
86 // |remote_service_| on the |remote_device|. The |adapter| must be already 77 // |remote_service_| on the |remote_device|. The |adapter| must be already
87 // initaalized and ready. The GATT connection may alreaady be established and 78 // initaalized and ready. The GATT connection may alreaady be established and
Kyle Horimoto 2016/06/27 18:06:02 s/initaalized/initialized/
jingxuy 2016/06/30 00:27:20 Done.
88 // pass through |gatt_connection|. A subsequent call to Connect() must be 79 // pass through |gatt_connection|. A subsequent call to Connect() must be
89 // made. 80 // made.
90 BluetoothLowEnergyConnection( 81 BluetoothLowEnergyWeaveClientConnection(
91 const RemoteDevice& remote_device, 82 const RemoteDevice& remote_device,
92 scoped_refptr<device::BluetoothAdapter> adapter, 83 scoped_refptr<device::BluetoothAdapter> adapter,
93 const device::BluetoothUUID remote_service_uuid, 84 const device::BluetoothUUID remote_service_uuid,
94 BluetoothThrottler* bluetooth_throttler, 85 BluetoothThrottler* bluetooth_throttler,
95 int max_number_of_write_attempts); 86 int max_number_of_write_attempts);
96 87
97 ~BluetoothLowEnergyConnection() override; 88 ~BluetoothLowEnergyWeaveClientConnection() override;
98 89
99 // proximity_auth::Connection: 90 // proximity_auth::Connection:
100 void Connect() override; 91 void Connect() override;
101 void Disconnect() override; 92 void Disconnect() override;
102 std::string GetDeviceAddress() override; 93 std::string GetDeviceAddress() override;
103 94
104 protected: 95 protected:
105 // Exposed for testing. 96 // Exposed for testing.
106 void SetSubStatus(SubStatus status); 97 void SetSubStatus(SubStatus status);
107 SubStatus sub_status() { return sub_status_; } 98 SubStatus sub_status() { return sub_status_; }
(...skipping 19 matching lines...) Expand all
127 void GattCharacteristicValueChanged( 118 void GattCharacteristicValueChanged(
128 device::BluetoothAdapter* adapter, 119 device::BluetoothAdapter* adapter,
129 device::BluetoothRemoteGattCharacteristic* characteristic, 120 device::BluetoothRemoteGattCharacteristic* characteristic,
130 const std::vector<uint8_t>& value) override; 121 const std::vector<uint8_t>& value) override;
131 122
132 private: 123 private:
133 // Represents a request to write |value| to a some characteristic. 124 // Represents a request to write |value| to a some characteristic.
134 // |is_last_write_for_wire_messsage| indicates whether this request 125 // |is_last_write_for_wire_messsage| indicates whether this request
135 // corresponds to the last write request for some wire message. 126 // corresponds to the last write request for some wire message.
136 // A WireMessage corresponds to exactly two WriteRequest: the first containing 127 // A WireMessage corresponds to exactly two WriteRequest: the first containing
137 // a kSendSignal + the size of the WireMessage, and the second containing a 128 // a kSendSignal + the size of the WireMessage, and the second containing a
Kyle Horimoto 2016/06/27 18:06:02 Doesn't use kSendSignal.
jingxuy 2016/06/30 00:27:21 Done.
138 // SendStatusSignal + the serialized WireMessage. 129 // SendStatusSignal + the serialized WireMessage.
139 struct WriteRequest { 130 struct WriteRequest {
140 WriteRequest(const std::vector<uint8_t>& val, bool flag); 131 WriteRequest(const std::vector<uint8_t>& val, bool flag);
Kyle Horimoto 2016/06/27 18:06:03 Use Packet. Same below, and same in the .cc, and s
jingxuy 2016/06/30 00:27:21 Done.
141 WriteRequest(const WriteRequest& other); 132 WriteRequest(const WriteRequest& other);
142 ~WriteRequest(); 133 ~WriteRequest();
143 134
144 std::vector<uint8_t> value; 135 std::vector<uint8_t> value;
145 bool is_last_write_for_wire_message; 136 bool is_last_write_for_wire_message;
146 int number_of_failed_attempts; 137 int number_of_failed_attempts;
147 }; 138 };
148 139
149 // Creates the GATT connection with |remote_device|. 140 // Creates the GATT connection with |remote_device|.
150 void CreateGattConnection(); 141 void CreateGattConnection();
151 142
152 // Called when a GATT connection is created. 143 // Called when a GATT connection is created.
153 void OnGattConnectionCreated( 144 void OnGattConnectionCreated(
154 std::unique_ptr<device::BluetoothGattConnection> gatt_connection); 145 std::unique_ptr<device::BluetoothGattConnection> gatt_connection);
155 146
156 // Callback called when there is an error creating the GATT connection. 147 // Callback called when there is an error creating the GATT connection.
157 void OnCreateGattConnectionError( 148 void OnCreateGattConnectionError(
158 device::BluetoothDevice::ConnectErrorCode error_code); 149 device::BluetoothDevice::ConnectErrorCode error_code);
159 150
160 // Callback called when |to_peripheral_char_| and |from_peripheral_char_| were 151 // Callback called when |tx_characteristic_| and |rx_characteristic_| were
161 // found. 152 // found.
162 void OnCharacteristicsFound(const RemoteAttribute& service, 153 void OnCharacteristicsFound(const RemoteAttribute& service,
163 const RemoteAttribute& to_peripheral_char, 154 const RemoteAttribute& to_peripheral_char,
sacomoto 2016/06/29 14:26:58 s/to_peripheral_char/tx_chararacteristic/.
jingxuy 2016/06/30 00:27:21 Done.
164 const RemoteAttribute& from_peripheral_char); 155 const RemoteAttribute& from_peripheral_char);
sacomoto 2016/06/29 14:26:58 s/from_peripheral_char/rx_characteristic/.
jingxuy 2016/06/30 00:27:20 Done.
165 156
166 // Callback called there was an error finding the characteristics. 157 // Callback called there was an error finding the characteristics.
167 void OnCharacteristicsFinderError( 158 void OnCharacteristicsFinderError(
168 const RemoteAttribute& to_peripheral_char, 159 const RemoteAttribute& to_peripheral_char,
169 const RemoteAttribute& from_peripheral_char); 160 const RemoteAttribute& from_peripheral_char);
sacomoto 2016/06/29 14:26:58 The same here.
jingxuy 2016/06/30 00:27:20 Done.
170 161
171 // Starts a notify session for |from_peripheral_char_| when ready 162 // Starts a notify session for |rx_characteristic_| when ready
172 // (SubStatus::CHARACTERISTICS_FOUND). 163 // (SubStatus::CHARACTERISTICS_FOUND).
173 void StartNotifySession(); 164 void StartNotifySession();
174 165
175 // Called when a notification session is successfully started for 166 // Called when a notification session is successfully started for
176 // |from_peripheral_char_| characteristic. 167 // |rx_characteristic_| characteristic.
177 void OnNotifySessionStarted( 168 void OnNotifySessionStarted(
178 std::unique_ptr<device::BluetoothGattNotifySession> notify_session); 169 std::unique_ptr<device::BluetoothGattNotifySession> notify_session);
179 170
180 // Called when there is an error starting a notification session for 171 // Called when there is an error starting a notification session for
181 // |from_peripheral_char_| characteristic. 172 // |rx_characteristic_| characteristic.
182 void OnNotifySessionError(device::BluetoothGattService::GattErrorCode); 173 void OnNotifySessionError(device::BluetoothGattService::GattErrorCode);
183 174
184 // Stops |notify_session_|. 175 // Stops |notify_session_|.
185 void StopNotifySession(); 176 void StopNotifySession();
186 177
187 // Sends an invite to connect signal to the peripheral if when ready 178 // Sends an invite to connect signal to the peripheral if when ready
188 // (SubStatus::NOTIFY_SESSION_READY). 179 // (SubStatus::NOTIFY_SESSION_READY).
189 void SendInviteToConnectSignal(); 180 void SendInviteToConnectSignal();
Kyle Horimoto 2016/06/27 18:06:02 Again, these "signal" values only exist in the leg
jingxuy 2016/06/30 00:27:20 Done.
190 181
191 // Completes and updates the status accordingly. 182 // Completes and updates the status accordingly.
192 void CompleteConnection(); 183 void CompleteConnection();
193 184
194 // This is the only entry point for WriteRequests, which are processed 185 // This is the only entry point for WriteRequests, which are processed
195 // accordingly the following flow: 186 // accordingly the following flow:
196 // 1) |request| is enqueued; 187 // 1) |request| is enqueued;
197 // 2) |request| will be processed by ProcessNextWriteRequest() when there is 188 // 2) |request| will be processed by ProcessNextWriteRequest() when there is
198 // no pending write request; 189 // no pending write request;
199 // 3) |request| will be dequeued when it's successfully processed 190 // 3) |request| will be dequeued when it's successfully processed
200 // (OnRemoteCharacteristicWritten()); 191 // (OnRemoteCharacteristicWritten());
201 // 4) |request| is not dequeued if it fails 192 // 4) |request| is not dequeued if it fails
202 // (OnWriteRemoteCharacteristicError()), it remains on the queue and will be 193 // (OnWriteRemoteCharacteristicError()), it remains on the queue and will be
203 // retried. |request| will remain on the queue until it succeeds or it 194 // retried. |request| will remain on the queue until it succeeds or it
204 // triggers a Disconnect() call (after |max_number_of_tries_|). 195 // triggers a Disconnect() call (after |max_number_of_tries_|).
205 void WriteRemoteCharacteristic(WriteRequest request); 196 void WriteRemoteCharacteristic(const WriteRequest& request);
206 197
207 // Processes the next request in |write_requests_queue_|. 198 // Processes the next request in |write_requests_queue_|.
208 void ProcessNextWriteRequest(); 199 void ProcessNextWriteRequest();
209 200
210 // Called when the 201 // Called when the
211 // BluetoothRemoteGattCharacteristic::RemoteCharacteristicWrite() is 202 // BluetoothRemoteGattCharacteristic::RemoteCharacteristicWrite() is
212 // successfully complete. 203 // successfully complete.
213 void OnRemoteCharacteristicWritten(bool run_did_send_message_callback); 204 void OnRemoteCharacteristicWritten(bool run_did_send_message_callback);
214 205
215 // Called when there is an error writing to the remote characteristic 206 // Called when there is an error writing to the remote characteristic
216 // |to_peripheral_char_|. 207 // |tx_characteristic_|.
217 void OnWriteRemoteCharacteristicError( 208 void OnWriteRemoteCharacteristicError(
218 bool run_did_send_message_callback, 209 bool run_did_send_message_callback,
219 device::BluetoothRemoteGattService::GattErrorCode error); 210 device::BluetoothRemoteGattService::GattErrorCode error);
220 211
221 // Builds the value to be written on |to_peripheral_char_|. The value
222 // corresponds to |signal| concatenated with |payload|.
223 WriteRequest BuildWriteRequest(const std::vector<uint8_t>& signal,
224 const std::vector<uint8_t>& bytes,
225 bool is_last_message_for_wire_message);
226
227 // Prints the time elapsed since |Connect()| was called. 212 // Prints the time elapsed since |Connect()| was called.
228 void PrintTimeElapsed(); 213 void PrintTimeElapsed();
229 214
230 // Returns the device corresponding to |remote_device_address_|. 215 // Returns the device corresponding to |remote_device_address_|.
231 device::BluetoothDevice* GetRemoteDevice(); 216 device::BluetoothDevice* GetRemoteDevice();
232 217
233 // Returns the service corresponding to |remote_service_| in the current 218 // Returns the service corresponding to |remote_service_| in the current
234 // device. 219 // device.
235 device::BluetoothRemoteGattService* GetRemoteService(); 220 device::BluetoothRemoteGattService* GetRemoteService();
236 221
237 // Returns the characteristic corresponding to |identifier| in the current 222 // Returns the characteristic corresponding to |identifier| in the current
238 // service. 223 // service.
239 device::BluetoothRemoteGattCharacteristic* GetGattCharacteristic( 224 device::BluetoothRemoteGattCharacteristic* GetGattCharacteristic(
240 const std::string& identifier); 225 const std::string& identifier);
241 226
242 // Convert the first 4 bytes from a byte vector to a uint32_t.
243 uint32_t ToUint32(const std::vector<uint8_t>& bytes);
244
245 // Convert an uint32_t to a byte vector.
246 const std::vector<uint8_t> ToByteVector(uint32_t value);
247
248 // The Bluetooth adapter over which the Bluetooth connection will be made. 227 // The Bluetooth adapter over which the Bluetooth connection will be made.
249 scoped_refptr<device::BluetoothAdapter> adapter_; 228 scoped_refptr<device::BluetoothAdapter> adapter_;
250 229
251 // Remote service the |gatt_connection_| was established with. 230 // Remote service the |gatt_connection_| was established with.
252 RemoteAttribute remote_service_; 231 RemoteAttribute remote_service_;
253 232
233 // uWeave packet generator.
234 std::unique_ptr<BluetoothLowEnergyWeavePacketGenerator> packet_generator_;
235
236 // uWeave packet receiver.
237 std::unique_ptr<BluetoothLowEnergyWeavePacketReceiver> packet_receiver_;
238
254 // Characteristic used to send data to the remote device. 239 // Characteristic used to send data to the remote device.
255 RemoteAttribute to_peripheral_char_; 240 RemoteAttribute tx_characteristic_;
256 241
257 // Characteristic used to receive data from the remote device. 242 // Characteristic used to receive data from the remote device.
258 RemoteAttribute from_peripheral_char_; 243 RemoteAttribute rx_characteristic_;
259 244
260 // Throttles repeated connection attempts to the same device. This is a 245 // Throttles repeated connection attempts to the same device. This is a
261 // workaround for crbug.com/508919. Not owned, must outlive this instance. 246 // workaround for crbug.com/508919. Not owned, must outlive this instance.
262 BluetoothThrottler* bluetooth_throttler_; 247 BluetoothThrottler* bluetooth_throttler_;
263 248
264 scoped_refptr<base::TaskRunner> task_runner_; 249 scoped_refptr<base::TaskRunner> task_runner_;
265 250
266 // The GATT connection with the remote device. 251 // The GATT connection with the remote device.
267 std::unique_ptr<device::BluetoothGattConnection> gatt_connection_; 252 std::unique_ptr<device::BluetoothGattConnection> gatt_connection_;
268 253
269 // The characteristics finder for remote device. 254 // The characteristics finder for remote device.
270 std::unique_ptr<BluetoothLowEnergyCharacteristicsFinder> 255 std::unique_ptr<BluetoothLowEnergyCharacteristicsFinder>
271 characteristic_finder_; 256 characteristic_finder_;
272 257
273 // The notify session for |from_peripheral_char|. 258 // The notify session for |from_peripheral_char|.
274 std::unique_ptr<device::BluetoothGattNotifySession> notify_session_; 259 std::unique_ptr<device::BluetoothGattNotifySession> notify_session_;
275 260
276 // Internal connection status 261 // Internal connection status
277 SubStatus sub_status_; 262 SubStatus sub_status_;
278 263
279 // Indicates a receiving operation is in progress. This is set after a
280 // ControlSignal::kSendSignal was received from the remote device.
281 bool receiving_bytes_;
282
283 // Total number of bytes expected for the current receive operation.
284 std::size_t expected_number_of_incoming_bytes_;
285
286 // Bytes already received for the current receive operation. 264 // Bytes already received for the current receive operation.
287 std::string incoming_bytes_buffer_; 265 std::string incoming_bytes_buffer_;
288 266
289 // Indicates there is a 267 // Indicates there is a
290 // BluetoothRemoteGattCharacteristic::WriteRemoteCharacteristic 268 // BluetoothRemoteGattCharacteristic::WriteRemoteCharacteristic
291 // operation pending. 269 // operation pending.
292 bool write_remote_characteristic_pending_; 270 bool write_remote_characteristic_pending_;
293 271
294 std::queue<WriteRequest> write_requests_queue_; 272 std::queue<WriteRequest> write_requests_queue_;
295 273
296 // Maximum number of tries to send any write request. 274 // Maximum number of tries to send any write request.
297 int max_number_of_write_attempts_; 275 int max_number_of_write_attempts_;
298 276
299 // Maximum number of bytes that fit in a single chunk to be written in
300 // |to_peripheral_char_|. Ideally, this should be the maximum value the
301 // peripheral supports and it should be agreed when the GATT connection is
302 // created. Currently, there is no API to find this value. The implementation
303 // uses a hard-coded constant.
304 int max_chunk_size_;
305
306 // Stores when the instace was created. 277 // Stores when the instace was created.
307 base::TimeTicks start_time_; 278 base::TimeTicks start_time_;
308 279
309 base::WeakPtrFactory<BluetoothLowEnergyConnection> weak_ptr_factory_; 280 base::WeakPtrFactory<BluetoothLowEnergyWeaveClientConnection>
281 weak_ptr_factory_;
310 282
311 DISALLOW_COPY_AND_ASSIGN(BluetoothLowEnergyConnection); 283 DISALLOW_COPY_AND_ASSIGN(BluetoothLowEnergyWeaveClientConnection);
312 }; 284 };
313 285
314 } // namespace proximity_auth 286 } // namespace proximity_auth
315 287
316 #endif // COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_CONNECTION_H_ 288 #endif // COMPONENTS_PROXIMITY_AUTH_BLE_BLUETOOTH_LOW_ENERGY_WEAVE_CLIENT_CONNE CTION_H_
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698