1 /*
2 * Copyright (C) 2015 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "adb_listeners.h"
18
19 #include <stdio.h>
20 #include <stdlib.h>
21
22 #include <base/stringprintf.h>
23
24 #include "sysdeps.h"
25 #include "transport.h"
26
27 int gListenAll = 0; /* Not static because it is used in commandline.c. */
28
29 alistener listener_list = {
30 .next = &listener_list,
31 .prev = &listener_list,
32 };
33
ss_listener_event_func(int _fd,unsigned ev,void * _l)34 void ss_listener_event_func(int _fd, unsigned ev, void *_l)
35 {
36 asocket *s;
37
38 if(ev & FDE_READ) {
39 struct sockaddr addr;
40 socklen_t alen;
41 int fd;
42
43 alen = sizeof(addr);
44 fd = adb_socket_accept(_fd, &addr, &alen);
45 if(fd < 0) return;
46
47 adb_socket_setbufsize(fd, CHUNK_SIZE);
48
49 s = create_local_socket(fd);
50 if(s) {
51 connect_to_smartsocket(s);
52 return;
53 }
54
55 adb_close(fd);
56 }
57 }
58
listener_event_func(int _fd,unsigned ev,void * _l)59 void listener_event_func(int _fd, unsigned ev, void* _l)
60 {
61 alistener* listener = reinterpret_cast<alistener*>(_l);
62 asocket *s;
63
64 if (ev & FDE_READ) {
65 struct sockaddr addr;
66 socklen_t alen;
67 int fd;
68
69 alen = sizeof(addr);
70 fd = adb_socket_accept(_fd, &addr, &alen);
71 if (fd < 0) {
72 return;
73 }
74
75 s = create_local_socket(fd);
76 if (s) {
77 s->transport = listener->transport;
78 connect_to_remote(s, listener->connect_to);
79 return;
80 }
81
82 adb_close(fd);
83 }
84 }
85
free_listener(alistener * l)86 static void free_listener(alistener* l)
87 {
88 if (l->next) {
89 l->next->prev = l->prev;
90 l->prev->next = l->next;
91 l->next = l->prev = l;
92 }
93
94 // closes the corresponding fd
95 fdevent_remove(&l->fde);
96
97 if (l->local_name)
98 free((char*)l->local_name);
99
100 if (l->connect_to)
101 free((char*)l->connect_to);
102
103 if (l->transport) {
104 remove_transport_disconnect(l->transport, &l->disconnect);
105 }
106 free(l);
107 }
108
listener_disconnect(void * listener,atransport * t)109 void listener_disconnect(void* listener, atransport* t)
110 {
111 free_listener(reinterpret_cast<alistener*>(listener));
112 }
113
local_name_to_fd(const char * name)114 int local_name_to_fd(const char *name)
115 {
116 int port;
117
118 if(!strncmp("tcp:", name, 4)){
119 int ret;
120 port = atoi(name + 4);
121
122 if (gListenAll > 0) {
123 ret = socket_inaddr_any_server(port, SOCK_STREAM);
124 } else {
125 ret = socket_loopback_server(port, SOCK_STREAM);
126 }
127
128 return ret;
129 }
130 #ifndef HAVE_WIN32_IPC /* no Unix-domain sockets on Win32 */
131 // It's non-sensical to support the "reserved" space on the adb host side
132 if(!strncmp(name, "local:", 6)) {
133 return socket_local_server(name + 6,
134 ANDROID_SOCKET_NAMESPACE_ABSTRACT, SOCK_STREAM);
135 } else if(!strncmp(name, "localabstract:", 14)) {
136 return socket_local_server(name + 14,
137 ANDROID_SOCKET_NAMESPACE_ABSTRACT, SOCK_STREAM);
138 } else if(!strncmp(name, "localfilesystem:", 16)) {
139 return socket_local_server(name + 16,
140 ANDROID_SOCKET_NAMESPACE_FILESYSTEM, SOCK_STREAM);
141 }
142
143 #endif
144 printf("unknown local portname '%s'\n", name);
145 return -1;
146 }
147
148 // Write the list of current listeners (network redirections) into a string.
format_listeners()149 std::string format_listeners() {
150 std::string result;
151 for (alistener* l = listener_list.next; l != &listener_list; l = l->next) {
152 // Ignore special listeners like those for *smartsocket*
153 if (l->connect_to[0] == '*') {
154 continue;
155 }
156 // <device-serial> " " <local-name> " " <remote-name> "\n"
157 android::base::StringAppendF(&result, "%s %s %s\n",
158 l->transport->serial, l->local_name, l->connect_to);
159 }
160 return result;
161 }
162
remove_listener(const char * local_name,atransport * transport)163 install_status_t remove_listener(const char *local_name, atransport* transport)
164 {
165 alistener *l;
166
167 for (l = listener_list.next; l != &listener_list; l = l->next) {
168 if (!strcmp(local_name, l->local_name)) {
169 listener_disconnect(l, l->transport);
170 return INSTALL_STATUS_OK;
171 }
172 }
173 return INSTALL_STATUS_LISTENER_NOT_FOUND;
174 }
175
remove_all_listeners(void)176 void remove_all_listeners(void)
177 {
178 alistener *l, *l_next;
179 for (l = listener_list.next; l != &listener_list; l = l_next) {
180 l_next = l->next;
181 // Never remove smart sockets.
182 if (l->connect_to[0] == '*')
183 continue;
184 listener_disconnect(l, l->transport);
185 }
186 }
187
install_listener(const std::string & local_name,const char * connect_to,atransport * transport,int no_rebind)188 install_status_t install_listener(const std::string& local_name,
189 const char *connect_to,
190 atransport* transport,
191 int no_rebind)
192 {
193 for (alistener* l = listener_list.next; l != &listener_list; l = l->next) {
194 if (local_name == l->local_name) {
195 char* cto;
196
197 /* can't repurpose a smartsocket */
198 if(l->connect_to[0] == '*') {
199 return INSTALL_STATUS_INTERNAL_ERROR;
200 }
201
202 /* can't repurpose a listener if 'no_rebind' is true */
203 if (no_rebind) {
204 return INSTALL_STATUS_CANNOT_REBIND;
205 }
206
207 cto = strdup(connect_to);
208 if(cto == 0) {
209 return INSTALL_STATUS_INTERNAL_ERROR;
210 }
211
212 free((void*) l->connect_to);
213 l->connect_to = cto;
214 if (l->transport != transport) {
215 remove_transport_disconnect(l->transport, &l->disconnect);
216 l->transport = transport;
217 add_transport_disconnect(l->transport, &l->disconnect);
218 }
219 return INSTALL_STATUS_OK;
220 }
221 }
222
223 alistener* listener = reinterpret_cast<alistener*>(
224 calloc(1, sizeof(alistener)));
225 if (listener == nullptr) {
226 goto nomem;
227 }
228
229 listener->local_name = strdup(local_name.c_str());
230 if (listener->local_name == nullptr) {
231 goto nomem;
232 }
233
234 listener->connect_to = strdup(connect_to);
235 if (listener->connect_to == nullptr) {
236 goto nomem;
237 }
238
239 listener->fd = local_name_to_fd(listener->local_name);
240 if (listener->fd < 0) {
241 printf("cannot bind '%s': %s\n", listener->local_name, strerror(errno));
242 free(listener->local_name);
243 free(listener->connect_to);
244 free(listener);
245 return INSTALL_STATUS_CANNOT_BIND;
246 }
247
248 close_on_exec(listener->fd);
249 if (!strcmp(listener->connect_to, "*smartsocket*")) {
250 fdevent_install(&listener->fde, listener->fd, ss_listener_event_func,
251 listener);
252 } else {
253 fdevent_install(&listener->fde, listener->fd, listener_event_func,
254 listener);
255 }
256 fdevent_set(&listener->fde, FDE_READ);
257
258 listener->next = &listener_list;
259 listener->prev = listener_list.prev;
260 listener->next->prev = listener;
261 listener->prev->next = listener;
262 listener->transport = transport;
263
264 if (transport) {
265 listener->disconnect.opaque = listener;
266 listener->disconnect.func = listener_disconnect;
267 add_transport_disconnect(transport, &listener->disconnect);
268 }
269 return INSTALL_STATUS_OK;
270
271 nomem:
272 fatal("cannot allocate listener");
273 return INSTALL_STATUS_INTERNAL_ERROR;
274 }
275