15f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian/*
25f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** Copyright 2011, The Android Open Source Project
35f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian **
45f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** Licensed under the Apache License, Version 2.0 (the "License");
55f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** you may not use this file except in compliance with the License.
65f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** You may obtain a copy of the License at
75f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian **
85f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian **     http://www.apache.org/licenses/LICENSE-2.0
95f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian **
105f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** Unless required by applicable law or agreed to in writing, software
115f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** distributed under the License is distributed on an "AS IS" BASIS,
125f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
135f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** See the License for the specific language governing permissions and
145f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian ** limitations under the License.
155f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian */
165f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
175f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian//#define LOG_NDEBUG 0
185f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
19b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian#include "BlobCache.h"
205f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
219f93afecb6e3467e43f919e8f20f3ab3d11f45b2Dan Albert#include <errno.h>
225f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#include <inttypes.h>
235f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
245f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#include <cutils/properties.h>
25b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian#include <log/log.h>
26b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian#include <chrono>
275f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
285f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopiannamespace android {
295f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
305f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian// BlobCache::Header::mMagicNumber value
315f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianstatic const uint32_t blobCacheMagic = ('_' << 24) + ('B' << 16) + ('b' << 8) + '$';
325f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
335f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian// BlobCache::Header::mBlobCacheVersion value
345f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianstatic const uint32_t blobCacheVersion = 3;
355f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
365f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian// BlobCache::Header::mDeviceVersion value
375f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianstatic const uint32_t blobCacheDeviceVersion = 1;
385f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
395f549b2089442cb80e8d7f4bd00ac69560375b2cMathias AgopianBlobCache::BlobCache(size_t maxKeySize, size_t maxValueSize, size_t maxTotalSize):
409e7cd07a30a0ed27852ad04d2997f00387b55dcfStan Iliev        mMaxTotalSize(maxTotalSize),
415f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mMaxKeySize(maxKeySize),
425f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mMaxValueSize(maxValueSize),
435f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mTotalSize(0) {
44b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    int64_t now = std::chrono::steady_clock::now().time_since_epoch().count();
455f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#ifdef _WIN32
465f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    srand(now);
475f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#else
485f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mRandState[0] = (now >> 0) & 0xFFFF;
495f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mRandState[1] = (now >> 16) & 0xFFFF;
505f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mRandState[2] = (now >> 32) & 0xFFFF;
515f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#endif
525f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    ALOGV("initializing random seed using %lld", (unsigned long long)now);
535f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
545f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
555f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianvoid BlobCache::set(const void* key, size_t keySize, const void* value,
565f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t valueSize) {
575f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mMaxKeySize < keySize) {
585f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("set: not caching because the key is too large: %zu (limit: %zu)",
595f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                keySize, mMaxKeySize);
605f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return;
615f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
625f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mMaxValueSize < valueSize) {
635f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("set: not caching because the value is too large: %zu (limit: %zu)",
645f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                valueSize, mMaxValueSize);
655f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return;
665f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
675f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mMaxTotalSize < keySize + valueSize) {
685f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("set: not caching because the combined key/value size is too "
695f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                "large: %zu (limit: %zu)", keySize + valueSize, mMaxTotalSize);
705f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return;
715f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
725f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (keySize == 0) {
735f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGW("set: not caching because keySize is 0");
745f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return;
755f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
765f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (valueSize <= 0) {
775f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGW("set: not caching because valueSize is 0");
785f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return;
795f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
805f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
81b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    std::shared_ptr<Blob> dummyKey(new Blob(key, keySize, false));
825f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    CacheEntry dummyEntry(dummyKey, NULL);
835f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
845f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    while (true) {
85b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        auto index = std::lower_bound(mCacheEntries.begin(), mCacheEntries.end(), dummyEntry);
86b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        if (index == mCacheEntries.end() || dummyEntry < *index) {
875f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            // Create a new cache entry.
88b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            std::shared_ptr<Blob> keyBlob(new Blob(key, keySize, true));
89b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            std::shared_ptr<Blob> valueBlob(new Blob(value, valueSize, true));
905f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            size_t newTotalSize = mTotalSize + keySize + valueSize;
915f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            if (mMaxTotalSize < newTotalSize) {
925f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                if (isCleanable()) {
935f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    // Clean the cache and try again.
945f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    clean();
955f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    continue;
965f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                } else {
975f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    ALOGV("set: not caching new key/value pair because the "
985f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                            "total cache size limit would be exceeded: %zu "
995f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                            "(limit: %zu)",
1005f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                            keySize + valueSize, mMaxTotalSize);
1015f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    break;
1025f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                }
1035f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            }
104b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            mCacheEntries.insert(index, CacheEntry(keyBlob, valueBlob));
1055f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            mTotalSize = newTotalSize;
1065f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            ALOGV("set: created new cache entry with %zu byte key and %zu byte value",
1075f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    keySize, valueSize);
1085f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        } else {
1095f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            // Update the existing cache entry.
110b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            std::shared_ptr<Blob> valueBlob(new Blob(value, valueSize, true));
111b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            std::shared_ptr<Blob> oldValueBlob(index->getValue());
1125f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            size_t newTotalSize = mTotalSize + valueSize - oldValueBlob->getSize();
1135f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            if (mMaxTotalSize < newTotalSize) {
1145f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                if (isCleanable()) {
1155f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    // Clean the cache and try again.
1165f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    clean();
1175f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    continue;
1185f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                } else {
1195f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    ALOGV("set: not caching new value because the total cache "
1205f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                            "size limit would be exceeded: %zu (limit: %zu)",
1215f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                            keySize + valueSize, mMaxTotalSize);
1225f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    break;
1235f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                }
1245f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            }
125b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            index->setValue(valueBlob);
1265f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            mTotalSize = newTotalSize;
1275f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            ALOGV("set: updated existing cache entry with %zu byte key and %zu byte "
1285f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                    "value", keySize, valueSize);
1295f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        }
1305f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        break;
1315f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
1325f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
1335f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
1345f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopiansize_t BlobCache::get(const void* key, size_t keySize, void* value,
1355f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t valueSize) {
1365f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mMaxKeySize < keySize) {
1375f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("get: not searching because the key is too large: %zu (limit %zu)",
1385f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                keySize, mMaxKeySize);
1395f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return 0;
1405f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
141b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    std::shared_ptr<Blob> dummyKey(new Blob(key, keySize, false));
1425f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    CacheEntry dummyEntry(dummyKey, NULL);
143b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    auto index = std::lower_bound(mCacheEntries.begin(), mCacheEntries.end(), dummyEntry);
144b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    if (index == mCacheEntries.end() || dummyEntry < *index) {
1455f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("get: no cache entry found for key of size %zu", keySize);
1465f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return 0;
1475f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
1485f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
1495f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // The key was found. Return the value if the caller's buffer is large
1505f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // enough.
151b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    std::shared_ptr<Blob> valueBlob(index->getValue());
1525f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    size_t valueBlobSize = valueBlob->getSize();
1535f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (valueBlobSize <= valueSize) {
1545f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("get: copying %zu bytes to caller's buffer", valueBlobSize);
1555f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        memcpy(value, valueBlob->getData(), valueBlobSize);
1565f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    } else {
1575f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGV("get: caller's buffer is too small for value: %zu (needs %zu)",
1585f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                valueSize, valueBlobSize);
1595f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
1605f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return valueBlobSize;
1615f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
1625f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
1635f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianstatic inline size_t align4(size_t size) {
1645f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return (size + 3) & ~3;
1655f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
1665f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
1675f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopiansize_t BlobCache::getFlattenedSize() const {
1685f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    size_t size = align4(sizeof(Header) + PROPERTY_VALUE_MAX);
169b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    for (const CacheEntry& e :  mCacheEntries) {
170b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        std::shared_ptr<Blob> const& keyBlob = e.getKey();
171b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        std::shared_ptr<Blob> const& valueBlob = e.getValue();
172b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        size += align4(sizeof(EntryHeader) + keyBlob->getSize() + valueBlob->getSize());
1735f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
1745f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return size;
1755f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
1765f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
177b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopianint BlobCache::flatten(void* buffer, size_t size) const {
1785f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // Write the cache header
1795f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (size < sizeof(Header)) {
1805f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGE("flatten: not enough room for cache header");
181b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        return 0;
1825f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
1835f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    Header* header = reinterpret_cast<Header*>(buffer);
1845f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    header->mMagicNumber = blobCacheMagic;
1855f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    header->mBlobCacheVersion = blobCacheVersion;
1865f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    header->mDeviceVersion = blobCacheDeviceVersion;
1875f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    header->mNumEntries = mCacheEntries.size();
1885f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    char buildId[PROPERTY_VALUE_MAX];
1895f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    header->mBuildIdLength = property_get("ro.build.id", buildId, "");
1905f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    memcpy(header->mBuildId, buildId, header->mBuildIdLength);
1915f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
1925f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // Write cache entries
1935f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    uint8_t* byteBuffer = reinterpret_cast<uint8_t*>(buffer);
1945f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    off_t byteOffset = align4(sizeof(Header) + header->mBuildIdLength);
195b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    for (const CacheEntry& e :  mCacheEntries) {
196b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        std::shared_ptr<Blob> const& keyBlob = e.getKey();
197b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        std::shared_ptr<Blob> const& valueBlob = e.getValue();
1985f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t keySize = keyBlob->getSize();
1995f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t valueSize = valueBlob->getSize();
2005f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2015f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t entrySize = sizeof(EntryHeader) + keySize + valueSize;
2025f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t totalSize = align4(entrySize);
2035f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        if (byteOffset + totalSize > size) {
2045f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            ALOGE("flatten: not enough room for cache entries");
205b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            return -EINVAL;
2065f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        }
2075f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
208b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        EntryHeader* eheader = reinterpret_cast<EntryHeader*>(&byteBuffer[byteOffset]);
2095f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        eheader->mKeySize = keySize;
2105f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        eheader->mValueSize = valueSize;
2115f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2125f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        memcpy(eheader->mData, keyBlob->getData(), keySize);
2135f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        memcpy(eheader->mData + keySize, valueBlob->getData(), valueSize);
2145f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2155f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        if (totalSize > entrySize) {
2165f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            // We have padding bytes. Those will get written to storage, and contribute to the CRC,
2175f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            // so make sure we zero-them to have reproducible results.
2185f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            memset(eheader->mData + keySize + valueSize, 0, totalSize - entrySize);
2195f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        }
2205f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2215f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        byteOffset += totalSize;
2225f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
2235f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
224b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    return 0;
2255f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
2265f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
227b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopianint BlobCache::unflatten(void const* buffer, size_t size) {
2285f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // All errors should result in the BlobCache being in an empty state.
2295f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mCacheEntries.clear();
2305f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2315f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // Read the cache header
2325f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (size < sizeof(Header)) {
2335f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGE("unflatten: not enough room for cache header");
234b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        return -EINVAL;
2355f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
2365f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    const Header* header = reinterpret_cast<const Header*>(buffer);
2375f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (header->mMagicNumber != blobCacheMagic) {
2385f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        ALOGE("unflatten: bad magic number: %" PRIu32, header->mMagicNumber);
239b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        return -EINVAL;
2405f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
2415f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    char buildId[PROPERTY_VALUE_MAX];
2425f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    int len = property_get("ro.build.id", buildId, "");
2435f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (header->mBlobCacheVersion != blobCacheVersion ||
2445f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            header->mDeviceVersion != blobCacheDeviceVersion ||
2455f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            len != header->mBuildIdLength ||
2465f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            strncmp(buildId, header->mBuildId, len)) {
2475f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        // We treat version mismatches as an empty cache.
248b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        return 0;
2495f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
2505f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2515f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // Read cache entries
2525f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    const uint8_t* byteBuffer = reinterpret_cast<const uint8_t*>(buffer);
2535f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    off_t byteOffset = align4(sizeof(Header) + header->mBuildIdLength);
2545f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    size_t numEntries = header->mNumEntries;
2555f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    for (size_t i = 0; i < numEntries; i++) {
2565f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        if (byteOffset + sizeof(EntryHeader) > size) {
2575f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            mCacheEntries.clear();
2585f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            ALOGE("unflatten: not enough room for cache entry headers");
259b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            return -EINVAL;
2605f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        }
2615f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2625f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        const EntryHeader* eheader = reinterpret_cast<const EntryHeader*>(
2635f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian                &byteBuffer[byteOffset]);
2645f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t keySize = eheader->mKeySize;
2655f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t valueSize = eheader->mValueSize;
2665f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t entrySize = sizeof(EntryHeader) + keySize + valueSize;
2675f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2685f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t totalSize = align4(entrySize);
2695f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        if (byteOffset + totalSize > size) {
2705f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            mCacheEntries.clear();
2715f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian            ALOGE("unflatten: not enough room for cache entry headers");
272b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian            return -EINVAL;
2735f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        }
2745f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2755f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        const uint8_t* data = eheader->mData;
2765f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        set(data, keySize, data + keySize, valueSize);
2775f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2785f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        byteOffset += totalSize;
2795f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
2805f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
281b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian    return 0;
2825f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
2835f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2845f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianlong int BlobCache::blob_random() {
2855f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#ifdef _WIN32
2865f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return rand();
2875f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#else
2885f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return nrand48(mRandState);
2895f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian#endif
2905f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
2915f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
2925f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianvoid BlobCache::clean() {
2935f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // Remove a random cache entry until the total cache size gets below half
2945f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    // the maximum total cache size.
2955f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    while (mTotalSize > mMaxTotalSize / 2) {
2965f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        size_t i = size_t(blob_random() % (mCacheEntries.size()));
2975f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        const CacheEntry& entry(mCacheEntries[i]);
2985f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mTotalSize -= entry.getKey()->getSize() + entry.getValue()->getSize();
299b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        mCacheEntries.erase(mCacheEntries.begin() + i);
3005f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
3015f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3025f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3035f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianbool BlobCache::isCleanable() const {
3045f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return mTotalSize > mMaxTotalSize / 2;
3055f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3065f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
307b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias AgopianBlobCache::Blob::Blob(const void* data, size_t size, bool copyData) :
3085f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mData(copyData ? malloc(size) : data),
3095f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mSize(size),
3105f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mOwnsData(copyData) {
3115f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (data != NULL && copyData) {
3125f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        memcpy(const_cast<void*>(mData), data, size);
3135f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
3145f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3155f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3165f549b2089442cb80e8d7f4bd00ac69560375b2cMathias AgopianBlobCache::Blob::~Blob() {
3175f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mOwnsData) {
3185f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        free(const_cast<void*>(mData));
3195f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
3205f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3215f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3225f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianbool BlobCache::Blob::operator<(const Blob& rhs) const {
3235f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    if (mSize == rhs.mSize) {
3245f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return memcmp(mData, rhs.mData, mSize) < 0;
3255f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    } else {
3265f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        return mSize < rhs.mSize;
3275f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    }
3285f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3295f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3305f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianconst void* BlobCache::Blob::getData() const {
3315f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return mData;
3325f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3335f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3345f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopiansize_t BlobCache::Blob::getSize() const {
3355f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return mSize;
3365f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3375f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3385f549b2089442cb80e8d7f4bd00ac69560375b2cMathias AgopianBlobCache::CacheEntry::CacheEntry() {
3395f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3405f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
341b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias AgopianBlobCache::CacheEntry::CacheEntry(
342b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopian        const std::shared_ptr<Blob>& key, const std::shared_ptr<Blob>& value):
3435f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mKey(key),
3445f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mValue(value) {
3455f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3465f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3475f549b2089442cb80e8d7f4bd00ac69560375b2cMathias AgopianBlobCache::CacheEntry::CacheEntry(const CacheEntry& ce):
3485f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mKey(ce.mKey),
3495f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian        mValue(ce.mValue) {
3505f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3515f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3525f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianbool BlobCache::CacheEntry::operator<(const CacheEntry& rhs) const {
3535f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return *mKey < *rhs.mKey;
3545f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3555f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3565f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopianconst BlobCache::CacheEntry& BlobCache::CacheEntry::operator=(const CacheEntry& rhs) {
3575f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mKey = rhs.mKey;
3585f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mValue = rhs.mValue;
3595f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return *this;
3605f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3615f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
362b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopianstd::shared_ptr<BlobCache::Blob> BlobCache::CacheEntry::getKey() const {
3635f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return mKey;
3645f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3655f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
366b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopianstd::shared_ptr<BlobCache::Blob> BlobCache::CacheEntry::getValue() const {
3675f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    return mValue;
3685f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3695f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
370b7f9a2400aaa2e0d29ffefd91576e90036d4cf83Mathias Agopianvoid BlobCache::CacheEntry::setValue(const std::shared_ptr<Blob>& value) {
3715f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian    mValue = value;
3725f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian}
3735f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian
3745f549b2089442cb80e8d7f4bd00ac69560375b2cMathias Agopian} // namespace android
375