2018-11-12 20:53:23 +01:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "../util/log/log.h"
|
|
|
|
|
|
|
|
#include "../util/util_error.h"
|
|
|
|
#include "../util/util_math.h"
|
|
|
|
#include "../util/util_string.h"
|
|
|
|
|
|
|
|
#include "vulkan_loader.h"
|
|
|
|
|
|
|
|
namespace dxvk::vk {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Presenter description
|
|
|
|
*
|
|
|
|
* Contains the desired properties of
|
|
|
|
* the swap chain. This is passed as
|
|
|
|
* an input during swap chain creation.
|
|
|
|
*/
|
|
|
|
struct PresenterDesc {
|
|
|
|
VkExtent2D imageExtent;
|
|
|
|
uint32_t imageCount;
|
|
|
|
uint32_t numFormats;
|
|
|
|
VkSurfaceFormatKHR formats[4];
|
|
|
|
uint32_t numPresentModes;
|
|
|
|
VkPresentModeKHR presentModes[4];
|
2019-12-03 01:28:28 +01:00
|
|
|
VkFullScreenExclusiveEXT fullscreenExclusive;
|
2018-11-12 20:53:23 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Presenter properties
|
|
|
|
*
|
|
|
|
* Contains the actual properties
|
|
|
|
* of the underlying swap chain.
|
|
|
|
*/
|
|
|
|
struct PresenterInfo {
|
|
|
|
VkSurfaceFormatKHR format;
|
|
|
|
VkPresentModeKHR presentMode;
|
|
|
|
VkExtent2D imageExtent;
|
|
|
|
uint32_t imageCount;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Adapter and queue
|
|
|
|
*/
|
|
|
|
struct PresenterDevice {
|
|
|
|
uint32_t queueFamily = 0;
|
|
|
|
VkQueue queue = VK_NULL_HANDLE;
|
|
|
|
VkPhysicalDevice adapter = VK_NULL_HANDLE;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Swap image and view
|
|
|
|
*/
|
|
|
|
struct PresenterImage {
|
|
|
|
VkImage image = VK_NULL_HANDLE;
|
|
|
|
VkImageView view = VK_NULL_HANDLE;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Presenter semaphores
|
|
|
|
*
|
2019-02-21 16:29:24 +01:00
|
|
|
* Pair of semaphores used for acquire and present
|
|
|
|
* operations, including the command buffers used
|
|
|
|
* in between. Also stores a fence to signal on
|
|
|
|
* image acquisition.
|
2018-11-12 20:53:23 +01:00
|
|
|
*/
|
|
|
|
struct PresenterSync {
|
2019-02-21 16:29:24 +01:00
|
|
|
VkFence fence;
|
2018-11-12 20:53:23 +01:00
|
|
|
VkSemaphore acquire;
|
|
|
|
VkSemaphore present;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Vulkan presenter
|
|
|
|
*
|
|
|
|
* Provides abstractions for some of the
|
|
|
|
* more complicated aspects of Vulkan's
|
|
|
|
* window system integration.
|
|
|
|
*/
|
|
|
|
class Presenter : public RcObject {
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Presenter(
|
|
|
|
HWND window,
|
|
|
|
const Rc<InstanceFn>& vki,
|
|
|
|
const Rc<DeviceFn>& vkd,
|
|
|
|
PresenterDevice device,
|
|
|
|
const PresenterDesc& desc);
|
|
|
|
|
|
|
|
~Presenter();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Actual presenter info
|
|
|
|
* \returns Swap chain properties
|
|
|
|
*/
|
|
|
|
PresenterInfo info() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \breif Retrieves a pair of semaphores
|
|
|
|
*
|
|
|
|
* These semaphores are meant to be used
|
|
|
|
* for acquire and present operations.
|
|
|
|
* \returns Pair of semaphores
|
|
|
|
*/
|
|
|
|
PresenterSync getSyncSemaphores() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Retrieves image by index
|
|
|
|
*
|
|
|
|
* Can be used to create per-image objects.
|
|
|
|
* \param [in] index Image index
|
|
|
|
* \returns Image handle
|
|
|
|
*/
|
|
|
|
PresenterImage getImage(
|
|
|
|
uint32_t index) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Acquires next image
|
|
|
|
*
|
|
|
|
* Potentially blocks the calling thread.
|
|
|
|
* If this returns an error, the swap chain
|
|
|
|
* must be recreated and a new image must
|
|
|
|
* be acquired before proceeding.
|
|
|
|
* \param [in] signal Semaphore to signal
|
2019-02-21 15:55:40 +01:00
|
|
|
* \param [in] fence Fence to signal (optional)
|
2018-11-12 20:53:23 +01:00
|
|
|
* \param [out] index Acquired image index
|
|
|
|
* \returns Status of the operation
|
|
|
|
*/
|
|
|
|
VkResult acquireNextImage(
|
|
|
|
VkSemaphore signal,
|
2019-02-21 15:55:40 +01:00
|
|
|
VkFence fence,
|
2018-11-12 20:53:23 +01:00
|
|
|
uint32_t& index);
|
|
|
|
|
2019-02-21 16:25:10 +01:00
|
|
|
/**
|
|
|
|
* \brief Waits for fence to get signaled
|
|
|
|
*
|
|
|
|
* Helper method that can be used in conjunction
|
|
|
|
* with the fence passed to \ref acquireNextImage.
|
|
|
|
* \param [in] fence Fence to wait on
|
|
|
|
* \returns Status of the operation
|
|
|
|
*/
|
|
|
|
VkResult waitForFence(
|
|
|
|
VkFence fence);
|
|
|
|
|
2018-11-12 20:53:23 +01:00
|
|
|
/**
|
|
|
|
* \brief Presents current image
|
|
|
|
*
|
|
|
|
* Presents the current image. If this returns
|
|
|
|
* an error, the swap chain must be recreated,
|
|
|
|
* but do not present before acquiring an image.
|
|
|
|
* \param [in] wait Semaphore to wait on
|
|
|
|
* \returns Status of the operation
|
|
|
|
*/
|
|
|
|
VkResult presentImage(
|
|
|
|
VkSemaphore wait);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Changes presenter properties
|
|
|
|
*
|
2019-11-19 23:12:07 +01:00
|
|
|
* Recreates the swap chain immediately. Note that
|
|
|
|
* no swap chain resources must be in use by the
|
|
|
|
* GPU at the time this is called.
|
2018-11-12 20:53:23 +01:00
|
|
|
* \param [in] desc Swap chain description
|
|
|
|
*/
|
|
|
|
VkResult recreateSwapChain(
|
|
|
|
const PresenterDesc& desc);
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
Rc<InstanceFn> m_vki;
|
|
|
|
Rc<DeviceFn> m_vkd;
|
|
|
|
|
|
|
|
PresenterDevice m_device;
|
|
|
|
PresenterInfo m_info;
|
|
|
|
|
2019-04-01 12:48:22 -05:00
|
|
|
HWND m_window = nullptr;
|
2018-11-12 20:53:23 +01:00
|
|
|
VkSurfaceKHR m_surface = VK_NULL_HANDLE;
|
|
|
|
VkSwapchainKHR m_swapchain = VK_NULL_HANDLE;
|
|
|
|
|
|
|
|
std::vector<PresenterImage> m_images;
|
|
|
|
std::vector<PresenterSync> m_semaphores;
|
|
|
|
|
|
|
|
uint32_t m_imageIndex = 0;
|
|
|
|
uint32_t m_frameIndex = 0;
|
|
|
|
|
|
|
|
VkResult getSupportedFormats(
|
|
|
|
std::vector<VkSurfaceFormatKHR>& formats);
|
|
|
|
|
|
|
|
VkResult getSupportedPresentModes(
|
|
|
|
std::vector<VkPresentModeKHR>& modes);
|
|
|
|
|
|
|
|
VkResult getSwapImages(
|
|
|
|
std::vector<VkImage>& images);
|
|
|
|
|
|
|
|
VkSurfaceFormatKHR pickFormat(
|
|
|
|
uint32_t numSupported,
|
|
|
|
const VkSurfaceFormatKHR* pSupported,
|
|
|
|
uint32_t numDesired,
|
|
|
|
const VkSurfaceFormatKHR* pDesired);
|
|
|
|
|
|
|
|
VkPresentModeKHR pickPresentMode(
|
|
|
|
uint32_t numSupported,
|
|
|
|
const VkPresentModeKHR* pSupported,
|
|
|
|
uint32_t numDesired,
|
|
|
|
const VkPresentModeKHR* pDesired);
|
|
|
|
|
|
|
|
VkExtent2D pickImageExtent(
|
|
|
|
const VkSurfaceCapabilitiesKHR& caps,
|
|
|
|
VkExtent2D desired);
|
|
|
|
|
|
|
|
uint32_t pickImageCount(
|
|
|
|
const VkSurfaceCapabilitiesKHR& caps,
|
|
|
|
VkPresentModeKHR presentMode,
|
|
|
|
uint32_t desired);
|
|
|
|
|
2019-04-01 12:48:22 -05:00
|
|
|
VkResult createSurface();
|
2018-11-12 20:53:23 +01:00
|
|
|
|
|
|
|
void destroySwapchain();
|
|
|
|
|
|
|
|
void destroySurface();
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2019-04-01 12:48:22 -05:00
|
|
|
}
|