blob: 0965910ca8539cbb809d09b4939954c99e7b5781 [file] [log] [blame]
Adam Lesinski59e04c62016-02-04 15:59:23 -08001/*
2 * Copyright (C) 2016 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
Adam Lesinskice5e56e2016-10-21 17:56:45 -070017#include <vector>
18
Adam Lesinskid5083f62017-01-16 15:07:21 -080019#include "androidfw/StringPiece.h"
20
Adam Lesinski59e04c62016-02-04 15:59:23 -080021#include "Debug.h"
22#include "Diagnostics.h"
23#include "Flags.h"
Adam Lesinski64587af2016-02-18 18:33:06 -080024#include "io/ZipArchive.h"
Adam Lesinski59e04c62016-02-04 15:59:23 -080025#include "process/IResourceTableConsumer.h"
26#include "proto/ProtoSerialize.h"
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070027#include "unflatten/BinaryResourceParser.h"
Adam Lesinski59e04c62016-02-04 15:59:23 -080028#include "util/Files.h"
Adam Lesinskid5083f62017-01-16 15:07:21 -080029
Adam Lesinski4ffea042017-08-10 15:37:28 -070030using ::android::StringPiece;
Adam Lesinski59e04c62016-02-04 15:59:23 -080031
Adam Lesinski59e04c62016-02-04 15:59:23 -080032namespace aapt {
33
Adam Lesinski4ffea042017-08-10 15:37:28 -070034bool DumpCompiledFile(const pb::internal::CompiledFile& pb_file, const void* data, size_t len,
Adam Lesinskid0f492d2017-04-03 18:12:45 -070035 const Source& source, IAaptContext* context) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -070036 std::unique_ptr<ResourceFile> file =
Adam Lesinskice5e56e2016-10-21 17:56:45 -070037 DeserializeCompiledFileFromPb(pb_file, source, context->GetDiagnostics());
Adam Lesinskicacb28f2016-10-19 12:18:14 -070038 if (!file) {
Adam Lesinskid0f492d2017-04-03 18:12:45 -070039 context->GetDiagnostics()->Warn(DiagMessage() << "failed to read compiled file");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010040 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070041 }
Adam Lesinski59e04c62016-02-04 15:59:23 -080042
Adam Lesinskicacb28f2016-10-19 12:18:14 -070043 std::cout << "Resource: " << file->name << "\n"
44 << "Config: " << file->config << "\n"
45 << "Source: " << file->source << "\n";
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010046 return true;
Adam Lesinski59e04c62016-02-04 15:59:23 -080047}
48
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010049bool TryDumpFile(IAaptContext* context, const std::string& file_path) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -070050 std::unique_ptr<ResourceTable> table;
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070051
Adam Lesinskicacb28f2016-10-19 12:18:14 -070052 std::string err;
Adam Lesinskid0f492d2017-04-03 18:12:45 -070053 std::unique_ptr<io::ZipFileCollection> zip = io::ZipFileCollection::Create(file_path, &err);
Adam Lesinskicacb28f2016-10-19 12:18:14 -070054 if (zip) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070055 io::IFile* file = zip->FindFile("resources.arsc.flat");
Adam Lesinskicacb28f2016-10-19 12:18:14 -070056 if (file) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070057 std::unique_ptr<io::IData> data = file->OpenAsData();
Adam Lesinskicacb28f2016-10-19 12:18:14 -070058 if (!data) {
Adam Lesinskid0f492d2017-04-03 18:12:45 -070059 context->GetDiagnostics()->Error(DiagMessage(file_path)
60 << "failed to open resources.arsc.flat");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010061 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070062 }
Adam Lesinski64587af2016-02-18 18:33:06 -080063
Adam Lesinskice5e56e2016-10-21 17:56:45 -070064 pb::ResourceTable pb_table;
65 if (!pb_table.ParseFromArray(data->data(), data->size())) {
Adam Lesinskid0f492d2017-04-03 18:12:45 -070066 context->GetDiagnostics()->Error(DiagMessage(file_path) << "invalid resources.arsc.flat");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010067 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070068 }
Adam Lesinski64587af2016-02-18 18:33:06 -080069
Adam Lesinskid0f492d2017-04-03 18:12:45 -070070 table = DeserializeTableFromPb(pb_table, Source(file_path), context->GetDiagnostics());
Adam Lesinskicacb28f2016-10-19 12:18:14 -070071 if (!table) {
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010072 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070073 }
Adam Lesinski64587af2016-02-18 18:33:06 -080074 }
75
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070076 if (!table) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070077 file = zip->FindFile("resources.arsc");
Adam Lesinskicacb28f2016-10-19 12:18:14 -070078 if (file) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070079 std::unique_ptr<io::IData> data = file->OpenAsData();
Adam Lesinskicacb28f2016-10-19 12:18:14 -070080 if (!data) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070081 context->GetDiagnostics()->Error(DiagMessage(file_path)
Adam Lesinskicacb28f2016-10-19 12:18:14 -070082 << "failed to open resources.arsc");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010083 return false;
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070084 }
85
Adam Lesinskicacb28f2016-10-19 12:18:14 -070086 table = util::make_unique<ResourceTable>();
Adam Lesinskid0f492d2017-04-03 18:12:45 -070087 BinaryResourceParser parser(context, table.get(), Source(file_path), data->data(),
88 data->size());
Adam Lesinskice5e56e2016-10-21 17:56:45 -070089 if (!parser.Parse()) {
Pierre Lecesneaadf27e2017-05-05 14:58:21 +010090 return false;
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070091 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -070092 }
93 }
94 }
Adam Lesinski5e8fa3a2016-06-27 16:21:42 -070095
Adam Lesinskicacb28f2016-10-19 12:18:14 -070096 if (!table) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070097 Maybe<android::FileMap> file = file::MmapPath(file_path, &err);
Adam Lesinskicacb28f2016-10-19 12:18:14 -070098 if (!file) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -070099 context->GetDiagnostics()->Error(DiagMessage(file_path) << err);
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100100 return false;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800101 }
102
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700103 android::FileMap* file_map = &file.value();
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700104
105 // Try as a compiled table.
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700106 pb::ResourceTable pb_table;
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700107 if (pb_table.ParseFromArray(file_map->getDataPtr(), file_map->getDataLength())) {
108 table = DeserializeTableFromPb(pb_table, Source(file_path), context->GetDiagnostics());
Adam Lesinski59e04c62016-02-04 15:59:23 -0800109 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700110
111 if (!table) {
112 // Try as a compiled file.
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700113 CompiledFileInputStream input(file_map->getDataPtr(), file_map->getDataLength());
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700114
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700115 uint32_t num_files = 0;
116 if (!input.ReadLittleEndian32(&num_files)) {
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100117 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700118 }
119
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700120 for (uint32_t i = 0; i < num_files; i++) {
Adam Lesinski4ffea042017-08-10 15:37:28 -0700121 pb::internal::CompiledFile compiled_file;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700122 if (!input.ReadCompiledFile(&compiled_file)) {
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700123 context->GetDiagnostics()->Warn(DiagMessage() << "failed to read compiled file");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100124 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700125 }
126
127 uint64_t offset, len;
128 if (!input.ReadDataMetaData(&offset, &len)) {
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700129 context->GetDiagnostics()->Warn(DiagMessage() << "failed to read meta data");
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100130 return false;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700131 }
132
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700133 const void* data = static_cast<const uint8_t*>(file_map->getDataPtr()) + offset;
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100134 if (!DumpCompiledFile(compiled_file, data, len, Source(file_path), context)) {
135 return false;
136 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700137 }
138 }
139 }
140
141 if (table) {
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700142 DebugPrintTableOptions options;
143 options.show_sources = true;
144 Debug::PrintTable(table.get(), options);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700145 }
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100146
147 return true;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800148}
149
150class DumpContext : public IAaptContext {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700151 public:
Adam Lesinskib522f042017-04-21 16:57:59 -0700152 PackageType GetPackageType() override {
153 // Doesn't matter.
154 return PackageType::kApp;
155 }
156
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700157 IDiagnostics* GetDiagnostics() override {
158 return &diagnostics_;
159 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800160
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700161 NameMangler* GetNameMangler() override {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700162 abort();
163 return nullptr;
164 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800165
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700166 const std::string& GetCompilationPackage() override {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700167 static std::string empty;
168 return empty;
169 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800170
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700171 uint8_t GetPackageId() override {
172 return 0;
173 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800174
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700175 SymbolTable* GetExternalSymbols() override {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700176 abort();
177 return nullptr;
178 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800179
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700180 bool IsVerbose() override {
181 return verbose_;
182 }
Adam Lesinski355f2852016-02-13 20:26:45 -0800183
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700184 void SetVerbose(bool val) {
185 verbose_ = val;
186 }
Adam Lesinski355f2852016-02-13 20:26:45 -0800187
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700188 int GetMinSdkVersion() override {
189 return 0;
190 }
Adam Lesinskifb6312f2016-06-28 14:40:32 -0700191
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700192 private:
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700193 StdErrDiagnostics diagnostics_;
194 bool verbose_ = false;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800195};
196
197/**
198 * Entry point for dump command.
199 */
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700200int Dump(const std::vector<StringPiece>& args) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700201 bool verbose = false;
Adam Lesinskid0f492d2017-04-03 18:12:45 -0700202 Flags flags = Flags().OptionalSwitch("-v", "increase verbosity of output", &verbose);
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700203 if (!flags.Parse("aapt2 dump", args, &std::cerr)) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700204 return 1;
205 }
Adam Lesinski59e04c62016-02-04 15:59:23 -0800206
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700207 DumpContext context;
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700208 context.SetVerbose(verbose);
Adam Lesinski59e04c62016-02-04 15:59:23 -0800209
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700210 for (const std::string& arg : flags.GetArgs()) {
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100211 if (!TryDumpFile(&context, arg)) {
212 return 1;
213 }
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700214 }
Pierre Lecesneaadf27e2017-05-05 14:58:21 +0100215
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700216 return 0;
Adam Lesinski59e04c62016-02-04 15:59:23 -0800217}
218
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700219} // namespace aapt