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 "sysdeps.h"
18 
19 #include <stdio.h>
20 
21 #include <android-base/file.h>
22 #include <android-base/logging.h>
23 #include <android-base/strings.h>
24 #include <cutils/sockets.h>
25 
26 #include "adb.h"
27 #include "adb_client.h"
28 #include "adb_io.h"
29 #include "adb_utils.h"
30 
31 // Return the console authentication command for the emulator, if needed
adb_construct_auth_command()32 static std::string adb_construct_auth_command() {
33     static const char auth_token_filename[] = ".emulator_console_auth_token";
34 
35     std::string auth_token_path = adb_get_homedir_path();
36     auth_token_path += OS_PATH_SEPARATOR;
37     auth_token_path += auth_token_filename;
38 
39     // read the token
40     std::string token;
41     if (!android::base::ReadFileToString(auth_token_path, &token)
42         || token.empty()) {
43         // we either can't read the file, or it doesn't exist, or it's empty -
44         // either way we won't add any authentication command.
45         return {};
46     }
47 
48     // now construct and return the actual command: "auth <token>\n"
49     std::string command = "auth ";
50     command += token;
51     command += '\n';
52     return command;
53 }
54 
55 // Return the console port of the currently connected emulator (if any) or -1 if
56 // there is no emulator, and -2 if there is more than one.
adb_get_emulator_console_port(const char * serial)57 static int adb_get_emulator_console_port(const char* serial) {
58     if (serial) {
59         // The user specified a serial number; is it an emulator?
60         int port;
61         return (sscanf(serial, "emulator-%d", &port) == 1) ? port : -1;
62     }
63 
64     // No specific device was given, so get the list of connected devices and
65     // search for emulators. If there's one, we'll take it. If there are more
66     // than one, that's an error.
67     std::string devices;
68     std::string error;
69     if (!adb_query("host:devices", &devices, &error)) {
70         fprintf(stderr, "error: no emulator connected: %s\n", error.c_str());
71         return -1;
72     }
73 
74     int port = -1;
75     size_t emulator_count = 0;
76     for (const auto& device : android::base::Split(devices, "\n")) {
77         if (sscanf(device.c_str(), "emulator-%d", &port) == 1) {
78             if (++emulator_count > 1) {
79                 fprintf(
80                     stderr, "error: more than one emulator detected; use -s\n");
81                 return -1;
82             }
83         }
84     }
85 
86     if (emulator_count == 0) {
87         fprintf(stderr, "error: no emulator detected\n");
88         return -1;
89     }
90 
91     return port;
92 }
93 
connect_to_console(const char * serial)94 static int connect_to_console(const char* serial) {
95     int port = adb_get_emulator_console_port(serial);
96     if (port == -1) {
97         return -1;
98     }
99 
100     std::string error;
101     int fd = network_loopback_client(port, SOCK_STREAM, &error);
102     if (fd == -1) {
103         fprintf(stderr, "error: could not connect to TCP port %d: %s\n", port,
104                 error.c_str());
105         return -1;
106     }
107     return fd;
108 }
109 
adb_send_emulator_command(int argc,const char ** argv,const char * serial)110 int adb_send_emulator_command(int argc, const char** argv, const char* serial) {
111     unique_fd fd(connect_to_console(serial));
112     if (fd == -1) {
113         return 1;
114     }
115 
116     std::string commands = adb_construct_auth_command();
117 
118     for (int i = 1; i < argc; i++) {
119         commands.append(argv[i]);
120         commands.push_back(i == argc - 1 ? '\n' : ' ');
121     }
122 
123     commands.append("quit\n");
124 
125     if (!WriteFdExactly(fd, commands)) {
126         fprintf(stderr, "error: cannot write to emulator: %s\n",
127                 strerror(errno));
128         return 1;
129     }
130 
131     // Drain output that the emulator console has sent us to prevent a problem
132     // on Windows where if adb closes the socket without reading all the data,
133     // the emulator's next call to recv() will have an ECONNABORTED error,
134     // preventing the emulator from reading the command that adb has sent.
135     // https://code.google.com/p/android/issues/detail?id=21021
136     int result;
137     std::string emulator_output;
138     do {
139         char buf[BUFSIZ];
140         result = adb_read(fd, buf, sizeof(buf));
141         // Keep reading until zero bytes (orderly/graceful shutdown) or an
142         // error. If 'adb emu kill' is executed, the emulator calls exit() with
143         // the socket open (and shutdown(SD_SEND) was not called), which causes
144         // Windows to send a TCP RST segment which causes adb to get ECONNRESET.
145         // Any other emu command is followed by the quit command that we
146         // appended above, and that causes the emulator to close the socket
147         // which should cause zero bytes (orderly/graceful shutdown) to be
148         // returned.
149         if (result > 0) emulator_output.append(buf, result);
150     } while (result > 0);
151 
152     // Note: the following messages are expected to be quite stable from emulator.
153     //
154     // Emulator console will send the following message upon connection:
155     //
156     // Android Console: Authentication required
157     // Android Console: type 'auth <auth_token>' to authenticate
158     // Android Console: you can find your <auth_token> in
159     // '/<path-to-home>/.emulator_console_auth_token'
160     // OK\r\n
161     //
162     // and the following after authentication:
163     // Android Console: type 'help' for a list of commands
164     // OK\r\n
165     //
166     // So try search and skip first two "OK\r\n", print the rest.
167     //
168     const std::string delims = "OK\r\n";
169     size_t found = 0;
170     for (int i = 0; i < 2; ++i) {
171         const size_t result = emulator_output.find(delims, found);
172         if (result == std::string::npos) {
173             break;
174         } else {
175             found = result + delims.size();
176         }
177     }
178 
179     printf("%s", emulator_output.c_str() + found);
180     return 0;
181 }
182