2018-05-18 16:18:16 +08:00
|
|
|
/*
|
2019-01-10 20:57:40 +08:00
|
|
|
* Copyright (C) 2018-2019 Intel Corporation
|
2018-05-18 16:18:16 +08:00
|
|
|
*
|
2018-09-18 15:11:08 +08:00
|
|
|
* SPDX-License-Identifier: MIT
|
2018-05-18 16:18:16 +08:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-12-10 17:30:39 +08:00
|
|
|
#include "runtime/command_stream/preemption.h"
|
2018-09-12 18:43:15 +08:00
|
|
|
#include "runtime/memory_manager/memory_constants.h"
|
2018-05-18 16:18:16 +08:00
|
|
|
#include "runtime/os_interface/windows/gdi_interface.h"
|
2018-07-16 22:37:17 +08:00
|
|
|
#include "unit_tests/fixtures/gmm_environment_fixture.h"
|
2018-06-21 18:25:59 +08:00
|
|
|
#include "unit_tests/helpers/debug_manager_state_restore.h"
|
2018-08-14 17:05:17 +08:00
|
|
|
#include "unit_tests/mocks/mock_wddm.h"
|
|
|
|
#include "unit_tests/mocks/mock_wddm_interface23.h"
|
2018-05-18 16:18:16 +08:00
|
|
|
#include "unit_tests/os_interface/windows/gdi_dll_fixture.h"
|
2018-08-21 23:36:08 +08:00
|
|
|
#include "runtime/os_interface/windows/os_context_win.h"
|
2018-08-27 21:48:29 +08:00
|
|
|
#include "runtime/os_interface/windows/os_interface.h"
|
2018-05-18 16:18:16 +08:00
|
|
|
#include "test.h"
|
|
|
|
|
|
|
|
using namespace OCLRT;
|
|
|
|
|
2018-08-24 15:27:00 +08:00
|
|
|
struct Wddm23TestsWithoutWddmInit : public ::testing::Test, GdiDllFixture, public GmmEnvironmentFixture {
|
2018-05-18 16:18:16 +08:00
|
|
|
void SetUp() override {
|
2018-07-16 22:37:17 +08:00
|
|
|
GmmEnvironmentFixture::SetUp();
|
2018-05-18 16:18:16 +08:00
|
|
|
GdiDllFixture::SetUp();
|
2018-08-14 15:45:57 +08:00
|
|
|
|
2018-08-27 21:48:29 +08:00
|
|
|
wddm = static_cast<WddmMock *>(Wddm::createWddm());
|
|
|
|
osInterface = std::make_unique<OSInterface>();
|
|
|
|
osInterface->get()->setWddm(wddm);
|
|
|
|
|
2018-08-14 15:45:57 +08:00
|
|
|
wddm->featureTable->ftrWddmHwQueues = true;
|
2018-08-10 22:41:44 +08:00
|
|
|
wddmMockInterface = new WddmMockInterface23(*wddm);
|
|
|
|
wddm->wddmInterface.reset(wddmMockInterface);
|
2018-06-21 18:25:59 +08:00
|
|
|
wddm->registryReader.reset(new RegistryReaderMock());
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-27 21:48:29 +08:00
|
|
|
void init() {
|
2018-12-10 17:30:39 +08:00
|
|
|
auto preemptionMode = PreemptionHelper::getDefaultPreemptionMode(*platformDevices[0]);
|
|
|
|
EXPECT_TRUE(wddm->init(preemptionMode));
|
2019-01-10 20:57:40 +08:00
|
|
|
osContext = std::make_unique<OsContext>(osInterface.get(), 0u, HwHelper::get(platformDevices[0]->pPlatform->eRenderCoreFamily).getGpgpuEngineInstances()[0], preemptionMode);
|
2018-08-27 21:48:29 +08:00
|
|
|
osContextWin = osContext->get();
|
|
|
|
}
|
|
|
|
|
2018-05-18 16:18:16 +08:00
|
|
|
void TearDown() override {
|
|
|
|
GdiDllFixture::TearDown();
|
2018-07-16 22:37:17 +08:00
|
|
|
GmmEnvironmentFixture::TearDown();
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-27 21:48:29 +08:00
|
|
|
std::unique_ptr<OSInterface> osInterface;
|
|
|
|
std::unique_ptr<OsContext> osContext;
|
|
|
|
OsContextWin *osContextWin = nullptr;
|
|
|
|
WddmMock *wddm = nullptr;
|
2018-08-10 22:41:44 +08:00
|
|
|
WddmMockInterface23 *wddmMockInterface = nullptr;
|
2018-05-18 16:18:16 +08:00
|
|
|
};
|
|
|
|
|
2018-08-24 15:27:00 +08:00
|
|
|
struct Wddm23Tests : public Wddm23TestsWithoutWddmInit {
|
|
|
|
using Wddm23TestsWithoutWddmInit::TearDown;
|
|
|
|
void SetUp() override {
|
|
|
|
Wddm23TestsWithoutWddmInit::SetUp();
|
2018-08-27 21:48:29 +08:00
|
|
|
init();
|
2018-08-24 15:27:00 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, whenCreateContextIsCalledThenEnableHwQueues) {
|
2018-08-10 22:41:44 +08:00
|
|
|
EXPECT_TRUE(wddm->wddmInterface->hwQueuesSupported());
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(1u, getCreateContextDataFcn()->Flags.HwQueueSupported);
|
|
|
|
}
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, givenPreemptionModeWhenCreateHwQueueCalledThenSetGpuTimeoutIfEnabled) {
|
2018-12-10 17:30:39 +08:00
|
|
|
wddm->wddmInterface->createHwQueue(PreemptionMode::Disabled, *osContextWin);
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(0u, getCreateHwQueueDataFcn()->Flags.DisableGpuTimeout);
|
|
|
|
|
2018-12-10 17:30:39 +08:00
|
|
|
wddm->wddmInterface->createHwQueue(PreemptionMode::MidBatch, *osContextWin);
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(1u, getCreateHwQueueDataFcn()->Flags.DisableGpuTimeout);
|
|
|
|
}
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, whenDestroyHwQueueCalledThenPassExistingHandle) {
|
2018-08-21 23:36:08 +08:00
|
|
|
D3DKMT_HANDLE hwQueue = 123;
|
2018-08-27 21:48:29 +08:00
|
|
|
osContextWin->setHwQueue(hwQueue);
|
|
|
|
wddmMockInterface->destroyHwQueue(osContextWin->getHwQueue());
|
2018-08-21 23:36:08 +08:00
|
|
|
EXPECT_EQ(hwQueue, getDestroyHwQueueDataFcn()->hHwQueue);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
2018-08-21 23:36:08 +08:00
|
|
|
hwQueue = 0;
|
2018-08-27 21:48:29 +08:00
|
|
|
osContextWin->setHwQueue(hwQueue);
|
|
|
|
wddmMockInterface->destroyHwQueue(osContextWin->getHwQueue());
|
2018-08-21 23:36:08 +08:00
|
|
|
EXPECT_NE(hwQueue, getDestroyHwQueueDataFcn()->hHwQueue); // gdi not called when 0
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, whenObjectIsDestructedThenDestroyHwQueue) {
|
2018-05-18 16:18:16 +08:00
|
|
|
D3DKMT_HANDLE hwQueue = 123;
|
2018-08-27 21:48:29 +08:00
|
|
|
osContextWin->setHwQueue(hwQueue);
|
|
|
|
osContext.reset();
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(hwQueue, getDestroyHwQueueDataFcn()->hHwQueue);
|
|
|
|
}
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, givenCmdBufferWhenSubmitCalledThenSetAllRequiredFiledsAndUpdateMonitoredFence) {
|
2018-05-18 16:18:16 +08:00
|
|
|
uint64_t cmdBufferAddress = 123;
|
|
|
|
size_t cmdSize = 456;
|
2018-08-27 21:48:29 +08:00
|
|
|
auto hwQueue = osContextWin->getHwQueue();
|
2018-05-18 16:18:16 +08:00
|
|
|
COMMAND_BUFFER_HEADER cmdBufferHeader = {};
|
|
|
|
|
2018-10-22 21:59:26 +08:00
|
|
|
EXPECT_EQ(1u, osContextWin->getResidencyController().getMonitoredFence().currentFenceValue);
|
|
|
|
EXPECT_EQ(0u, osContextWin->getResidencyController().getMonitoredFence().lastSubmittedFence);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
2018-08-27 21:48:29 +08:00
|
|
|
wddm->submit(cmdBufferAddress, cmdSize, &cmdBufferHeader, *osContextWin);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
|
|
|
EXPECT_EQ(cmdBufferAddress, getSubmitCommandToHwQueueDataFcn()->CommandBuffer);
|
|
|
|
EXPECT_EQ(static_cast<UINT>(cmdSize), getSubmitCommandToHwQueueDataFcn()->CommandLength);
|
2018-08-21 23:36:08 +08:00
|
|
|
EXPECT_EQ(hwQueue, getSubmitCommandToHwQueueDataFcn()->hHwQueue);
|
2018-10-22 21:59:26 +08:00
|
|
|
EXPECT_EQ(osContextWin->getResidencyController().getMonitoredFence().fenceHandle, getSubmitCommandToHwQueueDataFcn()->HwQueueProgressFenceId);
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(&cmdBufferHeader, getSubmitCommandToHwQueueDataFcn()->pPrivateDriverData);
|
2018-09-12 18:43:15 +08:00
|
|
|
EXPECT_EQ(static_cast<UINT>(MemoryConstants::pageSize), getSubmitCommandToHwQueueDataFcn()->PrivateDriverDataSize);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
2018-10-22 21:59:26 +08:00
|
|
|
EXPECT_EQ(osContextWin->getResidencyController().getMonitoredFence().gpuAddress, cmdBufferHeader.MonitorFenceVA);
|
|
|
|
EXPECT_EQ(osContextWin->getResidencyController().getMonitoredFence().lastSubmittedFence, cmdBufferHeader.MonitorFenceValue);
|
|
|
|
EXPECT_EQ(2u, osContextWin->getResidencyController().getMonitoredFence().currentFenceValue);
|
|
|
|
EXPECT_EQ(1u, osContextWin->getResidencyController().getMonitoredFence().lastSubmittedFence);
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-09-12 18:43:15 +08:00
|
|
|
TEST_F(Wddm23Tests, whenMonitoredFenceIsCreatedThenSetupAllRequiredFields) {
|
2018-10-26 18:22:47 +08:00
|
|
|
wddm->wddmInterface->createMonitoredFence(osContextWin->getResidencyController());
|
2018-09-12 18:43:15 +08:00
|
|
|
|
2018-10-22 21:59:26 +08:00
|
|
|
EXPECT_NE(nullptr, osContextWin->getResidencyController().getMonitoredFence().cpuAddress);
|
|
|
|
EXPECT_EQ(1u, osContextWin->getResidencyController().getMonitoredFence().currentFenceValue);
|
|
|
|
EXPECT_NE(static_cast<D3DKMT_HANDLE>(0), osContextWin->getResidencyController().getMonitoredFence().fenceHandle);
|
|
|
|
EXPECT_NE(static_cast<D3DGPU_VIRTUAL_ADDRESS>(0), osContextWin->getResidencyController().getMonitoredFence().gpuAddress);
|
|
|
|
EXPECT_EQ(0u, osContextWin->getResidencyController().getMonitoredFence().lastSubmittedFence);
|
2018-09-12 18:43:15 +08:00
|
|
|
}
|
|
|
|
|
2018-08-23 17:29:39 +08:00
|
|
|
TEST_F(Wddm23Tests, givenCurrentPendingFenceValueGreaterThanPendingFenceValueWhenSubmitCalledThenCallWaitOnGpu) {
|
2018-05-18 16:18:16 +08:00
|
|
|
uint64_t cmdBufferAddress = 123;
|
|
|
|
size_t cmdSize = 456;
|
|
|
|
COMMAND_BUFFER_HEADER cmdBufferHeader = {};
|
|
|
|
|
|
|
|
*wddm->pagingFenceAddress = 1;
|
|
|
|
wddm->currentPagingFenceValue = 1;
|
2018-08-27 21:48:29 +08:00
|
|
|
wddm->submit(cmdBufferAddress, cmdSize, &cmdBufferHeader, *osContextWin);
|
2018-08-14 17:05:17 +08:00
|
|
|
EXPECT_EQ(0u, wddm->waitOnGPUResult.called);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
|
|
|
wddm->currentPagingFenceValue = 2;
|
2018-08-27 21:48:29 +08:00
|
|
|
wddm->submit(cmdBufferAddress, cmdSize, &cmdBufferHeader, *osContextWin);
|
2018-08-14 17:05:17 +08:00
|
|
|
EXPECT_EQ(1u, wddm->waitOnGPUResult.called);
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-24 15:27:00 +08:00
|
|
|
TEST_F(Wddm23TestsWithoutWddmInit, whenInitCalledThenInitializeNewGdiDDIsAndCallToCreateHwQueue) {
|
2018-05-18 16:18:16 +08:00
|
|
|
EXPECT_EQ(nullptr, wddm->gdi->createHwQueue.mFunc);
|
|
|
|
EXPECT_EQ(nullptr, wddm->gdi->destroyHwQueue.mFunc);
|
|
|
|
EXPECT_EQ(nullptr, wddm->gdi->submitCommandToHwQueue.mFunc);
|
|
|
|
|
2018-08-27 21:48:29 +08:00
|
|
|
init();
|
2018-08-10 22:41:44 +08:00
|
|
|
EXPECT_EQ(1u, wddmMockInterface->createHwQueueCalled);
|
2018-05-18 16:18:16 +08:00
|
|
|
|
|
|
|
EXPECT_NE(nullptr, wddm->gdi->createHwQueue.mFunc);
|
|
|
|
EXPECT_NE(nullptr, wddm->gdi->destroyHwQueue.mFunc);
|
|
|
|
EXPECT_NE(nullptr, wddm->gdi->submitCommandToHwQueue.mFunc);
|
|
|
|
}
|
|
|
|
|
2018-08-24 15:27:00 +08:00
|
|
|
TEST_F(Wddm23TestsWithoutWddmInit, whenCreateHwQueueFailedThenReturnFalseFromInit) {
|
2018-08-10 22:41:44 +08:00
|
|
|
wddmMockInterface->forceCreateHwQueueFail = true;
|
2018-08-27 21:48:29 +08:00
|
|
|
init();
|
|
|
|
EXPECT_FALSE(osContextWin->isInitialized());
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-24 15:27:00 +08:00
|
|
|
TEST_F(Wddm23TestsWithoutWddmInit, givenFailureOnGdiInitializationWhenCreatingHwQueueThenReturnFailure) {
|
2018-05-18 16:18:16 +08:00
|
|
|
struct MyMockGdi : public Gdi {
|
|
|
|
bool setupHwQueueProcAddresses() override {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
2018-08-10 22:41:44 +08:00
|
|
|
auto myMockGdi = new MyMockGdi();
|
|
|
|
wddm->gdi.reset(myMockGdi);
|
2018-08-27 21:48:29 +08:00
|
|
|
init();
|
|
|
|
EXPECT_FALSE(osContextWin->isInitialized());
|
2018-08-10 22:41:44 +08:00
|
|
|
EXPECT_EQ(1u, wddmMockInterface->createHwQueueCalled);
|
|
|
|
EXPECT_FALSE(wddmMockInterface->createHwQueueResult);
|
2018-05-18 16:18:16 +08:00
|
|
|
}
|