Edit

kc3-lang/angle/src/tests/capture_replay_tests/CaptureReplayTests.cpp

Branch :

  • Show log

    Commit

  • Author : Manh Nguyen
    Date : 2020-07-31 10:52:21
    Hash : 6aed2832
    Message : Batch capture run + change how results are logged Before, the run stages of tests in a batch were run separately by multiple subprocesses. Now the run stages of tests in a batch are batched together and run by a single subprocess. Changes how results are logged. Tests in a batch are logged together. Within a batch, tests that fail at the same stage are also logged together. Bug: angleproject:4817 Change-Id: Ie3f992c081de914f1f1f521bec2d72f06ccca238 Reviewed-on: https://chromium-review.googlesource.com/c/angle/angle/+/2331738 Commit-Queue: Manh Nguyen <nguyenmh@google.com> Reviewed-by: Jamie Madill <jmadill@chromium.org> Reviewed-by: Cody Northrop <cnorthrop@google.com>

  • src/tests/capture_replay_tests/CaptureReplayTests.cpp
  • //
    // Copyright 2020 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.
    //
    // CaptureReplayTest.cpp:
    //   Application that runs replay for testing of capture replay
    //
    
    #include "common/system_utils.h"
    #include "libANGLE/Context.h"
    #include "libANGLE/frame_capture_utils.h"
    #include "util/EGLPlatformParameters.h"
    #include "util/EGLWindow.h"
    #include "util/OSWindow.h"
    
    #include <stdint.h>
    #include <string.h>
    #include <functional>
    #include <iostream>
    #include <list>
    #include <memory>
    #include <string>
    #include <utility>
    
    #include "util/frame_capture_test_utils.h"
    
    // Build the right context header based on replay ID
    // This will expand to "angle_capture_context<#>.h"
    #include ANGLE_MACRO_STRINGIZE(ANGLE_CAPTURE_REPLAY_COMPOSITE_TESTS_HEADER)
    
    const std::string resultTag = "*RESULT";
    
    class CaptureReplayTests
    {
      public:
        CaptureReplayTests(EGLint glesMajorVersion, EGLint glesMinorVersion)
            : mOSWindow(nullptr), mEGLWindow(nullptr)
        {
            mPlatformParams.renderer   = EGL_PLATFORM_ANGLE_TYPE_VULKAN_ANGLE;
            mPlatformParams.deviceType = EGL_PLATFORM_ANGLE_DEVICE_TYPE_HARDWARE_ANGLE;
    
            // Load EGL library so we can initialize the display.
            mEntryPointsLib.reset(
                angle::OpenSharedLibrary(ANGLE_EGL_LIBRARY_NAME, angle::SearchType::ApplicationDir));
            mEGLWindow = EGLWindow::New(glesMajorVersion, glesMinorVersion);
            mOSWindow  = OSWindow::New();
            mOSWindow->disableErrorMessageDialog();
        }
    
        ~CaptureReplayTests()
        {
            EGLWindow::Delete(&mEGLWindow);
            OSWindow::Delete(&mOSWindow);
        }
    
        bool initializeTest(uint32_t testIndex, TestTraceInfo &testTraceInfo)
        {
            if (!mOSWindow->initialize(testTraceInfo.testName, testTraceInfo.replayDrawSurfaceWidth,
                                       testTraceInfo.replayDrawSurfaceHeight))
            {
                return false;
            }
            mOSWindow->disableErrorMessageDialog();
            mOSWindow->setVisible(true);
    
            ConfigParameters configParams;
            configParams.redBits     = testTraceInfo.defaultFramebufferRedBits;
            configParams.greenBits   = testTraceInfo.defaultFramebufferGreenBits;
            configParams.blueBits    = testTraceInfo.defaultFramebufferBlueBits;
            configParams.alphaBits   = testTraceInfo.defaultFramebufferAlphaBits;
            configParams.depthBits   = testTraceInfo.defaultFramebufferDepthBits;
            configParams.stencilBits = testTraceInfo.defaultFramebufferStencilBits;
            if (!mEGLWindow->initializeGL(mOSWindow, mEntryPointsLib.get(),
                                          angle::GLESDriverType::AngleEGL, mPlatformParams,
                                          configParams))
            {
                mOSWindow->destroy();
                return false;
            }
            // Disable vsync
            if (!mEGLWindow->setSwapInterval(0))
            {
                cleanupTest();
                return false;
            }
            // Set CWD to executable directory.
            std::string exeDir = angle::GetExecutableDirectory();
            if (!angle::SetCWD(exeDir.c_str()))
            {
                cleanupTest();
                return false;
            }
            if (testTraceInfo.isBinaryDataCompressed)
            {
                SetBinaryDataDecompressCallback(testIndex, angle::DecompressBinaryData);
            }
            SetBinaryDataDir(testIndex, ANGLE_CAPTURE_REPLAY_TEST_DATA_DIR);
    
            SetupContextReplay(testIndex);
            return true;
        }
    
        void cleanupTest()
        {
            mEGLWindow->destroyGL();
            mOSWindow->destroy();
        }
    
        void swap() { mEGLWindow->swap(); }
    
        int runTest(uint32_t testIndex, TestTraceInfo &testTraceInfo)
        {
            if (!initializeTest(testIndex, testTraceInfo))
            {
                return -1;
            }
    
            for (uint32_t frame = testTraceInfo.replayFrameStart; frame <= testTraceInfo.replayFrameEnd;
                 frame++)
            {
                ReplayContextFrame(testIndex, frame);
                gl::Context *context = static_cast<gl::Context *>(mEGLWindow->getContext());
                gl::BinaryOutputStream bos;
                if (angle::SerializeContext(&bos, context) != angle::Result::Continue)
                {
                    cleanupTest();
                    return -1;
                }
                bool isEqual = compareSerializedStates(testIndex, frame, bos);
                if (!isEqual)
                {
                    cleanupTest();
                    return -1;
                }
                swap();
            }
            cleanupTest();
            return 0;
        }
    
        int run()
        {
            for (size_t i = 0; i < testTraceInfos.size(); i++)
            {
                int result = runTest(static_cast<uint32_t>(i), testTraceInfos[i]);
                std::cout << resultTag << " " << testTraceInfos[i].testName << " " << result << "\n";
            }
            return 0;
        }
    
      private:
        bool compareSerializedStates(uint32_t testIndex,
                                     uint32_t frame,
                                     const gl::BinaryOutputStream &replaySerializedContextData)
        {
            return GetSerializedContextStateData(testIndex, frame) ==
                   replaySerializedContextData.getData();
        }
    
        OSWindow *mOSWindow;
        EGLWindow *mEGLWindow;
        EGLPlatformParameters mPlatformParams;
        // Handle to the entry point binding library.
        std::unique_ptr<angle::Library> mEntryPointsLib;
    };
    
    int main(int argc, char **argv)
    {
        // TODO (nguyenmh): http://anglebug.com/4759: initialize app with arguments taken from cmdline
        const EGLint glesMajorVersion = 2;
        const GLint glesMinorVersion  = 0;
        CaptureReplayTests app(glesMajorVersion, glesMinorVersion);
        return app.run();
    }