1 #include "idmap.h"
2
3 #include <androidfw/AssetManager.h>
4 #include <androidfw/ResourceTypes.h>
5 #include <utils/ByteOrder.h>
6 #include <utils/String8.h>
7
8 #include <fcntl.h>
9 #include <sys/mman.h>
10 #include <sys/stat.h>
11
12 using namespace android;
13
14 namespace {
15 static const uint32_t IDMAP_MAGIC = 0x504D4449;
16 static const size_t PATH_LENGTH = 256;
17
18 void printe(const char *fmt, ...);
19
20 class IdmapBuffer {
21 private:
22 const char* buf_;
23 size_t len_;
24 size_t pos_;
25 public:
IdmapBuffer()26 IdmapBuffer() : buf_((const char *)MAP_FAILED), len_(0), pos_(0) {}
27
~IdmapBuffer()28 ~IdmapBuffer() {
29 if (buf_ != MAP_FAILED) {
30 munmap(const_cast<char*>(buf_), len_);
31 }
32 }
33
init(const char * idmap_path)34 status_t init(const char *idmap_path) {
35 struct stat st;
36 int fd;
37
38 if (stat(idmap_path, &st) < 0) {
39 printe("failed to stat idmap '%s': %s\n", idmap_path, strerror(errno));
40 return UNKNOWN_ERROR;
41 }
42 len_ = st.st_size;
43 if ((fd = TEMP_FAILURE_RETRY(open(idmap_path, O_RDONLY))) < 0) {
44 printe("failed to open idmap '%s': %s\n", idmap_path, strerror(errno));
45 return UNKNOWN_ERROR;
46 }
47 if ((buf_ = (const char*)mmap(NULL, len_, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) {
48 close(fd);
49 printe("failed to mmap idmap: %s\n", strerror(errno));
50 return UNKNOWN_ERROR;
51 }
52 close(fd);
53 return NO_ERROR;
54 }
55
nextUint32(uint32_t * i)56 status_t nextUint32(uint32_t* i) {
57 if (!buf_) {
58 printe("failed to read next uint32_t: buffer not initialized\n");
59 return UNKNOWN_ERROR;
60 }
61
62 if (pos_ + sizeof(uint32_t) > len_) {
63 printe("failed to read next uint32_t: end of buffer reached at pos=0x%08x\n",
64 pos_);
65 return UNKNOWN_ERROR;
66 }
67
68 if ((reinterpret_cast<uintptr_t>(buf_ + pos_) & 0x3) != 0) {
69 printe("failed to read next uint32_t: not aligned on 4-byte boundary\n");
70 return UNKNOWN_ERROR;
71 }
72
73 *i = dtohl(*reinterpret_cast<const uint32_t*>(buf_ + pos_));
74 pos_ += sizeof(uint32_t);
75 return NO_ERROR;
76 }
77
nextUint16(uint16_t * i)78 status_t nextUint16(uint16_t* i) {
79 if (!buf_) {
80 printe("failed to read next uint16_t: buffer not initialized\n");
81 return UNKNOWN_ERROR;
82 }
83
84 if (pos_ + sizeof(uint16_t) > len_) {
85 printe("failed to read next uint16_t: end of buffer reached at pos=0x%08x\n",
86 pos_);
87 return UNKNOWN_ERROR;
88 }
89
90 if ((reinterpret_cast<uintptr_t>(buf_ + pos_) & 0x1) != 0) {
91 printe("failed to read next uint32_t: not aligned on 2-byte boundary\n");
92 return UNKNOWN_ERROR;
93 }
94
95 *i = dtohs(*reinterpret_cast<const uint16_t*>(buf_ + pos_));
96 pos_ += sizeof(uint16_t);
97 return NO_ERROR;
98 }
99
nextPath(char * b)100 status_t nextPath(char *b) {
101 if (!buf_) {
102 printe("failed to read next path: buffer not initialized\n");
103 return UNKNOWN_ERROR;
104 }
105 if (pos_ + PATH_LENGTH > len_) {
106 printe("failed to read next path: end of buffer reached at pos=0x%08x\n", pos_);
107 return UNKNOWN_ERROR;
108 }
109 memcpy(b, buf_ + pos_, PATH_LENGTH);
110 pos_ += PATH_LENGTH;
111 return NO_ERROR;
112 }
113 };
114
printe(const char * fmt,...)115 void printe(const char *fmt, ...) {
116 va_list ap;
117
118 va_start(ap, fmt);
119 fprintf(stderr, "error: ");
120 vfprintf(stderr, fmt, ap);
121 va_end(ap);
122 }
123
print_header()124 void print_header() {
125 printf("SECTION ENTRY VALUE COMMENT\n");
126 }
127
print(const char * section,const char * subsection,uint32_t value,const char * fmt,...)128 void print(const char *section, const char *subsection, uint32_t value, const char *fmt, ...) {
129 va_list ap;
130
131 va_start(ap, fmt);
132 printf("%-12s %-12s 0x%08x ", section, subsection, value);
133 vprintf(fmt, ap);
134 printf("\n");
135 va_end(ap);
136 }
137
print_path(const char * section,const char * subsection,const char * fmt,...)138 void print_path(const char *section, const char *subsection, const char *fmt, ...) {
139 va_list ap;
140
141 va_start(ap, fmt);
142 printf("%-12s %-12s .......... ", section, subsection);
143 vprintf(fmt, ap);
144 printf("\n");
145 va_end(ap);
146 }
147
resource_metadata(const AssetManager & am,uint32_t res_id,String8 * package,String8 * type,String8 * name)148 status_t resource_metadata(const AssetManager& am, uint32_t res_id,
149 String8 *package, String8 *type, String8 *name) {
150 const ResTable& rt = am.getResources();
151 struct ResTable::resource_name data;
152 if (!rt.getResourceName(res_id, false, &data)) {
153 printe("failed to get resource name id=0x%08x\n", res_id);
154 return UNKNOWN_ERROR;
155 }
156 if (package != NULL) {
157 *package = String8(String16(data.package, data.packageLen));
158 }
159 if (type != NULL) {
160 *type = String8(String16(data.type, data.typeLen));
161 }
162 if (name != NULL) {
163 *name = String8(String16(data.name, data.nameLen));
164 }
165 return NO_ERROR;
166 }
167
parse_idmap_header(IdmapBuffer & buf,AssetManager & am)168 status_t parse_idmap_header(IdmapBuffer& buf, AssetManager& am) {
169 uint32_t i;
170 char path[PATH_LENGTH];
171
172 status_t err = buf.nextUint32(&i);
173 if (err != NO_ERROR) {
174 return err;
175 }
176
177 if (i != IDMAP_MAGIC) {
178 printe("not an idmap file: actual magic constant 0x%08x does not match expected magic "
179 "constant 0x%08x\n", i, IDMAP_MAGIC);
180 return UNKNOWN_ERROR;
181 }
182
183 print_header();
184 print("IDMAP HEADER", "magic", i, "");
185
186 err = buf.nextUint32(&i);
187 if (err != NO_ERROR) {
188 return err;
189 }
190 print("", "version", i, "");
191
192 err = buf.nextUint32(&i);
193 if (err != NO_ERROR) {
194 return err;
195 }
196 print("", "base crc", i, "");
197
198 err = buf.nextUint32(&i);
199 if (err != NO_ERROR) {
200 return err;
201 }
202 print("", "overlay crc", i, "");
203
204 err = buf.nextPath(path);
205 if (err != NO_ERROR) {
206 // printe done from IdmapBuffer::nextPath
207 return err;
208 }
209 print_path("", "base path", "%s", path);
210
211 if (!am.addAssetPath(String8(path), NULL)) {
212 printe("failed to add '%s' as asset path\n", path);
213 return UNKNOWN_ERROR;
214 }
215
216 err = buf.nextPath(path);
217 if (err != NO_ERROR) {
218 // printe done from IdmapBuffer::nextPath
219 return err;
220 }
221 print_path("", "overlay path", "%s", path);
222
223 return NO_ERROR;
224 }
225
parse_data(IdmapBuffer & buf,const AssetManager & am)226 status_t parse_data(IdmapBuffer& buf, const AssetManager& am) {
227 const uint32_t packageId = am.getResources().getBasePackageId(0);
228
229 uint16_t data16;
230 status_t err = buf.nextUint16(&data16);
231 if (err != NO_ERROR) {
232 return err;
233 }
234 print("DATA HEADER", "target pkg", static_cast<uint32_t>(data16), "");
235
236 err = buf.nextUint16(&data16);
237 if (err != NO_ERROR) {
238 return err;
239 }
240 print("", "types count", static_cast<uint32_t>(data16), "");
241
242 uint32_t typeCount = static_cast<uint32_t>(data16);
243 while (typeCount > 0) {
244 typeCount--;
245
246 err = buf.nextUint16(&data16);
247 if (err != NO_ERROR) {
248 return err;
249 }
250 const uint32_t targetTypeId = static_cast<uint32_t>(data16);
251 print("DATA BLOCK", "target type", targetTypeId, "");
252
253 err = buf.nextUint16(&data16);
254 if (err != NO_ERROR) {
255 return err;
256 }
257 print("", "overlay type", static_cast<uint32_t>(data16), "");
258
259 err = buf.nextUint16(&data16);
260 if (err != NO_ERROR) {
261 return err;
262 }
263 const uint32_t entryCount = static_cast<uint32_t>(data16);
264 print("", "entry count", entryCount, "");
265
266 err = buf.nextUint16(&data16);
267 if (err != NO_ERROR) {
268 return err;
269 }
270 const uint32_t entryOffset = static_cast<uint32_t>(data16);
271 print("", "entry offset", entryOffset, "");
272
273 for (uint32_t i = 0; i < entryCount; i++) {
274 uint32_t data32;
275 err = buf.nextUint32(&data32);
276 if (err != NO_ERROR) {
277 return err;
278 }
279
280 uint32_t resID = (packageId << 24) | (targetTypeId << 16) | (entryOffset + i);
281 String8 type;
282 String8 name;
283 err = resource_metadata(am, resID, NULL, &type, &name);
284 if (err != NO_ERROR) {
285 return err;
286 }
287 print("", "entry", data32, "%s/%s", type.string(), name.string());
288 }
289 }
290
291 return NO_ERROR;
292 }
293 }
294
idmap_inspect(const char * idmap_path)295 int idmap_inspect(const char *idmap_path) {
296 IdmapBuffer buf;
297 if (buf.init(idmap_path) < 0) {
298 // printe done from IdmapBuffer::init
299 return EXIT_FAILURE;
300 }
301 AssetManager am;
302 if (parse_idmap_header(buf, am) != NO_ERROR) {
303 // printe done from parse_idmap_header
304 return EXIT_FAILURE;
305 }
306 if (parse_data(buf, am) != NO_ERROR) {
307 // printe done from parse_data_header
308 return EXIT_FAILURE;
309 }
310 return EXIT_SUCCESS;
311 }
312