Package.cpp revision 8a39da80b33691b0c82458c3b7727e13ff71277e
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// Copyright 2006 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// Package assets into Zip files. 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project// 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "Main.h" 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "AaptAssets.h" 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "ResourceTable.h" 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 103b4062eeb01be33951ba214e027da523cf09f8b1Mathias Agopian#include <utils/Log.h> 113b4062eeb01be33951ba214e027da523cf09f8b1Mathias Agopian#include <utils/threads.h> 123b4062eeb01be33951ba214e027da523cf09f8b1Mathias Agopian#include <utils/List.h> 133b4062eeb01be33951ba214e027da523cf09f8b1Mathias Agopian#include <utils/Errors.h> 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <sys/types.h> 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <dirent.h> 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <ctype.h> 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <errno.h> 199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectusing namespace android; 219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const char* kExcludeExtension = ".EXCLUDE"; 239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* these formats are already compressed, or don't compress well */ 259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic const char* kNoCompressExt[] = { 269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".jpg", ".jpeg", ".png", ".gif", 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".wav", ".mp2", ".mp3", ".ogg", ".aac", 289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".mpg", ".mpeg", ".mid", ".midi", ".smf", ".jet", 299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".rtttl", ".imy", ".xmf", ".mp4", ".m4a", 309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".m4v", ".3gp", ".3gpp", ".3g2", ".3gpp2", 319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ".amr", ".awb", ".wma", ".wmv" 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}; 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* fwd decls, so I can write this downward */ 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processAssets(Bundle* bundle, ZipFile* zip, const sp<AaptAssets>& assets); 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processAssets(Bundle* bundle, ZipFile* zip, 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const sp<AaptDir>& dir, const AaptGroupEntry& ge); 389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool processFile(Bundle* bundle, ZipFile* zip, 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const sp<AaptGroup>& group, const sp<AaptFile>& file); 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool okayToCompress(Bundle* bundle, const String8& pathName); 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processJarFiles(Bundle* bundle, ZipFile* zip); 429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The directory hierarchy looks like this: 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * "outputDir" and "assetRoot" are existing directories. 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * On success, "bundle->numPackages" will be the number of Zip packages 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * we created. 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatus_t writeAPK(Bundle* bundle, const sp<AaptAssets>& assets, 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const String8& outputFile) 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 538a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin #if BENCHMARK 548a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin fprintf(stdout, "BENCHMARK: Starting APK Bundling \n"); 558a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin long startAPKTime = clock(); 568a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin #endif /* BENCHMARK */ 578a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin 589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t result = NO_ERROR; 599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ZipFile* zip = NULL; 609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count; 619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //bundle->setPackageCount(0); 639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* 659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Prep the Zip archive. 669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * If the file already exists, fail unless "update" or "force" is set. 689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * If "update" is set, update the contents of the existing archive. 699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Else, if "force" is set, remove the existing archive. 709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project FileType fileType = getFileType(outputFile.string()); 729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fileType == kFileTypeNonexistent) { 739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // okay, create it below 749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (fileType == kFileTypeRegular) { 759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getUpdate()) { 769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // okay, open it below 779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (bundle->getForce()) { 789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unlink(outputFile.string()) != 0) { 799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to remove '%s': %s\n", outputFile.string(), 809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project strerror(errno)); 819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: '%s' exists (use '-f' to force overwrite)\n", 859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project outputFile.string()); 869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: '%s' exists and is not a regular file\n", outputFile.string()); 909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("%s '%s'\n", (fileType == kFileTypeNonexistent) ? "Creating" : "Opening", 959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project outputFile.string()); 969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t status; 999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project zip = new ZipFile; 1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status = zip->open(outputFile.string(), ZipFile::kOpenReadWrite | ZipFile::kOpenCreate); 1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (status != NO_ERROR) { 1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to open '%s' as Zip file for writing\n", 1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project outputFile.string()); 1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Writing all files...\n"); 1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count = processAssets(bundle, zip, assets); 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count < 0) { 1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to process assets while packaging '%s'\n", 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project outputFile.string()); 1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = count; 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Generated %d file%s\n", count, (count==1) ? "" : "s"); 1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count = processJarFiles(bundle, zip); 1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (count < 0) { 1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to process jar files while packaging '%s'\n", 1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project outputFile.string()); 1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = count; 1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) 1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Included %d file%s from jar/zip files.\n", count, (count==1) ? "" : "s"); 1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = NO_ERROR; 1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* 1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Check for cruft. We set the "marked" flag on all entries we created 1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * or decided not to update. If the entry isn't already slated for 1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * deletion, remove it now. 1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project { 1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) 1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Checking for deleted files\n"); 1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int i, removed = 0; 1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i = 0; i < zip->getNumEntries(); i++) { 1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ZipEntry* entry = zip->getEntryByIndex(i); 1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!entry->getMarked() && entry->getDeleted()) { 1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" (removing crufty '%s')\n", 1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project entry->getFileName()); 1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project zip->remove(entry); 1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project removed++; 1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose() && removed > 0) 1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Removed %d file%s\n", removed, (removed==1) ? "" : "s"); 1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* tell Zip lib to process deletions and other pending changes */ 1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = zip->flush(); 1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result != NO_ERROR) { 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: Zip flush failed, archive may be hosed\n"); 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project goto bail; 1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* anything here? */ 1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (zip->getNumEntries() == 0) { 1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Archive is empty -- removing %s\n", outputFile.getPathLeaf().string()); 1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project delete zip; // close the file so we can remove it in Win32 1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project zip = NULL; 1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unlink(outputFile.string()) != 0) { 176dd931864209eac0b4182d7a0d1ca965fcc3b8c03Marco Nelissen fprintf(stderr, "warning: could not unlink '%s'\n", outputFile.string()); 1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 18003589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin if (bundle->getGenDependencies()) { 18103589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin // Add this file to the dependency file 18203589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin String8 dependencyFile = outputFile.getBasePath(); 18303589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin dependencyFile.append(".d"); 18403589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin 18503589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin FILE* fp = fopen(dependencyFile.string(), "a"); 18603589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin fprintf(fp, "%s \\\n", outputFile.string()); 18703589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin fclose(fp); 18803589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin } 18903589cc65355220e0a4a0c816189a9fa25cc81fcJosiah Gaskin 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project assert(result == NO_ERROR); 1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbail: 1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project delete zip; // must close before remove in Win32 1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result != NO_ERROR) { 1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Removing %s due to earlier failures\n", outputFile.string()); 1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (unlink(outputFile.string()) != 0) { 199dd931864209eac0b4182d7a0d1ca965fcc3b8c03Marco Nelissen fprintf(stderr, "warning: could not unlink '%s'\n", outputFile.string()); 2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result == NO_ERROR && bundle->getVerbose()) 2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf("Done!\n"); 2058a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin 2068a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin #if BENCHMARK 2078a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin fprintf(stdout, "BENCHMARK: End APK Bundling. Time Elapsed: %f ms \n",(clock() - startAPKTime)/1000.0); 2088a39da80b33691b0c82458c3b7727e13ff71277eJosiah Gaskin #endif /* BENCHMARK */ 2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return result; 2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processAssets(Bundle* bundle, ZipFile* zip, 2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const sp<AaptAssets>& assets) 2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ResourceFilter filter; 2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t status = filter.parse(bundle->getConfigurations()); 2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (status != NO_ERROR) { 2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t count = 0; 2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t N = assets->getGroupEntries().size(); 2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (size_t i=0; i<N; i++) { 2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const AaptGroupEntry& ge = assets->getGroupEntries()[i]; 2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!filter.match(ge.toParams())) { 2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t res = processAssets(bundle, zip, assets, ge); 2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (res < 0) { 2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return res; 2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count += res; 2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processAssets(Bundle* bundle, ZipFile* zip, 2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const sp<AaptDir>& dir, const AaptGroupEntry& ge) 2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t count = 0; 2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t ND = dir->getDirs().size(); 2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t i; 2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i=0; i<ND; i++) { 2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t res = processAssets(bundle, zip, dir->getDirs().valueAt(i), ge); 2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (res < 0) { 2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return res; 2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count += res; 2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const size_t NF = dir->getFiles().size(); 2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i=0; i<NF; i++) { 2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sp<AaptGroup> gp = dir->getFiles().valueAt(i); 2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t fi = gp->getFiles().indexOfKey(ge); 2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fi >= 0) { 2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sp<AaptFile> fl = gp->getFiles().valueAt(fi); 2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!processFile(bundle, zip, gp, fl)) { 2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return UNKNOWN_ERROR; 2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count++; 2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Process a regular file, adding it to the archive if appropriate. 2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * If we're in "update" mode, and the file already exists in the archive, 2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * delete the existing entry before adding the new one. 2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool processFile(Bundle* bundle, ZipFile* zip, 2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const sp<AaptGroup>& group, const sp<AaptFile>& file) 2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const bool hasData = file->hasData(); 2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 storageName(group->getPath()); 2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project storageName.convertToResPath(); 2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ZipEntry* entry; 2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project bool fromGzip = false; 2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t result; 2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* 2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See if the filename ends in ".EXCLUDE". We can't use 2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * String8::getPathExtension() because the length of what it considers 2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * to be an extension is capped. 2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The Asset Manager doesn't check for ".EXCLUDE" in Zip archives, 2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * so there's no value in adding them (and it makes life easier on 2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * the AssetManager lib if we don't). 2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * NOTE: this restriction has been removed. If you're in this code, you 2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * should clean this up, but I'm in here getting rid of Path Name, and I 2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * don't want to make other potentially breaking changes --joeo 2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int fileNameLen = storageName.length(); 3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int excludeExtensionLen = strlen(kExcludeExtension); 3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fileNameLen > excludeExtensionLen 3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project && (0 == strcmp(storageName.string() + (fileNameLen - excludeExtensionLen), 3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project kExcludeExtension))) { 305dd931864209eac0b4182d7a0d1ca965fcc3b8c03Marco Nelissen fprintf(stderr, "warning: '%s' not added to Zip\n", storageName.string()); 3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (strcasecmp(storageName.getPathExtension().string(), ".gz") == 0) { 3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fromGzip = true; 3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project storageName = storageName.getBasePath(); 3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getUpdate()) { 3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project entry = zip->getEntryByName(storageName.string()); 3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (entry != NULL) { 3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* file already exists in archive; there can be only one */ 3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (entry->getMarked()) { 3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, 3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project "ERROR: '%s' exists twice (check for with & w/o '.gz'?)\n", 3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project file->getPrintableSource().string()); 3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!hasData) { 3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const String8& srcName = file->getSourceFile(); 3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project time_t fileModWhen; 3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fileModWhen = getFileModDate(srcName.string()); 3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fileModWhen == (time_t) -1) { // file existence tested earlier, 3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; // not expecting an error here 3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fileModWhen > entry->getModWhen()) { 3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // mark as deleted so add() will succeed 3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" (removing old '%s')\n", storageName.string()); 3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project zip->remove(entry); 3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // version in archive is newer 3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" (not updating '%s')\n", storageName.string()); 3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project entry->setMarked(true); 3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Generated files are always replaced. 3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project zip->remove(entry); 3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //android_setMinPriority(NULL, ANDROID_LOG_VERBOSE); 3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fromGzip) { 3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = zip->addGzip(file->getSourceFile().string(), storageName.string(), &entry); 3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (!hasData) { 3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /* don't compress certain files, e.g. PNGs */ 3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int compressionMethod = bundle->getCompressionMethod(); 3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!okayToCompress(bundle, storageName)) { 3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project compressionMethod = ZipEntry::kCompressStored; 3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = zip->add(file->getSourceFile().string(), storageName.string(), compressionMethod, 3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project &entry); 3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project result = zip->add(file->getData(), file->getSize(), storageName.string(), 3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project file->getCompressionMethod(), &entry); 3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result == NO_ERROR) { 3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (bundle->getVerbose()) { 3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" '%s'%s", storageName.string(), fromGzip ? " (from .gz)" : ""); 3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (entry->getCompressionMethod() == ZipEntry::kCompressStored) { 3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" (not compressed)\n"); 3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project printf(" (compressed %d%%)\n", calcPercent(entry->getUncompressedLen(), 3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project entry->getCompressedLen())); 3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project entry->setMarked(true); 3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (result == ALREADY_EXISTS) { 3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, " Unable to add '%s': file already in archive (try '-u'?)\n", 3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project file->getPrintableSource().string()); 3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, " Unable to add '%s': Zip add failed\n", 3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project file->getPrintableSource().string()); 3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Determine whether or not we want to try to compress this file based 3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * on the file extension. 3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool okayToCompress(Bundle* bundle, const String8& pathName) 4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String8 ext = pathName.getPathExtension(); 4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int i; 4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (ext.length() == 0) 4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i = 0; i < NELEM(kNoCompressExt); i++) { 4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (strcasecmp(ext.string(), kNoCompressExt[i]) == 0) 4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const android::Vector<const char*>& others(bundle->getNoCompressExtensions()); 4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (i = 0; i < (int)others.size(); i++) { 4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* str = others[i]; 4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int pos = pathName.length() - strlen(str); 4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (pos < 0) { 4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* path = pathName.string(); 4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (strcasecmp(path + pos, str) == 0) { 4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectbool endsWith(const char* haystack, const char* needle) 4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t a = strlen(haystack); 4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t b = strlen(needle); 4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (a < b) return false; 4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return strcasecmp(haystack+(a-b), needle) == 0; 4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processJarFile(ZipFile* jar, ZipFile* out) 4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project status_t err; 4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t N = jar->getNumEntries(); 4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t count = 0; 4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (size_t i=0; i<N; i++) { 4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ZipEntry* entry = jar->getEntryByIndex(i); 4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const char* storageName = entry->getFileName(); 4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (endsWith(storageName, ".class")) { 4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int compressionMethod = entry->getCompressionMethod(); 4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t size = entry->getUncompressedLen(); 4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const void* data = jar->uncompress(entry); 4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (data == NULL) { 4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to uncompress entry '%s'\n", 4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project storageName); 4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return -1; 4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out->add(data, size, storageName, compressionMethod, NULL); 4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project free((void*)data); 4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count++; 4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectssize_t processJarFiles(Bundle* bundle, ZipFile* zip) 4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project{ 4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t err; 4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ssize_t count = 0; 4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project const android::Vector<const char*>& jars = bundle->getJarFiles(); 4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project size_t N = jars.size(); 4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (size_t i=0; i<N; i++) { 4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ZipFile jar; 4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project err = jar.open(jars[i], ZipFile::kOpenReadOnly); 4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (err != 0) { 4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to open '%s' as a zip file: %zd\n", 4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project jars[i], err); 4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return err; 4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project err += processJarFile(&jar, zip); 4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (err < 0) { 4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fprintf(stderr, "ERROR: unable to process '%s'\n", jars[i]); 4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return err; 4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project count += err; 4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return count; 4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 486