2018-11-12 20:53:23 +01:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "../util/log/log.h"
|
|
|
|
|
|
|
|
#include "../util/util_error.h"
|
2021-06-10 21:53:08 +02:00
|
|
|
#include "../util/util_fps_limiter.h"
|
2018-11-12 20:53:23 +01:00
|
|
|
#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:54:32 +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;
|
|
|
|
};
|
2019-12-03 02:03:22 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Presenter features
|
|
|
|
*/
|
|
|
|
struct PresenterFeatures {
|
|
|
|
bool fullScreenExclusive : 1;
|
|
|
|
};
|
2018-11-12 20:53:23 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Adapter and queue
|
|
|
|
*/
|
|
|
|
struct PresenterDevice {
|
|
|
|
uint32_t queueFamily = 0;
|
|
|
|
VkQueue queue = VK_NULL_HANDLE;
|
|
|
|
VkPhysicalDevice adapter = VK_NULL_HANDLE;
|
2019-12-03 02:03:22 +01:00
|
|
|
PresenterFeatures features = { };
|
2018-11-12 20:53:23 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \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 {
|
|
|
|
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;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \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.
|
2021-06-01 04:32:50 +02:00
|
|
|
* \param [out] sync Synchronization semaphores
|
2018-11-12 20:53:23 +01:00
|
|
|
* \param [out] index Acquired image index
|
|
|
|
* \returns Status of the operation
|
|
|
|
*/
|
|
|
|
VkResult acquireNextImage(
|
2021-06-01 04:32:50 +02:00
|
|
|
PresenterSync& sync,
|
2018-11-12 20:53:23 +01:00
|
|
|
uint32_t& index);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \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.
|
|
|
|
* \returns Status of the operation
|
|
|
|
*/
|
2021-06-01 04:32:50 +02:00
|
|
|
VkResult presentImage();
|
2018-11-12 20:53:23 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* \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);
|
|
|
|
|
2021-06-10 21:53:08 +02:00
|
|
|
/**
|
|
|
|
* \brief Changes maximum frame rate
|
|
|
|
*
|
|
|
|
* \param [in] frameRate Target frame rate. Set
|
|
|
|
* to 0 in order to disable the limiter.
|
|
|
|
*/
|
|
|
|
void setFrameRateLimit(double frameRate);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \brief Notifies frame rate limiter about the display refresh rate
|
|
|
|
*
|
|
|
|
* Used to dynamically disable the frame rate limiter in case
|
|
|
|
* vertical synchronization is used and the target frame rate
|
|
|
|
* roughly equals the display's refresh rate.
|
|
|
|
* \param [in] refresnRate Current refresh rate
|
|
|
|
*/
|
|
|
|
void setFrameRateLimiterRefreshRate(double refreshRate);
|
|
|
|
|
2019-12-15 11:32:43 +01:00
|
|
|
/**
|
|
|
|
* \brief Checks whether a Vulkan swap chain exists
|
|
|
|
*
|
|
|
|
* On Windows, there are situations where we cannot create
|
|
|
|
* a swap chain as the surface size can reach zero, and no
|
|
|
|
* presentation can be performed.
|
|
|
|
* \returns \c true if the presenter has a swap chain.
|
|
|
|
*/
|
|
|
|
bool hasSwapChain() const {
|
|
|
|
return m_swapchain;
|
|
|
|
}
|
|
|
|
|
2018-11-12 20:53:23 +01:00
|
|
|
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;
|
|
|
|
|
2021-06-01 04:44:02 +02:00
|
|
|
VkResult m_acquireStatus = VK_NOT_READY;
|
|
|
|
|
2021-06-10 21:53:08 +02:00
|
|
|
FpsLimiter m_fpsLimiter;
|
|
|
|
|
2018-11-12 20:53:23 +01:00
|
|
|
VkResult getSupportedFormats(
|
2019-12-03 01:54:32 +01:00
|
|
|
std::vector<VkSurfaceFormatKHR>& formats,
|
|
|
|
const PresenterDesc& desc);
|
2018-11-12 20:53:23 +01:00
|
|
|
|
|
|
|
VkResult getSupportedPresentModes(
|
2019-12-03 01:54:32 +01:00
|
|
|
std::vector<VkPresentModeKHR>& modes,
|
|
|
|
const PresenterDesc& desc);
|
2018-11-12 20:53:23 +01:00
|
|
|
|
|
|
|
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
|
|
|
}
|