summaryrefslogtreecommitdiffstats
path: root/examples/vulkan
diff options
context:
space:
mode:
Diffstat (limited to 'examples/vulkan')
-rw-r--r--examples/vulkan/CMakeLists.txt9
-rw-r--r--examples/vulkan/doc/images/hellovulkantexture.pngbin10259 -> 0 bytes
-rw-r--r--examples/vulkan/doc/images/hellovulkanwindow.pngbin2736 -> 0 bytes
-rw-r--r--examples/vulkan/doc/src/hellovulkancubes.qdoc1
-rw-r--r--examples/vulkan/doc/src/hellovulkantexture.qdoc18
-rw-r--r--examples/vulkan/doc/src/hellovulkantriangle.qdoc49
-rw-r--r--examples/vulkan/doc/src/hellovulkanwidget.qdoc1
-rw-r--r--examples/vulkan/doc/src/hellovulkanwindow.qdoc78
-rw-r--r--examples/vulkan/hellovulkancubes/CMakeLists.txt36
-rw-r--r--examples/vulkan/hellovulkancubes/mainwindow.cpp2
-rw-r--r--examples/vulkan/hellovulkancubes/mainwindow.h10
-rw-r--r--examples/vulkan/hellovulkancubes/renderer.cpp2
-rw-r--r--examples/vulkan/hellovulkantexture/CMakeLists.txt47
-rw-r--r--examples/vulkan/hellovulkantexture/hellovulkantexture.cpp792
-rw-r--r--examples/vulkan/hellovulkantexture/hellovulkantexture.h61
-rw-r--r--examples/vulkan/hellovulkantexture/hellovulkantexture.pro7
-rw-r--r--examples/vulkan/hellovulkantexture/hellovulkantexture.qrc7
-rw-r--r--examples/vulkan/hellovulkantexture/main.cpp32
-rw-r--r--examples/vulkan/hellovulkantexture/qt256.pngbin6208 -> 0 bytes
-rw-r--r--examples/vulkan/hellovulkantexture/texture.frag12
-rw-r--r--examples/vulkan/hellovulkantexture/texture.vert18
-rw-r--r--examples/vulkan/hellovulkantexture/texture_frag.spvbin556 -> 0 bytes
-rw-r--r--examples/vulkan/hellovulkantexture/texture_vert.spvbin968 -> 0 bytes
-rw-r--r--examples/vulkan/hellovulkantriangle/CMakeLists.txt32
-rw-r--r--examples/vulkan/hellovulkantriangle/main.cpp7
-rw-r--r--examples/vulkan/hellovulkanwidget/CMakeLists.txt34
-rw-r--r--examples/vulkan/hellovulkanwidget/hellovulkanwidget.h15
-rw-r--r--examples/vulkan/hellovulkanwindow/CMakeLists.txt33
-rw-r--r--examples/vulkan/hellovulkanwindow/hellovulkanwindow.cpp81
-rw-r--r--examples/vulkan/hellovulkanwindow/hellovulkanwindow.h30
-rw-r--r--examples/vulkan/hellovulkanwindow/hellovulkanwindow.pro6
-rw-r--r--examples/vulkan/hellovulkanwindow/main.cpp33
-rw-r--r--examples/vulkan/shared/trianglerenderer.h5
-rw-r--r--examples/vulkan/vulkan.pro4
34 files changed, 137 insertions, 1325 deletions
diff --git a/examples/vulkan/CMakeLists.txt b/examples/vulkan/CMakeLists.txt
index c5694435b6..98b0a31b5e 100644
--- a/examples/vulkan/CMakeLists.txt
+++ b/examples/vulkan/CMakeLists.txt
@@ -1,9 +1,10 @@
-qt_internal_add_example(hellovulkanwindow)
+# Copyright (C) 2022 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
qt_internal_add_example(hellovulkantriangle)
-qt_internal_add_example(hellovulkantexture)
-if(TARGET Qt::Widgets)
+if(TARGET Qt6::Widgets)
qt_internal_add_example(hellovulkanwidget)
endif()
-if(TARGET Qt::Concurrent AND TARGET Qt::Widgets)
+if(TARGET Qt6::Concurrent AND TARGET Qt6::Widgets)
qt_internal_add_example(hellovulkancubes)
endif()
diff --git a/examples/vulkan/doc/images/hellovulkantexture.png b/examples/vulkan/doc/images/hellovulkantexture.png
deleted file mode 100644
index 0cb47a70be..0000000000
--- a/examples/vulkan/doc/images/hellovulkantexture.png
+++ /dev/null
Binary files differ
diff --git a/examples/vulkan/doc/images/hellovulkanwindow.png b/examples/vulkan/doc/images/hellovulkanwindow.png
deleted file mode 100644
index c55029312c..0000000000
--- a/examples/vulkan/doc/images/hellovulkanwindow.png
+++ /dev/null
Binary files differ
diff --git a/examples/vulkan/doc/src/hellovulkancubes.qdoc b/examples/vulkan/doc/src/hellovulkancubes.qdoc
index dc60b8bb68..91149911df 100644
--- a/examples/vulkan/doc/src/hellovulkancubes.qdoc
+++ b/examples/vulkan/doc/src/hellovulkancubes.qdoc
@@ -7,6 +7,7 @@
\title Hello Vulkan Cubes Example
\ingroup examples-vulkan
\brief Shows the basics of using QVulkanWindow.
+ \examplecategory {Graphics}
The \e{Hello Vulkan Cubes Example} shows more advanced usage of QVulkanWindow.
diff --git a/examples/vulkan/doc/src/hellovulkantexture.qdoc b/examples/vulkan/doc/src/hellovulkantexture.qdoc
deleted file mode 100644
index 4945000f88..0000000000
--- a/examples/vulkan/doc/src/hellovulkantexture.qdoc
+++ /dev/null
@@ -1,18 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GFDL-1.3-no-invariants-only
-
-/*!
- \example hellovulkantexture
- \meta installpath vulkan
- \ingroup examples-vulkan
- \title Hello Vulkan Texture Vulkan Example
- \brief Shows the basics of rendering with textures in a QVulkanWindow.
-
- The \e{Hello Vulkan Texture Example} builds on \l hellovulkantriangle. Here
- instead of drawing a single triangle, a triangle strip is drawn in order to
- get a quad on the screen. This is then textured using a QImage loaded from
- a .png image file.
-
- \image hellovulkantexture.png
- \include examples-run.qdocinc
-*/
diff --git a/examples/vulkan/doc/src/hellovulkantriangle.qdoc b/examples/vulkan/doc/src/hellovulkantriangle.qdoc
index 71d451c63a..bafa026ccd 100644
--- a/examples/vulkan/doc/src/hellovulkantriangle.qdoc
+++ b/examples/vulkan/doc/src/hellovulkantriangle.qdoc
@@ -7,13 +7,56 @@
\ingroup examples-vulkan
\title Hello Vulkan Triangle Example
\brief Shows the basics of rendering with QVulkanWindow and the Vulkan API.
+ \examplecategory {Graphics}
- The \e{Hello Vulkan Triangle Example} builds on \l hellovulkanwindow. This
- time a full graphics pipeline is created, including a vertex and fragment
- shader. This pipeline is then used to render a triangle.
+ The \e{Hello Vulkan Triangle Example} creates a full graphics pipeline,
+ including a vertex and fragment shader, to render a triangle.
\image hellovulkantriangle.png
+ \section1 Startup
+
+ Each Qt application using Vulkan will have to have a \c{Vulkan instance}
+ which encapsulates application-level state and initializes a Vulkan library.
+
+ A QVulkanWindow must always be associated with a QVulkanInstance and hence
+ the example performs instance creation before the window. The
+ QVulkanInstance object must also outlive the window.
+
+ \snippet hellovulkantriangle/main.cpp 0
+
+ The example enables validation layers, when supported. When the requested
+ layers are not present, the request will be ignored. Additional layers and
+ extensions can be enabled in a similar manner.
+
+ \snippet hellovulkantriangle/main.cpp 1
+
+ Once the instance is ready, it is time to create a window. Note that \c w
+ lives on the stack and is declared after \c inst.
+
+ \section1 The QVulkanWindow Subclass
+
+ To add custom functionality to a QVulkanWindow, subclassing is used. This
+ follows the existing patterns from QOpenGLWindow and QOpenGLWidget.
+ However, QVulkanWindow utilizes a separate QVulkanWindowRenderer object.
+
+ The QVulkanWindow subclass reimplements the factory function
+ QVulkanWindow::createRenderer(). This simply returns a new instance of the
+ QVulkanWindowRenderer subclass. In order to be able to access various
+ Vulkan resources via the window object, a pointer to the window is passed
+ and stored via the constructor.
+
+ \snippet hellovulkantriangle/main.cpp 2
+
+ \section1 The Actual Rendering
+
+ QVulkanWindow subclasses queue their draw calls in their reimplementation
+ of QVulkanWindowRenderer::startNextFrame(). Once done, they are required to
+ call back QVulkanWindow::frameReady(). The example has no asynchronous
+ command generation, so the frameReady() call is made directly from
+ startNextFrame(). To get continuous updates, the example simply invokes
+ QWindow::requestUpdate() in order to schedule a repaint.
+
The example also demonstrates multisample antialiasing. Based on the
supported sample counts reported by QVulkanWindow::supportedSampleCounts()
the example chooses between 8x, 4x, or no multisampling. Once configured
diff --git a/examples/vulkan/doc/src/hellovulkanwidget.qdoc b/examples/vulkan/doc/src/hellovulkanwidget.qdoc
index 0166a361b1..2a71084fc4 100644
--- a/examples/vulkan/doc/src/hellovulkanwidget.qdoc
+++ b/examples/vulkan/doc/src/hellovulkanwidget.qdoc
@@ -7,6 +7,7 @@
\ingroup examples-vulkan
\title Hello Vulkan Widget Example
\brief Shows the usage of QVulkanWindow in QWidget applications.
+ \examplecategory {Graphics}
The \e{Hello Vulkan Widget Example} is a variant of \l hellovulkantriangle
that embeds the QVulkanWindow into a QWidget-based user interface using
diff --git a/examples/vulkan/doc/src/hellovulkanwindow.qdoc b/examples/vulkan/doc/src/hellovulkanwindow.qdoc
deleted file mode 100644
index 4b94457265..0000000000
--- a/examples/vulkan/doc/src/hellovulkanwindow.qdoc
+++ /dev/null
@@ -1,78 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GFDL-1.3-no-invariants-only
-
-/*!
- \example hellovulkanwindow
- \meta installpath vulkan
- \title Hello Vulkan Window Example
- \ingroup examples-vulkan
- \brief Shows the basics of using QVulkanWindow.
-
- The \e{Hello Vulkan Window Example} shows the basics of using QVulkanWindow
- in order to display rendering with the Vulkan graphics API on systems that
- support this.
-
- \image hellovulkanwindow.png
-
- In this example there will be no actual rendering: it simply begins and
- ends a render pass, which results in clearing the buffers to a fixed value.
- The color buffer clear value changes on every frame.
-
- \section1 Startup
-
- Each Qt application using Vulkan will have to have a \c{Vulkan instance}
- which encapsulates application-level state and initializes a Vulkan library.
-
- A QVulkanWindow must always be associated with a QVulkanInstance and hence
- the example performs instance creation before the window. The
- QVulkanInstance object must also outlive the window.
-
- \snippet hellovulkanwindow/main.cpp 0
-
- The example enables validation layers, when supported. When the requested
- layers are not present, the request will be ignored. Additional layers and
- extensions can be enabled in a similar manner.
-
- \snippet hellovulkanwindow/main.cpp 1
-
- Once the instance is ready, it is time to create a window. Note that \c w
- lives on the stack and is declared after \c inst.
-
- \section1 The QVulkanWindow Subclass
-
- To add custom functionality to a QVulkanWindow, subclassing is used. This
- follows the existing patterns from QOpenGLWindow and QOpenGLWidget.
- However, QVulkanWindow utilizes a separate QVulkanWindowRenderer object.
- This resembles QQuickFramebufferObject, and allows better separation of the
- functions that are supposed to be reimplemented.
-
- \snippet hellovulkanwindow/hellovulkanwindow.h 0
-
- The QVulkanWindow subclass reimplements the factory function
- QVulkanWindow::createRenderer(). This simply returns a new instance of the
- QVulkanWindowRenderer subclass. In order to be able to access various
- Vulkan resources via the window object, a pointer to the window is passed
- and stored via the constructor.
-
- \snippet hellovulkanwindow/hellovulkanwindow.cpp 0
-
- Graphics resource creation and destruction is typically done in one of the
- init - resource functions.
-
- \snippet hellovulkanwindow/hellovulkanwindow.cpp 1
-
- \section1 The Actual Rendering
-
- QVulkanWindow subclasses queue their draw calls in their reimplementation
- of QVulkanWindowRenderer::startNextFrame(). Once done, they are required to
- call back QVulkanWindow::frameReady(). The example has no asynchronous
- command generation, so the frameReady() call is made directly from
- startNextFrame().
-
- \snippet hellovulkanwindow/hellovulkanwindow.cpp 2
-
- To get continuous updates, the example simply invokes
- QWindow::requestUpdate() in order to schedule a repaint.
-
- \include examples-run.qdocinc
-*/
diff --git a/examples/vulkan/hellovulkancubes/CMakeLists.txt b/examples/vulkan/hellovulkancubes/CMakeLists.txt
index 61c4ec4aee..5debb4efaf 100644
--- a/examples/vulkan/hellovulkancubes/CMakeLists.txt
+++ b/examples/vulkan/hellovulkancubes/CMakeLists.txt
@@ -1,16 +1,13 @@
+# Copyright (C) 2022 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
cmake_minimum_required(VERSION 3.16)
project(hellovulkancubes LANGUAGES CXX)
-set(CMAKE_AUTOMOC ON)
-
-if(NOT DEFINED INSTALL_EXAMPLESDIR)
- set(INSTALL_EXAMPLESDIR "examples")
-endif()
-
-set(INSTALL_EXAMPLEDIR "${INSTALL_EXAMPLESDIR}/vulkan/hellovulkancubes")
-
find_package(Qt6 REQUIRED COMPONENTS Concurrent Core Gui Widgets)
+qt_standard_project_setup()
+
qt_add_executable(hellovulkancubes
camera.cpp camera.h
main.cpp
@@ -26,11 +23,11 @@ set_target_properties(hellovulkancubes PROPERTIES
MACOSX_BUNDLE TRUE
)
-target_link_libraries(hellovulkancubes PUBLIC
- Qt::Concurrent
- Qt::Core
- Qt::Gui
- Qt::Widgets
+target_link_libraries(hellovulkancubes PRIVATE
+ Qt6::Concurrent
+ Qt6::Core
+ Qt6::Gui
+ Qt6::Widgets
)
# Resources:
@@ -59,7 +56,14 @@ qt6_add_resources(hellovulkancubes "hellovulkancubes"
)
install(TARGETS hellovulkancubes
- RUNTIME DESTINATION "${INSTALL_EXAMPLEDIR}"
- BUNDLE DESTINATION "${INSTALL_EXAMPLEDIR}"
- LIBRARY DESTINATION "${INSTALL_EXAMPLEDIR}"
+ BUNDLE DESTINATION .
+ RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
+ LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
+)
+
+qt_generate_deploy_app_script(
+ TARGET hellovulkancubes
+ OUTPUT_SCRIPT deploy_script
+ NO_UNSUPPORTED_PLATFORM_ERROR
)
+install(SCRIPT ${deploy_script})
diff --git a/examples/vulkan/hellovulkancubes/mainwindow.cpp b/examples/vulkan/hellovulkancubes/mainwindow.cpp
index e84436bb1d..198a168bce 100644
--- a/examples/vulkan/hellovulkancubes/mainwindow.cpp
+++ b/examples/vulkan/hellovulkancubes/mainwindow.cpp
@@ -42,7 +42,7 @@ MainWindow::MainWindow(VulkanWindow *vulkanWindow)
pauseButton->setFocusPolicy(Qt::NoFocus);
connect(quitButton, &QPushButton::clicked, qApp, &QCoreApplication::quit);
- connect(newButton, &QPushButton::clicked, vulkanWindow, [=] {
+ connect(newButton, &QPushButton::clicked, vulkanWindow, [this, vulkanWindow] {
vulkanWindow->addNew();
m_count = vulkanWindow->instanceCount();
counterLcd->display(m_count);
diff --git a/examples/vulkan/hellovulkancubes/mainwindow.h b/examples/vulkan/hellovulkancubes/mainwindow.h
index 2e931c1a52..8ecb0002e7 100644
--- a/examples/vulkan/hellovulkancubes/mainwindow.h
+++ b/examples/vulkan/hellovulkancubes/mainwindow.h
@@ -6,12 +6,10 @@
#include <QWidget>
-QT_BEGIN_NAMESPACE
-class QLCDNumber;
-class QLabel;
-class QPushButton;
-class QCheckBox;
-QT_END_NAMESPACE
+QT_FORWARD_DECLARE_CLASS(QLCDNumber)
+QT_FORWARD_DECLARE_CLASS(QLabel)
+QT_FORWARD_DECLARE_CLASS(QPushButton)
+QT_FORWARD_DECLARE_CLASS(QCheckBox)
class VulkanWindow;
diff --git a/examples/vulkan/hellovulkancubes/renderer.cpp b/examples/vulkan/hellovulkancubes/renderer.cpp
index 34023eefdb..037d97014c 100644
--- a/examples/vulkan/hellovulkancubes/renderer.cpp
+++ b/examples/vulkan/hellovulkancubes/renderer.cpp
@@ -38,7 +38,7 @@ Renderer::Renderer(VulkanWindow *w, int initialCount)
m_blockMesh.load(QStringLiteral(":/block.buf"));
m_logoMesh.load(QStringLiteral(":/qt_logo.buf"));
- QObject::connect(&m_frameWatcher, &QFutureWatcherBase::finished, [this] {
+ QObject::connect(&m_frameWatcher, &QFutureWatcherBase::finished, m_window, [this] {
if (m_framePending) {
m_framePending = false;
m_window->frameReady();
diff --git a/examples/vulkan/hellovulkantexture/CMakeLists.txt b/examples/vulkan/hellovulkantexture/CMakeLists.txt
deleted file mode 100644
index fb9be834a3..0000000000
--- a/examples/vulkan/hellovulkantexture/CMakeLists.txt
+++ /dev/null
@@ -1,47 +0,0 @@
-cmake_minimum_required(VERSION 3.16)
-project(hellovulkantexture LANGUAGES CXX)
-
-set(CMAKE_AUTOMOC ON)
-
-if(NOT DEFINED INSTALL_EXAMPLESDIR)
- set(INSTALL_EXAMPLESDIR "examples")
-endif()
-
-set(INSTALL_EXAMPLEDIR "${INSTALL_EXAMPLESDIR}/vulkan/hellovulkantexture")
-
-find_package(Qt6 REQUIRED COMPONENTS Core Gui)
-
-qt_add_executable(hellovulkantexture
- hellovulkantexture.cpp hellovulkantexture.h
- main.cpp
-)
-
-set_target_properties(hellovulkantexture PROPERTIES
- WIN32_EXECUTABLE TRUE
- MACOSX_BUNDLE TRUE
-)
-
-target_link_libraries(hellovulkantexture PUBLIC
- Qt::Core
- Qt::Gui
-)
-
-# Resources:
-set(hellovulkantexture_resource_files
- "qt256.png"
- "texture_frag.spv"
- "texture_vert.spv"
-)
-
-qt_add_resources(hellovulkantexture "hellovulkantexture"
- PREFIX
- "/"
- FILES
- ${hellovulkantexture_resource_files}
-)
-
-install(TARGETS hellovulkantexture
- RUNTIME DESTINATION "${INSTALL_EXAMPLEDIR}"
- BUNDLE DESTINATION "${INSTALL_EXAMPLEDIR}"
- LIBRARY DESTINATION "${INSTALL_EXAMPLEDIR}"
-)
diff --git a/examples/vulkan/hellovulkantexture/hellovulkantexture.cpp b/examples/vulkan/hellovulkantexture/hellovulkantexture.cpp
deleted file mode 100644
index ed58c6caca..0000000000
--- a/examples/vulkan/hellovulkantexture/hellovulkantexture.cpp
+++ /dev/null
@@ -1,792 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include "hellovulkantexture.h"
-#include <QVulkanFunctions>
-#include <QCoreApplication>
-#include <QFile>
-
-// Use a triangle strip to get a quad.
-//
-// Note that the vertex data and the projection matrix assume OpenGL. With
-// Vulkan Y is negated in clip space and the near/far plane is at 0/1 instead
-// of -1/1. These will be corrected for by an extra transformation when
-// calculating the modelview-projection matrix.
-static float vertexData[] = { // Y up, front = CW
- // x, y, z, u, v
- -1, -1, 0, 0, 1,
- -1, 1, 0, 0, 0,
- 1, -1, 0, 1, 1,
- 1, 1, 0, 1, 0
-};
-
-static const int UNIFORM_DATA_SIZE = 16 * sizeof(float);
-
-static inline VkDeviceSize aligned(VkDeviceSize v, VkDeviceSize byteAlign)
-{
- return (v + byteAlign - 1) & ~(byteAlign - 1);
-}
-
-QVulkanWindowRenderer *VulkanWindow::createRenderer()
-{
- return new VulkanRenderer(this);
-}
-
-VulkanRenderer::VulkanRenderer(QVulkanWindow *w)
- : m_window(w)
-{
-}
-
-VkShaderModule VulkanRenderer::createShader(const QString &name)
-{
- QFile file(name);
- if (!file.open(QIODevice::ReadOnly)) {
- qWarning("Failed to read shader %s", qPrintable(name));
- return VK_NULL_HANDLE;
- }
- QByteArray blob = file.readAll();
- file.close();
-
- VkShaderModuleCreateInfo shaderInfo;
- memset(&shaderInfo, 0, sizeof(shaderInfo));
- shaderInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
- shaderInfo.codeSize = blob.size();
- shaderInfo.pCode = reinterpret_cast<const uint32_t *>(blob.constData());
- VkShaderModule shaderModule;
- VkResult err = m_devFuncs->vkCreateShaderModule(m_window->device(), &shaderInfo, nullptr, &shaderModule);
- if (err != VK_SUCCESS) {
- qWarning("Failed to create shader module: %d", err);
- return VK_NULL_HANDLE;
- }
-
- return shaderModule;
-}
-
-bool VulkanRenderer::createTexture(const QString &name)
-{
- QImage img(name);
- if (img.isNull()) {
- qWarning("Failed to load image %s", qPrintable(name));
- return false;
- }
-
- // Convert to byte ordered RGBA8. Use premultiplied alpha, see pColorBlendState in the pipeline.
- img = img.convertToFormat(QImage::Format_RGBA8888_Premultiplied);
-
- QVulkanFunctions *f = m_window->vulkanInstance()->functions();
- VkDevice dev = m_window->device();
-
- const bool srgb = QCoreApplication::arguments().contains(QStringLiteral("--srgb"));
- if (srgb)
- qDebug("sRGB swapchain was requested, making texture sRGB too");
-
- m_texFormat = srgb ? VK_FORMAT_R8G8B8A8_SRGB : VK_FORMAT_R8G8B8A8_UNORM;
-
- // Now we can either map and copy the image data directly, or have to go
- // through a staging buffer to copy and convert into the internal optimal
- // tiling format.
- VkFormatProperties props;
- f->vkGetPhysicalDeviceFormatProperties(m_window->physicalDevice(), m_texFormat, &props);
- const bool canSampleLinear = (props.linearTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT);
- const bool canSampleOptimal = (props.optimalTilingFeatures & VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT);
- if (!canSampleLinear && !canSampleOptimal) {
- qWarning("Neither linear nor optimal image sampling is supported for RGBA8");
- return false;
- }
-
- static bool alwaysStage = qEnvironmentVariableIntValue("QT_VK_FORCE_STAGE_TEX");
-
- if (canSampleLinear && !alwaysStage) {
- if (!createTextureImage(img.size(), &m_texImage, &m_texMem,
- VK_IMAGE_TILING_LINEAR, VK_IMAGE_USAGE_SAMPLED_BIT,
- m_window->hostVisibleMemoryIndex()))
- return false;
-
- if (!writeLinearImage(img, m_texImage, m_texMem))
- return false;
-
- m_texLayoutPending = true;
- } else {
- if (!createTextureImage(img.size(), &m_texStaging, &m_texStagingMem,
- VK_IMAGE_TILING_LINEAR, VK_IMAGE_USAGE_TRANSFER_SRC_BIT,
- m_window->hostVisibleMemoryIndex()))
- return false;
-
- if (!createTextureImage(img.size(), &m_texImage, &m_texMem,
- VK_IMAGE_TILING_OPTIMAL, VK_IMAGE_USAGE_TRANSFER_DST_BIT | VK_IMAGE_USAGE_SAMPLED_BIT,
- m_window->deviceLocalMemoryIndex()))
- return false;
-
- if (!writeLinearImage(img, m_texStaging, m_texStagingMem))
- return false;
-
- m_texStagingPending = true;
- }
-
- VkImageViewCreateInfo viewInfo;
- memset(&viewInfo, 0, sizeof(viewInfo));
- viewInfo.sType = VK_STRUCTURE_TYPE_IMAGE_VIEW_CREATE_INFO;
- viewInfo.image = m_texImage;
- viewInfo.viewType = VK_IMAGE_VIEW_TYPE_2D;
- viewInfo.format = m_texFormat;
- viewInfo.components.r = VK_COMPONENT_SWIZZLE_R;
- viewInfo.components.g = VK_COMPONENT_SWIZZLE_G;
- viewInfo.components.b = VK_COMPONENT_SWIZZLE_B;
- viewInfo.components.a = VK_COMPONENT_SWIZZLE_A;
- viewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
- viewInfo.subresourceRange.levelCount = viewInfo.subresourceRange.layerCount = 1;
-
- VkResult err = m_devFuncs->vkCreateImageView(dev, &viewInfo, nullptr, &m_texView);
- if (err != VK_SUCCESS) {
- qWarning("Failed to create image view for texture: %d", err);
- return false;
- }
-
- m_texSize = img.size();
-
- return true;
-}
-
-bool VulkanRenderer::createTextureImage(const QSize &size, VkImage *image, VkDeviceMemory *mem,
- VkImageTiling tiling, VkImageUsageFlags usage, uint32_t memIndex)
-{
- VkDevice dev = m_window->device();
-
- VkImageCreateInfo imageInfo;
- memset(&imageInfo, 0, sizeof(imageInfo));
- imageInfo.sType = VK_STRUCTURE_TYPE_IMAGE_CREATE_INFO;
- imageInfo.imageType = VK_IMAGE_TYPE_2D;
- imageInfo.format = m_texFormat;
- imageInfo.extent.width = size.width();
- imageInfo.extent.height = size.height();
- imageInfo.extent.depth = 1;
- imageInfo.mipLevels = 1;
- imageInfo.arrayLayers = 1;
- imageInfo.samples = VK_SAMPLE_COUNT_1_BIT;
- imageInfo.tiling = tiling;
- imageInfo.usage = usage;
- imageInfo.initialLayout = VK_IMAGE_LAYOUT_PREINITIALIZED;
-
- VkResult err = m_devFuncs->vkCreateImage(dev, &imageInfo, nullptr, image);
- if (err != VK_SUCCESS) {
- qWarning("Failed to create linear image for texture: %d", err);
- return false;
- }
-
- VkMemoryRequirements memReq;
- m_devFuncs->vkGetImageMemoryRequirements(dev, *image, &memReq);
-
- if (!(memReq.memoryTypeBits & (1 << memIndex))) {
- VkPhysicalDeviceMemoryProperties physDevMemProps;
- m_window->vulkanInstance()->functions()->vkGetPhysicalDeviceMemoryProperties(m_window->physicalDevice(), &physDevMemProps);
- for (uint32_t i = 0; i < physDevMemProps.memoryTypeCount; ++i) {
- if (!(memReq.memoryTypeBits & (1 << i)))
- continue;
- memIndex = i;
- }
- }
-
- VkMemoryAllocateInfo allocInfo = {
- VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO,
- nullptr,
- memReq.size,
- memIndex
- };
- qDebug("allocating %u bytes for texture image", uint32_t(memReq.size));
-
- err = m_devFuncs->vkAllocateMemory(dev, &allocInfo, nullptr, mem);
- if (err != VK_SUCCESS) {
- qWarning("Failed to allocate memory for linear image: %d", err);
- return false;
- }
-
- err = m_devFuncs->vkBindImageMemory(dev, *image, *mem, 0);
- if (err != VK_SUCCESS) {
- qWarning("Failed to bind linear image memory: %d", err);
- return false;
- }
-
- return true;
-}
-
-bool VulkanRenderer::writeLinearImage(const QImage &img, VkImage image, VkDeviceMemory memory)
-{
- VkDevice dev = m_window->device();
-
- VkImageSubresource subres = {
- VK_IMAGE_ASPECT_COLOR_BIT,
- 0, // mip level
- 0
- };
- VkSubresourceLayout layout;
- m_devFuncs->vkGetImageSubresourceLayout(dev, image, &subres, &layout);
-
- uchar *p;
- VkResult err = m_devFuncs->vkMapMemory(dev, memory, layout.offset, layout.size, 0, reinterpret_cast<void **>(&p));
- if (err != VK_SUCCESS) {
- qWarning("Failed to map memory for linear image: %d", err);
- return false;
- }
-
- for (int y = 0; y < img.height(); ++y) {
- const uchar *line = img.constScanLine(y);
- memcpy(p, line, img.width() * 4);
- p += layout.rowPitch;
- }
-
- m_devFuncs->vkUnmapMemory(dev, memory);
- return true;
-}
-
-void VulkanRenderer::ensureTexture()
-{
- if (!m_texLayoutPending && !m_texStagingPending)
- return;
-
- Q_ASSERT(m_texLayoutPending != m_texStagingPending);
- VkCommandBuffer cb = m_window->currentCommandBuffer();
-
- VkImageMemoryBarrier barrier;
- memset(&barrier, 0, sizeof(barrier));
- barrier.sType = VK_STRUCTURE_TYPE_IMAGE_MEMORY_BARRIER;
- barrier.subresourceRange.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
- barrier.subresourceRange.levelCount = barrier.subresourceRange.layerCount = 1;
-
- if (m_texLayoutPending) {
- m_texLayoutPending = false;
-
- barrier.oldLayout = VK_IMAGE_LAYOUT_PREINITIALIZED;
- barrier.newLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
- barrier.srcAccessMask = VK_ACCESS_HOST_WRITE_BIT;
- barrier.dstAccessMask = VK_ACCESS_SHADER_READ_BIT;
- barrier.image = m_texImage;
-
- m_devFuncs->vkCmdPipelineBarrier(cb,
- VK_PIPELINE_STAGE_HOST_BIT,
- VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT,
- 0, 0, nullptr, 0, nullptr,
- 1, &barrier);
- } else {
- m_texStagingPending = false;
-
- barrier.oldLayout = VK_IMAGE_LAYOUT_PREINITIALIZED;
- barrier.newLayout = VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL;
- barrier.srcAccessMask = VK_ACCESS_HOST_WRITE_BIT;
- barrier.dstAccessMask = VK_ACCESS_TRANSFER_READ_BIT;
- barrier.image = m_texStaging;
- m_devFuncs->vkCmdPipelineBarrier(cb,
- VK_PIPELINE_STAGE_HOST_BIT,
- VK_PIPELINE_STAGE_TRANSFER_BIT,
- 0, 0, nullptr, 0, nullptr,
- 1, &barrier);
-
- barrier.oldLayout = VK_IMAGE_LAYOUT_PREINITIALIZED;
- barrier.newLayout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
- barrier.srcAccessMask = 0;
- barrier.dstAccessMask = VK_ACCESS_TRANSFER_WRITE_BIT;
- barrier.image = m_texImage;
- m_devFuncs->vkCmdPipelineBarrier(cb,
- VK_PIPELINE_STAGE_TOP_OF_PIPE_BIT,
- VK_PIPELINE_STAGE_TRANSFER_BIT,
- 0, 0, nullptr, 0, nullptr,
- 1, &barrier);
-
- VkImageCopy copyInfo;
- memset(&copyInfo, 0, sizeof(copyInfo));
- copyInfo.srcSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
- copyInfo.srcSubresource.layerCount = 1;
- copyInfo.dstSubresource.aspectMask = VK_IMAGE_ASPECT_COLOR_BIT;
- copyInfo.dstSubresource.layerCount = 1;
- copyInfo.extent.width = m_texSize.width();
- copyInfo.extent.height = m_texSize.height();
- copyInfo.extent.depth = 1;
- m_devFuncs->vkCmdCopyImage(cb, m_texStaging, VK_IMAGE_LAYOUT_TRANSFER_SRC_OPTIMAL,
- m_texImage, VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL, 1, &copyInfo);
-
- barrier.oldLayout = VK_IMAGE_LAYOUT_TRANSFER_DST_OPTIMAL;
- barrier.newLayout = VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL;
- barrier.srcAccessMask = VK_ACCESS_TRANSFER_WRITE_BIT;
- barrier.dstAccessMask = VK_ACCESS_SHADER_READ_BIT;
- barrier.image = m_texImage;
- m_devFuncs->vkCmdPipelineBarrier(cb,
- VK_PIPELINE_STAGE_TRANSFER_BIT,
- VK_PIPELINE_STAGE_FRAGMENT_SHADER_BIT,
- 0, 0, nullptr, 0, nullptr,
- 1, &barrier);
- }
-}
-
-void VulkanRenderer::initResources()
-{
- qDebug("initResources");
-
- VkDevice dev = m_window->device();
- m_devFuncs = m_window->vulkanInstance()->deviceFunctions(dev);
-
- // The setup is similar to hellovulkantriangle. The difference is the
- // presence of a second vertex attribute (texcoord), a sampler, and that we
- // need blending.
-
- const int concurrentFrameCount = m_window->concurrentFrameCount();
- const VkPhysicalDeviceLimits *pdevLimits = &m_window->physicalDeviceProperties()->limits;
- const VkDeviceSize uniAlign = pdevLimits->minUniformBufferOffsetAlignment;
- qDebug("uniform buffer offset alignment is %u", (uint) uniAlign);
- VkBufferCreateInfo bufInfo;
- memset(&bufInfo, 0, sizeof(bufInfo));
- bufInfo.sType = VK_STRUCTURE_TYPE_BUFFER_CREATE_INFO;
- // Our internal layout is vertex, uniform, uniform, ... with each uniform buffer start offset aligned to uniAlign.
- const VkDeviceSize vertexAllocSize = aligned(sizeof(vertexData), uniAlign);
- const VkDeviceSize uniformAllocSize = aligned(UNIFORM_DATA_SIZE, uniAlign);
- bufInfo.size = vertexAllocSize + concurrentFrameCount * uniformAllocSize;
- bufInfo.usage = VK_BUFFER_USAGE_VERTEX_BUFFER_BIT | VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT;
-
- VkResult err = m_devFuncs->vkCreateBuffer(dev, &bufInfo, nullptr, &m_buf);
- if (err != VK_SUCCESS)
- qFatal("Failed to create buffer: %d", err);
-
- VkMemoryRequirements memReq;
- m_devFuncs->vkGetBufferMemoryRequirements(dev, m_buf, &memReq);
-
- VkMemoryAllocateInfo memAllocInfo = {
- VK_STRUCTURE_TYPE_MEMORY_ALLOCATE_INFO,
- nullptr,
- memReq.size,
- m_window->hostVisibleMemoryIndex()
- };
-
- err = m_devFuncs->vkAllocateMemory(dev, &memAllocInfo, nullptr, &m_bufMem);
- if (err != VK_SUCCESS)
- qFatal("Failed to allocate memory: %d", err);
-
- err = m_devFuncs->vkBindBufferMemory(dev, m_buf, m_bufMem, 0);
- if (err != VK_SUCCESS)
- qFatal("Failed to bind buffer memory: %d", err);
-
- quint8 *p;
- err = m_devFuncs->vkMapMemory(dev, m_bufMem, 0, memReq.size, 0, reinterpret_cast<void **>(&p));
- if (err != VK_SUCCESS)
- qFatal("Failed to map memory: %d", err);
- memcpy(p, vertexData, sizeof(vertexData));
- QMatrix4x4 ident;
- memset(m_uniformBufInfo, 0, sizeof(m_uniformBufInfo));
- for (int i = 0; i < concurrentFrameCount; ++i) {
- const VkDeviceSize offset = vertexAllocSize + i * uniformAllocSize;
- memcpy(p + offset, ident.constData(), 16 * sizeof(float));
- m_uniformBufInfo[i].buffer = m_buf;
- m_uniformBufInfo[i].offset = offset;
- m_uniformBufInfo[i].range = uniformAllocSize;
- }
- m_devFuncs->vkUnmapMemory(dev, m_bufMem);
-
- VkVertexInputBindingDescription vertexBindingDesc = {
- 0, // binding
- 5 * sizeof(float),
- VK_VERTEX_INPUT_RATE_VERTEX
- };
- VkVertexInputAttributeDescription vertexAttrDesc[] = {
- { // position
- 0, // location
- 0, // binding
- VK_FORMAT_R32G32B32_SFLOAT,
- 0
- },
- { // texcoord
- 1,
- 0,
- VK_FORMAT_R32G32_SFLOAT,
- 3 * sizeof(float)
- }
- };
-
- VkPipelineVertexInputStateCreateInfo vertexInputInfo;
- vertexInputInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO;
- vertexInputInfo.pNext = nullptr;
- vertexInputInfo.flags = 0;
- vertexInputInfo.vertexBindingDescriptionCount = 1;
- vertexInputInfo.pVertexBindingDescriptions = &vertexBindingDesc;
- vertexInputInfo.vertexAttributeDescriptionCount = 2;
- vertexInputInfo.pVertexAttributeDescriptions = vertexAttrDesc;
-
- // Sampler.
- VkSamplerCreateInfo samplerInfo;
- memset(&samplerInfo, 0, sizeof(samplerInfo));
- samplerInfo.sType = VK_STRUCTURE_TYPE_SAMPLER_CREATE_INFO;
- samplerInfo.magFilter = VK_FILTER_NEAREST;
- samplerInfo.minFilter = VK_FILTER_NEAREST;
- samplerInfo.addressModeU = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
- samplerInfo.addressModeV = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
- samplerInfo.addressModeW = VK_SAMPLER_ADDRESS_MODE_CLAMP_TO_EDGE;
- samplerInfo.maxAnisotropy = 1.0f;
- err = m_devFuncs->vkCreateSampler(dev, &samplerInfo, nullptr, &m_sampler);
- if (err != VK_SUCCESS)
- qFatal("Failed to create sampler: %d", err);
-
- // Texture.
- if (!createTexture(QStringLiteral(":/qt256.png")))
- qFatal("Failed to create texture");
-
- // Set up descriptor set and its layout.
- VkDescriptorPoolSize descPoolSizes[2] = {
- { VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, uint32_t(concurrentFrameCount) },
- { VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, uint32_t(concurrentFrameCount) }
- };
- VkDescriptorPoolCreateInfo descPoolInfo;
- memset(&descPoolInfo, 0, sizeof(descPoolInfo));
- descPoolInfo.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_POOL_CREATE_INFO;
- descPoolInfo.maxSets = concurrentFrameCount;
- descPoolInfo.poolSizeCount = 2;
- descPoolInfo.pPoolSizes = descPoolSizes;
- err = m_devFuncs->vkCreateDescriptorPool(dev, &descPoolInfo, nullptr, &m_descPool);
- if (err != VK_SUCCESS)
- qFatal("Failed to create descriptor pool: %d", err);
-
- VkDescriptorSetLayoutBinding layoutBinding[2] =
- {
- {
- 0, // binding
- VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER,
- 1, // descriptorCount
- VK_SHADER_STAGE_VERTEX_BIT,
- nullptr
- },
- {
- 1, // binding
- VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER,
- 1, // descriptorCount
- VK_SHADER_STAGE_FRAGMENT_BIT,
- nullptr
- }
- };
- VkDescriptorSetLayoutCreateInfo descLayoutInfo = {
- VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO,
- nullptr,
- 0,
- 2, // bindingCount
- layoutBinding
- };
- err = m_devFuncs->vkCreateDescriptorSetLayout(dev, &descLayoutInfo, nullptr, &m_descSetLayout);
- if (err != VK_SUCCESS)
- qFatal("Failed to create descriptor set layout: %d", err);
-
- for (int i = 0; i < concurrentFrameCount; ++i) {
- VkDescriptorSetAllocateInfo descSetAllocInfo = {
- VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO,
- nullptr,
- m_descPool,
- 1,
- &m_descSetLayout
- };
- err = m_devFuncs->vkAllocateDescriptorSets(dev, &descSetAllocInfo, &m_descSet[i]);
- if (err != VK_SUCCESS)
- qFatal("Failed to allocate descriptor set: %d", err);
-
- VkWriteDescriptorSet descWrite[2];
- memset(descWrite, 0, sizeof(descWrite));
- descWrite[0].sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
- descWrite[0].dstSet = m_descSet[i];
- descWrite[0].dstBinding = 0;
- descWrite[0].descriptorCount = 1;
- descWrite[0].descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER;
- descWrite[0].pBufferInfo = &m_uniformBufInfo[i];
-
- VkDescriptorImageInfo descImageInfo = {
- m_sampler,
- m_texView,
- VK_IMAGE_LAYOUT_SHADER_READ_ONLY_OPTIMAL
- };
-
- descWrite[1].sType = VK_STRUCTURE_TYPE_WRITE_DESCRIPTOR_SET;
- descWrite[1].dstSet = m_descSet[i];
- descWrite[1].dstBinding = 1;
- descWrite[1].descriptorCount = 1;
- descWrite[1].descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
- descWrite[1].pImageInfo = &descImageInfo;
-
- m_devFuncs->vkUpdateDescriptorSets(dev, 2, descWrite, 0, nullptr);
- }
-
- // Pipeline cache
- VkPipelineCacheCreateInfo pipelineCacheInfo;
- memset(&pipelineCacheInfo, 0, sizeof(pipelineCacheInfo));
- pipelineCacheInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_CACHE_CREATE_INFO;
- err = m_devFuncs->vkCreatePipelineCache(dev, &pipelineCacheInfo, nullptr, &m_pipelineCache);
- if (err != VK_SUCCESS)
- qFatal("Failed to create pipeline cache: %d", err);
-
- // Pipeline layout
- VkPipelineLayoutCreateInfo pipelineLayoutInfo;
- memset(&pipelineLayoutInfo, 0, sizeof(pipelineLayoutInfo));
- pipelineLayoutInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO;
- pipelineLayoutInfo.setLayoutCount = 1;
- pipelineLayoutInfo.pSetLayouts = &m_descSetLayout;
- err = m_devFuncs->vkCreatePipelineLayout(dev, &pipelineLayoutInfo, nullptr, &m_pipelineLayout);
- if (err != VK_SUCCESS)
- qFatal("Failed to create pipeline layout: %d", err);
-
- // Shaders
- VkShaderModule vertShaderModule = createShader(QStringLiteral(":/texture_vert.spv"));
- VkShaderModule fragShaderModule = createShader(QStringLiteral(":/texture_frag.spv"));
-
- // Graphics pipeline
- VkGraphicsPipelineCreateInfo pipelineInfo;
- memset(&pipelineInfo, 0, sizeof(pipelineInfo));
- pipelineInfo.sType = VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO;
-
- VkPipelineShaderStageCreateInfo shaderStages[2] = {
- {
- VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
- nullptr,
- 0,
- VK_SHADER_STAGE_VERTEX_BIT,
- vertShaderModule,
- "main",
- nullptr
- },
- {
- VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO,
- nullptr,
- 0,
- VK_SHADER_STAGE_FRAGMENT_BIT,
- fragShaderModule,
- "main",
- nullptr
- }
- };
- pipelineInfo.stageCount = 2;
- pipelineInfo.pStages = shaderStages;
-
- pipelineInfo.pVertexInputState = &vertexInputInfo;
-
- VkPipelineInputAssemblyStateCreateInfo ia;
- memset(&ia, 0, sizeof(ia));
- ia.sType = VK_STRUCTURE_TYPE_PIPELINE_INPUT_ASSEMBLY_STATE_CREATE_INFO;
- ia.topology = VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP;
- pipelineInfo.pInputAssemblyState = &ia;
-
- // The viewport and scissor will be set dynamically via vkCmdSetViewport/Scissor.
- // This way the pipeline does not need to be touched when resizing the window.
- VkPipelineViewportStateCreateInfo vp;
- memset(&vp, 0, sizeof(vp));
- vp.sType = VK_STRUCTURE_TYPE_PIPELINE_VIEWPORT_STATE_CREATE_INFO;
- vp.viewportCount = 1;
- vp.scissorCount = 1;
- pipelineInfo.pViewportState = &vp;
-
- VkPipelineRasterizationStateCreateInfo rs;
- memset(&rs, 0, sizeof(rs));
- rs.sType = VK_STRUCTURE_TYPE_PIPELINE_RASTERIZATION_STATE_CREATE_INFO;
- rs.polygonMode = VK_POLYGON_MODE_FILL;
- rs.cullMode = VK_CULL_MODE_BACK_BIT;
- rs.frontFace = VK_FRONT_FACE_CLOCKWISE;
- rs.lineWidth = 1.0f;
- pipelineInfo.pRasterizationState = &rs;
-
- VkPipelineMultisampleStateCreateInfo ms;
- memset(&ms, 0, sizeof(ms));
- ms.sType = VK_STRUCTURE_TYPE_PIPELINE_MULTISAMPLE_STATE_CREATE_INFO;
- ms.rasterizationSamples = VK_SAMPLE_COUNT_1_BIT;
- pipelineInfo.pMultisampleState = &ms;
-
- VkPipelineDepthStencilStateCreateInfo ds;
- memset(&ds, 0, sizeof(ds));
- ds.sType = VK_STRUCTURE_TYPE_PIPELINE_DEPTH_STENCIL_STATE_CREATE_INFO;
- ds.depthTestEnable = VK_TRUE;
- ds.depthWriteEnable = VK_TRUE;
- ds.depthCompareOp = VK_COMPARE_OP_LESS_OR_EQUAL;
- pipelineInfo.pDepthStencilState = &ds;
-
- VkPipelineColorBlendStateCreateInfo cb;
- memset(&cb, 0, sizeof(cb));
- cb.sType = VK_STRUCTURE_TYPE_PIPELINE_COLOR_BLEND_STATE_CREATE_INFO;
- // assume pre-multiplied alpha, blend, write out all of rgba
- VkPipelineColorBlendAttachmentState att;
- memset(&att, 0, sizeof(att));
- att.colorWriteMask = 0xF;
- att.blendEnable = VK_TRUE;
- att.srcColorBlendFactor = VK_BLEND_FACTOR_ONE;
- att.dstColorBlendFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
- att.colorBlendOp = VK_BLEND_OP_ADD;
- att.srcAlphaBlendFactor = VK_BLEND_FACTOR_ONE;
- att.dstAlphaBlendFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA;
- att.alphaBlendOp = VK_BLEND_OP_ADD;
- cb.attachmentCount = 1;
- cb.pAttachments = &att;
- pipelineInfo.pColorBlendState = &cb;
-
- VkDynamicState dynEnable[] = { VK_DYNAMIC_STATE_VIEWPORT, VK_DYNAMIC_STATE_SCISSOR };
- VkPipelineDynamicStateCreateInfo dyn;
- memset(&dyn, 0, sizeof(dyn));
- dyn.sType = VK_STRUCTURE_TYPE_PIPELINE_DYNAMIC_STATE_CREATE_INFO;
- dyn.dynamicStateCount = sizeof(dynEnable) / sizeof(VkDynamicState);
- dyn.pDynamicStates = dynEnable;
- pipelineInfo.pDynamicState = &dyn;
-
- pipelineInfo.layout = m_pipelineLayout;
- pipelineInfo.renderPass = m_window->defaultRenderPass();
-
- err = m_devFuncs->vkCreateGraphicsPipelines(dev, m_pipelineCache, 1, &pipelineInfo, nullptr, &m_pipeline);
- if (err != VK_SUCCESS)
- qFatal("Failed to create graphics pipeline: %d", err);
-
- if (vertShaderModule)
- m_devFuncs->vkDestroyShaderModule(dev, vertShaderModule, nullptr);
- if (fragShaderModule)
- m_devFuncs->vkDestroyShaderModule(dev, fragShaderModule, nullptr);
-}
-
-void VulkanRenderer::initSwapChainResources()
-{
- qDebug("initSwapChainResources");
-
- // Projection matrix
- m_proj = m_window->clipCorrectionMatrix(); // adjust for Vulkan-OpenGL clip space differences
- const QSize sz = m_window->swapChainImageSize();
- m_proj.perspective(45.0f, sz.width() / (float) sz.height(), 0.01f, 100.0f);
- m_proj.translate(0, 0, -4);
-}
-
-void VulkanRenderer::releaseSwapChainResources()
-{
- qDebug("releaseSwapChainResources");
-}
-
-void VulkanRenderer::releaseResources()
-{
- qDebug("releaseResources");
-
- VkDevice dev = m_window->device();
-
- if (m_sampler) {
- m_devFuncs->vkDestroySampler(dev, m_sampler, nullptr);
- m_sampler = VK_NULL_HANDLE;
- }
-
- if (m_texStaging) {
- m_devFuncs->vkDestroyImage(dev, m_texStaging, nullptr);
- m_texStaging = VK_NULL_HANDLE;
- }
-
- if (m_texStagingMem) {
- m_devFuncs->vkFreeMemory(dev, m_texStagingMem, nullptr);
- m_texStagingMem = VK_NULL_HANDLE;
- }
-
- if (m_texView) {
- m_devFuncs->vkDestroyImageView(dev, m_texView, nullptr);
- m_texView = VK_NULL_HANDLE;
- }
-
- if (m_texImage) {
- m_devFuncs->vkDestroyImage(dev, m_texImage, nullptr);
- m_texImage = VK_NULL_HANDLE;
- }
-
- if (m_texMem) {
- m_devFuncs->vkFreeMemory(dev, m_texMem, nullptr);
- m_texMem = VK_NULL_HANDLE;
- }
-
- if (m_pipeline) {
- m_devFuncs->vkDestroyPipeline(dev, m_pipeline, nullptr);
- m_pipeline = VK_NULL_HANDLE;
- }
-
- if (m_pipelineLayout) {
- m_devFuncs->vkDestroyPipelineLayout(dev, m_pipelineLayout, nullptr);
- m_pipelineLayout = VK_NULL_HANDLE;
- }
-
- if (m_pipelineCache) {
- m_devFuncs->vkDestroyPipelineCache(dev, m_pipelineCache, nullptr);
- m_pipelineCache = VK_NULL_HANDLE;
- }
-
- if (m_descSetLayout) {
- m_devFuncs->vkDestroyDescriptorSetLayout(dev, m_descSetLayout, nullptr);
- m_descSetLayout = VK_NULL_HANDLE;
- }
-
- if (m_descPool) {
- m_devFuncs->vkDestroyDescriptorPool(dev, m_descPool, nullptr);
- m_descPool = VK_NULL_HANDLE;
- }
-
- if (m_buf) {
- m_devFuncs->vkDestroyBuffer(dev, m_buf, nullptr);
- m_buf = VK_NULL_HANDLE;
- }
-
- if (m_bufMem) {
- m_devFuncs->vkFreeMemory(dev, m_bufMem, nullptr);
- m_bufMem = VK_NULL_HANDLE;
- }
-}
-
-void VulkanRenderer::startNextFrame()
-{
- VkDevice dev = m_window->device();
- VkCommandBuffer cb = m_window->currentCommandBuffer();
- const QSize sz = m_window->swapChainImageSize();
-
- // Add the necessary barriers and do the host-linear -> device-optimal copy, if not yet done.
- ensureTexture();
-
- VkClearColorValue clearColor = {{ 0, 0, 0, 1 }};
- VkClearDepthStencilValue clearDS = { 1, 0 };
- VkClearValue clearValues[2];
- memset(clearValues, 0, sizeof(clearValues));
- clearValues[0].color = clearColor;
- clearValues[1].depthStencil = clearDS;
-
- VkRenderPassBeginInfo rpBeginInfo;
- memset(&rpBeginInfo, 0, sizeof(rpBeginInfo));
- rpBeginInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO;
- rpBeginInfo.renderPass = m_window->defaultRenderPass();
- rpBeginInfo.framebuffer = m_window->currentFramebuffer();
- rpBeginInfo.renderArea.extent.width = sz.width();
- rpBeginInfo.renderArea.extent.height = sz.height();
- rpBeginInfo.clearValueCount = 2;
- rpBeginInfo.pClearValues = clearValues;
- VkCommandBuffer cmdBuf = m_window->currentCommandBuffer();
- m_devFuncs->vkCmdBeginRenderPass(cmdBuf, &rpBeginInfo, VK_SUBPASS_CONTENTS_INLINE);
-
- quint8 *p;
- VkResult err = m_devFuncs->vkMapMemory(dev, m_bufMem, m_uniformBufInfo[m_window->currentFrame()].offset,
- UNIFORM_DATA_SIZE, 0, reinterpret_cast<void **>(&p));
- if (err != VK_SUCCESS)
- qFatal("Failed to map memory: %d", err);
- QMatrix4x4 m = m_proj;
- m.rotate(m_rotation, 0, 0, 1);
- memcpy(p, m.constData(), 16 * sizeof(float));
- m_devFuncs->vkUnmapMemory(dev, m_bufMem);
-
- // Not exactly a real animation system, just advance on every frame for now.
- m_rotation += 1.0f;
-
- m_devFuncs->vkCmdBindPipeline(cb, VK_PIPELINE_BIND_POINT_GRAPHICS, m_pipeline);
- m_devFuncs->vkCmdBindDescriptorSets(cb, VK_PIPELINE_BIND_POINT_GRAPHICS, m_pipelineLayout, 0, 1,
- &m_descSet[m_window->currentFrame()], 0, nullptr);
- VkDeviceSize vbOffset = 0;
- m_devFuncs->vkCmdBindVertexBuffers(cb, 0, 1, &m_buf, &vbOffset);
-
- VkViewport viewport;
- viewport.x = viewport.y = 0;
- viewport.width = sz.width();
- viewport.height = sz.height();
- viewport.minDepth = 0;
- viewport.maxDepth = 1;
- m_devFuncs->vkCmdSetViewport(cb, 0, 1, &viewport);
-
- VkRect2D scissor;
- scissor.offset.x = scissor.offset.y = 0;
- scissor.extent.width = viewport.width;
- scissor.extent.height = viewport.height;
- m_devFuncs->vkCmdSetScissor(cb, 0, 1, &scissor);
-
- m_devFuncs->vkCmdDraw(cb, 4, 1, 0, 0);
-
- m_devFuncs->vkCmdEndRenderPass(cmdBuf);
-
- m_window->frameReady();
- m_window->requestUpdate(); // render continuously, throttled by the presentation rate
-}
diff --git a/examples/vulkan/hellovulkantexture/hellovulkantexture.h b/examples/vulkan/hellovulkantexture/hellovulkantexture.h
deleted file mode 100644
index 893d953b6c..0000000000
--- a/examples/vulkan/hellovulkantexture/hellovulkantexture.h
+++ /dev/null
@@ -1,61 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include <QVulkanWindow>
-#include <QImage>
-
-class VulkanRenderer : public QVulkanWindowRenderer
-{
-public:
- VulkanRenderer(QVulkanWindow *w);
-
- void initResources() override;
- void initSwapChainResources() override;
- void releaseSwapChainResources() override;
- void releaseResources() override;
-
- void startNextFrame() override;
-
-private:
- VkShaderModule createShader(const QString &name);
- bool createTexture(const QString &name);
- bool createTextureImage(const QSize &size, VkImage *image, VkDeviceMemory *mem,
- VkImageTiling tiling, VkImageUsageFlags usage, uint32_t memIndex);
- bool writeLinearImage(const QImage &img, VkImage image, VkDeviceMemory memory);
- void ensureTexture();
-
- QVulkanWindow *m_window;
- QVulkanDeviceFunctions *m_devFuncs;
-
- VkDeviceMemory m_bufMem = VK_NULL_HANDLE;
- VkBuffer m_buf = VK_NULL_HANDLE;
- VkDescriptorBufferInfo m_uniformBufInfo[QVulkanWindow::MAX_CONCURRENT_FRAME_COUNT];
-
- VkDescriptorPool m_descPool = VK_NULL_HANDLE;
- VkDescriptorSetLayout m_descSetLayout = VK_NULL_HANDLE;
- VkDescriptorSet m_descSet[QVulkanWindow::MAX_CONCURRENT_FRAME_COUNT];
-
- VkPipelineCache m_pipelineCache = VK_NULL_HANDLE;
- VkPipelineLayout m_pipelineLayout = VK_NULL_HANDLE;
- VkPipeline m_pipeline = VK_NULL_HANDLE;
-
- VkSampler m_sampler = VK_NULL_HANDLE;
- VkImage m_texImage = VK_NULL_HANDLE;
- VkDeviceMemory m_texMem = VK_NULL_HANDLE;
- bool m_texLayoutPending = false;
- VkImageView m_texView = VK_NULL_HANDLE;
- VkImage m_texStaging = VK_NULL_HANDLE;
- VkDeviceMemory m_texStagingMem = VK_NULL_HANDLE;
- bool m_texStagingPending = false;
- QSize m_texSize;
- VkFormat m_texFormat;
-
- QMatrix4x4 m_proj;
- float m_rotation = 0.0f;
-};
-
-class VulkanWindow : public QVulkanWindow
-{
-public:
- QVulkanWindowRenderer *createRenderer() override;
-};
diff --git a/examples/vulkan/hellovulkantexture/hellovulkantexture.pro b/examples/vulkan/hellovulkantexture/hellovulkantexture.pro
deleted file mode 100644
index 59bfcda715..0000000000
--- a/examples/vulkan/hellovulkantexture/hellovulkantexture.pro
+++ /dev/null
@@ -1,7 +0,0 @@
-HEADERS += hellovulkantexture.h
-SOURCES += hellovulkantexture.cpp main.cpp
-RESOURCES += hellovulkantexture.qrc
-
-# install
-target.path = $$[QT_INSTALL_EXAMPLES]/vulkan/hellovulkantexture
-INSTALLS += target
diff --git a/examples/vulkan/hellovulkantexture/hellovulkantexture.qrc b/examples/vulkan/hellovulkantexture/hellovulkantexture.qrc
deleted file mode 100644
index 04e7cda859..0000000000
--- a/examples/vulkan/hellovulkantexture/hellovulkantexture.qrc
+++ /dev/null
@@ -1,7 +0,0 @@
-<!DOCTYPE RCC><RCC version="1.0">
-<qresource>
- <file>texture_vert.spv</file>
- <file>texture_frag.spv</file>
- <file>qt256.png</file>
-</qresource>
-</RCC>
diff --git a/examples/vulkan/hellovulkantexture/main.cpp b/examples/vulkan/hellovulkantexture/main.cpp
deleted file mode 100644
index 59a3273303..0000000000
--- a/examples/vulkan/hellovulkantexture/main.cpp
+++ /dev/null
@@ -1,32 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include <QGuiApplication>
-#include <QVulkanInstance>
-#include <QLoggingCategory>
-#include "hellovulkantexture.h"
-
-Q_LOGGING_CATEGORY(lcVk, "qt.vulkan")
-
-int main(int argc, char *argv[])
-{
- QGuiApplication app(argc, argv);
-
- QLoggingCategory::setFilterRules(QStringLiteral("qt.vulkan=true"));
-
- QVulkanInstance inst;
- inst.setLayers({ "VK_LAYER_KHRONOS_validation" });
-
- if (!inst.create())
- qFatal("Failed to create Vulkan instance: %d", inst.errorCode());
-
- VulkanWindow w;
- w.setVulkanInstance(&inst);
- if (QCoreApplication::arguments().contains(QStringLiteral("--srgb")))
- w.setPreferredColorFormats(QList<VkFormat>() << VK_FORMAT_B8G8R8A8_SRGB);
-
- w.resize(1024, 768);
- w.show();
-
- return app.exec();
-}
diff --git a/examples/vulkan/hellovulkantexture/qt256.png b/examples/vulkan/hellovulkantexture/qt256.png
deleted file mode 100644
index 30c621c9c6..0000000000
--- a/examples/vulkan/hellovulkantexture/qt256.png
+++ /dev/null
Binary files differ
diff --git a/examples/vulkan/hellovulkantexture/texture.frag b/examples/vulkan/hellovulkantexture/texture.frag
deleted file mode 100644
index e6021fe905..0000000000
--- a/examples/vulkan/hellovulkantexture/texture.frag
+++ /dev/null
@@ -1,12 +0,0 @@
-#version 440
-
-layout(location = 0) in vec2 v_texcoord;
-
-layout(location = 0) out vec4 fragColor;
-
-layout(binding = 1) uniform sampler2D tex;
-
-void main()
-{
- fragColor = texture(tex, v_texcoord);
-}
diff --git a/examples/vulkan/hellovulkantexture/texture.vert b/examples/vulkan/hellovulkantexture/texture.vert
deleted file mode 100644
index de486cb772..0000000000
--- a/examples/vulkan/hellovulkantexture/texture.vert
+++ /dev/null
@@ -1,18 +0,0 @@
-#version 440
-
-layout(location = 0) in vec4 position;
-layout(location = 1) in vec2 texcoord;
-
-layout(location = 0) out vec2 v_texcoord;
-
-layout(std140, binding = 0) uniform buf {
- mat4 mvp;
-} ubuf;
-
-out gl_PerVertex { vec4 gl_Position; };
-
-void main()
-{
- v_texcoord = texcoord;
- gl_Position = ubuf.mvp * position;
-}
diff --git a/examples/vulkan/hellovulkantexture/texture_frag.spv b/examples/vulkan/hellovulkantexture/texture_frag.spv
deleted file mode 100644
index 7521ef6eef..0000000000
--- a/examples/vulkan/hellovulkantexture/texture_frag.spv
+++ /dev/null
Binary files differ
diff --git a/examples/vulkan/hellovulkantexture/texture_vert.spv b/examples/vulkan/hellovulkantexture/texture_vert.spv
deleted file mode 100644
index 6292c0de31..0000000000
--- a/examples/vulkan/hellovulkantexture/texture_vert.spv
+++ /dev/null
Binary files differ
diff --git a/examples/vulkan/hellovulkantriangle/CMakeLists.txt b/examples/vulkan/hellovulkantriangle/CMakeLists.txt
index 79979c416c..23789d4717 100644
--- a/examples/vulkan/hellovulkantriangle/CMakeLists.txt
+++ b/examples/vulkan/hellovulkantriangle/CMakeLists.txt
@@ -1,16 +1,13 @@
+# Copyright (C) 2022 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
cmake_minimum_required(VERSION 3.16)
project(hellovulkantriangle LANGUAGES CXX)
-set(CMAKE_AUTOMOC ON)
-
-if(NOT DEFINED INSTALL_EXAMPLESDIR)
- set(INSTALL_EXAMPLESDIR "examples")
-endif()
-
-set(INSTALL_EXAMPLEDIR "${INSTALL_EXAMPLESDIR}/vulkan/hellovulkantriangle")
-
find_package(Qt6 REQUIRED COMPONENTS Core Gui)
+qt_standard_project_setup()
+
qt_add_executable(hellovulkantriangle
../shared/trianglerenderer.cpp ../shared/trianglerenderer.h
main.cpp
@@ -21,9 +18,9 @@ set_target_properties(hellovulkantriangle PROPERTIES
MACOSX_BUNDLE TRUE
)
-target_link_libraries(hellovulkantriangle PUBLIC
- Qt::Core
- Qt::Gui
+target_link_libraries(hellovulkantriangle PRIVATE
+ Qt6::Core
+ Qt6::Gui
)
# Resources:
@@ -48,7 +45,14 @@ qt_add_resources(hellovulkantriangle "hellovulkantriangle"
)
install(TARGETS hellovulkantriangle
- RUNTIME DESTINATION "${INSTALL_EXAMPLEDIR}"
- BUNDLE DESTINATION "${INSTALL_EXAMPLEDIR}"
- LIBRARY DESTINATION "${INSTALL_EXAMPLEDIR}"
+ BUNDLE DESTINATION .
+ RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
+ LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
+)
+
+qt_generate_deploy_app_script(
+ TARGET hellovulkantriangle
+ OUTPUT_SCRIPT deploy_script
+ NO_UNSUPPORTED_PLATFORM_ERROR
)
+install(SCRIPT ${deploy_script})
diff --git a/examples/vulkan/hellovulkantriangle/main.cpp b/examples/vulkan/hellovulkantriangle/main.cpp
index 35ab28f235..3bf45e6b3a 100644
--- a/examples/vulkan/hellovulkantriangle/main.cpp
+++ b/examples/vulkan/hellovulkantriangle/main.cpp
@@ -8,11 +8,13 @@
Q_LOGGING_CATEGORY(lcVk, "qt.vulkan")
+//! [2]
class VulkanWindow : public QVulkanWindow
{
public:
QVulkanWindowRenderer *createRenderer() override;
};
+//! [2]
QVulkanWindowRenderer *VulkanWindow::createRenderer()
{
@@ -25,17 +27,20 @@ int main(int argc, char *argv[])
QLoggingCategory::setFilterRules(QStringLiteral("qt.vulkan=true"));
+//! [0]
QVulkanInstance inst;
inst.setLayers({ "VK_LAYER_KHRONOS_validation" });
-
if (!inst.create())
qFatal("Failed to create Vulkan instance: %d", inst.errorCode());
+//! [0]
+//! [1]
VulkanWindow w;
w.setVulkanInstance(&inst);
w.resize(1024, 768);
w.show();
+//! [1]
return app.exec();
}
diff --git a/examples/vulkan/hellovulkanwidget/CMakeLists.txt b/examples/vulkan/hellovulkanwidget/CMakeLists.txt
index 7f4f53708e..aed510b6d1 100644
--- a/examples/vulkan/hellovulkanwidget/CMakeLists.txt
+++ b/examples/vulkan/hellovulkanwidget/CMakeLists.txt
@@ -1,16 +1,13 @@
+# Copyright (C) 2022 The Qt Company Ltd.
+# SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+
cmake_minimum_required(VERSION 3.16)
project(hellovulkanwidget LANGUAGES CXX)
-set(CMAKE_AUTOMOC ON)
-
-if(NOT DEFINED INSTALL_EXAMPLESDIR)
- set(INSTALL_EXAMPLESDIR "examples")
-endif()
-
-set(INSTALL_EXAMPLEDIR "${INSTALL_EXAMPLESDIR}/vulkan/hellovulkanwidget")
-
find_package(Qt6 REQUIRED COMPONENTS Core Gui Widgets)
+qt_standard_project_setup()
+
qt_add_executable(hellovulkanwidget
../shared/trianglerenderer.cpp ../shared/trianglerenderer.h
hellovulkanwidget.cpp hellovulkanwidget.h
@@ -22,10 +19,10 @@ set_target_properties(hellovulkanwidget PROPERTIES
MACOSX_BUNDLE TRUE
)
-target_link_libraries(hellovulkanwidget PUBLIC
- Qt::Core
- Qt::Gui
- Qt::Widgets
+target_link_libraries(hellovulkanwidget PRIVATE
+ Qt6::Core
+ Qt6::Gui
+ Qt6::Widgets
)
# Resources:
@@ -50,7 +47,14 @@ qt_add_resources(hellovulkanwidget "hellovulkanwidget"
)
install(TARGETS hellovulkanwidget
- RUNTIME DESTINATION "${INSTALL_EXAMPLEDIR}"
- BUNDLE DESTINATION "${INSTALL_EXAMPLEDIR}"
- LIBRARY DESTINATION "${INSTALL_EXAMPLEDIR}"
+ BUNDLE DESTINATION .
+ RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
+ LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
+)
+
+qt_generate_deploy_app_script(
+ TARGET hellovulkanwidget
+ OUTPUT_SCRIPT deploy_script
+ NO_UNSUPPORTED_PLATFORM_ERROR
)
+install(SCRIPT ${deploy_script})
diff --git a/examples/vulkan/hellovulkanwidget/hellovulkanwidget.h b/examples/vulkan/hellovulkanwidget/hellovulkanwidget.h
index f4bcd5bb58..cbebe17f64 100644
--- a/examples/vulkan/hellovulkanwidget/hellovulkanwidget.h
+++ b/examples/vulkan/hellovulkanwidget/hellovulkanwidget.h
@@ -1,16 +1,17 @@
// Copyright (C) 2017 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+#ifndef HELLOVULKANWIDGET_H
+#define HELLOVULKANWIDGET_H
+
#include "../shared/trianglerenderer.h"
#include <QWidget>
-class VulkanWindow;
+QT_FORWARD_DECLARE_CLASS(QTabWidget)
+QT_FORWARD_DECLARE_CLASS(QPlainTextEdit)
+QT_FORWARD_DECLARE_CLASS(QLCDNumber)
-QT_BEGIN_NAMESPACE
-class QTabWidget;
-class QPlainTextEdit;
-class QLCDNumber;
-QT_END_NAMESPACE
+class VulkanWindow;
class MainWindow : public QWidget
{
@@ -51,3 +52,5 @@ signals:
void vulkanInfoReceived(const QString &text);
void frameQueued(int colorValue);
};
+
+#endif // HELLOVULKANWIDGET_H
diff --git a/examples/vulkan/hellovulkanwindow/CMakeLists.txt b/examples/vulkan/hellovulkanwindow/CMakeLists.txt
deleted file mode 100644
index 7323bab06c..0000000000
--- a/examples/vulkan/hellovulkanwindow/CMakeLists.txt
+++ /dev/null
@@ -1,33 +0,0 @@
-cmake_minimum_required(VERSION 3.16)
-project(hellovulkanwindow LANGUAGES CXX)
-
-set(CMAKE_AUTOMOC ON)
-
-if(NOT DEFINED INSTALL_EXAMPLESDIR)
- set(INSTALL_EXAMPLESDIR "examples")
-endif()
-
-set(INSTALL_EXAMPLEDIR "${INSTALL_EXAMPLESDIR}/vulkan/hellovulkanwindow")
-
-find_package(Qt6 REQUIRED COMPONENTS Core Gui)
-
-qt_add_executable(hellovulkanwindow
- hellovulkanwindow.cpp hellovulkanwindow.h
- main.cpp
-)
-
-set_target_properties(hellovulkanwindow PROPERTIES
- WIN32_EXECUTABLE TRUE
- MACOSX_BUNDLE TRUE
-)
-
-target_link_libraries(hellovulkanwindow PUBLIC
- Qt::Core
- Qt::Gui
-)
-
-install(TARGETS hellovulkanwindow
- RUNTIME DESTINATION "${INSTALL_EXAMPLEDIR}"
- BUNDLE DESTINATION "${INSTALL_EXAMPLEDIR}"
- LIBRARY DESTINATION "${INSTALL_EXAMPLEDIR}"
-)
diff --git a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.cpp b/examples/vulkan/hellovulkanwindow/hellovulkanwindow.cpp
deleted file mode 100644
index fba3a47547..0000000000
--- a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.cpp
+++ /dev/null
@@ -1,81 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include "hellovulkanwindow.h"
-#include <QVulkanFunctions>
-
-//! [0]
-QVulkanWindowRenderer *VulkanWindow::createRenderer()
-{
- return new VulkanRenderer(this);
-}
-
-VulkanRenderer::VulkanRenderer(QVulkanWindow *w)
- : m_window(w)
-{
-}
-//! [0]
-
-//! [1]
-void VulkanRenderer::initResources()
-{
- qDebug("initResources");
-
- m_devFuncs = m_window->vulkanInstance()->deviceFunctions(m_window->device());
-}
-//! [1]
-
-void VulkanRenderer::initSwapChainResources()
-{
- qDebug("initSwapChainResources");
-}
-
-void VulkanRenderer::releaseSwapChainResources()
-{
- qDebug("releaseSwapChainResources");
-}
-
-void VulkanRenderer::releaseResources()
-{
- qDebug("releaseResources");
-}
-
-//! [2]
-void VulkanRenderer::startNextFrame()
-{
- m_green += 0.005f;
- if (m_green > 1.0f)
- m_green = 0.0f;
-
- VkClearColorValue clearColor = {{ 0.0f, m_green, 0.0f, 1.0f }};
- VkClearDepthStencilValue clearDS = { 1.0f, 0 };
- VkClearValue clearValues[2];
- memset(clearValues, 0, sizeof(clearValues));
- clearValues[0].color = clearColor;
- clearValues[1].depthStencil = clearDS;
-
- VkRenderPassBeginInfo rpBeginInfo;
- memset(&rpBeginInfo, 0, sizeof(rpBeginInfo));
- rpBeginInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO;
- rpBeginInfo.renderPass = m_window->defaultRenderPass();
- rpBeginInfo.framebuffer = m_window->currentFramebuffer();
- const QSize sz = m_window->swapChainImageSize();
- rpBeginInfo.renderArea.extent.width = sz.width();
- rpBeginInfo.renderArea.extent.height = sz.height();
- rpBeginInfo.clearValueCount = 2;
- rpBeginInfo.pClearValues = clearValues;
- VkCommandBuffer cmdBuf = m_window->currentCommandBuffer();
- m_devFuncs->vkCmdBeginRenderPass(cmdBuf, &rpBeginInfo, VK_SUBPASS_CONTENTS_INLINE);
-
- // Do nothing else. We will just clear to green, changing the component on
- // every invocation. This also helps verifying the rate to which the thread
- // is throttled to. (The elapsed time between startNextFrame calls should
- // typically be around 16 ms. Note that rendering is 2 frames ahead of what
- // is displayed.)
-
- m_devFuncs->vkCmdEndRenderPass(cmdBuf);
-
- m_window->frameReady();
- m_window->requestUpdate(); // render continuously, throttled by the presentation rate
-}
-//! [2]
diff --git a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.h b/examples/vulkan/hellovulkanwindow/hellovulkanwindow.h
deleted file mode 100644
index 783abef2a9..0000000000
--- a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.h
+++ /dev/null
@@ -1,30 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include <QVulkanWindow>
-
-//! [0]
-class VulkanRenderer : public QVulkanWindowRenderer
-{
-public:
- VulkanRenderer(QVulkanWindow *w);
-
- void initResources() override;
- void initSwapChainResources() override;
- void releaseSwapChainResources() override;
- void releaseResources() override;
-
- void startNextFrame() override;
-
-private:
- QVulkanWindow *m_window;
- QVulkanDeviceFunctions *m_devFuncs;
- float m_green = 0;
-};
-
-class VulkanWindow : public QVulkanWindow
-{
-public:
- QVulkanWindowRenderer *createRenderer() override;
-};
-//! [0]
diff --git a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.pro b/examples/vulkan/hellovulkanwindow/hellovulkanwindow.pro
deleted file mode 100644
index 8f7d9494e2..0000000000
--- a/examples/vulkan/hellovulkanwindow/hellovulkanwindow.pro
+++ /dev/null
@@ -1,6 +0,0 @@
-HEADERS += hellovulkanwindow.h
-SOURCES += hellovulkanwindow.cpp main.cpp
-
-# install
-target.path = $$[QT_INSTALL_EXAMPLES]/vulkan/hellovulkanwindow
-INSTALLS += target
diff --git a/examples/vulkan/hellovulkanwindow/main.cpp b/examples/vulkan/hellovulkanwindow/main.cpp
deleted file mode 100644
index 46393b1cdb..0000000000
--- a/examples/vulkan/hellovulkanwindow/main.cpp
+++ /dev/null
@@ -1,33 +0,0 @@
-// Copyright (C) 2017 The Qt Company Ltd.
-// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
-
-#include <QGuiApplication>
-#include <QVulkanInstance>
-#include <QLoggingCategory>
-#include "hellovulkanwindow.h"
-
-Q_LOGGING_CATEGORY(lcVk, "qt.vulkan")
-
-int main(int argc, char *argv[])
-{
- QGuiApplication app(argc, argv);
-
- QLoggingCategory::setFilterRules(QStringLiteral("qt.vulkan=true"));
-
-//! [0]
- QVulkanInstance inst;
- inst.setLayers({ "VK_LAYER_KHRONOS_validation" });
- if (!inst.create())
- qFatal("Failed to create Vulkan instance: %d", inst.errorCode());
-//! [0]
-
-//! [1]
- VulkanWindow w;
- w.setVulkanInstance(&inst);
-
- w.resize(1024, 768);
- w.show();
-//! [1]
-
- return app.exec();
-}
diff --git a/examples/vulkan/shared/trianglerenderer.h b/examples/vulkan/shared/trianglerenderer.h
index 37fa608973..129c35aed9 100644
--- a/examples/vulkan/shared/trianglerenderer.h
+++ b/examples/vulkan/shared/trianglerenderer.h
@@ -1,6 +1,9 @@
// Copyright (C) 2017 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR BSD-3-Clause
+#ifndef TRIANGLERENDERER_H
+#define TRIANGLERENDERER_H
+
#include <QVulkanWindow>
class TriangleRenderer : public QVulkanWindowRenderer
@@ -36,3 +39,5 @@ protected:
QMatrix4x4 m_proj;
float m_rotation = 0.0f;
};
+
+#endif // TRIANGLERENDERER_H
diff --git a/examples/vulkan/vulkan.pro b/examples/vulkan/vulkan.pro
index 920762cb9e..4fe4dfda78 100644
--- a/examples/vulkan/vulkan.pro
+++ b/examples/vulkan/vulkan.pro
@@ -1,8 +1,6 @@
TEMPLATE = subdirs
-SUBDIRS = hellovulkanwindow \
- hellovulkantriangle \
- hellovulkantexture
+SUBDIRS = hellovulkantriangle
qtHaveModule(widgets) {
SUBDIRS += hellovulkanwidget