2017-12-29 22:20:31 +01:00
|
|
|
#include "d3d11_device.h"
|
|
|
|
#include "d3d11_query.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
D3D11Query::D3D11Query(
|
|
|
|
D3D11Device* device,
|
|
|
|
const D3D11_QUERY_DESC& desc)
|
2018-08-12 00:20:17 +02:00
|
|
|
: m_device(device), m_desc(desc), m_d3d10(this) {
|
2018-02-18 22:34:23 +01:00
|
|
|
switch (m_desc.Query) {
|
2018-01-24 21:28:43 -08:00
|
|
|
case D3D11_QUERY_EVENT:
|
2018-02-18 22:34:23 +01:00
|
|
|
m_event = new DxvkEvent();
|
|
|
|
break;
|
|
|
|
|
2018-01-13 23:35:33 -08:00
|
|
|
case D3D11_QUERY_OCCLUSION:
|
2018-02-19 11:27:14 +01:00
|
|
|
m_query = new DxvkQuery(
|
|
|
|
VK_QUERY_TYPE_OCCLUSION,
|
|
|
|
VK_QUERY_CONTROL_PRECISE_BIT);
|
|
|
|
break;
|
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
case D3D11_QUERY_OCCLUSION_PREDICATE:
|
2018-02-19 11:27:14 +01:00
|
|
|
m_query = new DxvkQuery(
|
|
|
|
VK_QUERY_TYPE_OCCLUSION, 0);
|
2018-02-18 22:34:23 +01:00
|
|
|
break;
|
|
|
|
|
2018-01-22 21:39:54 -08:00
|
|
|
case D3D11_QUERY_TIMESTAMP:
|
2018-02-19 11:27:14 +01:00
|
|
|
m_query = new DxvkQuery(
|
|
|
|
VK_QUERY_TYPE_TIMESTAMP, 0);
|
2018-02-18 22:34:23 +01:00
|
|
|
break;
|
|
|
|
|
2018-01-22 21:39:54 -08:00
|
|
|
case D3D11_QUERY_TIMESTAMP_DISJOINT:
|
2018-01-13 23:35:33 -08:00
|
|
|
break;
|
2018-02-18 22:34:23 +01:00
|
|
|
|
|
|
|
case D3D11_QUERY_PIPELINE_STATISTICS:
|
2018-02-19 11:27:14 +01:00
|
|
|
m_query = new DxvkQuery(
|
|
|
|
VK_QUERY_TYPE_PIPELINE_STATISTICS, 0);
|
2018-02-18 22:34:23 +01:00
|
|
|
break;
|
|
|
|
|
2018-01-13 23:35:33 -08:00
|
|
|
default:
|
2018-02-18 22:34:23 +01:00
|
|
|
throw DxvkError(str::format("D3D11: Unhandled query type: ", desc.Query));
|
2018-01-13 23:35:33 -08:00
|
|
|
}
|
2017-12-29 22:20:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
D3D11Query::~D3D11Query() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE D3D11Query::QueryInterface(REFIID riid, void** ppvObject) {
|
2018-04-02 12:52:02 +02:00
|
|
|
*ppvObject = nullptr;
|
2017-12-29 22:20:31 +01:00
|
|
|
|
2018-04-02 12:52:02 +02:00
|
|
|
if (riid == __uuidof(IUnknown)
|
|
|
|
|| riid == __uuidof(ID3D11DeviceChild)
|
|
|
|
|| riid == __uuidof(ID3D11Asynchronous)
|
|
|
|
|| riid == __uuidof(ID3D11Query)) {
|
|
|
|
*ppvObject = ref(this);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2018-08-12 00:20:17 +02:00
|
|
|
if (riid == __uuidof(IUnknown)
|
|
|
|
|| riid == __uuidof(ID3D10DeviceChild)
|
|
|
|
|| riid == __uuidof(ID3D10Asynchronous)
|
|
|
|
|| riid == __uuidof(ID3D10Query)) {
|
|
|
|
*ppvObject = ref(&m_d3d10);
|
2018-04-02 12:52:02 +02:00
|
|
|
return S_OK;
|
|
|
|
}
|
2018-02-18 22:34:23 +01:00
|
|
|
|
2018-08-12 00:20:17 +02:00
|
|
|
if (m_desc.Query == D3D11_QUERY_OCCLUSION_PREDICATE) {
|
|
|
|
if (riid == __uuidof(ID3D10Predicate)) {
|
|
|
|
*ppvObject = ref(this);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (riid == __uuidof(ID3D10Predicate)) {
|
|
|
|
*ppvObject = ref(&m_d3d10);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-29 22:20:31 +01:00
|
|
|
Logger::warn("D3D11Query: Unknown interface query");
|
2018-03-12 14:05:43 +03:00
|
|
|
Logger::warn(str::format(riid));
|
2017-12-29 22:20:31 +01:00
|
|
|
return E_NOINTERFACE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void STDMETHODCALLTYPE D3D11Query::GetDevice(ID3D11Device **ppDevice) {
|
|
|
|
*ppDevice = ref(m_device);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
UINT STDMETHODCALLTYPE D3D11Query::GetDataSize() {
|
|
|
|
switch (m_desc.Query) {
|
|
|
|
case D3D11_QUERY_EVENT:
|
|
|
|
return sizeof(BOOL);
|
|
|
|
|
|
|
|
case D3D11_QUERY_OCCLUSION:
|
|
|
|
return sizeof(UINT64);
|
|
|
|
|
|
|
|
case D3D11_QUERY_TIMESTAMP:
|
|
|
|
return sizeof(UINT64);
|
|
|
|
|
|
|
|
case D3D11_QUERY_TIMESTAMP_DISJOINT:
|
|
|
|
return sizeof(D3D11_QUERY_DATA_TIMESTAMP_DISJOINT);
|
|
|
|
|
|
|
|
case D3D11_QUERY_PIPELINE_STATISTICS:
|
|
|
|
return sizeof(D3D11_QUERY_DATA_PIPELINE_STATISTICS);
|
|
|
|
|
|
|
|
case D3D11_QUERY_OCCLUSION_PREDICATE:
|
|
|
|
return sizeof(BOOL);
|
|
|
|
|
|
|
|
case D3D11_QUERY_SO_STATISTICS:
|
|
|
|
case D3D11_QUERY_SO_STATISTICS_STREAM0:
|
|
|
|
case D3D11_QUERY_SO_STATISTICS_STREAM1:
|
|
|
|
case D3D11_QUERY_SO_STATISTICS_STREAM2:
|
|
|
|
case D3D11_QUERY_SO_STATISTICS_STREAM3:
|
|
|
|
return sizeof(D3D11_QUERY_DATA_SO_STATISTICS);
|
|
|
|
|
|
|
|
case D3D11_QUERY_SO_OVERFLOW_PREDICATE:
|
|
|
|
case D3D11_QUERY_SO_OVERFLOW_PREDICATE_STREAM0:
|
|
|
|
case D3D11_QUERY_SO_OVERFLOW_PREDICATE_STREAM1:
|
|
|
|
case D3D11_QUERY_SO_OVERFLOW_PREDICATE_STREAM2:
|
|
|
|
case D3D11_QUERY_SO_OVERFLOW_PREDICATE_STREAM3:
|
|
|
|
return sizeof(BOOL);
|
|
|
|
}
|
|
|
|
|
|
|
|
Logger::err("D3D11Query: Failed to query data size");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void STDMETHODCALLTYPE D3D11Query::GetDesc(D3D11_QUERY_DESC *pDesc) {
|
|
|
|
*pDesc = m_desc;
|
|
|
|
}
|
|
|
|
|
2018-01-01 20:59:54 +01:00
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
uint32_t D3D11Query::Reset() {
|
|
|
|
if (m_query != nullptr)
|
|
|
|
return m_query->reset();
|
2018-01-01 20:59:54 +01:00
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
if (m_event != nullptr)
|
|
|
|
return m_event->reset();
|
2018-01-01 20:59:54 +01:00
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool D3D11Query::HasBeginEnabled() const {
|
|
|
|
return m_desc.Query == D3D11_QUERY_OCCLUSION
|
|
|
|
|| m_desc.Query == D3D11_QUERY_OCCLUSION_PREDICATE
|
|
|
|
|| m_desc.Query == D3D11_QUERY_PIPELINE_STATISTICS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11Query::Begin(DxvkContext* ctx, uint32_t revision) {
|
|
|
|
m_revision = revision;
|
2018-01-01 20:59:54 +01:00
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
if (m_query != nullptr) {
|
|
|
|
DxvkQueryRevision rev = { m_query, revision };
|
|
|
|
ctx->beginQuery(rev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11Query::End(DxvkContext* ctx) {
|
|
|
|
if (m_query != nullptr) {
|
|
|
|
DxvkQueryRevision rev = { m_query, m_revision };
|
|
|
|
ctx->endQuery(rev);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D3D11Query::Signal(DxvkContext* ctx, uint32_t revision) {
|
2018-01-01 20:59:54 +01:00
|
|
|
switch (m_desc.Query) {
|
2018-02-18 22:57:45 +01:00
|
|
|
case D3D11_QUERY_EVENT: {
|
|
|
|
DxvkEventRevision rev = { m_event, revision };
|
|
|
|
ctx->signalEvent(rev);
|
|
|
|
} break;
|
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
case D3D11_QUERY_TIMESTAMP: {
|
2018-02-18 22:57:45 +01:00
|
|
|
DxvkQueryRevision rev = { m_query, revision };
|
2018-02-18 22:34:23 +01:00
|
|
|
ctx->writeTimestamp(rev);
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HRESULT STDMETHODCALLTYPE D3D11Query::GetData(
|
|
|
|
void* pData,
|
|
|
|
UINT GetDataFlags) {
|
|
|
|
if (m_desc.Query == D3D11_QUERY_EVENT) {
|
2018-02-26 14:33:27 +01:00
|
|
|
const bool signaled = m_event->getStatus() == DxvkEventStatus::Signaled;
|
2018-02-27 10:14:53 +01:00
|
|
|
|
2018-02-26 14:33:27 +01:00
|
|
|
if (pData != nullptr)
|
|
|
|
*static_cast<BOOL*>(pData) = signaled;
|
|
|
|
|
|
|
|
return signaled ? S_OK : S_FALSE;
|
2018-02-18 22:34:23 +01:00
|
|
|
} else {
|
2018-02-26 14:33:27 +01:00
|
|
|
DxvkQueryData queryData = {};
|
|
|
|
|
|
|
|
if (m_query != nullptr
|
|
|
|
&& m_query->getData(queryData) != DxvkQueryStatus::Available)
|
|
|
|
return S_FALSE;
|
2018-02-18 22:34:23 +01:00
|
|
|
|
|
|
|
if (pData == nullptr)
|
2018-01-01 20:59:54 +01:00
|
|
|
return S_OK;
|
2018-02-18 22:34:23 +01:00
|
|
|
|
|
|
|
switch (m_desc.Query) {
|
|
|
|
case D3D11_QUERY_OCCLUSION:
|
2018-02-26 14:33:27 +01:00
|
|
|
*static_cast<UINT64*>(pData) = queryData.occlusion.samplesPassed;
|
2018-02-18 22:34:23 +01:00
|
|
|
return S_OK;
|
|
|
|
|
|
|
|
case D3D11_QUERY_OCCLUSION_PREDICATE:
|
2018-02-26 14:33:27 +01:00
|
|
|
*static_cast<BOOL*>(pData) = queryData.occlusion.samplesPassed != 0;
|
2018-02-18 22:34:23 +01:00
|
|
|
return S_OK;
|
|
|
|
|
|
|
|
case D3D11_QUERY_TIMESTAMP:
|
2018-02-26 14:33:27 +01:00
|
|
|
*static_cast<UINT64*>(pData) = queryData.timestamp.time;
|
2018-02-18 22:34:23 +01:00
|
|
|
return S_OK;
|
|
|
|
|
|
|
|
case D3D11_QUERY_TIMESTAMP_DISJOINT: {
|
|
|
|
auto data = static_cast<D3D11_QUERY_DATA_TIMESTAMP_DISJOINT*>(pData);
|
2018-06-13 14:29:58 +02:00
|
|
|
data->Frequency = GetTimestampQueryFrequency();
|
2018-02-18 22:34:23 +01:00
|
|
|
data->Disjoint = FALSE;
|
|
|
|
} return S_OK;
|
2018-01-01 20:59:54 +01:00
|
|
|
|
2018-02-18 22:34:23 +01:00
|
|
|
case D3D11_QUERY_PIPELINE_STATISTICS: {
|
|
|
|
auto data = static_cast<D3D11_QUERY_DATA_PIPELINE_STATISTICS*>(pData);
|
2018-02-26 14:33:27 +01:00
|
|
|
data->IAVertices = queryData.statistic.iaVertices;
|
|
|
|
data->IAPrimitives = queryData.statistic.iaPrimitives;
|
|
|
|
data->VSInvocations = queryData.statistic.vsInvocations;
|
|
|
|
data->GSInvocations = queryData.statistic.gsInvocations;
|
|
|
|
data->GSPrimitives = queryData.statistic.gsPrimitives;
|
|
|
|
data->CInvocations = queryData.statistic.clipInvocations;
|
|
|
|
data->CPrimitives = queryData.statistic.clipPrimitives;
|
|
|
|
data->PSInvocations = queryData.statistic.fsInvocations;
|
|
|
|
data->HSInvocations = queryData.statistic.tcsPatches;
|
|
|
|
data->DSInvocations = queryData.statistic.tesInvocations;
|
|
|
|
data->CSInvocations = queryData.statistic.csInvocations;
|
2018-02-18 22:34:23 +01:00
|
|
|
} return S_OK;
|
|
|
|
|
|
|
|
default:
|
|
|
|
Logger::err(str::format("D3D11: Unhandled query type in GetData: ", m_desc.Query));
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
2018-01-01 20:59:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-13 14:29:58 +02:00
|
|
|
|
|
|
|
UINT64 D3D11Query::GetTimestampQueryFrequency() const {
|
|
|
|
Rc<DxvkDevice> device = m_device->GetDXVKDevice();
|
|
|
|
Rc<DxvkAdapter> adapter = device->adapter();
|
|
|
|
|
|
|
|
VkPhysicalDeviceLimits limits = adapter->deviceProperties().limits;
|
|
|
|
return uint64_t(1'000'000'000.0f / limits.timestampPeriod);
|
|
|
|
}
|
|
|
|
|
2017-12-29 22:20:31 +01:00
|
|
|
}
|