/* * Copyright (C) 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ //#define LOG_NDEBUG 0 #define LOG_TAG "MidiIoWrapper" #include #include #include #include #include #include #include static int readAt(void *handle, void *buffer, int pos, int size) { return ((android::MidiIoWrapper*)handle)->readAt(buffer, pos, size); } static int size(void *handle) { return ((android::MidiIoWrapper*)handle)->size(); } namespace android { int MidiIoWrapper::sCacheBufferSize = 0; Mutex MidiIoWrapper::mCacheLock; MidiIoWrapper::MidiIoWrapper(const char *path) { ALOGV("MidiIoWrapper(%s)", path); mFd = open(path, O_RDONLY | O_LARGEFILE); mBase = 0; mLength = lseek(mFd, 0, SEEK_END); mDataSource = nullptr; mCacheBuffer = NULL; mCacheBufRangeLength = 0; } MidiIoWrapper::MidiIoWrapper(int fd, off64_t offset, int64_t size) { ALOGV("MidiIoWrapper(fd=%d)", fd); mFd = fd < 0 ? -1 : dup(fd); mBase = offset; mLength = size; mDataSource = nullptr; mCacheBuffer = NULL; mCacheBufRangeLength = 0; } class MidiIoWrapper::DataSourceUnwrapper { public: explicit DataSourceUnwrapper(CDataSource *csource) { mSource = csource; } virtual ~DataSourceUnwrapper() {} virtual status_t initCheck() const { return OK; } // Returns the number of bytes read, or -1 on failure. It's not an error if // this returns zero; it just means the given offset is equal to, or // beyond, the end of the source. virtual ssize_t readAt(off64_t offset, void *data, size_t size) { return mSource->readAt(mSource->handle, offset, data, size); } // May return ERROR_UNSUPPORTED. virtual status_t getSize(off64_t *size) { return mSource->getSize(mSource->handle, size); } virtual bool getUri(char * /*uriString*/, size_t /*bufferSize*/) { return false; } virtual uint32_t flags() { return 0; } virtual void close() {}; private: CDataSource *mSource; }; MidiIoWrapper::MidiIoWrapper(CDataSource *csource) { ALOGV("MidiIoWrapper(CDataSource)"); mFd = -1; mBase = 0; mDataSource = new DataSourceUnwrapper(csource); off64_t l; if (mDataSource->getSize(&l) == OK) { mLength = l; } else { mLength = 0; } mCacheBuffer = NULL; mCacheBufRangeLength = 0; } MidiIoWrapper::~MidiIoWrapper() { ALOGV("~MidiIoWrapper"); if (mFd >= 0) { close(mFd); } delete mDataSource; if (NULL != mCacheBuffer) { delete [] mCacheBuffer; mCacheBuffer = NULL; { Mutex::Autolock _l(mCacheLock); sCacheBufferSize -= mLength; } } } int MidiIoWrapper::readAt(void *buffer, int offset, int size) { ALOGV("readAt(%p, %d, %d)", buffer, offset, size); if (offset < 0) { return UNKNOWN_ERROR; } if (offset + size > mLength) { size = mLength - offset; } if (mCacheBuffer == NULL) { Mutex::Autolock _l(mCacheLock); if (sCacheBufferSize + mLength <= kTotalCacheSize) { mCacheBuffer = new (std::nothrow) unsigned char[mLength]; if (NULL != mCacheBuffer) { sCacheBufferSize += mLength; ALOGV("sCacheBufferSize : %d", sCacheBufferSize); } else { ALOGE("failed to allocate memory for mCacheBuffer"); } } else { ALOGV("not allocate memory for mCacheBuffer"); } } if (mCacheBuffer != NULL) { if (mCacheBufRangeLength > 0 && mCacheBufRangeLength >= (offset + size)) { /* Use buffered data */ memcpy(buffer, (void*)(mCacheBuffer + offset), size); return size; } else { /* Buffer new data */ int64_t beyondCacheBufRangeLength = (offset + size) - mCacheBufRangeLength; int64_t numRequiredBytesToCache = std::max((int64_t)kSingleCacheSize, beyondCacheBufRangeLength); int64_t availableReadLength = mLength - mCacheBufRangeLength; int64_t readSize = std::min(availableReadLength, numRequiredBytesToCache); int actualNumBytesRead = unbufferedReadAt(mCacheBuffer + mCacheBufRangeLength, mCacheBufRangeLength, readSize); if(actualNumBytesRead > 0) { mCacheBufRangeLength += actualNumBytesRead; if (offset >= mCacheBufRangeLength) { return 0; } else if (offset + size >= mCacheBufRangeLength) { memcpy(buffer, (void*)(mCacheBuffer + offset), mCacheBufRangeLength - offset); return mCacheBufRangeLength - offset; } else { memcpy(buffer, (void*)(mCacheBuffer + offset), size); return size; } } else { return actualNumBytesRead; } } } else { return unbufferedReadAt(buffer, offset, size); } } int MidiIoWrapper::unbufferedReadAt(void *buffer, int offset, int size) { ALOGV("unbufferedReadAt(%p, %d, %d)", buffer, offset, size); if (mDataSource != NULL) { return mDataSource->readAt(offset, buffer, size); } if (mFd < 0) { errno = EBADF; return -1; // as per failed read. } lseek(mFd, mBase + offset, SEEK_SET); if (offset + size > mLength) { size = mLength - offset; } return read(mFd, buffer, size); } int MidiIoWrapper::size() { ALOGV("size() = %d", int(mLength)); return mLength; } EAS_FILE_LOCATOR MidiIoWrapper::getLocator() { mEasFile.handle = this; mEasFile.readAt = ::readAt; mEasFile.size = ::size; return &mEasFile; } } // namespace android