You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
o3de/Gems/Atom/RHI/Code/Tests/QueryTests.cpp

492 lines
20 KiB
C++

/*
* Copyright (c) Contributors to the Open 3D Engine Project.
* For complete copyright and license terms please see the LICENSE at the root of this distribution.
*
* SPDX-License-Identifier: Apache-2.0 OR MIT
*
*/
#include "RHITestFixture.h"
#include <AzCore/std/containers/bitset.h>
#include <Tests/Factory.h>
#include <Tests/Device.h>
#include <Atom/RHI/FrameEventBus.h>
namespace UnitTest
{
using namespace AZ;
class QueryTests
: public RHITestFixture
{
public:
QueryTests()
: RHITestFixture()
{
}
~QueryTests()
{
}
private:
void SetUp() override
{
RHITestFixture::SetUp();
m_factory.reset(aznew Factory());
m_device = MakeTestDevice();
}
void TearDown() override
{
m_factory.reset();
m_device.reset();
RHITestFixture::TearDown();
}
protected:
AZStd::unique_ptr<Factory> m_factory;
RHI::Ptr<RHI::Device> m_device;
};
TEST_F(QueryTests, TestNoop)
{
RHI::Ptr<RHI::Query> noopQuery;
noopQuery = RHI::Factory::Get().CreateQuery();
AZ_TEST_ASSERT(noopQuery);
RHI::Ptr<RHI::QueryPool> noopQueryPool;
noopQueryPool = RHI::Factory::Get().CreateQueryPool();
AZ_TEST_ASSERT(noopQueryPool);
}
TEST_F(QueryTests, Test)
{
RHI::Ptr<RHI::Query> queryA;
queryA = RHI::Factory::Get().CreateQuery();
queryA->SetName(Name("QueryA"));
AZ_TEST_ASSERT(queryA->GetName().GetStringView() == "QueryA");
AZ_TEST_ASSERT(queryA->use_count() == 1);
{
RHI::Ptr<RHI::QueryPool> queryPool;
queryPool = RHI::Factory::Get().CreateQueryPool();
EXPECT_EQ(1, queryPool->use_count());
RHI::Ptr<RHI::Query> queryB;
queryB = RHI::Factory::Get().CreateQuery();
EXPECT_EQ(1, queryB->use_count());
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = 2;
queryPoolDesc.m_type = RHI::QueryType::Occlusion;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::None;
queryPool->Init(*m_device, queryPoolDesc);
EXPECT_FALSE(queryA->IsInitialized());
EXPECT_FALSE(queryB->IsInitialized());
queryPool->InitQuery(queryA.get());
EXPECT_EQ(1, queryA->use_count());
EXPECT_TRUE(queryA->IsInitialized());
queryPool->InitQuery(queryB.get());
EXPECT_TRUE(queryB->IsInitialized());
EXPECT_EQ(queryA->GetPool(), queryPool.get());
EXPECT_EQ(queryB->GetPool(), queryPool.get());
EXPECT_EQ(queryPool->GetResourceCount(), 2);
{
uint32_t queryIndex = 0;
const RHI::Query* queries[] =
{
queryA.get(),
queryB.get()
};
queryPool->ForEach<RHI::Query>([&queryIndex, &queries]([[maybe_unused]] RHI::Query& query)
{
AZ_UNUSED(queries); // Prevent unused warning in release builds
AZ_Assert(queries[queryIndex] == &query, "Queries don't match");
queryIndex++;
});
}
queryB->Shutdown();
EXPECT_EQ(queryB->GetPool(), nullptr);
RHI::Ptr<RHI::QueryPool> queryPoolB;
queryPoolB = RHI::Factory::Get().CreateQueryPool();
queryPoolB->Init(*m_device, queryPoolDesc);
queryPoolB->InitQuery(queryB.get());
EXPECT_EQ(queryB->GetPool(), queryPoolB.get());
//Since we are switching queryPool for queryB it adds a refcount and invalidates the views.
//We need this to ensure the views are fully invalidated in order to release the refcount and avoid a leak.
RHI::ResourceInvalidateBus::ExecuteQueuedEvents();
queryPoolB->Shutdown();
EXPECT_EQ(queryPoolB->GetResourceCount(), 0);
}
EXPECT_EQ(queryA->GetPool(), nullptr);
EXPECT_EQ(queryA->use_count(), 1);
}
TEST_F(QueryTests, TestAllocations)
{
static const uint32_t numQueries = 10;
AZStd::array<RHI::Ptr<RHI::Query>, numQueries> queries;
for (auto& query : queries)
{
query = RHI::Factory::Get().CreateQuery();
}
RHI::Ptr<RHI::QueryPool> queryPool;
queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = numQueries;
queryPoolDesc.m_type = RHI::QueryType::Occlusion;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::None;
queryPool->Init(*m_device, queryPoolDesc);
AZStd::vector<RHI::Query*> queriesToInitialize(numQueries);
for (size_t i = 0; i < queries.size(); ++i)
{
queriesToInitialize[i] = queries[i].get();
}
RHI::ResultCode result = queryPool->InitQuery(queriesToInitialize.data(), static_cast<uint32_t>(queriesToInitialize.size()));
EXPECT_EQ(result, RHI::ResultCode::Success);
auto checkSlotsFunc = [](const AZStd::vector<RHI::Query*>& queries)
{
if (queries.size() < 2)
{
return true;
}
AZStd::vector<uint32_t> indices;
for (auto& query : queries)
{
indices.push_back(query->GetHandle().GetIndex());
}
AZStd::sort(indices.begin(), indices.end());
for (size_t i = 0; i < indices.size() - 1; ++i)
{
if (indices[i] != (indices[i + 1] + 1))
{
return false;
}
}
return true;
};
checkSlotsFunc(queriesToInitialize);
auto extraQuery = RHI::Factory::Get().CreateQuery();
EXPECT_EQ(queryPool->InitQuery(extraQuery.get()), RHI::ResultCode::OutOfMemory);
AZ_TEST_ASSERT(!extraQuery->IsInitialized());
AZStd::vector<uint32_t> queriesIndicesToShutdown = { 5, 6 };
AZStd::vector<RHI::Query*> queriesToShutdown;
for (auto& index : queriesIndicesToShutdown)
{
queries[index]->Shutdown();
queriesToShutdown.push_back(queries[index].get());
}
EXPECT_EQ(queryPool->GetResourceCount(), numQueries - static_cast<uint32_t>(queriesIndicesToShutdown.size()));
result = queryPool->InitQuery(queriesToShutdown.data(), static_cast<uint32_t>(queriesIndicesToShutdown.size()));
EXPECT_EQ(result, RHI::ResultCode::Success);
checkSlotsFunc(queriesToShutdown);
queriesIndicesToShutdown = {2, 5, 9};
queriesToShutdown.clear();
for (auto& index : queriesIndicesToShutdown)
{
queries[index]->Shutdown();
queriesToShutdown.push_back(queries[index].get());
}
EXPECT_EQ(queryPool->GetResourceCount(), (numQueries - static_cast<uint32_t>(queriesIndicesToShutdown.size())));
result = queryPool->InitQuery(queriesToShutdown.data(), static_cast<uint32_t>(queriesToShutdown.size()));
//Since we are switching queryPools for some queries it adds a refcount and invalidates the views.
//We need to ensure the views are fully invalidated in order to release the refcount and avoid leaks.
RHI::ResourceInvalidateBus::ExecuteQueuedEvents();
checkSlotsFunc(queriesToInitialize);
}
TEST_F(QueryTests, TestIntervals)
{
static const uint32_t numQueries = 10;
AZStd::array<RHI::Ptr<RHI::Query>, numQueries> queries;
for (auto& query : queries)
{
query = RHI::Factory::Get().CreateQuery();
}
RHI::Ptr<RHI::QueryPool> queryPool;
queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = numQueries;
queryPoolDesc.m_type = RHI::QueryType::Occlusion;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::None;
queryPool->Init(*m_device, queryPoolDesc);
AZStd::vector<RHI::Query*> queriesToInitialize(numQueries);
for (size_t i = 0; i < queries.size(); ++i)
{
queriesToInitialize[i] = queries[i].get();
}
RHI::ResultCode result = queryPool->InitQuery(queriesToInitialize.data(), static_cast<uint32_t>(queriesToInitialize.size()));
EXPECT_EQ(result, RHI::ResultCode::Success);
auto* testQueryPool = static_cast<UnitTest::QueryPool*>(queryPool.get());
auto& testQueryPoolIntervals = testQueryPool->m_calledIntervals;
uint64_t results[numQueries] = {};
EXPECT_EQ(queryPool->GetResults(results, numQueries, RHI::QueryResultFlagBits::None), RHI::ResultCode::Success);
EXPECT_EQ(testQueryPoolIntervals.size(), 1);
EXPECT_EQ(testQueryPoolIntervals.front(), RHI::Interval(0, numQueries - 1));
testQueryPoolIntervals.clear();
auto* queryToTest = queries[5].get();
EXPECT_EQ(queryPool->GetResults(queryToTest, results, 1, RHI::QueryResultFlagBits::None), RHI::ResultCode::Success);
EXPECT_EQ(testQueryPoolIntervals.size(), 1);
EXPECT_EQ(testQueryPoolIntervals.front(), RHI::Interval(queryToTest->GetHandle().GetIndex(), queryToTest->GetHandle().GetIndex()));
AZStd::vector<RHI::Interval> intervalsToTest = { RHI::Interval(5, 5), RHI::Interval(0, 3), RHI::Interval(8, 9) };
AZStd::vector<RHI::Query*> queriesToTest;
for (auto& interval : intervalsToTest)
{
for (uint32_t i = interval.m_min; i <= interval.m_max; ++i)
{
queriesToTest.push_back(queries[i].get());
}
}
testQueryPoolIntervals.clear();
EXPECT_EQ(queryPool->GetResults(queriesToTest.data(), static_cast<uint32_t>(queriesToTest.size()), results, numQueries, RHI::QueryResultFlagBits::None), RHI::ResultCode::Success);
EXPECT_EQ(testQueryPoolIntervals.size(), intervalsToTest.size());
for (auto& interval : intervalsToTest)
{
auto foundIt = AZStd::find(testQueryPoolIntervals.begin(), testQueryPoolIntervals.end(), interval);
EXPECT_NE(foundIt, testQueryPoolIntervals.end());
}
}
TEST_F(QueryTests, TestQuery)
{
AZStd::array<RHI::Ptr<RHI::QueryPool>, RHI::QueryTypeCount> queryPools;
for (size_t i = 0; i < queryPools.size(); ++i)
{
auto& queryPool = queryPools[i];
queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = 1;
queryPoolDesc.m_type = static_cast<RHI::QueryType>(i);
queryPoolDesc.m_pipelineStatisticsMask = queryPoolDesc.m_type == RHI::QueryType::PipelineStatistics ? RHI::PipelineStatisticsFlags::CInvocations : RHI::PipelineStatisticsFlags::None;
queryPool->Init(*m_device, queryPoolDesc);
}
auto& occlusionQueryPool = queryPools[static_cast<uint32_t>(RHI::QueryType::Occlusion)];
auto& timestampQueryPool = queryPools[static_cast<uint32_t>(RHI::QueryType::Timestamp)];
auto& statisticsQueryPool = queryPools[static_cast<uint32_t>(RHI::QueryType::PipelineStatistics)];
uint64_t data;
RHI::CommandList& dummyCommandList = reinterpret_cast<RHI::CommandList&>(data);
// Correct begin and end for occlusion
{
auto query = RHI::Factory::Get().CreateQuery();
EXPECT_EQ(occlusionQueryPool->InitQuery(query.get()), RHI::ResultCode::Success);
EXPECT_EQ(query->Begin(dummyCommandList), RHI::ResultCode::Success);
EXPECT_EQ(query->End(dummyCommandList), RHI::ResultCode::Success);
}
// Double Begin
{
auto query = RHI::Factory::Get().CreateQuery();
occlusionQueryPool->InitQuery(query.get());
query->Begin(dummyCommandList);
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::Fail, query->Begin(dummyCommandList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// End without Begin
{
auto query = RHI::Factory::Get().CreateQuery();
occlusionQueryPool->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::Fail, query->End(dummyCommandList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// End with another command list
{
auto query = RHI::Factory::Get().CreateQuery();
occlusionQueryPool->InitQuery(query.get());
uint64_t anotherData;
RHI::CommandList& anotherDummyCmdList = reinterpret_cast<RHI::CommandList&>(anotherData);
EXPECT_EQ(RHI::ResultCode::Success, query->Begin(dummyCommandList));
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::InvalidArgument, query->End(anotherDummyCmdList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// Invalid flag
{
auto query = RHI::Factory::Get().CreateQuery();
statisticsQueryPool->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::InvalidArgument, query->Begin(dummyCommandList, RHI::QueryControlFlags::PreciseOcclusion));
AZ_TEST_STOP_ASSERTTEST(1);
}
// Invalid Begin for Timestamp
{
auto query = RHI::Factory::Get().CreateQuery();
timestampQueryPool->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::Fail, query->Begin(dummyCommandList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// Invalid End for Timestamp
{
auto query = RHI::Factory::Get().CreateQuery();
timestampQueryPool->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::Fail, query->End(dummyCommandList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// Invalid WriteTimestamp
{
auto query = RHI::Factory::Get().CreateQuery();
occlusionQueryPool->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(RHI::ResultCode::Fail, query->WriteTimestamp(dummyCommandList));
AZ_TEST_STOP_ASSERTTEST(1);
}
// Correct WriteTimestamp
{
auto query = RHI::Factory::Get().CreateQuery();
timestampQueryPool->InitQuery(query.get());
EXPECT_EQ(RHI::ResultCode::Success, query->WriteTimestamp(dummyCommandList));
}
}
TEST_F(QueryTests, TestQueryPoolInitialization)
{
RHI::Ptr<RHI::QueryPool> queryPool;
queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = 0;
queryPoolDesc.m_type = RHI::QueryType::Occlusion;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::None;
// Count of 0
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPool->Init(*m_device, queryPoolDesc), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(1);
// valid m_pipelineStatisticsMask for Occlusion QueryType
queryPoolDesc.m_queriesCount = 1;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::CInvocations;
EXPECT_EQ(queryPool->Init(*m_device, queryPoolDesc), RHI::ResultCode::Success);
// invalid m_pipelineStatisticsMask for PipelineStatistics QueryType
queryPoolDesc.m_type = RHI::QueryType::PipelineStatistics;
queryPoolDesc.m_pipelineStatisticsMask = RHI::PipelineStatisticsFlags::None;
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPool->Init(*m_device, queryPoolDesc), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(1);
}
TEST_F(QueryTests, TestResults)
{
AZStd::array<RHI::Ptr<RHI::QueryPool>, 2> queryPools;
RHI::PipelineStatisticsFlags mask = RHI::PipelineStatisticsFlags::CInvocations | RHI::PipelineStatisticsFlags::CPrimitives | RHI::PipelineStatisticsFlags::IAPrimitives;
for (auto& queryPool : queryPools)
{
queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = 2;
queryPoolDesc.m_type = RHI::QueryType::PipelineStatistics;
queryPoolDesc.m_pipelineStatisticsMask = mask;
EXPECT_EQ(queryPool->Init(*m_device, queryPoolDesc), RHI::ResultCode::Success);
}
auto query = RHI::Factory::Get().CreateQuery();
uint32_t numPipelineStatistics = RHI::CountBitsSet(static_cast<uint64_t>(mask));
AZStd::vector<uint64_t> results(numPipelineStatistics * 2);
// Using uninitialized query
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPools[0]->GetResults(results.data(), numPipelineStatistics, RHI::QueryResultFlagBits::None), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(3);
// Wrong size for results count.
queryPools[0]->InitQuery(query.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPools[0]->GetResults(results.data(), 1, RHI::QueryResultFlagBits::None), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(1);
// Using a query from another pool
auto anotherQuery = RHI::Factory::Get().CreateQuery();
queryPools[1]->InitQuery(anotherQuery.get());
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPools[0]->GetResults(anotherQuery.get(), results.data(), numPipelineStatistics, RHI::QueryResultFlagBits::None), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(1);
// Results count is too small
anotherQuery->Shutdown();
queryPools[0]->InitQuery(anotherQuery.get());
RHI::Query* queries[] = { query.get(), anotherQuery.get() };
AZ_TEST_START_ASSERTTEST;
EXPECT_EQ(queryPools[0]->GetResults(queries, 2, results.data(), numPipelineStatistics, RHI::QueryResultFlagBits::None), RHI::ResultCode::InvalidArgument);
AZ_TEST_STOP_ASSERTTEST(1);
// Correct usage
EXPECT_EQ(queryPools[0]->GetResults(queries, 2, results.data(), numPipelineStatistics * 5, RHI::QueryResultFlagBits::None), RHI::ResultCode::Success);
// Unsorted queries
{
const size_t numQueries = 5;
AZStd::array<RHI::Ptr<AZ::RHI::Query>, numQueries> queries2;
AZStd::vector<uint64_t> results2(numQueries);
RHI::Ptr<RHI::QueryPool> queryPool = RHI::Factory::Get().CreateQueryPool();
RHI::QueryPoolDescriptor queryPoolDesc;
queryPoolDesc.m_queriesCount = 5;
queryPoolDesc.m_type = RHI::QueryType::Occlusion;
EXPECT_EQ(queryPool->Init(*m_device, queryPoolDesc), RHI::ResultCode::Success);
for (size_t i = 0; i < queries2.size(); ++i)
{
queries2[i] = RHI::Factory::Get().CreateQuery();
queryPool->InitQuery(queries2[i].get());
}
AZStd::array<RHI::Query*, numQueries> queriesPtr = { queries2[2].get(), queries2[0].get(), queries2[1].get(), queries2[3].get(), queries2[4].get() };
EXPECT_EQ(queryPool->GetResults(queriesPtr.data(), numQueries, results2.data(), numQueries, RHI::QueryResultFlagBits::None), RHI::ResultCode::Success);
for (uint32_t i = 0; i < numQueries; ++i)
{
EXPECT_EQ(results2[i], queriesPtr[i]->GetHandle().GetIndex());
}
}
//Since we are switching queryPools for some queries it adds a refcount and invalidates the views.
//We need to ensure the views are fully invalidated in order to release the refcount and avoid leaks.
RHI::ResourceInvalidateBus::ExecuteQueuedEvents();
}
}