Edit

kc3-lang/angle/src/tests/egl_tests/EGLBlobCacheTest.cpp

Branch :

  • Show log

    Commit

  • Author : Jamie Madill
    Date : 2018-09-23 17:01:20
    Hash : f668a4b5
    Message : Skip Texture::syncState when no dirty bits. We sometimes generate local dirty bits in TextureGL. To make sure the local dirty bits don't get skipped we use a Subject/Observer pattern between the TextureGL and gl::Texture. This allows us to skip syncState in the hot path. Also inlines a couple of other texture functions. And fixes a stray header in EGLBlobCacheTest. Bug: angleproject:2763 Change-Id: Ie1d8a5865deaf2a563a358c31ae28bef6b2458b1 Reviewed-on: https://chromium-review.googlesource.com/1228374 Commit-Queue: Jamie Madill <jmadill@chromium.org> Reviewed-by: Yuly Novikov <ynovikov@chromium.org>

  • src/tests/egl_tests/EGLBlobCacheTest.cpp
  • //
    // Copyright 2018 The ANGLE Project Authors. All rights reserved.
    // Use of this source code is governed by a BSD-style license that can be
    // found in the LICENSE file.
    //
    // EGLBlobCacheTest:
    //   Unit tests for the EGL_ANDROID_blob_cache extension.
    
    #include <map>
    #include <vector>
    #include "common/angleutils.h"
    #include "test_utils/ANGLETest.h"
    #include "test_utils/gl_raii.h"
    
    using namespace angle;
    
    constexpr char kEGLExtName[] = "EGL_ANDROID_blob_cache";
    
    enum class CacheOpResult
    {
        SET_SUCCESS,
        GET_NOT_FOUND,
        GET_MEMORY_TOO_SMALL,
        GET_SUCCESS,
    
        VALUE_NOT_SET,
    };
    
    namespace
    {
    std::map<std::vector<uint8_t>, std::vector<uint8_t>> gApplicationCache;
    CacheOpResult gLastCacheOpResult = CacheOpResult::VALUE_NOT_SET;
    
    void SetBlob(const void *key, EGLsizeiANDROID keySize, const void *value, EGLsizeiANDROID valueSize)
    {
        std::vector<uint8_t> keyVec(keySize);
        memcpy(keyVec.data(), key, keySize);
    
        std::vector<uint8_t> valueVec(valueSize);
        memcpy(valueVec.data(), value, valueSize);
    
        gApplicationCache[keyVec] = valueVec;
    
        gLastCacheOpResult = CacheOpResult::SET_SUCCESS;
    }
    
    EGLsizeiANDROID GetBlob(const void *key,
                            EGLsizeiANDROID keySize,
                            void *value,
                            EGLsizeiANDROID valueSize)
    {
        std::vector<uint8_t> keyVec(keySize);
        memcpy(keyVec.data(), key, keySize);
    
        auto entry = gApplicationCache.find(keyVec);
        if (entry == gApplicationCache.end())
        {
            gLastCacheOpResult = CacheOpResult::GET_NOT_FOUND;
            return 0;
        }
    
        if (entry->second.size() <= static_cast<size_t>(valueSize))
        {
            memcpy(value, entry->second.data(), entry->second.size());
            gLastCacheOpResult = CacheOpResult::GET_SUCCESS;
        }
        else
        {
            gLastCacheOpResult = CacheOpResult::GET_MEMORY_TOO_SMALL;
        }
    
        return entry->second.size();
    }
    }  // anonymous namespace
    
    class EGLBlobCacheTest : public ANGLETest
    {
      protected:
        EGLBlobCacheTest() : mHasProgramCache(false) { setDeferContextInit(true); }
    
        void SetUp() override
        {
            ANGLETest::SetUp();
    
            // Enable the program cache so that angle would do caching, which eventually lands in the
            // BlobCache
            EGLDisplay display = getEGLWindow()->getDisplay();
            if (eglDisplayExtensionEnabled(display, "EGL_ANGLE_program_cache_control"))
            {
                mHasProgramCache = true;
                setContextProgramCacheEnabled(true);
                eglProgramCacheResizeANGLE(display, 0x10000, EGL_PROGRAM_CACHE_RESIZE_ANGLE);
            }
    
            getEGLWindow()->initializeContext();
        }
    
        void TearDown() override { ANGLETest::TearDown(); }
    
        bool extensionAvailable()
        {
            EGLDisplay display = getEGLWindow()->getDisplay();
            return eglDisplayExtensionEnabled(display, kEGLExtName);
        }
    
        bool programBinaryAvailable()
        {
            return (getClientMajorVersion() >= 3 || extensionEnabled("GL_OES_get_program_binary"));
        }
    
        bool mHasProgramCache;
    };
    
    // Makes sure the extension exists and works
    TEST_P(EGLBlobCacheTest, Functional)
    {
        EGLDisplay display = getEGLWindow()->getDisplay();
    
        EXPECT_EQ(true, extensionAvailable());
        eglSetBlobCacheFuncsANDROID(display, SetBlob, GetBlob);
        ASSERT_EGL_SUCCESS();
    
        constexpr char kVertexShaderSrc[] = R"(attribute vec4 aTest;
    attribute vec2 aPosition;
    varying vec4 vTest;
    void main()
    {
        vTest        = aTest;
        gl_Position  = vec4(aPosition, 0.0, 1.0);
        gl_PointSize = 1.0;
    })";
    
        constexpr char kFragmentShaderSrc[] = R"(precision mediump float;
    varying vec4 vTest;
    void main()
    {
        gl_FragColor = vTest;
    })";
    
        constexpr char kVertexShaderSrc2[] = R"(attribute vec4 aTest;
    attribute vec2 aPosition;
    varying vec4 vTest;
    void main()
    {
        vTest        = aTest;
        gl_Position  = vec4(aPosition, 1.0, 1.0);
        gl_PointSize = 1.0;
    })";
    
        constexpr char kFragmentShaderSrc2[] = R"(precision mediump float;
    varying vec4 vTest;
    void main()
    {
        gl_FragColor = vTest - vec4(0.0, 1.0, 0.0, 0.0);
    })";
    
        // Compile a shader so it puts something in the cache
        if (mHasProgramCache && programBinaryAvailable())
        {
            GLuint program = CompileProgram(kVertexShaderSrc, kFragmentShaderSrc);
            ASSERT_NE(0u, program);
            EXPECT_EQ(CacheOpResult::SET_SUCCESS, gLastCacheOpResult);
            gLastCacheOpResult = CacheOpResult::VALUE_NOT_SET;
    
            // Compile the same shader again, so it would try to retrieve it from the cache
            program = CompileProgram(kVertexShaderSrc, kFragmentShaderSrc);
            ASSERT_NE(0u, program);
            EXPECT_EQ(CacheOpResult::GET_SUCCESS, gLastCacheOpResult);
            gLastCacheOpResult = CacheOpResult::VALUE_NOT_SET;
    
            // Compile another shader, which should create a new entry
            program = CompileProgram(kVertexShaderSrc2, kFragmentShaderSrc2);
            ASSERT_NE(0u, program);
            EXPECT_EQ(CacheOpResult::SET_SUCCESS, gLastCacheOpResult);
            gLastCacheOpResult = CacheOpResult::VALUE_NOT_SET;
    
            // Compile the first shader again, which should still reside in the cache
            program = CompileProgram(kVertexShaderSrc, kFragmentShaderSrc);
            ASSERT_NE(0u, program);
            EXPECT_EQ(CacheOpResult::GET_SUCCESS, gLastCacheOpResult);
            gLastCacheOpResult = CacheOpResult::VALUE_NOT_SET;
        }
    }
    
    // Tests error conditions of the APIs.
    TEST_P(EGLBlobCacheTest, NegativeAPI)
    {
        EXPECT_EQ(true, extensionAvailable());
    
        // Test bad display
        eglSetBlobCacheFuncsANDROID(EGL_NO_DISPLAY, nullptr, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_DISPLAY);
    
        eglSetBlobCacheFuncsANDROID(EGL_NO_DISPLAY, SetBlob, GetBlob);
        EXPECT_EGL_ERROR(EGL_BAD_DISPLAY);
    
        EGLDisplay display = getEGLWindow()->getDisplay();
    
        // Test bad arguments
        eglSetBlobCacheFuncsANDROID(display, nullptr, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        eglSetBlobCacheFuncsANDROID(display, SetBlob, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        eglSetBlobCacheFuncsANDROID(display, nullptr, GetBlob);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        // Set the arguments once and test setting them again (which should fail)
        eglSetBlobCacheFuncsANDROID(display, SetBlob, GetBlob);
        ASSERT_EGL_SUCCESS();
    
        eglSetBlobCacheFuncsANDROID(display, SetBlob, GetBlob);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        // Try again with bad parameters
        eglSetBlobCacheFuncsANDROID(EGL_NO_DISPLAY, nullptr, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_DISPLAY);
    
        eglSetBlobCacheFuncsANDROID(display, nullptr, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        eglSetBlobCacheFuncsANDROID(display, SetBlob, nullptr);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    
        eglSetBlobCacheFuncsANDROID(display, nullptr, GetBlob);
        EXPECT_EGL_ERROR(EGL_BAD_PARAMETER);
    }
    
    ANGLE_INSTANTIATE_TEST(EGLBlobCacheTest, ES2_D3D9(), ES2_D3D11(), ES2_OPENGL());