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