1dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project/*
2dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
3dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project *
4dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * you may not use this file except in compliance with the License.
6dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * You may obtain a copy of the License at
7dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project *
8dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
9dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project *
10dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * See the License for the specific language governing permissions and
14dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project * limitations under the License.
15dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project */
16dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
17dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include <stdlib.h>
18dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include <stdio.h>
19dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include <unistd.h>
20dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include <string.h>
21dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include <errno.h>
22dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
23dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include "sysdeps.h"
24dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
25408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#define  TRACE_TAG  TRACE_SERVICES
26dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include "adb.h"
27dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#include "file_sync_service.h"
28dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
29dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
30dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#  ifndef HAVE_WINSOCK
31dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#    include <netinet/in.h>
32dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#    include <netdb.h>
33408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#    include <sys/ioctl.h>
34dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#  endif
35cc1de48dcdf06c76ee14abbe2a237aa51b5b3badMike Lockwood#else
36e3aeeb4de34dbb93e832e6554f494122ba633f3bKen Sumrall#  include <cutils/android_reboot.h>
37dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
38dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
39dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projecttypedef struct stinfo stinfo;
40dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
41dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstruct stinfo {
42dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    void (*func)(int fd, void *cookie);
43dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int fd;
44dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    void *cookie;
45dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project};
46dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
47dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
48dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectvoid *service_bootstrap_func(void *x)
49dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
50dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    stinfo *sti = x;
51dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    sti->func(sti->fd, sti->cookie);
52dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    free(sti);
53dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    return 0;
54dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
55dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
56dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
57dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source ProjectADB_MUTEX_DEFINE( dns_lock );
58dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
59dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstatic void dns_service(int fd, void *cookie)
60dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
61dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char *hostname = cookie;
62dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    struct hostent *hp;
63dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    unsigned zero = 0;
64dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
65dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_mutex_lock(&dns_lock);
66dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    hp = gethostbyname(hostname);
67b6b40079e6d5b32026cc2abfcb71adfd49e8d954Mike Lockwood    free(cookie);
68dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(hp == 0) {
69dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        writex(fd, &zero, 4);
70dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else {
71dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        writex(fd, hp->h_addr, 4);
72dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
73dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_mutex_unlock(&dns_lock);
74dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_close(fd);
75dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
76dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#else
77dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectextern int recovery_mode;
78dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
79dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstatic void recover_service(int s, void *cookie)
80dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
81dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    unsigned char buf[4096];
82dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    unsigned count = (unsigned) cookie;
83dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int fd;
84dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
85dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    fd = adb_creat("/tmp/update", 0644);
86dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(fd < 0) {
87dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        adb_close(s);
88dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return;
89dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
90dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
91dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    while(count > 0) {
92dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        unsigned xfer = (count > 4096) ? 4096 : count;
93dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(readx(s, buf, xfer)) break;
94dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(writex(fd, buf, xfer)) break;
95dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        count -= xfer;
96dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
97dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
98dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(count == 0) {
99dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        writex(s, "OKAY", 4);
100dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else {
101dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        writex(s, "FAIL", 4);
102dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
103dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_close(fd);
104dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_close(s);
105dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
106dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    fd = adb_creat("/tmp/update.begin", 0644);
107dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_close(fd);
108dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
109dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
110e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Projectvoid restart_root_service(int fd, void *cookie)
111e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project{
112e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    char buf[100];
113e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    char value[PROPERTY_VALUE_MAX];
114e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project
115e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    if (getuid() == 0) {
116e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        snprintf(buf, sizeof(buf), "adbd is already running as root\n");
117e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        writex(fd, buf, strlen(buf));
118e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        adb_close(fd);
119e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    } else {
120e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        property_get("ro.debuggable", value, "");
121e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        if (strcmp(value, "1") != 0) {
122e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project            snprintf(buf, sizeof(buf), "adbd cannot run as root in production builds\n");
123e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project            writex(fd, buf, strlen(buf));
124ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood            adb_close(fd);
125e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project            return;
126e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        }
127e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project
128e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        snprintf(buf, sizeof(buf), "restarting adbd as root\n");
129e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        writex(fd, buf, strlen(buf));
130e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        adb_close(fd);
131e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project
132535164e9d9649a83d4d63829f3389f2bea339fe1Mike Lockwood        // This will cause a property trigger in init.rc to restart us
133535164e9d9649a83d4d63829f3389f2bea339fe1Mike Lockwood        property_set("service.adb.root", "1");
134e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    }
135e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project}
136e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project
137ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwoodvoid restart_tcp_service(int fd, void *cookie)
138ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood{
139ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    char buf[100];
140ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    char value[PROPERTY_VALUE_MAX];
141ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    int port = (int)cookie;
142ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
143ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    if (port <= 0) {
144ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        snprintf(buf, sizeof(buf), "invalid port\n");
145ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        writex(fd, buf, strlen(buf));
146ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        adb_close(fd);
147ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        return;
148ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    }
149ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
150ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    snprintf(value, sizeof(value), "%d", port);
151ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    property_set("service.adb.tcp.port", value);
152ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    snprintf(buf, sizeof(buf), "restarting in TCP mode port: %d\n", port);
153ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    writex(fd, buf, strlen(buf));
154ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    adb_close(fd);
155ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
156ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    // quit, and init will restart us in TCP mode
157ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    sleep(1);
158ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    exit(1);
159ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood}
160ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
161ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwoodvoid restart_usb_service(int fd, void *cookie)
162ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood{
163ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    char buf[100];
164ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
165ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    property_set("service.adb.tcp.port", "0");
166ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    snprintf(buf, sizeof(buf), "restarting in USB mode\n");
167ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    writex(fd, buf, strlen(buf));
168ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    adb_close(fd);
169ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
170ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    // quit, and init will restart us in USB mode
171ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    sleep(1);
172ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    exit(1);
173ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood}
174ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood
175ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwoodvoid reboot_service(int fd, void *arg)
176ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood{
177ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    char buf[100];
178d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    int pid, ret;
179ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood
180ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    sync();
181d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood
182d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    /* Attempt to unmount the SD card first.
183d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood     * No need to bother checking for errors.
184d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood     */
185d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    pid = fork();
186d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    if (pid == 0) {
187d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood        /* ask vdc to unmount it */
188d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood        execl("/system/bin/vdc", "/system/bin/vdc", "volume", "unmount",
189d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood                getenv("EXTERNAL_STORAGE"), "force", NULL);
190d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    } else if (pid > 0) {
191d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood        /* wait until vdc succeeds or fails */
192d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood        waitpid(pid, &ret, 0);
193d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood    }
194d969faa161310d0a3792766320daa3200b84bd74Mike Lockwood
195e3aeeb4de34dbb93e832e6554f494122ba633f3bKen Sumrall    ret = android_reboot(ANDROID_RB_RESTART2, 0, (char *) arg);
196ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    if (ret < 0) {
197ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood        snprintf(buf, sizeof(buf), "reboot failed: %s\n", strerror(errno));
198ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood        writex(fd, buf, strlen(buf));
199ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    }
200b6b40079e6d5b32026cc2abfcb71adfd49e8d954Mike Lockwood    free(arg);
201ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    adb_close(fd);
202ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood}
203ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood
204dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
205dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
206dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if 0
207dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstatic void echo_service(int fd, void *cookie)
208dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
209dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char buf[4096];
210dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int r;
211dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char *p;
212dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int c;
213dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
214dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    for(;;) {
215dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        r = read(fd, buf, 4096);
216dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(r == 0) goto done;
217dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(r < 0) {
218dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            if(errno == EINTR) continue;
219dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            else goto done;
220dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        }
221dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
222dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        c = r;
223dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        p = buf;
224dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        while(c > 0) {
225dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            r = write(fd, p, c);
226dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            if(r > 0) {
227dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                c -= r;
228dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                p += r;
229dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                continue;
230dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            }
231dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            if((r < 0) && (errno == EINTR)) continue;
232dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            goto done;
233dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        }
234dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
235dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectdone:
236dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    close(fd);
237dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
238dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
239dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
240dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstatic int create_service_thread(void (*func)(int, void *), void *cookie)
241dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
242dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    stinfo *sti;
243dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_thread_t t;
244dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int s[2];
245dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
246dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(adb_socketpair(s)) {
247dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        printf("cannot create service socket pair\n");
248dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return -1;
249dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
250dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
251dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    sti = malloc(sizeof(stinfo));
252dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(sti == 0) fatal("cannot allocate stinfo");
253dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    sti->func = func;
254dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    sti->cookie = cookie;
255dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    sti->fd = s[1];
256dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
257dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(adb_thread_create( &t, service_bootstrap_func, sti)){
258dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        free(sti);
259dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        adb_close(s[0]);
260dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        adb_close(s[1]);
261dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        printf("cannot create service thread\n");
262dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return -1;
263dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
264dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
265dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    D("service thread started, %d:%d\n",s[0], s[1]);
266dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    return s[0];
267dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
268dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
269408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#if !ADB_HOST
270408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrallstatic int create_subprocess(const char *cmd, const char *arg0, const char *arg1, pid_t *pid)
271dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
27291acb14877e7138879057f794a61ee2fd424a41dJoe Onorato#ifdef HAVE_WIN32_PROC
273408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    D("create_subprocess(cmd=%s, arg0=%s, arg1=%s)\n", cmd, arg0, arg1);
274408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    fprintf(stderr, "error: create_subprocess not implemented on Win32 (%s %s %s)\n", cmd, arg0, arg1);
275408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    return -1;
27691acb14877e7138879057f794a61ee2fd424a41dJoe Onorato#else /* !HAVE_WIN32_PROC */
277dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char *devname;
278dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int ptm;
279dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
280dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    ptm = unix_open("/dev/ptmx", O_RDWR); // | O_NOCTTY);
281dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(ptm < 0){
282dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        printf("[ cannot open /dev/ptmx - %s ]\n",strerror(errno));
283dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return -1;
284dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
285dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    fcntl(ptm, F_SETFD, FD_CLOEXEC);
286dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
287dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(grantpt(ptm) || unlockpt(ptm) ||
288dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project       ((devname = (char*) ptsname(ptm)) == 0)){
289dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        printf("[ trouble with /dev/ptmx - %s ]\n", strerror(errno));
290408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        adb_close(ptm);
291dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return -1;
292dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
293dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
294408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    *pid = fork();
295408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if(*pid < 0) {
296dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        printf("- fork failed: %s -\n", strerror(errno));
297408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        adb_close(ptm);
298dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return -1;
299dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
300dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
301408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if(*pid == 0){
302dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        int pts;
303dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
304dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        setsid();
305dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
306dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        pts = unix_open(devname, O_RDWR);
307408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        if(pts < 0) {
308408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            fprintf(stderr, "child failed to open pseudo-term slave: %s\n", devname);
309408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            exit(-1);
310408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        }
311dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
312dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        dup2(pts, 0);
313dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        dup2(pts, 1);
314dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        dup2(pts, 2);
315dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
31695ef82866c7a922bf588027aa38c58a45eb84d9cBenoit Goby        adb_close(pts);
317dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        adb_close(ptm);
318dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
319408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        // set OOM adjustment to zero
320249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood        char text[64];
321408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        snprintf(text, sizeof text, "/proc/%d/oom_adj", getpid());
322249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood        int fd = adb_open(text, O_WRONLY);
323249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood        if (fd >= 0) {
324249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood            adb_write(fd, "0", 1);
325249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood            adb_close(fd);
326249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood        } else {
327249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood           D("adb: unable to open %s\n", text);
328249ad57a887680538d1dc0195e746b1d877ebd6aMike Lockwood        }
329408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        execl(cmd, cmd, arg0, arg1, NULL);
330408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        fprintf(stderr, "- exec '%s' failed: %s (%d) -\n",
331408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                cmd, strerror(errno), errno);
332408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        exit(-1);
333408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    } else {
334408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        // Don't set child's OOM adjustment to zero.
335408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        // Let the child do it itself, as sometimes the parent starts
336408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        // running before the child has a /proc/pid/oom_adj.
337408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        // """adb: unable to open /proc/644/oom_adj""" seen in some logs.
338dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return ptm;
339dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
34091acb14877e7138879057f794a61ee2fd424a41dJoe Onorato#endif /* !HAVE_WIN32_PROC */
341dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
342408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#endif  /* !ABD_HOST */
343dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
344dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
345dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#define SHELL_COMMAND "/bin/sh"
346dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#else
347dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#define SHELL_COMMAND "/system/bin/sh"
348dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
349dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
350408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#if !ADB_HOST
351408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrallstatic void subproc_waiter_service(int fd, void *cookie)
352408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall{
353408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    pid_t pid = (pid_t)cookie;
354408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
355408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    D("entered. fd=%d of pid=%d\n", fd, pid);
356408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    for (;;) {
357408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        int status;
358408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        pid_t p = waitpid(pid, &status, 0);
359408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        if (p == pid) {
360408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            D("fd=%d, post waitpid(pid=%d) status=%04x\n", fd, p, status);
361408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            if (WIFSIGNALED(status)) {
362408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                D("*** Killed by signal %d\n", WTERMSIG(status));
363408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                break;
364408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            } else if (!WIFEXITED(status)) {
365408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                D("*** Didn't exit!!. status %d\n", status);
366408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                break;
367408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            } else if (WEXITSTATUS(status) >= 0) {
368408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                D("*** Exit code %d\n", WEXITSTATUS(status));
369408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall                break;
370408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            }
371408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall         }
372408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        usleep(100000);  // poll every 0.1 sec
373408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    }
374408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    D("shell exited fd=%d of pid=%d err=%d\n", fd, pid, errno);
375408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if (SHELL_EXIT_NOTIFY_FD >=0) {
376408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall      int res;
377408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall      res = writex(SHELL_EXIT_NOTIFY_FD, &fd, sizeof(fd));
378408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall      D("notified shell exit via fd=%d for pid=%d res=%d errno=%d\n",
379408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        SHELL_EXIT_NOTIFY_FD, pid, res, errno);
380408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    }
381408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall}
382408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
383408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrallstatic int create_subproc_thread(const char *name)
384408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall{
385408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    stinfo *sti;
386408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    adb_thread_t t;
387408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    int ret_fd;
388408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    pid_t pid;
389408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if(name) {
390408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        ret_fd = create_subprocess(SHELL_COMMAND, "-c", name, &pid);
391408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    } else {
392408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        ret_fd = create_subprocess(SHELL_COMMAND, "-", 0, &pid);
393408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    }
394408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    D("create_subprocess() ret_fd=%d pid=%d\n", ret_fd, pid);
395408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
396408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    sti = malloc(sizeof(stinfo));
397408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if(sti == 0) fatal("cannot allocate stinfo");
398408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    sti->func = subproc_waiter_service;
399408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    sti->cookie = (void*)pid;
400408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    sti->fd = ret_fd;
401408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
402408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    if(adb_thread_create( &t, service_bootstrap_func, sti)){
403408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        free(sti);
404408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        adb_close(ret_fd);
405408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        printf("cannot create service thread\n");
406408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall        return -1;
407408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    }
408408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
409408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    D("service thread started, fd=%d pid=%d\n",ret_fd, pid);
410408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall    return ret_fd;
411408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall}
412408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall#endif
413408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall
414dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectint service_to_fd(const char *name)
415dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
416dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int ret = -1;
417dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
418dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(!strncmp(name, "tcp:", 4)) {
419dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        int port = atoi(name + 4);
420dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        name = strchr(name + 4, ':');
421dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(name == 0) {
422dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            ret = socket_loopback_client(port, SOCK_STREAM);
423dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            if (ret >= 0)
424dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                disable_tcp_nagle(ret);
425dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        } else {
426dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
427dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            adb_mutex_lock(&dns_lock);
428dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            ret = socket_network_client(name + 1, port, SOCK_STREAM);
429dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            adb_mutex_unlock(&dns_lock);
430dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#else
431dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            return -1;
432dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
433dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        }
434dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#ifndef HAVE_WINSOCK   /* winsock doesn't implement unix domain sockets */
435dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "local:", 6)) {
436dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = socket_local_client(name + 6,
437dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                ANDROID_SOCKET_NAMESPACE_RESERVED, SOCK_STREAM);
438dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "localreserved:", 14)) {
439dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = socket_local_client(name + 14,
440dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                ANDROID_SOCKET_NAMESPACE_RESERVED, SOCK_STREAM);
441dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "localabstract:", 14)) {
442dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = socket_local_client(name + 14,
443dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                ANDROID_SOCKET_NAMESPACE_ABSTRACT, SOCK_STREAM);
444dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "localfilesystem:", 16)) {
445dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = socket_local_client(name + 16,
446dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project                ANDROID_SOCKET_NAMESPACE_FILESYSTEM, SOCK_STREAM);
447dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
448dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
449dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp("dns:", name, 4)){
450dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        char *n = strdup(name + 4);
451dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(n == 0) return -1;
452dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(dns_service, n);
453dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#else /* !ADB_HOST */
454dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp("dev:", name, 4)) {
455dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = unix_open(name + 4, O_RDWR);
456dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "framebuffer:", 12)) {
457dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(framebuffer_service, 0);
458dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(recovery_mode && !strncmp(name, "recover:", 8)) {
459dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(recover_service, (void*) atoi(name + 8));
460dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if (!strncmp(name, "jdwp:", 5)) {
461dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_jdwp_connection_fd(atoi(name+5));
462dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if (!strncmp(name, "log:", 4)) {
463dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(log_service, get_log_file_path(name + 4));
464dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!HOST && !strncmp(name, "shell:", 6)) {
465dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if(name[6]) {
466408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            ret = create_subproc_thread(name + 6);
467dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        } else {
468408fa57864c01113deaa213e5c1848a9c594ae92JP Abgrall            ret = create_subproc_thread(0);
469dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        }
470dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "sync:", 5)) {
471dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(file_sync_service, NULL);
472dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "remount:", 8)) {
473dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(remount_service, NULL);
474ee15662140632672a6878787a1ff26f5ebdc1e68Mike Lockwood    } else if(!strncmp(name, "reboot:", 7)) {
475b6b40079e6d5b32026cc2abfcb71adfd49e8d954Mike Lockwood        void* arg = strdup(name + 7);
476b6b40079e6d5b32026cc2abfcb71adfd49e8d954Mike Lockwood        if(arg == 0) return -1;
477b6b40079e6d5b32026cc2abfcb71adfd49e8d954Mike Lockwood        ret = create_service_thread(reboot_service, arg);
478e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project    } else if(!strncmp(name, "root:", 5)) {
479e037fd7e193ecccbb5c0888e49f6d58c224bc11dThe Android Open Source Project        ret = create_service_thread(restart_root_service, NULL);
480d2f5415c603f7d9961f7a0b05579a0768e071410Christopher Tate    } else if(!strncmp(name, "backup:", 7)) {
481d2f5415c603f7d9961f7a0b05579a0768e071410Christopher Tate        char* arg = strdup(name+7);
482d2f5415c603f7d9961f7a0b05579a0768e071410Christopher Tate        if (arg == NULL) return -1;
483702967afb1bebc97c0b8a23c075d4932820ef7a3Christopher Tate        ret = backup_service(BACKUP, arg);
484702967afb1bebc97c0b8a23c075d4932820ef7a3Christopher Tate    } else if(!strncmp(name, "restore:", 8)) {
485702967afb1bebc97c0b8a23c075d4932820ef7a3Christopher Tate        ret = backup_service(RESTORE, NULL);
486ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    } else if(!strncmp(name, "tcpip:", 6)) {
487ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        int port;
488ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        if (sscanf(name + 6, "%d", &port) == 0) {
489ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood            port = 0;
490ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        }
491ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        ret = create_service_thread(restart_tcp_service, (void *)port);
492ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood    } else if(!strncmp(name, "usb:", 4)) {
493ff19670d485fd5ee296e23b00b035ac22975c647Mike Lockwood        ret = create_service_thread(restart_usb_service, NULL);
494dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
495dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if 0
496dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if(!strncmp(name, "echo:", 5)){
497dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        ret = create_service_thread(echo_service, 0);
498dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
499dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
500dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if (ret >= 0) {
501dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        close_on_exec(ret);
502dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
503dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    return ret;
504dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
505dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
506dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
507dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstruct state_info {
508dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    transport_type transport;
509dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char* serial;
510dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    int state;
511dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project};
512dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
513dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectstatic void wait_for_state(int fd, void* cookie)
514dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
515dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    struct state_info* sinfo = cookie;
516dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    char* err = "unknown error";
517dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
518dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    D("wait_for_state %d\n", sinfo->state);
519dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
520dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    atransport *t = acquire_one_transport(sinfo->state, sinfo->transport, sinfo->serial, &err);
521dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if(t != 0) {
522dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        writex(fd, "OKAY", 4);
523dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else {
524dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        sendfailmsg(fd, err);
525dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
526dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
527dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if (sinfo->serial)
528dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        free(sinfo->serial);
529dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    free(sinfo);
530dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    adb_close(fd);
531dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    D("wait_for_state is done\n");
532dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
533dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif
534dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
535dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#if ADB_HOST
536dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Projectasocket*  host_service_to_socket(const char*  name, const char *serial)
537dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project{
538dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    if (!strcmp(name,"track-devices")) {
539dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return create_device_tracker();
540dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    } else if (!strncmp(name, "wait-for-", strlen("wait-for-"))) {
541dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        struct state_info* sinfo = malloc(sizeof(struct state_info));
542dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
543dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if (serial)
544dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->serial = strdup(serial);
545dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        else
546dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->serial = NULL;
547dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
548dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        name += strlen("wait-for-");
549dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
550dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        if (!strncmp(name, "local", strlen("local"))) {
551dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->transport = kTransportLocal;
552dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->state = CS_DEVICE;
553dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        } else if (!strncmp(name, "usb", strlen("usb"))) {
554dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->transport = kTransportUsb;
555dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->state = CS_DEVICE;
556dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        } else if (!strncmp(name, "any", strlen("any"))) {
557dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->transport = kTransportAny;
558dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            sinfo->state = CS_DEVICE;
559dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        } else {
560dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            free(sinfo);
561dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project            return NULL;
562dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        }
563dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project
564dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        int fd = create_service_thread(wait_for_state, sinfo);
565dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project        return create_local_socket(fd);
566dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    }
567dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project    return NULL;
568dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project}
569dd7bc3319deb2b77c5d07a51b7d6cd7e11b5beb0The Android Open Source Project#endif /* ADB_HOST */
570