1 //
2 // Copyright 2006 The Android Open Source Project
3 //
4 // Information about assets being operated on.
5 //
6 #ifndef __AAPT_ASSETS_H
7 #define __AAPT_ASSETS_H
8
9 #include <stdlib.h>
10 #include <utils/AssetManager.h>
11 #include <utils/KeyedVector.h>
12 #include <utils/String8.h>
13 #include <utils/ResourceTypes.h>
14 #include <utils/SortedVector.h>
15 #include <utils/String8.h>
16 #include <utils/Vector.h>
17 #include <utils/RefBase.h>
18 #include "ZipFile.h"
19
20 #include "Bundle.h"
21 #include "SourcePos.h"
22
23 using namespace android;
24
25 bool valid_symbol_name(const String8& str);
26
27 enum {
28 AXIS_NONE = 0,
29 AXIS_MCC = 1,
30 AXIS_MNC,
31 AXIS_LANGUAGE,
32 AXIS_REGION,
33 AXIS_SCREENLAYOUTSIZE,
34 AXIS_SCREENLAYOUTLONG,
35 AXIS_ORIENTATION,
36 AXIS_DENSITY,
37 AXIS_TOUCHSCREEN,
38 AXIS_KEYSHIDDEN,
39 AXIS_KEYBOARD,
40 AXIS_NAVHIDDEN,
41 AXIS_NAVIGATION,
42 AXIS_SCREENSIZE,
43 AXIS_VERSION
44 };
45
46 /**
47 * This structure contains a specific variation of a single file out
48 * of all the variations it can have that we can have.
49 */
50 struct AaptGroupEntry
51 {
52 public:
AaptGroupEntryAaptGroupEntry53 AaptGroupEntry() { }
AaptGroupEntryAaptGroupEntry54 AaptGroupEntry(const String8& _locale, const String8& _vendor)
55 : locale(_locale), vendor(_vendor) { }
56
57 String8 mcc;
58 String8 mnc;
59 String8 locale;
60 String8 vendor;
61 String8 screenLayoutSize;
62 String8 screenLayoutLong;
63 String8 orientation;
64 String8 density;
65 String8 touchscreen;
66 String8 keysHidden;
67 String8 keyboard;
68 String8 navHidden;
69 String8 navigation;
70 String8 screenSize;
71 String8 version;
72
73 bool initFromDirName(const char* dir, String8* resType);
74
75 static status_t parseNamePart(const String8& part, int* axis, uint32_t* value);
76
77 static bool getMccName(const char* name, ResTable_config* out = NULL);
78 static bool getMncName(const char* name, ResTable_config* out = NULL);
79 static bool getLocaleName(const char* name, ResTable_config* out = NULL);
80 static bool getScreenLayoutSizeName(const char* name, ResTable_config* out = NULL);
81 static bool getScreenLayoutLongName(const char* name, ResTable_config* out = NULL);
82 static bool getOrientationName(const char* name, ResTable_config* out = NULL);
83 static bool getDensityName(const char* name, ResTable_config* out = NULL);
84 static bool getTouchscreenName(const char* name, ResTable_config* out = NULL);
85 static bool getKeysHiddenName(const char* name, ResTable_config* out = NULL);
86 static bool getKeyboardName(const char* name, ResTable_config* out = NULL);
87 static bool getNavigationName(const char* name, ResTable_config* out = NULL);
88 static bool getNavHiddenName(const char* name, ResTable_config* out = NULL);
89 static bool getScreenSizeName(const char* name, ResTable_config* out = NULL);
90 static bool getVersionName(const char* name, ResTable_config* out = NULL);
91
92 int compare(const AaptGroupEntry& o) const;
93
94 ResTable_config toParams() const;
95
96 inline bool operator<(const AaptGroupEntry& o) const { return compare(o) < 0; }
97 inline bool operator<=(const AaptGroupEntry& o) const { return compare(o) <= 0; }
98 inline bool operator==(const AaptGroupEntry& o) const { return compare(o) == 0; }
99 inline bool operator!=(const AaptGroupEntry& o) const { return compare(o) != 0; }
100 inline bool operator>=(const AaptGroupEntry& o) const { return compare(o) >= 0; }
101 inline bool operator>(const AaptGroupEntry& o) const { return compare(o) > 0; }
102
103 String8 toString() const;
104 String8 toDirName(const String8& resType) const;
105 };
106
compare_type(const AaptGroupEntry & lhs,const AaptGroupEntry & rhs)107 inline int compare_type(const AaptGroupEntry& lhs, const AaptGroupEntry& rhs)
108 {
109 return lhs.compare(rhs);
110 }
111
strictly_order_type(const AaptGroupEntry & lhs,const AaptGroupEntry & rhs)112 inline int strictly_order_type(const AaptGroupEntry& lhs, const AaptGroupEntry& rhs)
113 {
114 return compare_type(lhs, rhs) < 0;
115 }
116
117 class AaptGroup;
118
119 /**
120 * A single asset file we know about.
121 */
122 class AaptFile : public RefBase
123 {
124 public:
AaptFile(const String8 & sourceFile,const AaptGroupEntry & groupEntry,const String8 & resType)125 AaptFile(const String8& sourceFile, const AaptGroupEntry& groupEntry,
126 const String8& resType)
127 : mGroupEntry(groupEntry)
128 , mResourceType(resType)
129 , mSourceFile(sourceFile)
130 , mData(NULL)
131 , mDataSize(0)
132 , mBufferSize(0)
133 , mCompression(ZipEntry::kCompressStored)
134 {
135 //printf("new AaptFile created %s\n", (const char*)sourceFile);
136 }
~AaptFile()137 virtual ~AaptFile() {
138 free(mData);
139 }
140
getPath()141 const String8& getPath() const { return mPath; }
getGroupEntry()142 const AaptGroupEntry& getGroupEntry() const { return mGroupEntry; }
143
144 // Data API. If there is data attached to the file,
145 // getSourceFile() is not used.
hasData()146 bool hasData() const { return mData != NULL; }
getData()147 const void* getData() const { return mData; }
getSize()148 size_t getSize() const { return mDataSize; }
149 void* editData(size_t size);
150 void* editData(size_t* outSize = NULL);
151 void* padData(size_t wordSize);
152 status_t writeData(const void* data, size_t size);
153 void clearData();
154
getResourceType()155 const String8& getResourceType() const { return mResourceType; }
156
157 // File API. If the file does not hold raw data, this is
158 // a full path to a file on the filesystem that holds its data.
getSourceFile()159 const String8& getSourceFile() const { return mSourceFile; }
160
161 String8 getPrintableSource() const;
162
163 // Desired compression method, as per utils/ZipEntry.h. For example,
164 // no compression is ZipEntry::kCompressStored.
getCompressionMethod()165 int getCompressionMethod() const { return mCompression; }
setCompressionMethod(int c)166 void setCompressionMethod(int c) { mCompression = c; }
167 private:
168 friend class AaptGroup;
169
170 String8 mPath;
171 AaptGroupEntry mGroupEntry;
172 String8 mResourceType;
173 String8 mSourceFile;
174 void* mData;
175 size_t mDataSize;
176 size_t mBufferSize;
177 int mCompression;
178 };
179
180 /**
181 * A group of related files (the same file, with different
182 * vendor/locale variations).
183 */
184 class AaptGroup : public RefBase
185 {
186 public:
AaptGroup(const String8 & leaf,const String8 & path)187 AaptGroup(const String8& leaf, const String8& path)
188 : mLeaf(leaf), mPath(path) { }
~AaptGroup()189 virtual ~AaptGroup() { }
190
getLeaf()191 const String8& getLeaf() const { return mLeaf; }
192
193 // Returns the relative path after the AaptGroupEntry dirs.
getPath()194 const String8& getPath() const { return mPath; }
195
getFiles()196 const DefaultKeyedVector<AaptGroupEntry, sp<AaptFile> >& getFiles() const
197 { return mFiles; }
198
199 status_t addFile(const sp<AaptFile>& file);
200 void removeFile(size_t index);
201
202 void print() const;
203
204 String8 getPrintableSource() const;
205
206 private:
207 String8 mLeaf;
208 String8 mPath;
209
210 DefaultKeyedVector<AaptGroupEntry, sp<AaptFile> > mFiles;
211 };
212
213 /**
214 * A single directory of assets, which can contain for files and other
215 * sub-directories.
216 */
217 class AaptDir : public RefBase
218 {
219 public:
AaptDir(const String8 & leaf,const String8 & path)220 AaptDir(const String8& leaf, const String8& path)
221 : mLeaf(leaf), mPath(path) { }
~AaptDir()222 virtual ~AaptDir() { }
223
getLeaf()224 const String8& getLeaf() const { return mLeaf; }
225
getPath()226 const String8& getPath() const { return mPath; }
227
getFiles()228 const DefaultKeyedVector<String8, sp<AaptGroup> >& getFiles() const { return mFiles; }
getDirs()229 const DefaultKeyedVector<String8, sp<AaptDir> >& getDirs() const { return mDirs; }
230
231 status_t addFile(const String8& name, const sp<AaptGroup>& file);
232 status_t addDir(const String8& name, const sp<AaptDir>& dir);
233
234 sp<AaptDir> makeDir(const String8& name);
235
236 void removeFile(const String8& name);
237 void removeDir(const String8& name);
238
239 status_t renameFile(const sp<AaptFile>& file, const String8& newName);
240
241 status_t addLeafFile(const String8& leafName,
242 const sp<AaptFile>& file);
243
244 virtual ssize_t slurpFullTree(Bundle* bundle,
245 const String8& srcDir,
246 const AaptGroupEntry& kind,
247 const String8& resType);
248
249 /*
250 * Perform some sanity checks on the names of files and directories here.
251 * In particular:
252 * - Check for illegal chars in filenames.
253 * - Check filename length.
254 * - Check for presence of ".gz" and non-".gz" copies of same file.
255 * - Check for multiple files whose names match in a case-insensitive
256 * fashion (problematic for some systems).
257 *
258 * Comparing names against all other names is O(n^2). We could speed
259 * it up some by sorting the entries and being smarter about what we
260 * compare against, but I'm not expecting to have enough files in a
261 * single directory to make a noticeable difference in speed.
262 *
263 * Note that sorting here is not enough to guarantee that the package
264 * contents are sorted -- subsequent updates can rearrange things.
265 */
266 status_t validate() const;
267
268 void print() const;
269
270 String8 getPrintableSource() const;
271
272 private:
273 String8 mLeaf;
274 String8 mPath;
275
276 DefaultKeyedVector<String8, sp<AaptGroup> > mFiles;
277 DefaultKeyedVector<String8, sp<AaptDir> > mDirs;
278 };
279
280 /**
281 * All information we know about a particular symbol.
282 */
283 class AaptSymbolEntry
284 {
285 public:
AaptSymbolEntry()286 AaptSymbolEntry()
287 : isPublic(false), typeCode(TYPE_UNKNOWN)
288 {
289 }
AaptSymbolEntry(const String8 & _name)290 AaptSymbolEntry(const String8& _name)
291 : name(_name), isPublic(false), typeCode(TYPE_UNKNOWN)
292 {
293 }
AaptSymbolEntry(const AaptSymbolEntry & o)294 AaptSymbolEntry(const AaptSymbolEntry& o)
295 : name(o.name), sourcePos(o.sourcePos), isPublic(o.isPublic)
296 , comment(o.comment), typeComment(o.typeComment)
297 , typeCode(o.typeCode), int32Val(o.int32Val), stringVal(o.stringVal)
298 {
299 }
300 AaptSymbolEntry operator=(const AaptSymbolEntry& o)
301 {
302 sourcePos = o.sourcePos;
303 isPublic = o.isPublic;
304 comment = o.comment;
305 typeComment = o.typeComment;
306 typeCode = o.typeCode;
307 int32Val = o.int32Val;
308 stringVal = o.stringVal;
309 return *this;
310 }
311
312 const String8 name;
313
314 SourcePos sourcePos;
315 bool isPublic;
316
317 String16 comment;
318 String16 typeComment;
319
320 enum {
321 TYPE_UNKNOWN = 0,
322 TYPE_INT32,
323 TYPE_STRING
324 };
325
326 int typeCode;
327
328 // Value. May be one of these.
329 int32_t int32Val;
330 String8 stringVal;
331 };
332
333 /**
334 * A group of related symbols (such as indices into a string block)
335 * that have been generated from the assets.
336 */
337 class AaptSymbols : public RefBase
338 {
339 public:
AaptSymbols()340 AaptSymbols() { }
~AaptSymbols()341 virtual ~AaptSymbols() { }
342
addSymbol(const String8 & name,int32_t value,const SourcePos & pos)343 status_t addSymbol(const String8& name, int32_t value, const SourcePos& pos) {
344 if (!check_valid_symbol_name(name, pos, "symbol")) {
345 return BAD_VALUE;
346 }
347 AaptSymbolEntry& sym = edit_symbol(name, &pos);
348 sym.typeCode = AaptSymbolEntry::TYPE_INT32;
349 sym.int32Val = value;
350 return NO_ERROR;
351 }
352
addStringSymbol(const String8 & name,const String8 & value,const SourcePos & pos)353 status_t addStringSymbol(const String8& name, const String8& value,
354 const SourcePos& pos) {
355 if (!check_valid_symbol_name(name, pos, "symbol")) {
356 return BAD_VALUE;
357 }
358 AaptSymbolEntry& sym = edit_symbol(name, &pos);
359 sym.typeCode = AaptSymbolEntry::TYPE_STRING;
360 sym.stringVal = value;
361 return NO_ERROR;
362 }
363
makeSymbolPublic(const String8 & name,const SourcePos & pos)364 status_t makeSymbolPublic(const String8& name, const SourcePos& pos) {
365 if (!check_valid_symbol_name(name, pos, "symbol")) {
366 return BAD_VALUE;
367 }
368 AaptSymbolEntry& sym = edit_symbol(name, &pos);
369 sym.isPublic = true;
370 return NO_ERROR;
371 }
372
appendComment(const String8 & name,const String16 & comment,const SourcePos & pos)373 void appendComment(const String8& name, const String16& comment, const SourcePos& pos) {
374 if (comment.size() <= 0) {
375 return;
376 }
377 AaptSymbolEntry& sym = edit_symbol(name, &pos);
378 if (sym.comment.size() == 0) {
379 sym.comment = comment;
380 } else {
381 sym.comment.append(String16("\n"));
382 sym.comment.append(comment);
383 }
384 }
385
appendTypeComment(const String8 & name,const String16 & comment)386 void appendTypeComment(const String8& name, const String16& comment) {
387 if (comment.size() <= 0) {
388 return;
389 }
390 AaptSymbolEntry& sym = edit_symbol(name, NULL);
391 if (sym.typeComment.size() == 0) {
392 sym.typeComment = comment;
393 } else {
394 sym.typeComment.append(String16("\n"));
395 sym.typeComment.append(comment);
396 }
397 }
398
addNestedSymbol(const String8 & name,const SourcePos & pos)399 sp<AaptSymbols> addNestedSymbol(const String8& name, const SourcePos& pos) {
400 if (!check_valid_symbol_name(name, pos, "nested symbol")) {
401 return NULL;
402 }
403
404 sp<AaptSymbols> sym = mNestedSymbols.valueFor(name);
405 if (sym == NULL) {
406 sym = new AaptSymbols();
407 mNestedSymbols.add(name, sym);
408 }
409
410 return sym;
411 }
412
getSymbols()413 const KeyedVector<String8, AaptSymbolEntry>& getSymbols() const
414 { return mSymbols; }
getNestedSymbols()415 const DefaultKeyedVector<String8, sp<AaptSymbols> >& getNestedSymbols() const
416 { return mNestedSymbols; }
417
getComment(const String8 & name)418 const String16& getComment(const String8& name) const
419 { return get_symbol(name).comment; }
getTypeComment(const String8 & name)420 const String16& getTypeComment(const String8& name) const
421 { return get_symbol(name).typeComment; }
422
423 private:
check_valid_symbol_name(const String8 & symbol,const SourcePos & pos,const char * label)424 bool check_valid_symbol_name(const String8& symbol, const SourcePos& pos, const char* label) {
425 if (valid_symbol_name(symbol)) {
426 return true;
427 }
428 pos.error("invalid %s: '%s'\n", label, symbol.string());
429 return false;
430 }
edit_symbol(const String8 & symbol,const SourcePos * pos)431 AaptSymbolEntry& edit_symbol(const String8& symbol, const SourcePos* pos) {
432 ssize_t i = mSymbols.indexOfKey(symbol);
433 if (i < 0) {
434 i = mSymbols.add(symbol, AaptSymbolEntry(symbol));
435 }
436 AaptSymbolEntry& sym = mSymbols.editValueAt(i);
437 if (pos != NULL && sym.sourcePos.line < 0) {
438 sym.sourcePos = *pos;
439 }
440 return sym;
441 }
get_symbol(const String8 & symbol)442 const AaptSymbolEntry& get_symbol(const String8& symbol) const {
443 ssize_t i = mSymbols.indexOfKey(symbol);
444 if (i >= 0) {
445 return mSymbols.valueAt(i);
446 }
447 return mDefSymbol;
448 }
449
450 KeyedVector<String8, AaptSymbolEntry> mSymbols;
451 DefaultKeyedVector<String8, sp<AaptSymbols> > mNestedSymbols;
452 AaptSymbolEntry mDefSymbol;
453 };
454
455 class ResourceTypeSet : public RefBase,
456 public KeyedVector<String8,sp<AaptGroup> >
457 {
458 public:
459 ResourceTypeSet();
460 };
461
462
463 /**
464 * Asset hierarchy being operated on.
465 */
466 class AaptAssets : public AaptDir
467 {
468 public:
AaptAssets()469 AaptAssets() : AaptDir(String8(), String8()), mHaveIncludedAssets(false), mRes(NULL) { }
~AaptAssets()470 virtual ~AaptAssets() { delete mRes; }
471
getPackage()472 const String8& getPackage() const { return mPackage; }
setPackage(const String8 & package)473 void setPackage(const String8& package) { mPackage = package; mSymbolsPrivatePackage = package; }
474
getGroupEntries()475 const SortedVector<AaptGroupEntry>& getGroupEntries() const { return mGroupEntries; }
476
477 sp<AaptFile> addFile(const String8& filePath,
478 const AaptGroupEntry& entry,
479 const String8& srcDir,
480 sp<AaptGroup>* outGroup,
481 const String8& resType);
482
483 void addResource(const String8& leafName,
484 const String8& path,
485 const sp<AaptFile>& file,
486 const String8& resType);
487
addGroupEntry(const AaptGroupEntry & entry)488 void addGroupEntry(const AaptGroupEntry& entry) { mGroupEntries.add(entry); }
489
490 ssize_t slurpFromArgs(Bundle* bundle);
491
492 virtual ssize_t slurpFullTree(Bundle* bundle,
493 const String8& srcDir,
494 const AaptGroupEntry& kind,
495 const String8& resType);
496
497 ssize_t slurpResourceTree(Bundle* bundle, const String8& srcDir);
498 ssize_t slurpResourceZip(Bundle* bundle, const char* filename);
499
500 sp<AaptSymbols> getSymbolsFor(const String8& name);
501
getSymbols()502 const DefaultKeyedVector<String8, sp<AaptSymbols> >& getSymbols() const { return mSymbols; }
503
getSymbolsPrivatePackage()504 String8 getSymbolsPrivatePackage() const { return mSymbolsPrivatePackage; }
setSymbolsPrivatePackage(const String8 & pkg)505 void setSymbolsPrivatePackage(const String8& pkg) { mSymbolsPrivatePackage = pkg; }
506
507 status_t buildIncludedResources(Bundle* bundle);
508 status_t addIncludedResources(const sp<AaptFile>& file);
509 const ResTable& getIncludedResources() const;
510
511 void print() const;
512
resDirs()513 inline const Vector<sp<AaptDir> >& resDirs() { return mDirs; }
514 sp<AaptDir> resDir(const String8& name);
515
getOverlay()516 inline sp<AaptAssets> getOverlay() { return mOverlay; }
setOverlay(sp<AaptAssets> & overlay)517 inline void setOverlay(sp<AaptAssets>& overlay) { mOverlay = overlay; }
518
getResources()519 inline KeyedVector<String8, sp<ResourceTypeSet> >* getResources() { return mRes; }
520 inline void
setResources(KeyedVector<String8,sp<ResourceTypeSet>> * res)521 setResources(KeyedVector<String8, sp<ResourceTypeSet> >* res) { delete mRes; mRes = res; }
522
523 private:
524 String8 mPackage;
525 SortedVector<AaptGroupEntry> mGroupEntries;
526 DefaultKeyedVector<String8, sp<AaptSymbols> > mSymbols;
527 String8 mSymbolsPrivatePackage;
528
529 Vector<sp<AaptDir> > mDirs;
530
531 bool mHaveIncludedAssets;
532 AssetManager mIncludedAssets;
533
534 sp<AaptAssets> mOverlay;
535 KeyedVector<String8, sp<ResourceTypeSet> >* mRes;
536 };
537
538 #endif // __AAPT_ASSETS_H
539
540