diff options
Diffstat (limited to 'libs/utils/tests/BlobCache_test.cpp')
| -rw-r--r-- | libs/utils/tests/BlobCache_test.cpp | 164 | 
1 files changed, 164 insertions, 0 deletions
| diff --git a/libs/utils/tests/BlobCache_test.cpp b/libs/utils/tests/BlobCache_test.cpp index 653ea5e..b64cc39 100644 --- a/libs/utils/tests/BlobCache_test.cpp +++ b/libs/utils/tests/BlobCache_test.cpp @@ -14,9 +14,13 @@   ** limitations under the License.   */ +#include <fcntl.h> +#include <stdio.h> +  #include <gtest/gtest.h>  #include <utils/BlobCache.h> +#include <utils/Errors.h>  namespace android { @@ -254,4 +258,164 @@ TEST_F(BlobCacheTest, ExceedingTotalLimitHalvesCacheSize) {      ASSERT_EQ(maxEntries/2 + 1, numCached);  } +class BlobCacheFlattenTest : public BlobCacheTest { +protected: +    virtual void SetUp() { +        BlobCacheTest::SetUp(); +        mBC2 = new BlobCache(MAX_KEY_SIZE, MAX_VALUE_SIZE, MAX_TOTAL_SIZE); +    } + +    virtual void TearDown() { +        mBC2.clear(); +        BlobCacheTest::TearDown(); +    } + +    void roundTrip() { +        size_t size = mBC->getFlattenedSize(); +        uint8_t* flat = new uint8_t[size]; +        ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); +        ASSERT_EQ(OK, mBC2->unflatten(flat, size, NULL, 0)); +        delete[] flat; +    } + +    sp<BlobCache> mBC2; +}; + +TEST_F(BlobCacheFlattenTest, FlattenOneValue) { +    char buf[4] = { 0xee, 0xee, 0xee, 0xee }; +    mBC->set("abcd", 4, "efgh", 4); +    roundTrip(); +    ASSERT_EQ(size_t(4), mBC2->get("abcd", 4, buf, 4)); +    ASSERT_EQ('e', buf[0]); +    ASSERT_EQ('f', buf[1]); +    ASSERT_EQ('g', buf[2]); +    ASSERT_EQ('h', buf[3]); +} + +TEST_F(BlobCacheFlattenTest, FlattenFullCache) { +    // Fill up the entire cache with 1 char key/value pairs. +    const int maxEntries = MAX_TOTAL_SIZE / 2; +    for (int i = 0; i < maxEntries; i++) { +        uint8_t k = i; +        mBC->set(&k, 1, &k, 1); +    } + +    roundTrip(); + +    // Verify the deserialized cache +    for (int i = 0; i < maxEntries; i++) { +        uint8_t k = i; +        uint8_t v = 0xee; +        ASSERT_EQ(size_t(1), mBC2->get(&k, 1, &v, 1)); +        ASSERT_EQ(k, v); +    } +} + +TEST_F(BlobCacheFlattenTest, FlattenDoesntChangeCache) { +    // Fill up the entire cache with 1 char key/value pairs. +    const int maxEntries = MAX_TOTAL_SIZE / 2; +    for (int i = 0; i < maxEntries; i++) { +        uint8_t k = i; +        mBC->set(&k, 1, &k, 1); +    } + +    size_t size = mBC->getFlattenedSize(); +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); +    delete[] flat; + +    // Verify the cache that we just serialized +    for (int i = 0; i < maxEntries; i++) { +        uint8_t k = i; +        uint8_t v = 0xee; +        ASSERT_EQ(size_t(1), mBC->get(&k, 1, &v, 1)); +        ASSERT_EQ(k, v); +    } +} + +TEST_F(BlobCacheFlattenTest, FlattenCatchesBufferTooSmall) { +    // Fill up the entire cache with 1 char key/value pairs. +    const int maxEntries = MAX_TOTAL_SIZE / 2; +    for (int i = 0; i < maxEntries; i++) { +        uint8_t k = i; +        mBC->set(&k, 1, &k, 1); +    } + +    size_t size = mBC->getFlattenedSize() - 1; +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(BAD_VALUE, mBC->flatten(flat, size, NULL, 0)); +    delete[] flat; +} + +TEST_F(BlobCacheFlattenTest, UnflattenCatchesBadMagic) { +    char buf[4] = { 0xee, 0xee, 0xee, 0xee }; +    mBC->set("abcd", 4, "efgh", 4); + +    size_t size = mBC->getFlattenedSize(); +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); +    flat[1] = ~flat[1]; + +    // Bad magic should cause an error. +    ASSERT_EQ(BAD_VALUE, mBC2->unflatten(flat, size, NULL, 0)); +    delete[] flat; + +    // The error should cause the unflatten to result in an empty cache +    ASSERT_EQ(size_t(0), mBC2->get("abcd", 4, buf, 4)); +} + +TEST_F(BlobCacheFlattenTest, UnflattenCatchesBadBlobCacheVersion) { +    char buf[4] = { 0xee, 0xee, 0xee, 0xee }; +    mBC->set("abcd", 4, "efgh", 4); + +    size_t size = mBC->getFlattenedSize(); +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); +    flat[5] = ~flat[5]; + +    // Version mismatches shouldn't cause errors, but should not use the +    // serialized entries +    ASSERT_EQ(OK, mBC2->unflatten(flat, size, NULL, 0)); +    delete[] flat; + +    // The version mismatch should cause the unflatten to result in an empty +    // cache +    ASSERT_EQ(size_t(0), mBC2->get("abcd", 4, buf, 4)); +} + +TEST_F(BlobCacheFlattenTest, UnflattenCatchesBadBlobCacheDeviceVersion) { +    char buf[4] = { 0xee, 0xee, 0xee, 0xee }; +    mBC->set("abcd", 4, "efgh", 4); + +    size_t size = mBC->getFlattenedSize(); +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); +    flat[10] = ~flat[10]; + +    // Version mismatches shouldn't cause errors, but should not use the +    // serialized entries +    ASSERT_EQ(OK, mBC2->unflatten(flat, size, NULL, 0)); +    delete[] flat; + +    // The version mismatch should cause the unflatten to result in an empty +    // cache +    ASSERT_EQ(size_t(0), mBC2->get("abcd", 4, buf, 4)); +} + +TEST_F(BlobCacheFlattenTest, UnflattenCatchesBufferTooSmall) { +    char buf[4] = { 0xee, 0xee, 0xee, 0xee }; +    mBC->set("abcd", 4, "efgh", 4); + +    size_t size = mBC->getFlattenedSize(); +    uint8_t* flat = new uint8_t[size]; +    ASSERT_EQ(OK, mBC->flatten(flat, size, NULL, 0)); + +    // A buffer truncation shouldt cause an error +    ASSERT_EQ(BAD_VALUE, mBC2->unflatten(flat, size-1, NULL, 0)); +    delete[] flat; + +    // The error should cause the unflatten to result in an empty cache +    ASSERT_EQ(size_t(0), mBC2->get("abcd", 4, buf, 4)); +} +  } // namespace android | 
