Check surface support when choosing device
This commit is contained in:
@@ -9,12 +9,16 @@
|
|||||||
#include <nova/api.h>
|
#include <nova/api.h>
|
||||||
#include <nova/types.h>
|
#include <nova/types.h>
|
||||||
|
|
||||||
|
#include <initializer_list>
|
||||||
|
#include <span>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace Nova {
|
namespace Nova {
|
||||||
struct RenderDevice {
|
using SurfaceID = uptr;
|
||||||
static constexpr u32 AUTO = static_cast<u32>(-1);
|
|
||||||
|
class RenderDriver;
|
||||||
|
|
||||||
|
struct NOVA_API RenderDevice {
|
||||||
enum class Vendor { UNKNOWN = 0, INTEL = 0x8086, AMD = 0x1002, NVIDIA = 0x10de };
|
enum class Vendor { UNKNOWN = 0, INTEL = 0x8086, AMD = 0x1002, NVIDIA = 0x10de };
|
||||||
enum class Type { OTHER = 0, INTEGRATED = 1, DISCRETE = 2, VIRTUAL = 3, CPU = 4 };
|
enum class Type { OTHER = 0, INTEGRATED = 1, DISCRETE = 2, VIRTUAL = 3, CPU = 4 };
|
||||||
|
|
||||||
@@ -24,6 +28,12 @@ namespace Nova {
|
|||||||
u32 deviceID;
|
u32 deviceID;
|
||||||
void* handle;
|
void* handle;
|
||||||
|
|
||||||
NOVA_API static u32 choose_device(const std::vector<RenderDevice>& devices);
|
static u32 choose_device(RenderDriver* driver, std::span<const SurfaceID> surfaces = {});
|
||||||
|
static u32 choose_device(RenderDriver* driver, std::initializer_list<SurfaceID> surfaces) {
|
||||||
|
return choose_device(driver, {surfaces.begin(), surfaces.end()});
|
||||||
|
}
|
||||||
|
static u32 choose_device(RenderDriver* driver, SurfaceID surface) {
|
||||||
|
return choose_device(driver, {surface});
|
||||||
|
}
|
||||||
};
|
};
|
||||||
} // namespace Nova
|
} // namespace Nova
|
||||||
|
|||||||
@@ -8,7 +8,6 @@
|
|||||||
|
|
||||||
#include <nova/api.h>
|
#include <nova/api.h>
|
||||||
#include <nova/render/render_api.h>
|
#include <nova/render/render_api.h>
|
||||||
#include <nova/render/render_device.h>
|
|
||||||
#include <nova/types.h>
|
#include <nova/types.h>
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
@@ -17,6 +16,7 @@ namespace Nova {
|
|||||||
using WindowID = uptr;
|
using WindowID = uptr;
|
||||||
using SurfaceID = uptr;
|
using SurfaceID = uptr;
|
||||||
|
|
||||||
|
struct RenderDevice;
|
||||||
class WindowDriver;
|
class WindowDriver;
|
||||||
|
|
||||||
class NOVA_API RenderDriver {
|
class NOVA_API RenderDriver {
|
||||||
@@ -31,6 +31,7 @@ namespace Nova {
|
|||||||
|
|
||||||
[[nodiscard]] virtual u32 get_device_count() const = 0;
|
[[nodiscard]] virtual u32 get_device_count() const = 0;
|
||||||
[[nodiscard]] virtual const RenderDevice& get_device(u32 index) const = 0;
|
[[nodiscard]] virtual const RenderDevice& get_device(u32 index) const = 0;
|
||||||
|
[[nodiscard]] virtual bool get_device_supports_surface(u32 index, SurfaceID surface) const = 0;
|
||||||
virtual void select_device(u32 index) = 0;
|
virtual void select_device(u32 index) = 0;
|
||||||
|
|
||||||
[[nodiscard]] virtual SurfaceID create_surface(WindowID window) = 0;
|
[[nodiscard]] virtual SurfaceID create_surface(WindowID window) = 0;
|
||||||
|
|||||||
@@ -10,6 +10,7 @@
|
|||||||
|
|
||||||
#include <nova/core/debug.h>
|
#include <nova/core/debug.h>
|
||||||
#include <nova/platform/window_driver.h>
|
#include <nova/platform/window_driver.h>
|
||||||
|
#include <nova/render/render_device.h>
|
||||||
#include <nova/version.h>
|
#include <nova/version.h>
|
||||||
#include <vulkan/vulkan.h>
|
#include <vulkan/vulkan.h>
|
||||||
|
|
||||||
@@ -73,15 +74,26 @@ const RenderDevice& VulkanRenderDriver::get_device(const u32 index) const {
|
|||||||
return m_devices[index];
|
return m_devices[index];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool VulkanRenderDriver::get_device_supports_surface(const u32 index, const SurfaceID surface) const {
|
||||||
|
NOVA_AUTO_TRACE();
|
||||||
|
NOVA_ASSERT(index < m_devices.size());
|
||||||
|
NOVA_ASSERT(surface);
|
||||||
|
|
||||||
|
// TODO: Check other queue families?
|
||||||
|
|
||||||
|
SurfaceData* data = reinterpret_cast<SurfaceData*>(surface);
|
||||||
|
VkBool32 supported = false;
|
||||||
|
if (vkGetPhysicalDeviceSurfaceSupportKHR(static_cast<VkPhysicalDevice>(m_devices[index].handle), 0, data->handle, &supported)
|
||||||
|
!= VK_SUCCESS) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return supported;
|
||||||
|
}
|
||||||
|
|
||||||
void VulkanRenderDriver::select_device(u32 index) {
|
void VulkanRenderDriver::select_device(u32 index) {
|
||||||
NOVA_AUTO_TRACE();
|
NOVA_AUTO_TRACE();
|
||||||
NOVA_ASSERT(!m_device);
|
NOVA_ASSERT(!m_device);
|
||||||
|
|
||||||
if (index == RenderDevice::AUTO) {
|
|
||||||
index = RenderDevice::choose_device(m_devices);
|
|
||||||
} else {
|
|
||||||
NOVA_ASSERT(index < m_devices.size());
|
NOVA_ASSERT(index < m_devices.size());
|
||||||
}
|
|
||||||
|
|
||||||
NOVA_LOG("Using device: {}", m_devices[index].name);
|
NOVA_LOG("Using device: {}", m_devices[index].name);
|
||||||
m_physical_device = static_cast<VkPhysicalDevice>(m_devices[index].handle);
|
m_physical_device = static_cast<VkPhysicalDevice>(m_devices[index].handle);
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ namespace Nova {
|
|||||||
|
|
||||||
[[nodiscard]] u32 get_device_count() const override;
|
[[nodiscard]] u32 get_device_count() const override;
|
||||||
[[nodiscard]] const RenderDevice& get_device(u32 index) const override;
|
[[nodiscard]] const RenderDevice& get_device(u32 index) const override;
|
||||||
|
[[nodiscard]] bool get_device_supports_surface(u32 index, SurfaceID surface) const override;
|
||||||
void select_device(u32 index) override;
|
void select_device(u32 index) override;
|
||||||
|
|
||||||
[[nodiscard]] SurfaceID create_surface(WindowID window) override;
|
[[nodiscard]] SurfaceID create_surface(WindowID window) override;
|
||||||
|
|||||||
@@ -6,18 +6,31 @@
|
|||||||
|
|
||||||
#include <nova/core/debug.h>
|
#include <nova/core/debug.h>
|
||||||
#include <nova/render/render_device.h>
|
#include <nova/render/render_device.h>
|
||||||
|
#include <nova/render/render_driver.h>
|
||||||
|
|
||||||
using namespace Nova;
|
using namespace Nova;
|
||||||
|
|
||||||
u32 RenderDevice::choose_device(const std::vector<RenderDevice>& devices) {
|
u32 RenderDevice::choose_device(RenderDriver* driver, std::span<const SurfaceID> surfaces) {
|
||||||
NOVA_AUTO_TRACE();
|
NOVA_AUTO_TRACE();
|
||||||
|
|
||||||
u32 best_index = 0;
|
u32 best_index = -1;
|
||||||
u32 best_score = 0;
|
u32 best_score = 0;
|
||||||
|
|
||||||
for (u32 i = 0; i < devices.size(); i++) {
|
for (u32 i = 0; i < driver->get_device_count(); i++) {
|
||||||
u32 score = 0;
|
auto& device = driver->get_device(i);
|
||||||
switch (devices[i].type) {
|
u32 score = 1;
|
||||||
|
|
||||||
|
for (SurfaceID surface : surfaces) {
|
||||||
|
if (!driver->get_device_supports_surface(i, surface)) {
|
||||||
|
score = 0;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (score == 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (device.type) {
|
||||||
case Type::DISCRETE:
|
case Type::DISCRETE:
|
||||||
score += 4;
|
score += 4;
|
||||||
break;
|
break;
|
||||||
@@ -33,11 +46,16 @@ u32 RenderDevice::choose_device(const std::vector<RenderDevice>& devices) {
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (score > best_score) {
|
if (score > best_score) {
|
||||||
best_index = i;
|
best_index = i;
|
||||||
best_score = score;
|
best_score = score;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (best_index == -1U) {
|
||||||
|
throw std::runtime_error("No suitable render device found");
|
||||||
|
}
|
||||||
|
|
||||||
return best_index;
|
return best_index;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user