Chromium Code Reviews| 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 |