1 /* 2 ** Copyright 2011, 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 //#define LOG_NDEBUG 0 18 19 #include "BlobCache.h" 20 21 #include <errno.h> 22 #include <inttypes.h> 23 24 #if defined(__ANDROID__) 25 #include <cutils/properties.h> 26 #else 27 #include <string.h> 28 #include <algorithm> 29 static const char property_value[] = "[HOST]"; 30 #define PROPERTY_VALUE_MAX (sizeof(property_value) - 1) 31 static int property_get(const char *key, char *value, const char *default_value) { 32 if (!strcmp(key, "ro.build.id")) { 33 memcpy(value, property_value, PROPERTY_VALUE_MAX); 34 return PROPERTY_VALUE_MAX; 35 } 36 if (default_value) { 37 const size_t len = std::max(strlen(default_value) + 1, size_t(PROPERTY_VALUE_MAX)); 38 memcpy(value, default_value, len); 39 } 40 return 0; 41 } 42 #endif 43 44 #include <log/log.h> 45 46 #include <algorithm> 47 #include <chrono> 48 49 namespace android { 50 51 // BlobCache::Header::mMagicNumber value 52 static const uint32_t blobCacheMagic = ('_' << 24) + ('B' << 16) + ('b' << 8) + '$'; 53 54 // BlobCache::Header::mBlobCacheVersion value 55 static const uint32_t blobCacheVersion = 3; 56 57 // BlobCache::Header::mDeviceVersion value 58 static const uint32_t blobCacheDeviceVersion = 1; 59 60 BlobCache::BlobCache(size_t maxKeySize, size_t maxValueSize, size_t maxTotalSize, Policy policy): 61 mMaxKeySize(maxKeySize), 62 mMaxValueSize(maxValueSize), 63 mMaxTotalSize(maxTotalSize), 64 mPolicySelect(policy.first), 65 mPolicyCapacity(policy.second), 66 mTotalSize(0), 67 mAccessCount(0) { 68 int64_t now = std::chrono::steady_clock::now().time_since_epoch().count(); 69 #ifdef _WIN32 70 srand(now); 71 #else 72 mRandState[0] = (now >> 0) & 0xFFFF; 73 mRandState[1] = (now >> 16) & 0xFFFF; 74 mRandState[2] = (now >> 32) & 0xFFFF; 75 #endif 76 ALOGV("initializing random seed using %lld", (unsigned long long)now); 77 } 78 79 void BlobCache::set(const void* key, size_t keySize, const void* value, 80 size_t valueSize) { 81 if (mMaxKeySize < keySize) { 82 ALOGV("set: not caching because the key is too large: %zu (limit: %zu)", 83 keySize, mMaxKeySize); 84 return; 85 } 86 if (mMaxValueSize < valueSize) { 87 ALOGV("set: not caching because the value is too large: %zu (limit: %zu)", 88 valueSize, mMaxValueSize); 89 return; 90 } 91 if (mMaxTotalSize < keySize + valueSize) { 92 ALOGV("set: not caching because the combined key/value size is too " 93 "large: %zu (limit: %zu)", keySize + valueSize, mMaxTotalSize); 94 return; 95 } 96 if (keySize == 0) { 97 ALOGW("set: not caching because keySize is 0"); 98 return; 99 } 100 if (valueSize <= 0) { 101 ALOGW("set: not caching because valueSize is 0"); 102 return; 103 } 104 105 std::shared_ptr<Blob> dummyKey(new Blob(key, keySize, false)); 106 CacheEntry dummyEntry(dummyKey, NULL, 0); 107 108 while (true) { 109 auto index = std::lower_bound(mCacheEntries.begin(), mCacheEntries.end(), dummyEntry); 110 if (index == mCacheEntries.end() || dummyEntry < *index) { 111 // Create a new cache entry. 112 std::shared_ptr<Blob> keyBlob(new Blob(key, keySize, true)); 113 std::shared_ptr<Blob> valueBlob(new Blob(value, valueSize, true)); 114 size_t newEntrySize = keySize + valueSize; 115 size_t newTotalSize = mTotalSize + newEntrySize; 116 if (mMaxTotalSize < newTotalSize) { 117 if (isCleanable()) { 118 // Clean the cache and try again. 119 if (!clean(newEntrySize, NoEntry)) { 120 // We have some kind of logic error -- perhaps 121 // an inconsistency between isCleanable() and 122 // findDownTo(). 123 ALOGE("set: not caching new key/value pair because " 124 "cleaning failed"); 125 break; 126 } 127 continue; 128 } else { 129 ALOGV("set: not caching new key/value pair because the " 130 "total cache size limit would be exceeded: %zu " 131 "(limit: %zu)", 132 keySize + valueSize, mMaxTotalSize); 133 break; 134 } 135 } 136 mCacheEntries.insert(index, CacheEntry(keyBlob, valueBlob, ++mAccessCount)); 137 mTotalSize = newTotalSize; 138 ALOGV("set: created new cache entry with %zu byte key and %zu byte value", 139 keySize, valueSize); 140 } else { 141 // Update the existing cache entry. 142 std::shared_ptr<Blob> valueBlob(new Blob(value, valueSize, true)); 143 std::shared_ptr<Blob> oldValueBlob(index->getValue()); 144 size_t newTotalSize = mTotalSize + valueSize - oldValueBlob->getSize(); 145 if (mMaxTotalSize < newTotalSize) { 146 if (isCleanable()) { 147 // Clean the cache and try again. 148 if (!clean(index->getKey()->getSize() + valueSize, 149 index - mCacheEntries.begin())) { 150 // We have some kind of logic error -- perhaps 151 // an inconsistency between isCleanable() and 152 // findDownTo(). 153 ALOGE("set: not caching new value because " 154 "cleaning failed"); 155 break; 156 } 157 continue; 158 } else { 159 ALOGV("set: not caching new value because the total cache " 160 "size limit would be exceeded: %zu (limit: %zu)", 161 keySize + valueSize, mMaxTotalSize); 162 break; 163 } 164 } 165 index->setValue(valueBlob); 166 index->setRecency(++mAccessCount); 167 mTotalSize = newTotalSize; 168 ALOGV("set: updated existing cache entry with %zu byte key and %zu byte " 169 "value", keySize, valueSize); 170 } 171 break; 172 } 173 } 174 175 size_t BlobCache::get(const void* key, size_t keySize, void* value, 176 size_t valueSize) { 177 void *dummy; 178 return get(key, keySize, &dummy, 179 [value, valueSize](size_t allocSize) { 180 return (allocSize <= valueSize ? value : nullptr); 181 }); 182 } 183 184 size_t BlobCache::get(const void* key, size_t keySize, void** value, 185 std::function<void*(size_t)> alloc) { 186 if (mMaxKeySize < keySize) { 187 ALOGV("get: not searching because the key is too large: %zu (limit %zu)", 188 keySize, mMaxKeySize); 189 *value = nullptr; 190 return 0; 191 } 192 std::shared_ptr<Blob> dummyKey(new Blob(key, keySize, false)); 193 CacheEntry dummyEntry(dummyKey, NULL, 0); 194 auto index = std::lower_bound(mCacheEntries.begin(), mCacheEntries.end(), dummyEntry); 195 if (index == mCacheEntries.end() || dummyEntry < *index) { 196 ALOGV("get: no cache entry found for key of size %zu", keySize); 197 *value = nullptr; 198 return 0; 199 } 200 201 // The key was found. Return the value if we can allocate a buffer. 202 std::shared_ptr<Blob> valueBlob(index->getValue()); 203 size_t valueBlobSize = valueBlob->getSize(); 204 void *buf = alloc(valueBlobSize); 205 if (buf != nullptr) { 206 ALOGV("get: copying %zu bytes to caller's buffer", valueBlobSize); 207 memcpy(buf, valueBlob->getData(), valueBlobSize); 208 *value = buf; 209 index->setRecency(++mAccessCount); 210 } else { 211 ALOGV("get: cannot allocate caller's buffer: needs %zu", valueBlobSize); 212 *value = nullptr; 213 } 214 return valueBlobSize; 215 } 216 217 static inline size_t align4(size_t size) { 218 return (size + 3) & ~3; 219 } 220 221 size_t BlobCache::getFlattenedSize() const { 222 size_t size = align4(sizeof(Header) + PROPERTY_VALUE_MAX); 223 for (const CacheEntry& e : mCacheEntries) { 224 std::shared_ptr<Blob> const& keyBlob = e.getKey(); 225 std::shared_ptr<Blob> const& valueBlob = e.getValue(); 226 size += align4(sizeof(EntryHeader) + keyBlob->getSize() + valueBlob->getSize()); 227 } 228 return size; 229 } 230 231 int BlobCache::flatten(void* buffer, size_t size) const { 232 // Write the cache header 233 if (size < sizeof(Header)) { 234 ALOGE("flatten: not enough room for cache header"); 235 return 0; 236 } 237 Header* header = reinterpret_cast<Header*>(buffer); 238 header->mMagicNumber = blobCacheMagic; 239 header->mBlobCacheVersion = blobCacheVersion; 240 header->mDeviceVersion = blobCacheDeviceVersion; 241 header->mNumEntries = mCacheEntries.size(); 242 char buildId[PROPERTY_VALUE_MAX]; 243 header->mBuildIdLength = property_get("ro.build.id", buildId, ""); 244 memcpy(header->mBuildId, buildId, header->mBuildIdLength); 245 246 // Write cache entries 247 uint8_t* byteBuffer = reinterpret_cast<uint8_t*>(buffer); 248 off_t byteOffset = align4(sizeof(Header) + header->mBuildIdLength); 249 for (const CacheEntry& e : mCacheEntries) { 250 std::shared_ptr<Blob> const& keyBlob = e.getKey(); 251 std::shared_ptr<Blob> const& valueBlob = e.getValue(); 252 size_t keySize = keyBlob->getSize(); 253 size_t valueSize = valueBlob->getSize(); 254 255 size_t entrySize = sizeof(EntryHeader) + keySize + valueSize; 256 size_t totalSize = align4(entrySize); 257 if (byteOffset + totalSize > size) { 258 ALOGE("flatten: not enough room for cache entries"); 259 return -EINVAL; 260 } 261 262 EntryHeader* eheader = reinterpret_cast<EntryHeader*>(&byteBuffer[byteOffset]); 263 eheader->mKeySize = keySize; 264 eheader->mValueSize = valueSize; 265 266 memcpy(eheader->mData, keyBlob->getData(), keySize); 267 memcpy(eheader->mData + keySize, valueBlob->getData(), valueSize); 268 269 if (totalSize > entrySize) { 270 // We have padding bytes. Those will get written to storage, and contribute to the CRC, 271 // so make sure we zero-them to have reproducible results. 272 memset(eheader->mData + keySize + valueSize, 0, totalSize - entrySize); 273 } 274 275 byteOffset += totalSize; 276 } 277 278 return 0; 279 } 280 281 int BlobCache::unflatten(void const* buffer, size_t size) { 282 // All errors should result in the BlobCache being in an empty state. 283 mCacheEntries.clear(); 284 285 // Read the cache header 286 if (size < sizeof(Header)) { 287 ALOGE("unflatten: not enough room for cache header"); 288 return -EINVAL; 289 } 290 const Header* header = reinterpret_cast<const Header*>(buffer); 291 if (header->mMagicNumber != blobCacheMagic) { 292 ALOGE("unflatten: bad magic number: %" PRIu32, header->mMagicNumber); 293 return -EINVAL; 294 } 295 char buildId[PROPERTY_VALUE_MAX]; 296 int len = property_get("ro.build.id", buildId, ""); 297 if (header->mBlobCacheVersion != blobCacheVersion || 298 header->mDeviceVersion != blobCacheDeviceVersion || 299 len != header->mBuildIdLength || 300 strncmp(buildId, header->mBuildId, len)) { 301 // We treat version mismatches as an empty cache. 302 return 0; 303 } 304 305 // Read cache entries 306 const uint8_t* byteBuffer = reinterpret_cast<const uint8_t*>(buffer); 307 off_t byteOffset = align4(sizeof(Header) + header->mBuildIdLength); 308 size_t numEntries = header->mNumEntries; 309 for (size_t i = 0; i < numEntries; i++) { 310 if (byteOffset + sizeof(EntryHeader) > size) { 311 mCacheEntries.clear(); 312 ALOGE("unflatten: not enough room for cache entry header"); 313 return -EINVAL; 314 } 315 316 const EntryHeader* eheader = reinterpret_cast<const EntryHeader*>( 317 &byteBuffer[byteOffset]); 318 size_t keySize = eheader->mKeySize; 319 size_t valueSize = eheader->mValueSize; 320 size_t entrySize = sizeof(EntryHeader) + keySize + valueSize; 321 322 size_t totalSize = align4(entrySize); 323 if (byteOffset + totalSize > size) { 324 mCacheEntries.clear(); 325 ALOGE("unflatten: not enough room for cache entry"); 326 return -EINVAL; 327 } 328 329 const uint8_t* data = eheader->mData; 330 set(data, keySize, data + keySize, valueSize); 331 332 byteOffset += totalSize; 333 } 334 335 return 0; 336 } 337 338 long int BlobCache::blob_random() { 339 #ifdef _WIN32 340 return rand(); 341 #else 342 return nrand48(mRandState); 343 #endif 344 } 345 346 size_t BlobCache::findVictim() { 347 switch (mPolicySelect) { 348 case Select::RANDOM: 349 return size_t(blob_random() % (mCacheEntries.size())); 350 case Select::LRU: 351 return std::min_element(mCacheEntries.begin(), mCacheEntries.end(), 352 [](const CacheEntry &a, const CacheEntry &b) { 353 return a.getRecency() < b.getRecency(); 354 }) - mCacheEntries.begin(); 355 default: 356 ALOGE("findVictim: unknown mPolicySelect: %d", mPolicySelect); 357 return 0; 358 } 359 } 360 361 size_t BlobCache::findDownTo(size_t newEntrySize, size_t onBehalfOf) { 362 auto oldEntrySize = [this, onBehalfOf]() -> size_t { 363 if (onBehalfOf == NoEntry) 364 return 0; 365 const auto &entry = mCacheEntries[onBehalfOf]; 366 return entry.getKey()->getSize() + entry.getValue()->getSize(); 367 }; 368 switch (mPolicyCapacity) { 369 case Capacity::HALVE: 370 return mMaxTotalSize / 2; 371 case Capacity::FIT: 372 return mMaxTotalSize - (newEntrySize - oldEntrySize()); 373 case Capacity::FIT_HALVE: 374 return std::min(mMaxTotalSize - (newEntrySize - oldEntrySize()), mMaxTotalSize / 2); 375 default: 376 ALOGE("findDownTo: unknown mPolicyCapacity: %d", mPolicyCapacity); 377 return 0; 378 } 379 } 380 381 bool BlobCache::isFit(Capacity capacity) { 382 switch (capacity) { 383 case Capacity::HALVE: 384 return false; 385 case Capacity::FIT: 386 case Capacity::FIT_HALVE: 387 return true; 388 default: 389 ALOGE("isFit: unknown capacity: %d", capacity); 390 return false; 391 } 392 } 393 394 bool BlobCache::clean(size_t newEntrySize, size_t onBehalfOf) { 395 // Remove a selected cache entry until the total cache size does 396 // not exceed downTo. 397 const size_t downTo = findDownTo(newEntrySize, onBehalfOf); 398 399 bool cleaned = false; 400 while (mTotalSize > downTo) { 401 const size_t i = findVictim(); 402 const CacheEntry& entry(mCacheEntries[i]); 403 const size_t entrySize = entry.getKey()->getSize() + entry.getValue()->getSize(); 404 mTotalSize -= entrySize; 405 mCacheEntries.erase(mCacheEntries.begin() + i); 406 cleaned = true; 407 } 408 return cleaned; 409 } 410 411 bool BlobCache::isCleanable() const { 412 switch (mPolicyCapacity) { 413 case Capacity::HALVE: 414 return mTotalSize > mMaxTotalSize / 2; 415 default: 416 ALOGE("isCleanable: unknown mPolicyCapacity: %d", mPolicyCapacity); 417 [[fallthrough]]; 418 case Capacity::FIT: 419 case Capacity::FIT_HALVE: 420 return mTotalSize > 0; 421 } 422 } 423 424 BlobCache::Blob::Blob(const void* data, size_t size, bool copyData) : 425 mData(copyData ? malloc(size) : data), 426 mSize(size), 427 mOwnsData(copyData) { 428 if (data != NULL && copyData) { 429 memcpy(const_cast<void*>(mData), data, size); 430 } 431 } 432 433 BlobCache::Blob::~Blob() { 434 if (mOwnsData) { 435 free(const_cast<void*>(mData)); 436 } 437 } 438 439 bool BlobCache::Blob::operator<(const Blob& rhs) const { 440 if (mSize == rhs.mSize) { 441 return memcmp(mData, rhs.mData, mSize) < 0; 442 } else { 443 return mSize < rhs.mSize; 444 } 445 } 446 447 const void* BlobCache::Blob::getData() const { 448 return mData; 449 } 450 451 size_t BlobCache::Blob::getSize() const { 452 return mSize; 453 } 454 455 BlobCache::CacheEntry::CacheEntry(): mRecency(0) { 456 } 457 458 BlobCache::CacheEntry::CacheEntry( 459 const std::shared_ptr<Blob>& key, const std::shared_ptr<Blob>& value, uint32_t recency): 460 mKey(key), 461 mValue(value), 462 mRecency(recency) { 463 } 464 465 BlobCache::CacheEntry::CacheEntry(const CacheEntry& ce): 466 mKey(ce.mKey), 467 mValue(ce.mValue), 468 mRecency(ce.mRecency) { 469 } 470 471 bool BlobCache::CacheEntry::operator<(const CacheEntry& rhs) const { 472 return *mKey < *rhs.mKey; 473 } 474 475 const BlobCache::CacheEntry& BlobCache::CacheEntry::operator=(const CacheEntry& rhs) { 476 mKey = rhs.mKey; 477 mValue = rhs.mValue; 478 mRecency = rhs.mRecency; 479 return *this; 480 } 481 482 std::shared_ptr<BlobCache::Blob> BlobCache::CacheEntry::getKey() const { 483 return mKey; 484 } 485 486 std::shared_ptr<BlobCache::Blob> BlobCache::CacheEntry::getValue() const { 487 return mValue; 488 } 489 490 void BlobCache::CacheEntry::setValue(const std::shared_ptr<Blob>& value) { 491 mValue = value; 492 } 493 494 uint32_t BlobCache::CacheEntry::getRecency() const { 495 return mRecency; 496 } 497 498 void BlobCache::CacheEntry::setRecency(uint32_t recency) { 499 mRecency = recency; 500 } 501 502 } // namespace android 503