Skip to content

Commit

Permalink
GS/D3D12: Disable DX12 if feature level is not detected.
Browse files Browse the repository at this point in the history
Intel Haswell doesn't actually support DX12 even tho the device is created which results in a crash,
to get around this check if device can be created using feature level 12 (skylake+).
  • Loading branch information
lightningterror committed Oct 22, 2024
1 parent cce6580 commit 85f1fa5
Show file tree
Hide file tree
Showing 2 changed files with 15 additions and 11 deletions.
22 changes: 13 additions & 9 deletions pcsx2/GS/Renderers/DX12/GSDevice12.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -149,7 +149,7 @@ u32 GSDevice12::GetAdapterVendorID() const
return desc.VendorId;
}

bool GSDevice12::CreateDevice()
bool GSDevice12::CreateDevice(u32& vendor_id)
{
bool enable_debug_layer = GSConfig.UseDebugDevice;

Expand All @@ -158,6 +158,7 @@ bool GSDevice12::CreateDevice()
return false;

m_adapter = D3D::GetAdapterByName(m_dxgi_factory.get(), GSConfig.Adapter);
vendor_id = GetAdapterVendorID();

HRESULT hr;

Expand All @@ -177,8 +178,11 @@ bool GSDevice12::CreateDevice()
}
}

// Intel Haswell doesn't actually support DX12 even tho the device is created which results in a crash,
// to get around this check if device can be created using feature level 12 (skylake+).
const bool isIntel = (vendor_id == 0x163C || vendor_id == 0x8086 || vendor_id == 0x8087);
// Create the actual device.
hr = D3D12CreateDevice(m_adapter.get(), D3D_FEATURE_LEVEL_11_0, IID_PPV_ARGS(&m_device));
hr = D3D12CreateDevice(m_adapter.get(), isIntel ? D3D_FEATURE_LEVEL_12_0 : D3D_FEATURE_LEVEL_11_0, IID_PPV_ARGS(&m_device));
if (FAILED(hr))
{
Console.Error("Failed to create D3D12 device: %08X", hr);
Expand Down Expand Up @@ -681,10 +685,11 @@ bool GSDevice12::Create(GSVSyncMode vsync_mode, bool allow_present_throttle)
if (!GSDevice::Create(vsync_mode, allow_present_throttle))
return false;

if (!CreateDevice())
u32 vendor_id = 0;
if (!CreateDevice(vendor_id))
return false;

if (!CheckFeatures())
if (!CheckFeatures(vendor_id))
{
Console.Error("Your GPU does not support the required D3D12 features.");
return false;
Expand Down Expand Up @@ -1010,7 +1015,7 @@ std::string GSDevice12::GetDriverInfo() const
{
std::string ret = "Unknown Feature Level";

static constexpr std::array<std::tuple<D3D_FEATURE_LEVEL, const char*>, 4> feature_level_names = {{
static constexpr std::array<std::tuple<D3D_FEATURE_LEVEL, const char*>, 2> feature_level_names = {{
{D3D_FEATURE_LEVEL_11_0, "D3D_FEATURE_LEVEL_11_0"},
{D3D_FEATURE_LEVEL_11_1, "D3D_FEATURE_LEVEL_11_1"},
}};
Expand Down Expand Up @@ -1206,10 +1211,9 @@ void GSDevice12::InsertDebugMessage(DebugMessageCategory category, const char* f
#endif
}

bool GSDevice12::CheckFeatures()
bool GSDevice12::CheckFeatures(const u32& vendor_id)
{
const u32 vendorID = GetAdapterVendorID();
const bool isAMD = (vendorID == 0x1002 || vendorID == 0x1022);
const bool isAMD = (vendor_id == 0x1002 || vendor_id == 0x1022);

m_features.texture_barrier = false;
m_features.broken_point_sampler = isAMD;
Expand All @@ -1232,7 +1236,7 @@ bool GSDevice12::CheckFeatures()
m_max_texture_size = D3D12_REQ_TEXTURE2D_U_OR_V_DIMENSION;

BOOL allow_tearing_supported = false;
HRESULT hr = m_dxgi_factory->CheckFeatureSupport(
const HRESULT hr = m_dxgi_factory->CheckFeatureSupport(
DXGI_FEATURE_PRESENT_ALLOW_TEARING, &allow_tearing_supported, sizeof(allow_tearing_supported));
m_allow_tearing_supported = (SUCCEEDED(hr) && allow_tearing_supported == TRUE);

Expand Down
4 changes: 2 additions & 2 deletions pcsx2/GS/Renderers/DX12/GSDevice12.h
Original file line number Diff line number Diff line change
Expand Up @@ -144,7 +144,7 @@ class GSDevice12 final : public GSDevice
bool has_timestamp_query = false;
};

bool CreateDevice();
bool CreateDevice(u32& vendor_id);
bool CreateDescriptorHeaps();
bool CreateCommandLists();
bool CreateTimestampQuery();
Expand Down Expand Up @@ -373,7 +373,7 @@ class GSDevice12 final : public GSDevice
ComPtr<ID3DBlob> GetUtilityVertexShader(const std::string& source, const char* entry_point);
ComPtr<ID3DBlob> GetUtilityPixelShader(const std::string& source, const char* entry_point);

bool CheckFeatures();
bool CheckFeatures(const u32& vendor_id);
bool CreateNullTexture();
bool CreateBuffers();
bool CreateRootSignatures();
Expand Down

0 comments on commit 85f1fa5

Please sign in to comment.