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 "tools/android/common/adb_connection.h"
11 #include <sys/socket.h>
12 #include <sys/types.h>
15 #include "base/logging.h"
16 #include "base/posix/eintr_wrapper.h"
17 #include "tools/android/common/net.h"
22 void CloseSocket(int fd
) {
24 int old_errno
= errno
;
32 int ConnectAdbHostSocket(const char* forward_to
) {
33 // ADB port forward request format: HHHHtcp:port:address.
34 // HHHH is the hexidecimal length of the "tcp:port:address" part.
35 const size_t kBufferMaxLength
= 30;
36 const size_t kLengthOfLength
= 4;
38 const char kAddressPrefix
[] = { 't', 'c', 'p', ':' };
39 size_t address_length
= arraysize(kAddressPrefix
) + strlen(forward_to
);
40 if (address_length
> kBufferMaxLength
- kLengthOfLength
) {
41 LOG(ERROR
) << "Forward to address is too long: " << forward_to
;
45 char request
[kBufferMaxLength
];
46 memcpy(request
+ kLengthOfLength
, kAddressPrefix
, arraysize(kAddressPrefix
));
47 memcpy(request
+ kLengthOfLength
+ arraysize(kAddressPrefix
),
48 forward_to
, strlen(forward_to
));
50 char length_buffer
[kLengthOfLength
+ 1];
51 snprintf(length_buffer
, arraysize(length_buffer
), "%04X",
52 static_cast<int>(address_length
));
53 memcpy(request
, length_buffer
, kLengthOfLength
);
55 int host_socket
= socket(AF_INET
, SOCK_STREAM
, 0);
56 if (host_socket
< 0) {
57 LOG(ERROR
) << "Failed to create adb socket: " << strerror(errno
);
61 DisableNagle(host_socket
);
63 const int kAdbPort
= 5037;
65 memset(&addr
, 0, sizeof(addr
));
66 addr
.sin_family
= AF_INET
;
67 addr
.sin_addr
.s_addr
= htonl(INADDR_LOOPBACK
);
68 addr
.sin_port
= htons(kAdbPort
);
69 if (HANDLE_EINTR(connect(host_socket
, reinterpret_cast<sockaddr
*>(&addr
),
71 LOG(ERROR
) << "Failed to connect adb socket: " << strerror(errno
);
72 CloseSocket(host_socket
);
76 size_t bytes_remaining
= address_length
+ kLengthOfLength
;
77 size_t bytes_sent
= 0;
78 while (bytes_remaining
> 0) {
79 int ret
= HANDLE_EINTR(send(host_socket
, request
+ bytes_sent
,
82 LOG(ERROR
) << "Failed to send request: " << strerror(errno
);
83 CloseSocket(host_socket
);
88 bytes_remaining
-= ret
;
91 const int kAdbStatusLength
= 4;
92 char response
[kBufferMaxLength
];
93 int response_length
= HANDLE_EINTR(recv(host_socket
, response
,
94 kBufferMaxLength
, 0));
95 if (response_length
< kAdbStatusLength
||
96 strncmp("OKAY", response
, kAdbStatusLength
) != 0) {
97 LOG(ERROR
) << "Bad response from ADB: length: " << response_length
98 << " data: " << DumpBinary(response
, response_length
);
99 CloseSocket(host_socket
);