1 /* 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 17 #include "Debug.h" 18 #include "Diagnostics.h" 19 #include "Flags.h" 20 #include "io/ZipArchive.h" 21 #include "process/IResourceTableConsumer.h" 22 #include "proto/ProtoSerialize.h" 23 #include "util/Files.h" 24 #include "util/StringPiece.h" 25 26 #include <vector> 27 28 namespace aapt { 29 30 //struct DumpOptions { 31 // 32 //}; 33 34 void dumpCompiledFile(const pb::CompiledFile& pbFile, const void* data, size_t len, 35 const Source& source, IAaptContext* context) { 36 std::unique_ptr<ResourceFile> file = deserializeCompiledFileFromPb(pbFile, source, 37 context->getDiagnostics()); 38 if (!file) { 39 return; 40 } 41 42 std::cout << "Resource: " << file->name << "\n" 43 << "Config: " << file->config << "\n" 44 << "Source: " << file->source << "\n"; 45 } 46 47 void dumpCompiledTable(const pb::ResourceTable& pbTable, const Source& source, 48 IAaptContext* context) { 49 std::unique_ptr<ResourceTable> table = deserializeTableFromPb(pbTable, source, 50 context->getDiagnostics()); 51 if (!table) { 52 return; 53 } 54 55 Debug::printTable(table.get()); 56 } 57 58 void tryDumpFile(IAaptContext* context, const std::string& filePath) { 59 std::string err; 60 std::unique_ptr<io::ZipFileCollection> zip = io::ZipFileCollection::create(filePath, &err); 61 if (zip) { 62 io::IFile* file = zip->findFile("resources.arsc.flat"); 63 if (file) { 64 std::unique_ptr<io::IData> data = file->openAsData(); 65 if (!data) { 66 context->getDiagnostics()->error(DiagMessage(filePath) 67 << "failed to open resources.arsc.flat"); 68 return; 69 } 70 71 pb::ResourceTable pbTable; 72 if (!pbTable.ParseFromArray(data->data(), data->size())) { 73 context->getDiagnostics()->error(DiagMessage(filePath) 74 << "invalid resources.arsc.flat"); 75 return; 76 } 77 78 std::unique_ptr<ResourceTable> table = deserializeTableFromPb( 79 pbTable, Source(filePath), context->getDiagnostics()); 80 if (table) { 81 DebugPrintTableOptions debugPrintTableOptions; 82 debugPrintTableOptions.showSources = true; 83 Debug::printTable(table.get(), debugPrintTableOptions); 84 } 85 } 86 return; 87 } 88 89 Maybe<android::FileMap> file = file::mmapPath(filePath, &err); 90 if (!file) { 91 context->getDiagnostics()->error(DiagMessage(filePath) << err); 92 return; 93 } 94 95 android::FileMap* fileMap = &file.value(); 96 97 // Try as a compiled table. 98 pb::ResourceTable pbTable; 99 if (pbTable.ParseFromArray(fileMap->getDataPtr(), fileMap->getDataLength())) { 100 dumpCompiledTable(pbTable, Source(filePath), context); 101 return; 102 } 103 104 // Try as a compiled file. 105 CompiledFileInputStream input(fileMap->getDataPtr(), fileMap->getDataLength()); 106 if (const pb::CompiledFile* pbFile = input.CompiledFile()) { 107 dumpCompiledFile(*pbFile, input.data(), input.size(), Source(filePath), context); 108 return; 109 } 110 } 111 112 class DumpContext : public IAaptContext { 113 public: 114 IDiagnostics* getDiagnostics() override { 115 return &mDiagnostics; 116 } 117 118 NameMangler* getNameMangler() override { 119 abort(); 120 return nullptr; 121 } 122 123 const std::u16string& getCompilationPackage() override { 124 static std::u16string empty; 125 return empty; 126 } 127 128 uint8_t getPackageId() override { 129 return 0; 130 } 131 132 SymbolTable* getExternalSymbols() override { 133 abort(); 134 return nullptr; 135 } 136 137 bool verbose() override { 138 return mVerbose; 139 } 140 141 void setVerbose(bool val) { 142 mVerbose = val; 143 } 144 145 private: 146 StdErrDiagnostics mDiagnostics; 147 bool mVerbose = false; 148 }; 149 150 /** 151 * Entry point for dump command. 152 */ 153 int dump(const std::vector<StringPiece>& args) { 154 bool verbose = false; 155 Flags flags = Flags() 156 .optionalSwitch("-v", "increase verbosity of output", &verbose); 157 if (!flags.parse("aapt2 dump", args, &std::cerr)) { 158 return 1; 159 } 160 161 DumpContext context; 162 context.setVerbose(verbose); 163 164 for (const std::string& arg : flags.getArgs()) { 165 tryDumpFile(&context, arg); 166 } 167 return 0; 168 } 169 170 } // namespace aapt 171