OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2014 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 #include "components/proximity_auth/bluetooth_connection.h" | |
6 | |
7 #include "base/bind.h" | |
8 #include "base/message_loop/message_loop.h" | |
9 #include "base/numerics/safe_conversions.h" | |
10 #include "components/proximity_auth/bluetooth_util.h" | |
11 #include "components/proximity_auth/remote_device.h" | |
12 #include "components/proximity_auth/wire_message.h" | |
13 #include "device/bluetooth/bluetooth_adapter_factory.h" | |
14 #include "device/bluetooth/bluetooth_device.h" | |
15 #include "net/base/io_buffer.h" | |
16 | |
17 namespace proximity_auth { | |
18 namespace { | |
19 const int kReceiveBufferSizeBytes = 1024; | |
20 } | |
21 | |
22 BluetoothConnection::BluetoothConnection(const RemoteDevice& remote_device, | |
23 const device::BluetoothUUID& uuid) | |
24 : Connection(remote_device), uuid_(uuid), weak_ptr_factory_(this) { | |
25 } | |
26 | |
27 BluetoothConnection::~BluetoothConnection() { | |
28 Disconnect(); | |
29 } | |
30 | |
31 void BluetoothConnection::Connect() { | |
32 if (status() != DISCONNECTED) { | |
33 VLOG(1) | |
34 << "[BC] Ignoring attempt to connect a non-disconnected connection."; | |
35 return; | |
36 } | |
37 | |
38 if (!device::BluetoothAdapterFactory::IsBluetoothAdapterAvailable()) { | |
39 VLOG(1) | |
40 << "[BC] Connection failed: Bluetooth is unsupported on this platform."; | |
41 return; | |
42 } | |
43 | |
44 SetStatus(IN_PROGRESS); | |
45 device::BluetoothAdapterFactory::GetAdapter( | |
46 base::Bind(&BluetoothConnection::OnAdapterInitialized, | |
47 weak_ptr_factory_.GetWeakPtr())); | |
48 } | |
49 | |
50 void BluetoothConnection::Disconnect() { | |
51 if (status() == DISCONNECTED) { | |
52 VLOG(1) | |
53 << "[BC] Ignoring attempt to disconnect a non-connected connection."; | |
54 return; | |
55 } | |
56 | |
57 // Set status as disconnected now, rather than after the socket closes, so | |
58 // this connection is not reused. | |
59 SetStatus(DISCONNECTED); | |
60 if (socket_.get()) { | |
61 socket_->Disconnect(base::Bind(&base::DoNothing)); | |
62 socket_ = NULL; | |
63 } | |
64 if (adapter_.get()) { | |
65 adapter_->RemoveObserver(this); | |
66 adapter_ = NULL; | |
67 } | |
68 } | |
69 | |
70 void BluetoothConnection::SendMessageImpl(scoped_ptr<WireMessage> message) { | |
71 DCHECK_EQ(status(), CONNECTED); | |
72 | |
73 // Serialize the message. | |
74 std::string serialized_message = message->Serialize(); | |
75 int message_length = base::checked_cast<int>(serialized_message.size()); | |
76 scoped_refptr<net::IOBuffer> buffer = new net::IOBuffer(message_length); | |
77 memcpy(buffer->data(), serialized_message.c_str(), message_length); | |
78 | |
79 // Send it. | |
80 pending_message_ = message.Pass(); | |
81 base::WeakPtr<BluetoothConnection> weak_this = weak_ptr_factory_.GetWeakPtr(); | |
82 socket_->Send(buffer, | |
83 message_length, | |
84 base::Bind(&BluetoothConnection::OnSend, weak_this), | |
85 base::Bind(&BluetoothConnection::OnSendError, weak_this)); | |
86 } | |
87 | |
88 void BluetoothConnection::DeviceRemoved(device::BluetoothAdapter* adapter, | |
89 device::BluetoothDevice* device) { | |
90 DCHECK_EQ(adapter, adapter_.get()); | |
91 if (device->GetAddress() != remote_device().bluetooth_address) | |
92 return; | |
93 | |
94 DCHECK_NE(status(), DISCONNECTED); | |
95 VLOG(1) << "[BC] Device disconnected..."; | |
96 Disconnect(); | |
97 } | |
98 | |
99 void BluetoothConnection::ConnectToService( | |
100 device::BluetoothDevice* device, | |
101 const device::BluetoothUUID& uuid, | |
102 const device::BluetoothDevice::ConnectToServiceCallback& callback, | |
103 const device::BluetoothDevice::ConnectToServiceErrorCallback& | |
104 error_callback) { | |
105 bluetooth_util::ConnectToServiceInsecurely( | |
106 device, uuid, callback, error_callback); | |
107 } | |
108 | |
109 void BluetoothConnection::StartReceive() { | |
110 base::WeakPtr<BluetoothConnection> weak_this = weak_ptr_factory_.GetWeakPtr(); | |
111 socket_->Receive(kReceiveBufferSizeBytes, | |
112 base::Bind(&BluetoothConnection::OnReceive, weak_this), | |
113 base::Bind(&BluetoothConnection::OnReceiveError, weak_this)); | |
114 } | |
115 | |
116 void BluetoothConnection::OnAdapterInitialized( | |
117 scoped_refptr<device::BluetoothAdapter> adapter) { | |
118 const std::string address = remote_device().bluetooth_address; | |
119 device::BluetoothDevice* bluetooth_device = adapter->GetDevice(address); | |
120 if (!bluetooth_device) { | |
Tim Song
2014/09/22 23:30:38
Reading this code gave me the thought that it woul
Ilya Sherman
2014/09/23 05:24:22
Yeah, I think that makes sense. I'll go ahead and
| |
121 VLOG(1) << "[BC] Device with address " << address | |
122 << " is not known to the system Bluetooth daemon."; | |
123 Disconnect(); | |
124 return; | |
125 } | |
126 | |
127 adapter_ = adapter; | |
128 adapter_->AddObserver(this); | |
129 | |
130 base::WeakPtr<BluetoothConnection> weak_this = weak_ptr_factory_.GetWeakPtr(); | |
131 ConnectToService( | |
132 bluetooth_device, | |
133 uuid_, | |
134 base::Bind(&BluetoothConnection::OnConnected, weak_this), | |
135 base::Bind(&BluetoothConnection::OnConnectionError, weak_this)); | |
136 } | |
137 | |
138 void BluetoothConnection::OnConnected( | |
139 scoped_refptr<device::BluetoothSocket> socket) { | |
140 if (status() != IN_PROGRESS) { | |
141 // This case is reachable if the client of |this| connection called | |
142 // |Disconnect()| while the backing Bluetooth connection was pending. | |
143 DCHECK_EQ(status(), DISCONNECTED); | |
144 VLOG(1) << "[BC] Ignoring successful backend Bluetooth connection to an " | |
145 << "already disconnected logical connection."; | |
146 return; | |
147 } | |
148 | |
149 VLOG(1) << "[BC] Connection established with " | |
150 << remote_device().bluetooth_address; | |
151 socket_ = socket; | |
152 SetStatus(CONNECTED); | |
153 StartReceive(); | |
154 } | |
155 | |
156 void BluetoothConnection::OnConnectionError(const std::string& error_message) { | |
157 VLOG(1) << "[BC] Connection failed: " << error_message; | |
158 Disconnect(); | |
159 } | |
160 | |
161 void BluetoothConnection::OnSend(int bytes_sent) { | |
162 VLOG(1) << "[BC] Successfully sent " << bytes_sent << " bytes."; | |
163 OnDidSendMessage(*pending_message_, true); | |
164 pending_message_.reset(); | |
165 } | |
166 | |
167 void BluetoothConnection::OnSendError(const std::string& error_message) { | |
168 VLOG(1) << "[BC] Error when sending bytes: " << error_message; | |
169 OnDidSendMessage(*pending_message_, false); | |
170 pending_message_.reset(); | |
171 | |
172 Disconnect(); | |
173 } | |
174 | |
175 void BluetoothConnection::OnReceive(int bytes_received, | |
176 scoped_refptr<net::IOBuffer> buffer) { | |
177 VLOG(1) << "[BC] Received " << bytes_received << " bytes."; | |
178 OnBytesReceived(std::string(buffer->data(), bytes_received)); | |
179 | |
180 // Post a task to delay the read until the socket is available, as | |
181 // calling StartReceive at this point would error with ERR_IO_PENDING. | |
182 base::MessageLoop::current()->PostTask( | |
183 FROM_HERE, | |
184 base::Bind(&BluetoothConnection::StartReceive, | |
185 weak_ptr_factory_.GetWeakPtr())); | |
186 } | |
187 | |
188 void BluetoothConnection::OnReceiveError( | |
189 device::BluetoothSocket::ErrorReason error_reason, | |
190 const std::string& error_message) { | |
191 VLOG(1) << "[BC] Error receiving bytes: " << error_message; | |
192 | |
193 // Post a task to delay the read until the socket is available, as | |
194 // calling StartReceive at this point would error with ERR_IO_PENDING. | |
195 base::MessageLoop::current()->PostTask( | |
196 FROM_HERE, | |
197 base::Bind(&BluetoothConnection::StartReceive, | |
198 weak_ptr_factory_.GetWeakPtr())); | |
199 } | |
200 | |
201 } // namespace proximity_auth | |
OLD | NEW |