1 /* 2 * Copyright (C) 2006 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 // 18 // Provide access to a read-only asset. 19 // 20 21 #define LOG_TAG "asset" 22 //#define NDEBUG 0 23 24 #include <androidfw/Asset.h> 25 #include <androidfw/StreamingZipInflater.h> 26 #include <androidfw/Util.h> 27 #include <androidfw/ZipFileRO.h> 28 #include <androidfw/ZipUtils.h> 29 #include <cutils/atomic.h> 30 #include <utils/FileMap.h> 31 #include <utils/Log.h> 32 #include <utils/threads.h> 33 34 #include <assert.h> 35 #include <errno.h> 36 #include <fcntl.h> 37 #include <memory.h> 38 #include <string.h> 39 #include <sys/stat.h> 40 #include <sys/types.h> 41 #include <unistd.h> 42 43 using namespace android; 44 45 #ifndef O_BINARY 46 # define O_BINARY 0 47 #endif 48 49 static const bool kIsDebug = false; 50 51 static Mutex gAssetLock; 52 static int32_t gCount = 0; 53 static Asset* gHead = NULL; 54 static Asset* gTail = NULL; 55 56 void Asset::registerAsset(Asset* asset) 57 { 58 AutoMutex _l(gAssetLock); 59 gCount++; 60 asset->mNext = asset->mPrev = NULL; 61 if (gTail == NULL) { 62 gHead = gTail = asset; 63 } else { 64 asset->mPrev = gTail; 65 gTail->mNext = asset; 66 gTail = asset; 67 } 68 69 if (kIsDebug) { 70 ALOGI("Creating Asset %p #%d\n", asset, gCount); 71 } 72 } 73 74 void Asset::unregisterAsset(Asset* asset) 75 { 76 AutoMutex _l(gAssetLock); 77 gCount--; 78 if (gHead == asset) { 79 gHead = asset->mNext; 80 } 81 if (gTail == asset) { 82 gTail = asset->mPrev; 83 } 84 if (asset->mNext != NULL) { 85 asset->mNext->mPrev = asset->mPrev; 86 } 87 if (asset->mPrev != NULL) { 88 asset->mPrev->mNext = asset->mNext; 89 } 90 asset->mNext = asset->mPrev = NULL; 91 92 if (kIsDebug) { 93 ALOGI("Destroying Asset in %p #%d\n", asset, gCount); 94 } 95 } 96 97 int32_t Asset::getGlobalCount() 98 { 99 AutoMutex _l(gAssetLock); 100 return gCount; 101 } 102 103 String8 Asset::getAssetAllocations() 104 { 105 AutoMutex _l(gAssetLock); 106 String8 res; 107 Asset* cur = gHead; 108 while (cur != NULL) { 109 if (cur->isAllocated()) { 110 res.append(" "); 111 res.append(cur->getAssetSource()); 112 off64_t size = (cur->getLength()+512)/1024; 113 char buf[64]; 114 snprintf(buf, sizeof(buf), ": %dK\n", (int)size); 115 res.append(buf); 116 } 117 cur = cur->mNext; 118 } 119 120 return res; 121 } 122 123 Asset::Asset(void) 124 : mAccessMode(ACCESS_UNKNOWN), mNext(NULL), mPrev(NULL) 125 { 126 } 127 128 /* 129 * Create a new Asset from a file on disk. There is a fair chance that 130 * the file doesn't actually exist. 131 * 132 * We can use "mode" to decide how we want to go about it. 133 */ 134 /*static*/ Asset* Asset::createFromFile(const char* fileName, AccessMode mode) 135 { 136 _FileAsset* pAsset; 137 status_t result; 138 off64_t length; 139 int fd; 140 141 fd = open(fileName, O_RDONLY | O_BINARY); 142 if (fd < 0) 143 return NULL; 144 145 /* 146 * Under Linux, the lseek fails if we actually opened a directory. To 147 * be correct we should test the file type explicitly, but since we 148 * always open things read-only it doesn't really matter, so there's 149 * no value in incurring the extra overhead of an fstat() call. 150 */ 151 // TODO(kroot): replace this with fstat despite the plea above. 152 #if 1 153 length = lseek64(fd, 0, SEEK_END); 154 if (length < 0) { 155 ::close(fd); 156 return NULL; 157 } 158 (void) lseek64(fd, 0, SEEK_SET); 159 #else 160 struct stat st; 161 if (fstat(fd, &st) < 0) { 162 ::close(fd); 163 return NULL; 164 } 165 166 if (!S_ISREG(st.st_mode)) { 167 ::close(fd); 168 return NULL; 169 } 170 #endif 171 172 pAsset = new _FileAsset; 173 result = pAsset->openChunk(fileName, fd, 0, length); 174 if (result != NO_ERROR) { 175 delete pAsset; 176 return NULL; 177 } 178 179 pAsset->mAccessMode = mode; 180 return pAsset; 181 } 182 183 184 /* 185 * Create a new Asset from a compressed file on disk. There is a fair chance 186 * that the file doesn't actually exist. 187 * 188 * We currently support gzip files. We might want to handle .bz2 someday. 189 */ 190 /*static*/ Asset* Asset::createFromCompressedFile(const char* fileName, 191 AccessMode mode) 192 { 193 _CompressedAsset* pAsset; 194 status_t result; 195 off64_t fileLen; 196 bool scanResult; 197 long offset; 198 int method; 199 long uncompressedLen, compressedLen; 200 int fd; 201 202 fd = open(fileName, O_RDONLY | O_BINARY); 203 if (fd < 0) 204 return NULL; 205 206 fileLen = lseek(fd, 0, SEEK_END); 207 if (fileLen < 0) { 208 ::close(fd); 209 return NULL; 210 } 211 (void) lseek(fd, 0, SEEK_SET); 212 213 /* want buffered I/O for the file scan; must dup so fclose() is safe */ 214 FILE* fp = fdopen(dup(fd), "rb"); 215 if (fp == NULL) { 216 ::close(fd); 217 return NULL; 218 } 219 220 unsigned long crc32; 221 scanResult = ZipUtils::examineGzip(fp, &method, &uncompressedLen, 222 &compressedLen, &crc32); 223 offset = ftell(fp); 224 fclose(fp); 225 if (!scanResult) { 226 ALOGD("File '%s' is not in gzip format\n", fileName); 227 ::close(fd); 228 return NULL; 229 } 230 231 pAsset = new _CompressedAsset; 232 result = pAsset->openChunk(fd, offset, method, uncompressedLen, 233 compressedLen); 234 if (result != NO_ERROR) { 235 delete pAsset; 236 return NULL; 237 } 238 239 pAsset->mAccessMode = mode; 240 return pAsset; 241 } 242 243 244 #if 0 245 /* 246 * Create a new Asset from part of an open file. 247 */ 248 /*static*/ Asset* Asset::createFromFileSegment(int fd, off64_t offset, 249 size_t length, AccessMode mode) 250 { 251 _FileAsset* pAsset; 252 status_t result; 253 254 pAsset = new _FileAsset; 255 result = pAsset->openChunk(NULL, fd, offset, length); 256 if (result != NO_ERROR) 257 return NULL; 258 259 pAsset->mAccessMode = mode; 260 return pAsset; 261 } 262 263 /* 264 * Create a new Asset from compressed data in an open file. 265 */ 266 /*static*/ Asset* Asset::createFromCompressedData(int fd, off64_t offset, 267 int compressionMethod, size_t uncompressedLen, size_t compressedLen, 268 AccessMode mode) 269 { 270 _CompressedAsset* pAsset; 271 status_t result; 272 273 pAsset = new _CompressedAsset; 274 result = pAsset->openChunk(fd, offset, compressionMethod, 275 uncompressedLen, compressedLen); 276 if (result != NO_ERROR) 277 return NULL; 278 279 pAsset->mAccessMode = mode; 280 return pAsset; 281 } 282 #endif 283 284 /* 285 * Create a new Asset from a memory mapping. 286 */ 287 /*static*/ Asset* Asset::createFromUncompressedMap(FileMap* dataMap, 288 AccessMode mode) 289 { 290 _FileAsset* pAsset; 291 status_t result; 292 293 pAsset = new _FileAsset; 294 result = pAsset->openChunk(dataMap); 295 if (result != NO_ERROR) { 296 delete pAsset; 297 return NULL; 298 } 299 300 pAsset->mAccessMode = mode; 301 return pAsset; 302 } 303 304 /*static*/ std::unique_ptr<Asset> Asset::createFromUncompressedMap(std::unique_ptr<FileMap> dataMap, 305 AccessMode mode) 306 { 307 std::unique_ptr<_FileAsset> pAsset = util::make_unique<_FileAsset>(); 308 309 status_t result = pAsset->openChunk(dataMap.get()); 310 if (result != NO_ERROR) { 311 return NULL; 312 } 313 314 // We succeeded, so relinquish control of dataMap 315 (void) dataMap.release(); 316 pAsset->mAccessMode = mode; 317 return std::move(pAsset); 318 } 319 320 /* 321 * Create a new Asset from compressed data in a memory mapping. 322 */ 323 /*static*/ Asset* Asset::createFromCompressedMap(FileMap* dataMap, 324 size_t uncompressedLen, AccessMode mode) 325 { 326 _CompressedAsset* pAsset; 327 status_t result; 328 329 pAsset = new _CompressedAsset; 330 result = pAsset->openChunk(dataMap, uncompressedLen); 331 if (result != NO_ERROR) 332 return NULL; 333 334 pAsset->mAccessMode = mode; 335 return pAsset; 336 } 337 338 /*static*/ std::unique_ptr<Asset> Asset::createFromCompressedMap(std::unique_ptr<FileMap> dataMap, 339 size_t uncompressedLen, AccessMode mode) 340 { 341 std::unique_ptr<_CompressedAsset> pAsset = util::make_unique<_CompressedAsset>(); 342 343 status_t result = pAsset->openChunk(dataMap.get(), uncompressedLen); 344 if (result != NO_ERROR) { 345 return NULL; 346 } 347 348 // We succeeded, so relinquish control of dataMap 349 (void) dataMap.release(); 350 pAsset->mAccessMode = mode; 351 return std::move(pAsset); 352 } 353 354 /* 355 * Do generic seek() housekeeping. Pass in the offset/whence values from 356 * the seek request, along with the current chunk offset and the chunk 357 * length. 358 * 359 * Returns the new chunk offset, or -1 if the seek is illegal. 360 */ 361 off64_t Asset::handleSeek(off64_t offset, int whence, off64_t curPosn, off64_t maxPosn) 362 { 363 off64_t newOffset; 364 365 switch (whence) { 366 case SEEK_SET: 367 newOffset = offset; 368 break; 369 case SEEK_CUR: 370 newOffset = curPosn + offset; 371 break; 372 case SEEK_END: 373 newOffset = maxPosn + offset; 374 break; 375 default: 376 ALOGW("unexpected whence %d\n", whence); 377 // this was happening due to an off64_t size mismatch 378 assert(false); 379 return (off64_t) -1; 380 } 381 382 if (newOffset < 0 || newOffset > maxPosn) { 383 ALOGW("seek out of range: want %ld, end=%ld\n", 384 (long) newOffset, (long) maxPosn); 385 return (off64_t) -1; 386 } 387 388 return newOffset; 389 } 390 391 392 /* 393 * =========================================================================== 394 * _FileAsset 395 * =========================================================================== 396 */ 397 398 /* 399 * Constructor. 400 */ 401 _FileAsset::_FileAsset(void) 402 : mStart(0), mLength(0), mOffset(0), mFp(NULL), mFileName(NULL), mMap(NULL), mBuf(NULL) 403 { 404 // Register the Asset with the global list here after it is fully constructed and its 405 // vtable pointer points to this concrete type. b/31113965 406 registerAsset(this); 407 } 408 409 /* 410 * Destructor. Release resources. 411 */ 412 _FileAsset::~_FileAsset(void) 413 { 414 close(); 415 416 // Unregister the Asset from the global list here before it is destructed and while its vtable 417 // pointer still points to this concrete type. b/31113965 418 unregisterAsset(this); 419 } 420 421 /* 422 * Operate on a chunk of an uncompressed file. 423 * 424 * Zero-length chunks are allowed. 425 */ 426 status_t _FileAsset::openChunk(const char* fileName, int fd, off64_t offset, size_t length) 427 { 428 assert(mFp == NULL); // no reopen 429 assert(mMap == NULL); 430 assert(fd >= 0); 431 assert(offset >= 0); 432 433 /* 434 * Seek to end to get file length. 435 */ 436 off64_t fileLength; 437 fileLength = lseek64(fd, 0, SEEK_END); 438 if (fileLength == (off64_t) -1) { 439 // probably a bad file descriptor 440 ALOGD("failed lseek (errno=%d)\n", errno); 441 return UNKNOWN_ERROR; 442 } 443 444 if ((off64_t) (offset + length) > fileLength) { 445 ALOGD("start (%ld) + len (%ld) > end (%ld)\n", 446 (long) offset, (long) length, (long) fileLength); 447 return BAD_INDEX; 448 } 449 450 /* after fdopen, the fd will be closed on fclose() */ 451 mFp = fdopen(fd, "rb"); 452 if (mFp == NULL) 453 return UNKNOWN_ERROR; 454 455 mStart = offset; 456 mLength = length; 457 assert(mOffset == 0); 458 459 /* seek the FILE* to the start of chunk */ 460 if (fseek(mFp, mStart, SEEK_SET) != 0) { 461 assert(false); 462 } 463 464 mFileName = fileName != NULL ? strdup(fileName) : NULL; 465 466 return NO_ERROR; 467 } 468 469 /* 470 * Create the chunk from the map. 471 */ 472 status_t _FileAsset::openChunk(FileMap* dataMap) 473 { 474 assert(mFp == NULL); // no reopen 475 assert(mMap == NULL); 476 assert(dataMap != NULL); 477 478 mMap = dataMap; 479 mStart = -1; // not used 480 mLength = dataMap->getDataLength(); 481 assert(mOffset == 0); 482 483 return NO_ERROR; 484 } 485 486 /* 487 * Read a chunk of data. 488 */ 489 ssize_t _FileAsset::read(void* buf, size_t count) 490 { 491 size_t maxLen; 492 size_t actual; 493 494 assert(mOffset >= 0 && mOffset <= mLength); 495 496 if (getAccessMode() == ACCESS_BUFFER) { 497 /* 498 * On first access, read or map the entire file. The caller has 499 * requested buffer access, either because they're going to be 500 * using the buffer or because what they're doing has appropriate 501 * performance needs and access patterns. 502 */ 503 if (mBuf == NULL) 504 getBuffer(false); 505 } 506 507 /* adjust count if we're near EOF */ 508 maxLen = mLength - mOffset; 509 if (count > maxLen) 510 count = maxLen; 511 512 if (!count) 513 return 0; 514 515 if (mMap != NULL) { 516 /* copy from mapped area */ 517 //printf("map read\n"); 518 memcpy(buf, (char*)mMap->getDataPtr() + mOffset, count); 519 actual = count; 520 } else if (mBuf != NULL) { 521 /* copy from buffer */ 522 //printf("buf read\n"); 523 memcpy(buf, (char*)mBuf + mOffset, count); 524 actual = count; 525 } else { 526 /* read from the file */ 527 //printf("file read\n"); 528 if (ftell(mFp) != mStart + mOffset) { 529 ALOGE("Hosed: %ld != %ld+%ld\n", 530 ftell(mFp), (long) mStart, (long) mOffset); 531 assert(false); 532 } 533 534 /* 535 * This returns 0 on error or eof. We need to use ferror() or feof() 536 * to tell the difference, but we don't currently have those on the 537 * device. However, we know how much data is *supposed* to be in the 538 * file, so if we don't read the full amount we know something is 539 * hosed. 540 */ 541 actual = fread(buf, 1, count, mFp); 542 if (actual == 0) // something failed -- I/O error? 543 return -1; 544 545 assert(actual == count); 546 } 547 548 mOffset += actual; 549 return actual; 550 } 551 552 /* 553 * Seek to a new position. 554 */ 555 off64_t _FileAsset::seek(off64_t offset, int whence) 556 { 557 off64_t newPosn; 558 off64_t actualOffset; 559 560 // compute new position within chunk 561 newPosn = handleSeek(offset, whence, mOffset, mLength); 562 if (newPosn == (off64_t) -1) 563 return newPosn; 564 565 actualOffset = mStart + newPosn; 566 567 if (mFp != NULL) { 568 if (fseek(mFp, (long) actualOffset, SEEK_SET) != 0) 569 return (off64_t) -1; 570 } 571 572 mOffset = actualOffset - mStart; 573 return mOffset; 574 } 575 576 /* 577 * Close the asset. 578 */ 579 void _FileAsset::close(void) 580 { 581 if (mMap != NULL) { 582 delete mMap; 583 mMap = NULL; 584 } 585 if (mBuf != NULL) { 586 delete[] mBuf; 587 mBuf = NULL; 588 } 589 590 if (mFileName != NULL) { 591 free(mFileName); 592 mFileName = NULL; 593 } 594 595 if (mFp != NULL) { 596 // can only be NULL when called from destructor 597 // (otherwise we would never return this object) 598 fclose(mFp); 599 mFp = NULL; 600 } 601 } 602 603 /* 604 * Return a read-only pointer to a buffer. 605 * 606 * We can either read the whole thing in or map the relevant piece of 607 * the source file. Ideally a map would be established at a higher 608 * level and we'd be using a different object, but we didn't, so we 609 * deal with it here. 610 */ 611 const void* _FileAsset::getBuffer(bool wordAligned) 612 { 613 /* subsequent requests just use what we did previously */ 614 if (mBuf != NULL) 615 return mBuf; 616 if (mMap != NULL) { 617 if (!wordAligned) { 618 return mMap->getDataPtr(); 619 } 620 return ensureAlignment(mMap); 621 } 622 623 assert(mFp != NULL); 624 625 if (mLength < kReadVsMapThreshold) { 626 unsigned char* buf; 627 long allocLen; 628 629 /* zero-length files are allowed; not sure about zero-len allocs */ 630 /* (works fine with gcc + x86linux) */ 631 allocLen = mLength; 632 if (mLength == 0) 633 allocLen = 1; 634 635 buf = new unsigned char[allocLen]; 636 if (buf == NULL) { 637 ALOGE("alloc of %ld bytes failed\n", (long) allocLen); 638 return NULL; 639 } 640 641 ALOGV("Asset %p allocating buffer size %d (smaller than threshold)", this, (int)allocLen); 642 if (mLength > 0) { 643 long oldPosn = ftell(mFp); 644 fseek(mFp, mStart, SEEK_SET); 645 if (fread(buf, 1, mLength, mFp) != (size_t) mLength) { 646 ALOGE("failed reading %ld bytes\n", (long) mLength); 647 delete[] buf; 648 return NULL; 649 } 650 fseek(mFp, oldPosn, SEEK_SET); 651 } 652 653 ALOGV(" getBuffer: loaded into buffer\n"); 654 655 mBuf = buf; 656 return mBuf; 657 } else { 658 FileMap* map; 659 660 map = new FileMap; 661 if (!map->create(NULL, fileno(mFp), mStart, mLength, true)) { 662 delete map; 663 return NULL; 664 } 665 666 ALOGV(" getBuffer: mapped\n"); 667 668 mMap = map; 669 if (!wordAligned) { 670 return mMap->getDataPtr(); 671 } 672 return ensureAlignment(mMap); 673 } 674 } 675 676 int _FileAsset::openFileDescriptor(off64_t* outStart, off64_t* outLength) const 677 { 678 if (mMap != NULL) { 679 const char* fname = mMap->getFileName(); 680 if (fname == NULL) { 681 fname = mFileName; 682 } 683 if (fname == NULL) { 684 return -1; 685 } 686 *outStart = mMap->getDataOffset(); 687 *outLength = mMap->getDataLength(); 688 return open(fname, O_RDONLY | O_BINARY); 689 } 690 if (mFileName == NULL) { 691 return -1; 692 } 693 *outStart = mStart; 694 *outLength = mLength; 695 return open(mFileName, O_RDONLY | O_BINARY); 696 } 697 698 const void* _FileAsset::ensureAlignment(FileMap* map) 699 { 700 void* data = map->getDataPtr(); 701 if ((((size_t)data)&0x3) == 0) { 702 // We can return this directly if it is aligned on a word 703 // boundary. 704 ALOGV("Returning aligned FileAsset %p (%s).", this, 705 getAssetSource()); 706 return data; 707 } 708 // If not aligned on a word boundary, then we need to copy it into 709 // our own buffer. 710 ALOGV("Copying FileAsset %p (%s) to buffer size %d to make it aligned.", this, 711 getAssetSource(), (int)mLength); 712 unsigned char* buf = new unsigned char[mLength]; 713 if (buf == NULL) { 714 ALOGE("alloc of %ld bytes failed\n", (long) mLength); 715 return NULL; 716 } 717 memcpy(buf, data, mLength); 718 mBuf = buf; 719 return buf; 720 } 721 722 /* 723 * =========================================================================== 724 * _CompressedAsset 725 * =========================================================================== 726 */ 727 728 /* 729 * Constructor. 730 */ 731 _CompressedAsset::_CompressedAsset(void) 732 : mStart(0), mCompressedLen(0), mUncompressedLen(0), mOffset(0), 733 mMap(NULL), mFd(-1), mZipInflater(NULL), mBuf(NULL) 734 { 735 // Register the Asset with the global list here after it is fully constructed and its 736 // vtable pointer points to this concrete type. b/31113965 737 registerAsset(this); 738 } 739 740 /* 741 * Destructor. Release resources. 742 */ 743 _CompressedAsset::~_CompressedAsset(void) 744 { 745 close(); 746 747 // Unregister the Asset from the global list here before it is destructed and while its vtable 748 // pointer still points to this concrete type. b/31113965 749 unregisterAsset(this); 750 } 751 752 /* 753 * Open a chunk of compressed data inside a file. 754 * 755 * This currently just sets up some values and returns. On the first 756 * read, we expand the entire file into a buffer and return data from it. 757 */ 758 status_t _CompressedAsset::openChunk(int fd, off64_t offset, 759 int compressionMethod, size_t uncompressedLen, size_t compressedLen) 760 { 761 assert(mFd < 0); // no re-open 762 assert(mMap == NULL); 763 assert(fd >= 0); 764 assert(offset >= 0); 765 assert(compressedLen > 0); 766 767 if (compressionMethod != ZipFileRO::kCompressDeflated) { 768 assert(false); 769 return UNKNOWN_ERROR; 770 } 771 772 mStart = offset; 773 mCompressedLen = compressedLen; 774 mUncompressedLen = uncompressedLen; 775 assert(mOffset == 0); 776 mFd = fd; 777 assert(mBuf == NULL); 778 779 if (uncompressedLen > StreamingZipInflater::OUTPUT_CHUNK_SIZE) { 780 mZipInflater = new StreamingZipInflater(mFd, offset, uncompressedLen, compressedLen); 781 } 782 783 return NO_ERROR; 784 } 785 786 /* 787 * Open a chunk of compressed data in a mapped region. 788 * 789 * Nothing is expanded until the first read call. 790 */ 791 status_t _CompressedAsset::openChunk(FileMap* dataMap, size_t uncompressedLen) 792 { 793 assert(mFd < 0); // no re-open 794 assert(mMap == NULL); 795 assert(dataMap != NULL); 796 797 mMap = dataMap; 798 mStart = -1; // not used 799 mCompressedLen = dataMap->getDataLength(); 800 mUncompressedLen = uncompressedLen; 801 assert(mOffset == 0); 802 803 if (uncompressedLen > StreamingZipInflater::OUTPUT_CHUNK_SIZE) { 804 mZipInflater = new StreamingZipInflater(dataMap, uncompressedLen); 805 } 806 return NO_ERROR; 807 } 808 809 /* 810 * Read data from a chunk of compressed data. 811 * 812 * [For now, that's just copying data out of a buffer.] 813 */ 814 ssize_t _CompressedAsset::read(void* buf, size_t count) 815 { 816 size_t maxLen; 817 size_t actual; 818 819 assert(mOffset >= 0 && mOffset <= mUncompressedLen); 820 821 /* If we're relying on a streaming inflater, go through that */ 822 if (mZipInflater) { 823 actual = mZipInflater->read(buf, count); 824 } else { 825 if (mBuf == NULL) { 826 if (getBuffer(false) == NULL) 827 return -1; 828 } 829 assert(mBuf != NULL); 830 831 /* adjust count if we're near EOF */ 832 maxLen = mUncompressedLen - mOffset; 833 if (count > maxLen) 834 count = maxLen; 835 836 if (!count) 837 return 0; 838 839 /* copy from buffer */ 840 //printf("comp buf read\n"); 841 memcpy(buf, (char*)mBuf + mOffset, count); 842 actual = count; 843 } 844 845 mOffset += actual; 846 return actual; 847 } 848 849 /* 850 * Handle a seek request. 851 * 852 * If we're working in a streaming mode, this is going to be fairly 853 * expensive, because it requires plowing through a bunch of compressed 854 * data. 855 */ 856 off64_t _CompressedAsset::seek(off64_t offset, int whence) 857 { 858 off64_t newPosn; 859 860 // compute new position within chunk 861 newPosn = handleSeek(offset, whence, mOffset, mUncompressedLen); 862 if (newPosn == (off64_t) -1) 863 return newPosn; 864 865 if (mZipInflater) { 866 mZipInflater->seekAbsolute(newPosn); 867 } 868 mOffset = newPosn; 869 return mOffset; 870 } 871 872 /* 873 * Close the asset. 874 */ 875 void _CompressedAsset::close(void) 876 { 877 if (mMap != NULL) { 878 delete mMap; 879 mMap = NULL; 880 } 881 882 delete[] mBuf; 883 mBuf = NULL; 884 885 delete mZipInflater; 886 mZipInflater = NULL; 887 888 if (mFd > 0) { 889 ::close(mFd); 890 mFd = -1; 891 } 892 } 893 894 /* 895 * Get a pointer to a read-only buffer of data. 896 * 897 * The first time this is called, we expand the compressed data into a 898 * buffer. 899 */ 900 const void* _CompressedAsset::getBuffer(bool) 901 { 902 unsigned char* buf = NULL; 903 904 if (mBuf != NULL) 905 return mBuf; 906 907 /* 908 * Allocate a buffer and read the file into it. 909 */ 910 buf = new unsigned char[mUncompressedLen]; 911 if (buf == NULL) { 912 ALOGW("alloc %ld bytes failed\n", (long) mUncompressedLen); 913 goto bail; 914 } 915 916 if (mMap != NULL) { 917 if (!ZipUtils::inflateToBuffer(mMap->getDataPtr(), buf, 918 mUncompressedLen, mCompressedLen)) 919 goto bail; 920 } else { 921 assert(mFd >= 0); 922 923 /* 924 * Seek to the start of the compressed data. 925 */ 926 if (lseek(mFd, mStart, SEEK_SET) != mStart) 927 goto bail; 928 929 /* 930 * Expand the data into it. 931 */ 932 if (!ZipUtils::inflateToBuffer(mFd, buf, mUncompressedLen, 933 mCompressedLen)) 934 goto bail; 935 } 936 937 /* 938 * Success - now that we have the full asset in RAM we 939 * no longer need the streaming inflater 940 */ 941 delete mZipInflater; 942 mZipInflater = NULL; 943 944 mBuf = buf; 945 buf = NULL; 946 947 bail: 948 delete[] buf; 949 return mBuf; 950 } 951 952