xref: /aosp_15_r20/frameworks/base/tools/aapt2/Main.cpp (revision d57664e9bc4670b3ecf6748a746a57c557b6bc9e)
1*d57664e9SAndroid Build Coastguard Worker /*
2*d57664e9SAndroid Build Coastguard Worker  * Copyright (C) 2015 The Android Open Source Project
3*d57664e9SAndroid Build Coastguard Worker  *
4*d57664e9SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*d57664e9SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*d57664e9SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*d57664e9SAndroid Build Coastguard Worker  *
8*d57664e9SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*d57664e9SAndroid Build Coastguard Worker  *
10*d57664e9SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*d57664e9SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*d57664e9SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*d57664e9SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*d57664e9SAndroid Build Coastguard Worker  * limitations under the License.
15*d57664e9SAndroid Build Coastguard Worker  */
16*d57664e9SAndroid Build Coastguard Worker 
17*d57664e9SAndroid Build Coastguard Worker #ifdef _WIN32
18*d57664e9SAndroid Build Coastguard Worker // clang-format off
19*d57664e9SAndroid Build Coastguard Worker #include <windows.h>
20*d57664e9SAndroid Build Coastguard Worker #include <shellapi.h>
21*d57664e9SAndroid Build Coastguard Worker // clang-format on
22*d57664e9SAndroid Build Coastguard Worker #endif
23*d57664e9SAndroid Build Coastguard Worker 
24*d57664e9SAndroid Build Coastguard Worker #include <iostream>
25*d57664e9SAndroid Build Coastguard Worker #include <vector>
26*d57664e9SAndroid Build Coastguard Worker 
27*d57664e9SAndroid Build Coastguard Worker #include "Diagnostics.h"
28*d57664e9SAndroid Build Coastguard Worker #include "android-base/stringprintf.h"
29*d57664e9SAndroid Build Coastguard Worker #include "android-base/utf8.h"
30*d57664e9SAndroid Build Coastguard Worker #include "androidfw/FileStream.h"
31*d57664e9SAndroid Build Coastguard Worker #include "androidfw/IDiagnostics.h"
32*d57664e9SAndroid Build Coastguard Worker #include "androidfw/StringPiece.h"
33*d57664e9SAndroid Build Coastguard Worker #include "cmd/ApkInfo.h"
34*d57664e9SAndroid Build Coastguard Worker #include "cmd/Command.h"
35*d57664e9SAndroid Build Coastguard Worker #include "cmd/Compile.h"
36*d57664e9SAndroid Build Coastguard Worker #include "cmd/Convert.h"
37*d57664e9SAndroid Build Coastguard Worker #include "cmd/Diff.h"
38*d57664e9SAndroid Build Coastguard Worker #include "cmd/Dump.h"
39*d57664e9SAndroid Build Coastguard Worker #include "cmd/Link.h"
40*d57664e9SAndroid Build Coastguard Worker #include "cmd/Optimize.h"
41*d57664e9SAndroid Build Coastguard Worker #include "trace/TraceBuffer.h"
42*d57664e9SAndroid Build Coastguard Worker #include "util/Files.h"
43*d57664e9SAndroid Build Coastguard Worker #include "util/Util.h"
44*d57664e9SAndroid Build Coastguard Worker 
45*d57664e9SAndroid Build Coastguard Worker using ::android::StringPiece;
46*d57664e9SAndroid Build Coastguard Worker using ::android::base::StringPrintf;
47*d57664e9SAndroid Build Coastguard Worker 
48*d57664e9SAndroid Build Coastguard Worker namespace aapt {
49*d57664e9SAndroid Build Coastguard Worker 
50*d57664e9SAndroid Build Coastguard Worker /** Prints the version information of AAPT2. */
51*d57664e9SAndroid Build Coastguard Worker class VersionCommand : public Command {
52*d57664e9SAndroid Build Coastguard Worker  public:
VersionCommand()53*d57664e9SAndroid Build Coastguard Worker   explicit VersionCommand() : Command("version") {
54*d57664e9SAndroid Build Coastguard Worker     SetDescription("Prints the version of aapt.");
55*d57664e9SAndroid Build Coastguard Worker   }
56*d57664e9SAndroid Build Coastguard Worker 
Action(const std::vector<std::string> &)57*d57664e9SAndroid Build Coastguard Worker   int Action(const std::vector<std::string>& /* args */) override {
58*d57664e9SAndroid Build Coastguard Worker     std::cerr << StringPrintf("%s %s", util::GetToolName(), util::GetToolFingerprint().c_str())
59*d57664e9SAndroid Build Coastguard Worker               << std::endl;
60*d57664e9SAndroid Build Coastguard Worker     return 0;
61*d57664e9SAndroid Build Coastguard Worker   }
62*d57664e9SAndroid Build Coastguard Worker };
63*d57664e9SAndroid Build Coastguard Worker 
64*d57664e9SAndroid Build Coastguard Worker /** The main entry point of AAPT. */
65*d57664e9SAndroid Build Coastguard Worker class MainCommand : public Command {
66*d57664e9SAndroid Build Coastguard Worker  public:
MainCommand(text::Printer * printer,android::IDiagnostics * diagnostics)67*d57664e9SAndroid Build Coastguard Worker   explicit MainCommand(text::Printer* printer, android::IDiagnostics* diagnostics)
68*d57664e9SAndroid Build Coastguard Worker       : Command("aapt2"), diagnostics_(diagnostics) {
69*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<CompileCommand>(diagnostics));
70*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<LinkCommand>(diagnostics));
71*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<DumpCommand>(printer, diagnostics));
72*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<DiffCommand>());
73*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<OptimizeCommand>());
74*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<ConvertCommand>());
75*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<VersionCommand>());
76*d57664e9SAndroid Build Coastguard Worker     AddOptionalSubcommand(util::make_unique<ApkInfoCommand>(diagnostics));
77*d57664e9SAndroid Build Coastguard Worker   }
78*d57664e9SAndroid Build Coastguard Worker 
Action(const std::vector<std::string> & args)79*d57664e9SAndroid Build Coastguard Worker   int Action(const std::vector<std::string>& args) override {
80*d57664e9SAndroid Build Coastguard Worker     if (args.size() == 0) {
81*d57664e9SAndroid Build Coastguard Worker       diagnostics_->Error(android::DiagMessage() << "no subcommand specified");
82*d57664e9SAndroid Build Coastguard Worker     } else {
83*d57664e9SAndroid Build Coastguard Worker       diagnostics_->Error(android::DiagMessage() << "unknown subcommand '" << args[0] << "'");
84*d57664e9SAndroid Build Coastguard Worker     }
85*d57664e9SAndroid Build Coastguard Worker 
86*d57664e9SAndroid Build Coastguard Worker     Usage(&std::cerr);
87*d57664e9SAndroid Build Coastguard Worker     return -1;
88*d57664e9SAndroid Build Coastguard Worker   }
89*d57664e9SAndroid Build Coastguard Worker 
90*d57664e9SAndroid Build Coastguard Worker  private:
91*d57664e9SAndroid Build Coastguard Worker   android::IDiagnostics* diagnostics_;
92*d57664e9SAndroid Build Coastguard Worker };
93*d57664e9SAndroid Build Coastguard Worker 
94*d57664e9SAndroid Build Coastguard Worker /*
95*d57664e9SAndroid Build Coastguard Worker  * Run in daemon mode. The first line of input is the command. This can be 'quit' which ends
96*d57664e9SAndroid Build Coastguard Worker  * the daemon mode. Each subsequent line is a single parameter to the command. The end of a
97*d57664e9SAndroid Build Coastguard Worker  * invocation is signaled by providing an empty line. At any point, an EOF signal or the
98*d57664e9SAndroid Build Coastguard Worker  * command 'quit' will end the daemon mode.
99*d57664e9SAndroid Build Coastguard Worker  */
100*d57664e9SAndroid Build Coastguard Worker class DaemonCommand : public Command {
101*d57664e9SAndroid Build Coastguard Worker  public:
DaemonCommand(android::FileOutputStream * out,android::IDiagnostics * diagnostics)102*d57664e9SAndroid Build Coastguard Worker   explicit DaemonCommand(android::FileOutputStream* out, android::IDiagnostics* diagnostics)
103*d57664e9SAndroid Build Coastguard Worker       : Command("daemon", "m"), out_(out), diagnostics_(diagnostics) {
104*d57664e9SAndroid Build Coastguard Worker     SetDescription("Runs aapt in daemon mode. Each subsequent line is a single parameter to the\n"
105*d57664e9SAndroid Build Coastguard Worker         "command. The end of an invocation is signaled by providing an empty line.");
106*d57664e9SAndroid Build Coastguard Worker     AddOptionalFlag("--trace_folder", "Generate systrace json trace fragment to specified folder.",
107*d57664e9SAndroid Build Coastguard Worker                     &trace_folder_);
108*d57664e9SAndroid Build Coastguard Worker   }
109*d57664e9SAndroid Build Coastguard Worker 
Action(const std::vector<std::string> & arguments)110*d57664e9SAndroid Build Coastguard Worker   int Action(const std::vector<std::string>& arguments) override {
111*d57664e9SAndroid Build Coastguard Worker     TRACE_FLUSH_ARGS(trace_folder_ ? trace_folder_.value() : "", "daemon", arguments);
112*d57664e9SAndroid Build Coastguard Worker     text::Printer printer(out_);
113*d57664e9SAndroid Build Coastguard Worker     std::cout << "Ready" << std::endl;
114*d57664e9SAndroid Build Coastguard Worker 
115*d57664e9SAndroid Build Coastguard Worker     while (true) {
116*d57664e9SAndroid Build Coastguard Worker       std::vector<std::string> raw_args;
117*d57664e9SAndroid Build Coastguard Worker       for (std::string line; std::getline(std::cin, line) && !line.empty();) {
118*d57664e9SAndroid Build Coastguard Worker         raw_args.push_back(line);
119*d57664e9SAndroid Build Coastguard Worker       }
120*d57664e9SAndroid Build Coastguard Worker 
121*d57664e9SAndroid Build Coastguard Worker       if (!std::cin) {
122*d57664e9SAndroid Build Coastguard Worker         break;
123*d57664e9SAndroid Build Coastguard Worker       }
124*d57664e9SAndroid Build Coastguard Worker 
125*d57664e9SAndroid Build Coastguard Worker       // An empty command does nothing.
126*d57664e9SAndroid Build Coastguard Worker       if (raw_args.empty()) {
127*d57664e9SAndroid Build Coastguard Worker         continue;
128*d57664e9SAndroid Build Coastguard Worker       }
129*d57664e9SAndroid Build Coastguard Worker 
130*d57664e9SAndroid Build Coastguard Worker       // End the dameon
131*d57664e9SAndroid Build Coastguard Worker       if (raw_args[0] == "quit") {
132*d57664e9SAndroid Build Coastguard Worker         break;
133*d57664e9SAndroid Build Coastguard Worker       }
134*d57664e9SAndroid Build Coastguard Worker 
135*d57664e9SAndroid Build Coastguard Worker       std::vector<StringPiece> args;
136*d57664e9SAndroid Build Coastguard Worker       args.insert(args.end(), raw_args.begin(), raw_args.end());
137*d57664e9SAndroid Build Coastguard Worker       int result = MainCommand(&printer, diagnostics_).Execute(args, &std::cerr);
138*d57664e9SAndroid Build Coastguard Worker       out_->Flush();
139*d57664e9SAndroid Build Coastguard Worker       if (result != 0) {
140*d57664e9SAndroid Build Coastguard Worker         std::cerr << "Error" << std::endl;
141*d57664e9SAndroid Build Coastguard Worker       }
142*d57664e9SAndroid Build Coastguard Worker       std::cerr << "Done" << std::endl;
143*d57664e9SAndroid Build Coastguard Worker     }
144*d57664e9SAndroid Build Coastguard Worker     std::cout << "Exiting daemon" << std::endl;
145*d57664e9SAndroid Build Coastguard Worker 
146*d57664e9SAndroid Build Coastguard Worker     return 0;
147*d57664e9SAndroid Build Coastguard Worker   }
148*d57664e9SAndroid Build Coastguard Worker 
149*d57664e9SAndroid Build Coastguard Worker  private:
150*d57664e9SAndroid Build Coastguard Worker   android::FileOutputStream* out_;
151*d57664e9SAndroid Build Coastguard Worker   android::IDiagnostics* diagnostics_;
152*d57664e9SAndroid Build Coastguard Worker   std::optional<std::string> trace_folder_;
153*d57664e9SAndroid Build Coastguard Worker };
154*d57664e9SAndroid Build Coastguard Worker 
155*d57664e9SAndroid Build Coastguard Worker }  // namespace aapt
156*d57664e9SAndroid Build Coastguard Worker 
MainImpl(int argc,char ** argv)157*d57664e9SAndroid Build Coastguard Worker int MainImpl(int argc, char** argv) {
158*d57664e9SAndroid Build Coastguard Worker   if (argc < 1) {
159*d57664e9SAndroid Build Coastguard Worker     return -1;
160*d57664e9SAndroid Build Coastguard Worker   }
161*d57664e9SAndroid Build Coastguard Worker 
162*d57664e9SAndroid Build Coastguard Worker   // Collect the arguments starting after the program name and command name.
163*d57664e9SAndroid Build Coastguard Worker   std::vector<StringPiece> args;
164*d57664e9SAndroid Build Coastguard Worker   for (int i = 1; i < argc; i++) {
165*d57664e9SAndroid Build Coastguard Worker     args.push_back(argv[i]);
166*d57664e9SAndroid Build Coastguard Worker   }
167*d57664e9SAndroid Build Coastguard Worker 
168*d57664e9SAndroid Build Coastguard Worker   // Use a smaller buffer so that there is less latency for printing to stdout.
169*d57664e9SAndroid Build Coastguard Worker   constexpr size_t kStdOutBufferSize = 1024u;
170*d57664e9SAndroid Build Coastguard Worker   android::FileOutputStream fout(STDOUT_FILENO, kStdOutBufferSize);
171*d57664e9SAndroid Build Coastguard Worker   aapt::text::Printer printer(&fout);
172*d57664e9SAndroid Build Coastguard Worker 
173*d57664e9SAndroid Build Coastguard Worker   aapt::StdErrDiagnostics diagnostics;
174*d57664e9SAndroid Build Coastguard Worker   aapt::MainCommand main_command(&printer, &diagnostics);
175*d57664e9SAndroid Build Coastguard Worker 
176*d57664e9SAndroid Build Coastguard Worker   // Add the daemon subcommand here so it cannot be called while executing the daemon
177*d57664e9SAndroid Build Coastguard Worker   main_command.AddOptionalSubcommand(
178*d57664e9SAndroid Build Coastguard Worker       aapt::util::make_unique<aapt::DaemonCommand>(&fout, &diagnostics));
179*d57664e9SAndroid Build Coastguard Worker   return main_command.Execute(args, &std::cerr);
180*d57664e9SAndroid Build Coastguard Worker }
181*d57664e9SAndroid Build Coastguard Worker 
main(int argc,char ** argv)182*d57664e9SAndroid Build Coastguard Worker int main(int argc, char** argv) {
183*d57664e9SAndroid Build Coastguard Worker #ifdef _WIN32
184*d57664e9SAndroid Build Coastguard Worker   LPWSTR* wide_argv = CommandLineToArgvW(GetCommandLineW(), &argc);
185*d57664e9SAndroid Build Coastguard Worker   CHECK(wide_argv != nullptr) << "invalid command line parameters passed to process";
186*d57664e9SAndroid Build Coastguard Worker 
187*d57664e9SAndroid Build Coastguard Worker   std::vector<std::string> utf8_args;
188*d57664e9SAndroid Build Coastguard Worker   for (int i = 0; i < argc; i++) {
189*d57664e9SAndroid Build Coastguard Worker     std::string utf8_arg;
190*d57664e9SAndroid Build Coastguard Worker     if (!::android::base::WideToUTF8(wide_argv[i], &utf8_arg)) {
191*d57664e9SAndroid Build Coastguard Worker       std::cerr << "error converting input arguments to UTF-8" << std::endl;
192*d57664e9SAndroid Build Coastguard Worker       return 1;
193*d57664e9SAndroid Build Coastguard Worker     }
194*d57664e9SAndroid Build Coastguard Worker     utf8_args.push_back(std::move(utf8_arg));
195*d57664e9SAndroid Build Coastguard Worker   }
196*d57664e9SAndroid Build Coastguard Worker   LocalFree(wide_argv);
197*d57664e9SAndroid Build Coastguard Worker 
198*d57664e9SAndroid Build Coastguard Worker   std::unique_ptr<char* []> utf8_argv(new char*[utf8_args.size()]);
199*d57664e9SAndroid Build Coastguard Worker   for (int i = 0; i < argc; i++) {
200*d57664e9SAndroid Build Coastguard Worker     utf8_argv[i] = const_cast<char*>(utf8_args[i].c_str());
201*d57664e9SAndroid Build Coastguard Worker   }
202*d57664e9SAndroid Build Coastguard Worker   argv = utf8_argv.get();
203*d57664e9SAndroid Build Coastguard Worker #endif
204*d57664e9SAndroid Build Coastguard Worker   return MainImpl(argc, argv);
205*d57664e9SAndroid Build Coastguard Worker }
206