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.
5 #include "device/bluetooth/bluetooth_socket_win.h"
11 #include "base/logging.h"
12 #include "base/memory/ref_counted.h"
13 #include "base/sequenced_task_runner.h"
14 #include "base/strings/sys_string_conversions.h"
15 #include "base/strings/utf_string_conversions.h"
16 #include "base/threading/thread_restrictions.h"
17 #include "device/bluetooth/bluetooth_init_win.h"
18 #include "device/bluetooth/bluetooth_service_record_win.h"
19 #include "device/bluetooth/bluetooth_socket_thread.h"
20 #include "net/base/io_buffer.h"
21 #include "net/base/ip_endpoint.h"
22 #include "net/base/net_errors.h"
23 #include "net/base/winsock_init.h"
27 const char kL2CAPNotSupported
[] = "Bluetooth L2CAP protocal is not supported";
28 const char kSocketAlreadyConnected
[] = "Socket is already connected.";
29 const char kInvalidRfcommPort
[] = "Invalid RFCCOMM port.";
30 const char kFailedToCreateSocket
[] = "Failed to create socket.";
31 const char kFailedToBindSocket
[] = "Failed to bind socket.";
32 const char kFailedToListenOnSocket
[] = "Failed to listen on socket.";
33 const char kFailedToGetSockNameForSocket
[] = "Failed to getsockname.";
34 const char kBadUuid
[] = "Bad uuid.";
35 const char kWsaSetServiceError
[] = "WSASetService error.";
41 struct BluetoothSocketWin::ServiceRegData
{
43 ZeroMemory(&address
, sizeof(address
));
44 ZeroMemory(&address_info
, sizeof(address_info
));
45 ZeroMemory(&uuid
, sizeof(uuid
));
46 ZeroMemory(&service
, sizeof(service
));
50 CSADDR_INFO address_info
;
57 scoped_refptr
<BluetoothSocketWin
>
58 BluetoothSocketWin::CreateBluetoothSocket(
59 scoped_refptr
<base::SequencedTaskRunner
> ui_task_runner
,
60 scoped_refptr
<device::BluetoothSocketThread
> socket_thread
,
62 const net::NetLog::Source
& source
) {
63 DCHECK(ui_task_runner
->RunsTasksOnCurrentThread());
65 return make_scoped_refptr(
66 new BluetoothSocketWin(ui_task_runner
, socket_thread
, net_log
, source
));
69 BluetoothSocketWin::BluetoothSocketWin(
70 scoped_refptr
<base::SequencedTaskRunner
> ui_task_runner
,
71 scoped_refptr
<BluetoothSocketThread
> socket_thread
,
73 const net::NetLog::Source
& source
)
74 : BluetoothSocketNet(ui_task_runner
, socket_thread
, net_log
, source
),
75 supports_rfcomm_(false),
77 bth_addr_(BTH_ADDR_NULL
) {
80 BluetoothSocketWin::~BluetoothSocketWin() {
83 void BluetoothSocketWin::StartService(
84 const BluetoothUUID
& uuid
,
85 const std::string
& name
,
87 const base::Closure
& success_callback
,
88 const ErrorCompletionCallback
& error_callback
,
89 const OnNewConnectionCallback
& new_connection_callback
) {
90 DCHECK(ui_task_runner()->RunsTasksOnCurrentThread());
92 socket_thread()->task_runner()->PostTask(
94 base::Bind(&BluetoothSocketWin::DoStartService
,
101 new_connection_callback
));
104 void BluetoothSocketWin::Connect(
105 const BluetoothServiceRecord
& service_record
,
106 const base::Closure
& success_callback
,
107 const ErrorCompletionCallback
& error_callback
) {
108 DCHECK(ui_task_runner()->RunsTasksOnCurrentThread());
110 const BluetoothServiceRecordWin
* service_record_win
=
111 static_cast<const BluetoothServiceRecordWin
*>(&service_record
);
112 device_address_
= service_record_win
->address();
113 if (service_record
.SupportsRfcomm()) {
114 supports_rfcomm_
= true;
115 rfcomm_channel_
= service_record_win
->rfcomm_channel();
116 bth_addr_
= service_record_win
->bth_addr();
119 socket_thread()->task_runner()->PostTask(
122 &BluetoothSocketWin::DoConnect
,
124 base::Bind(&BluetoothSocketWin::PostSuccess
, this, success_callback
),
126 &BluetoothSocketWin::PostErrorCompletion
, this, error_callback
)));
130 void BluetoothSocketWin::ResetData() {
131 if (service_reg_data_
) {
132 if (WSASetService(&service_reg_data_
->service
,RNRSERVICE_DELETE
, 0) ==
134 LOG(WARNING
) << "Failed to unregister service.";
136 service_reg_data_
.reset();
140 void BluetoothSocketWin::Accept(
141 const AcceptCompletionCallback
& success_callback
,
142 const ErrorCompletionCallback
& error_callback
) {
146 void BluetoothSocketWin::DoConnect(
147 const base::Closure
& success_callback
,
148 const ErrorCompletionCallback
& error_callback
) {
149 DCHECK(socket_thread()->task_runner()->RunsTasksOnCurrentThread());
150 base::ThreadRestrictions::AssertIOAllowed();
153 error_callback
.Run(kSocketAlreadyConnected
);
157 if (!supports_rfcomm_
) {
158 // TODO(youngki) add support for L2CAP sockets as well.
159 error_callback
.Run(kL2CAPNotSupported
);
164 net::EnsureWinsockInit();
165 SOCKET socket_fd
= socket(AF_BTH
, SOCK_STREAM
, BTHPROTO_RFCOMM
);
167 ZeroMemory(&sa
, sizeof(sa
));
168 sa
.addressFamily
= AF_BTH
;
169 sa
.port
= rfcomm_channel_
;
170 sa
.btAddr
= bth_addr_
;
172 // TODO(rpaquay): Condider making this call non-blocking.
173 int status
= connect(socket_fd
, reinterpret_cast<SOCKADDR
*>(&sa
), sizeof(sa
));
174 DWORD error_code
= WSAGetLastError();
175 if (!(status
== 0 || error_code
== WSAEINPROGRESS
)) {
176 LOG(ERROR
) << "Failed to connect bluetooth socket "
177 << "(" << device_address_
<< "): "
178 << logging::SystemErrorCodeToString(error_code
);
179 error_callback
.Run("Error connecting to socket: " +
180 logging::SystemErrorCodeToString(error_code
));
181 closesocket(socket_fd
);
185 // Note: We don't have a meaningful |IPEndPoint|, but that is ok since the
186 // TCPSocket implementation does not actually require one.
188 tcp_socket()->AdoptConnectedSocket(socket_fd
, net::IPEndPoint());
189 if (net_result
!= net::OK
) {
190 error_callback
.Run("Error connecting to socket: " +
191 std::string(net::ErrorToString(net_result
)));
192 closesocket(socket_fd
);
196 success_callback
.Run();
199 void BluetoothSocketWin::DoStartService(
200 const BluetoothUUID
& uuid
,
201 const std::string
& name
,
203 const base::Closure
& success_callback
,
204 const ErrorCompletionCallback
& error_callback
,
205 const OnNewConnectionCallback
& new_connection_callback
) {
206 DCHECK(socket_thread()->task_runner()->RunsTasksOnCurrentThread());
207 DCHECK(!tcp_socket() &&
208 !service_reg_data_
&&
209 on_new_connection_callback_
.is_null());
211 // The valid range is 0-30. 0 means BT_PORT_ANY and 1-30 are the
212 // valid RFCOMM port numbers of SOCKADDR_BTH.
213 if (rfcomm_channel
< 0 || rfcomm_channel
> 30) {
214 LOG(WARNING
) << "Failed to start service: "
215 << "Invalid RFCCOMM port " << rfcomm_channel
216 << ", uuid=" << uuid
.value();
217 PostErrorCompletion(error_callback
, kInvalidRfcommPort
);
221 SOCKET socket_fd
= socket(AF_BTH
, SOCK_STREAM
, BTHPROTO_RFCOMM
);
222 if (socket_fd
== INVALID_SOCKET
) {
223 LOG(WARNING
) << "Failed to start service: create socket, "
224 << "winsock err=" << WSAGetLastError();
225 PostErrorCompletion(error_callback
, kFailedToCreateSocket
);
229 // Note that |socket_fd| belongs to a non-TCP address family (i.e. AF_BTH),
230 // TCPSocket methods that involve address could not be called. So bind()
231 // is called on |socket_fd| directly.
232 scoped_ptr
<net::TCPSocket
> scoped_socket(
233 new net::TCPSocket(NULL
, net::NetLog::Source()));
234 scoped_socket
->AdoptListenSocket(socket_fd
);
237 struct sockaddr
* sock_addr
= reinterpret_cast<struct sockaddr
*>(&sa
);
238 int sock_addr_len
= sizeof(sa
);
239 ZeroMemory(&sa
, sock_addr_len
);
240 sa
.addressFamily
= AF_BTH
;
241 sa
.port
= rfcomm_channel
? rfcomm_channel
: BT_PORT_ANY
;
242 if (bind(socket_fd
, sock_addr
, sock_addr_len
) < 0) {
243 LOG(WARNING
) << "Failed to start service: create socket, "
244 << "winsock err=" << WSAGetLastError();
245 PostErrorCompletion(error_callback
, kFailedToBindSocket
);
249 const int kListenBacklog
= 5;
250 if (scoped_socket
->Listen(kListenBacklog
) < 0) {
251 LOG(WARNING
) << "Failed to start service: Listen"
252 << "winsock err=" << WSAGetLastError();
253 PostErrorCompletion(error_callback
, kFailedToListenOnSocket
);
257 scoped_ptr
<ServiceRegData
> reg_data(new ServiceRegData
);
258 reg_data
->name
= base::UTF8ToUTF16(name
);
260 if (getsockname(socket_fd
, sock_addr
, &sock_addr_len
)) {
261 LOG(WARNING
) << "Failed to start service: getsockname, "
262 << "winsock err=" << WSAGetLastError();
263 PostErrorCompletion(error_callback
, kFailedToGetSockNameForSocket
);
266 reg_data
->address
= sa
;
268 reg_data
->address_info
.LocalAddr
.iSockaddrLength
= sizeof(sa
);
269 reg_data
->address_info
.LocalAddr
.lpSockaddr
=
270 reinterpret_cast<struct sockaddr
*>(®_data
->address
);
271 reg_data
->address_info
.iSocketType
= SOCK_STREAM
;
272 reg_data
->address_info
.iProtocol
= BTHPROTO_RFCOMM
;
274 base::string16 cannonical_uuid
= L
"{" + base::ASCIIToUTF16(
275 uuid
.canonical_value()) + L
"}";
276 if (!SUCCEEDED(CLSIDFromString(cannonical_uuid
.c_str(), ®_data
->uuid
))) {
277 LOG(WARNING
) << "Failed to start service: "
278 << ", bad uuid=" << cannonical_uuid
;
279 PostErrorCompletion(error_callback
, kBadUuid
);
283 reg_data
->service
.dwSize
= sizeof(WSAQUERYSET
);
284 reg_data
->service
.lpszServiceInstanceName
=
285 const_cast<LPWSTR
>(reg_data
->name
.c_str());
286 reg_data
->service
.lpServiceClassId
= ®_data
->uuid
;
287 reg_data
->service
.dwNameSpace
= NS_BTH
;
288 reg_data
->service
.dwNumberOfCsAddrs
= 1;
289 reg_data
->service
.lpcsaBuffer
= ®_data
->address_info
;
291 if (WSASetService(®_data
->service
,
292 RNRSERVICE_REGISTER
, 0) == SOCKET_ERROR
) {
293 LOG(WARNING
) << "Failed to register profile: WSASetService"
294 << "winsock err=" << WSAGetLastError();
295 PostErrorCompletion(error_callback
, kWsaSetServiceError
);
299 SetTCPSocket(scoped_socket
.Pass());
300 service_reg_data_
= reg_data
.Pass();
301 on_new_connection_callback_
= new_connection_callback
;
304 PostSuccess(success_callback
);
307 void BluetoothSocketWin::DoAccept() {
308 DCHECK(socket_thread()->task_runner()->RunsTasksOnCurrentThread());
309 int result
= tcp_socket()->Accept(
312 base::Bind(&BluetoothSocketWin::OnAcceptOnSocketThread
, this));
313 if (result
!= net::OK
&& result
!= net::ERR_IO_PENDING
)
314 LOG(WARNING
) << "Failed to accept, net err=" << result
;
317 void BluetoothSocketWin::OnAcceptOnSocketThread(int accept_result
) {
318 DCHECK(socket_thread()->task_runner()->RunsTasksOnCurrentThread());
319 if (accept_result
!= net::OK
) {
320 LOG(WARNING
) << "OnAccept error, net err=" << accept_result
;
324 ui_task_runner()->PostTask(
326 base::Bind(&BluetoothSocketWin::OnAcceptOnUI
,
328 base::Passed(&accept_socket_
),
333 void BluetoothSocketWin::OnAcceptOnUI(
334 scoped_ptr
<net::TCPSocket
> accept_socket
,
335 const net::IPEndPoint
& peer_address
) {
336 DCHECK(ui_task_runner()->RunsTasksOnCurrentThread());
338 scoped_refptr
<BluetoothSocketWin
> peer
= CreateBluetoothSocket(
343 peer
->SetTCPSocket(accept_socket
.Pass());
345 on_new_connection_callback_
.Run(peer
, peer_address
);
348 } // namespace device