// Copyright (C) 2019 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include #include #include #include #include #include #include #include #include #include #if QT_CONFIG(opengl) # include # include # include # include # include # define TST_GL #endif #if QT_CONFIG(vulkan) # include # include # include # define TST_VK #endif #ifdef Q_OS_WIN #include # define TST_D3D11 #endif #if defined(Q_OS_MACOS) || defined(Q_OS_IOS) # include # define TST_MTL #endif Q_DECLARE_METATYPE(QRhi::Implementation) Q_DECLARE_METATYPE(QRhiInitParams *) class tst_QRhi : public QObject { Q_OBJECT private slots: void initTestCase(); void cleanupTestCase(); void rhiTestData(); void create_data(); void create(); void stats_data(); void stats(); void nativeHandles_data(); void nativeHandles(); void nativeHandlesImportVulkan(); void nativeHandlesImportD3D11(); void nativeHandlesImportOpenGL(); void nativeTexture_data(); void nativeTexture(); void nativeBuffer_data(); void nativeBuffer(); void resourceUpdateBatchBuffer_data(); void resourceUpdateBatchBuffer(); void resourceUpdateBatchRGBATextureUpload_data(); void resourceUpdateBatchRGBATextureUpload(); void resourceUpdateBatchRGBATextureCopy_data(); void resourceUpdateBatchRGBATextureCopy(); void resourceUpdateBatchRGBATextureMip_data(); void resourceUpdateBatchRGBATextureMip(); void resourceUpdateBatchTextureRawDataStride_data(); void resourceUpdateBatchTextureRawDataStride(); void resourceUpdateBatchLotsOfResources_data(); void resourceUpdateBatchLotsOfResources(); void invalidPipeline_data(); void invalidPipeline(); void srbLayoutCompatibility_data(); void srbLayoutCompatibility(); void srbWithNoResource_data(); void srbWithNoResource(); void renderPassDescriptorCompatibility_data(); void renderPassDescriptorCompatibility(); void renderPassDescriptorClone_data(); void renderPassDescriptorClone(); void renderToTextureSimple_data(); void renderToTextureSimple(); void renderToTextureMip_data(); void renderToTextureMip(); void renderToTextureCubemapFace_data(); void renderToTextureCubemapFace(); void renderToTextureTextureArray_data(); void renderToTextureTextureArray(); void renderToTextureTexturedQuad_data(); void renderToTextureTexturedQuad(); void renderToTextureSampleWithSeparateTextureAndSampler_data(); void renderToTextureSampleWithSeparateTextureAndSampler(); void renderToTextureArrayOfTexturedQuad_data(); void renderToTextureArrayOfTexturedQuad(); void renderToTextureTexturedQuadAndUniformBuffer_data(); void renderToTextureTexturedQuadAndUniformBuffer(); void renderToTextureTexturedQuadAllDynamicBuffers_data(); void renderToTextureTexturedQuadAllDynamicBuffers(); void renderToTextureDeferredSrb_data(); void renderToTextureDeferredSrb(); void renderToTextureMultipleUniformBuffersAndDynamicOffset_data(); void renderToTextureMultipleUniformBuffersAndDynamicOffset(); void renderToTextureSrbReuse_data(); void renderToTextureSrbReuse(); void renderToTextureIndexedDraw_data(); void renderToTextureIndexedDraw(); void renderToWindowSimple_data(); void renderToWindowSimple(); void finishWithinSwapchainFrame_data(); void finishWithinSwapchainFrame(); void resourceUpdateBatchBufferTextureWithSwapchainFrames_data(); void resourceUpdateBatchBufferTextureWithSwapchainFrames(); void textureRenderTargetAutoRebuild_data(); void textureRenderTargetAutoRebuild(); void pipelineCache_data(); void pipelineCache(); void textureImportOpenGL(); void renderbufferImportOpenGL(); void threeDimTexture_data(); void threeDimTexture(); void oneDimTexture_data(); void oneDimTexture(); void leakedResourceDestroy_data(); void leakedResourceDestroy(); void renderToFloatTexture_data(); void renderToFloatTexture(); void renderToRgb10Texture_data(); void renderToRgb10Texture(); void tessellation_data(); void tessellation(); private: void setWindowType(QWindow *window, QRhi::Implementation impl); struct { QRhiNullInitParams null; #ifdef TST_GL QRhiGles2InitParams gl; #endif #ifdef TST_VK QRhiVulkanInitParams vk; #endif #ifdef TST_D3D11 QRhiD3D11InitParams d3d; #endif #ifdef TST_MTL QRhiMetalInitParams mtl; #endif } initParams; #ifdef TST_VK QVulkanInstance vulkanInstance; #endif QOffscreenSurface *fallbackSurface = nullptr; }; void tst_QRhi::initTestCase() { #ifdef TST_GL QSurfaceFormat fmt; fmt.setDepthBufferSize(24); fmt.setStencilBufferSize(8); QSurfaceFormat::setDefaultFormat(fmt); initParams.gl.format = QSurfaceFormat::defaultFormat(); fallbackSurface = QRhiGles2InitParams::newFallbackSurface(); initParams.gl.fallbackSurface = fallbackSurface; #endif #ifdef TST_VK const QVersionNumber supportedVersion = vulkanInstance.supportedApiVersion(); if (supportedVersion >= QVersionNumber(1, 2)) vulkanInstance.setApiVersion(QVersionNumber(1, 2)); else if (supportedVersion >= QVersionNumber(1, 1)) vulkanInstance.setApiVersion(QVersionNumber(1, 1)); vulkanInstance.setLayers({ "VK_LAYER_KHRONOS_validation" }); vulkanInstance.setExtensions(QRhiVulkanInitParams::preferredInstanceExtensions()); vulkanInstance.create(); initParams.vk.inst = &vulkanInstance; #endif #ifdef TST_D3D11 initParams.d3d.enableDebugLayer = true; #endif } void tst_QRhi::cleanupTestCase() { #ifdef TST_VK vulkanInstance.destroy(); #endif delete fallbackSurface; } void tst_QRhi::rhiTestData() { QTest::addColumn("impl"); QTest::addColumn("initParams"); // webOS does not support raster (software) pipeline #ifndef Q_OS_WEBOS QTest::newRow("Null") << QRhi::Null << static_cast(&initParams.null); #endif #ifdef TST_GL if (QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::OpenGL)) QTest::newRow("OpenGL") << QRhi::OpenGLES2 << static_cast(&initParams.gl); #endif #ifdef TST_VK if (vulkanInstance.isValid()) QTest::newRow("Vulkan") << QRhi::Vulkan << static_cast(&initParams.vk); #endif #ifdef TST_D3D11 QTest::newRow("Direct3D 11") << QRhi::D3D11 << static_cast(&initParams.d3d); #endif #ifdef TST_MTL QTest::newRow("Metal") << QRhi::Metal << static_cast(&initParams.mtl); #endif } void tst_QRhi::create_data() { rhiTestData(); } static int aligned(int v, int a) { return (v + a - 1) & ~(a - 1); } void tst_QRhi::create() { // Merely attempting to create a QRhi should survive, with an error when // not supported. (of course, there is always a chance we encounter a crash // due to some random graphics stack...) QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (rhi) { QVERIFY(QRhi::probe(impl, initParams)); qDebug() << rhi->driverInfo(); QCOMPARE(rhi->backend(), impl); QVERIFY(strcmp(rhi->backendName(), "")); QVERIFY(!strcmp(rhi->backendName(), QRhi::backendName(rhi->backend()))); QVERIFY(!rhi->driverInfo().deviceName.isEmpty()); QCOMPARE(rhi->thread(), QThread::currentThread()); // do a basic smoke test for the apis that do not directly render anything int cleanupOk = 0; QRhi *rhiPtr = rhi.data(); auto cleanupFunc = [rhiPtr, &cleanupOk](QRhi *dyingRhi) { if (rhiPtr == dyingRhi) cleanupOk += 1; }; rhi->addCleanupCallback(cleanupFunc); rhi->runCleanup(); QCOMPARE(cleanupOk, 1); cleanupOk = 0; rhi->addCleanupCallback(cleanupFunc); QRhiResourceUpdateBatch *resUpd = rhi->nextResourceUpdateBatch(); QVERIFY(resUpd); resUpd->release(); QRhiResourceUpdateBatch *resUpdArray[64]; for (int i = 0; i < 64; ++i) { resUpdArray[i] = rhi->nextResourceUpdateBatch(); QVERIFY(resUpdArray[i]); } resUpd = rhi->nextResourceUpdateBatch(); QVERIFY(!resUpd); for (int i = 0; i < 64; ++i) resUpdArray[i]->release(); resUpd = rhi->nextResourceUpdateBatch(); QVERIFY(resUpd); resUpd->release(); QVERIFY(!rhi->supportedSampleCounts().isEmpty()); QVERIFY(rhi->supportedSampleCounts().contains(1)); QVERIFY(rhi->ubufAlignment() > 0); QCOMPARE(rhi->ubufAligned(123), aligned(123, rhi->ubufAlignment())); QCOMPARE(rhi->mipLevelsForSize(QSize(512, 300)), 10); QCOMPARE(rhi->sizeForMipLevel(0, QSize(512, 300)), QSize(512, 300)); QCOMPARE(rhi->sizeForMipLevel(1, QSize(512, 300)), QSize(256, 150)); QCOMPARE(rhi->sizeForMipLevel(2, QSize(512, 300)), QSize(128, 75)); QCOMPARE(rhi->sizeForMipLevel(9, QSize(512, 300)), QSize(1, 1)); const bool fbUp = rhi->isYUpInFramebuffer(); const bool ndcUp = rhi->isYUpInNDC(); const bool d0to1 = rhi->isClipDepthZeroToOne(); const QMatrix4x4 corrMat = rhi->clipSpaceCorrMatrix(); if (impl == QRhi::OpenGLES2) { QVERIFY(fbUp); QVERIFY(ndcUp); QVERIFY(!d0to1); QVERIFY(corrMat.isIdentity()); } else if (impl == QRhi::Vulkan) { QVERIFY(!fbUp); QVERIFY(!ndcUp); QVERIFY(d0to1); QVERIFY(!corrMat.isIdentity()); } else if (impl == QRhi::D3D11) { QVERIFY(!fbUp); QVERIFY(ndcUp); QVERIFY(d0to1); QVERIFY(!corrMat.isIdentity()); } else if (impl == QRhi::Metal) { QVERIFY(!fbUp); QVERIFY(ndcUp); QVERIFY(d0to1); QVERIFY(!corrMat.isIdentity()); } const int texMin = rhi->resourceLimit(QRhi::TextureSizeMin); const int texMax = rhi->resourceLimit(QRhi::TextureSizeMax); const int maxAtt = rhi->resourceLimit(QRhi::MaxColorAttachments); const int framesInFlight = rhi->resourceLimit(QRhi::FramesInFlight); const int texArrayMax = rhi->resourceLimit(QRhi::TextureArraySizeMax); const int uniBufRangeMax = rhi->resourceLimit(QRhi::MaxUniformBufferRange); const int maxVertexInputs = rhi->resourceLimit(QRhi::MaxVertexInputs); const int maxVertexOutputs = rhi->resourceLimit(QRhi::MaxVertexOutputs); QVERIFY(texMin >= 1); QVERIFY(texMax >= texMin); QVERIFY(maxAtt >= 1); QVERIFY(framesInFlight >= 1); if (rhi->isFeatureSupported(QRhi::TextureArrays)) QVERIFY(texArrayMax > 1); QVERIFY(uniBufRangeMax >= 224 * 4 * 4); QVERIFY(maxVertexInputs >= 8); QVERIFY(maxVertexOutputs >= 8); QVERIFY(rhi->nativeHandles()); const QRhi::Feature features[] = { QRhi::MultisampleTexture, QRhi::MultisampleRenderBuffer, QRhi::DebugMarkers, QRhi::Timestamps, QRhi::Instancing, QRhi::CustomInstanceStepRate, QRhi::PrimitiveRestart, QRhi::NonDynamicUniformBuffers, QRhi::NonFourAlignedEffectiveIndexBufferOffset, QRhi::NPOTTextureRepeat, QRhi::RedOrAlpha8IsRed, QRhi::ElementIndexUint, QRhi::Compute, QRhi::WideLines, QRhi::VertexShaderPointSize, QRhi::BaseVertex, QRhi::BaseInstance, QRhi::TriangleFanTopology, QRhi::ReadBackNonUniformBuffer, QRhi::ReadBackNonBaseMipLevel, QRhi::TexelFetch, QRhi::RenderToNonBaseMipLevel, QRhi::IntAttributes, QRhi::ScreenSpaceDerivatives, QRhi::ReadBackAnyTextureFormat, QRhi::PipelineCacheDataLoadSave, QRhi::ImageDataStride, QRhi::RenderBufferImport, QRhi::ThreeDimensionalTextures, QRhi::RenderTo3DTextureSlice, QRhi::TextureArrays, QRhi::Tessellation, QRhi::GeometryShader, QRhi::TextureArrayRange, QRhi::NonFillPolygonMode }; for (size_t i = 0; i isFeatureSupported(features[i]); QVERIFY(rhi->isTextureFormatSupported(QRhiTexture::RGBA8)); rhi->releaseCachedResources(); QVERIFY(!rhi->isDeviceLost()); rhi.reset(); QCOMPARE(cleanupOk, 1); } } void tst_QRhi::stats_data() { rhiTestData(); } void tst_QRhi::stats() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing statistics getter"); QRhiStats stats = rhi->statistics(); qDebug() << stats; QCOMPARE(stats.totalPipelineCreationTime, 0); if (impl == QRhi::Vulkan) { QScopedPointer buf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, 32768)); QVERIFY(buf->create()); QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(1024, 1024))); QVERIFY(tex->create()); stats = rhi->statistics(); qDebug() << stats; QVERIFY(stats.allocCount > 0); QVERIFY(stats.blockCount > 0); QVERIFY(stats.usedBytes > 0); } } void tst_QRhi::nativeHandles_data() { rhiTestData(); } void tst_QRhi::nativeHandles() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing native handles"); // QRhi::nativeHandles() { const QRhiNativeHandles *rhiHandles = rhi->nativeHandles(); Q_ASSERT(rhiHandles); switch (impl) { case QRhi::Null: break; #ifdef TST_VK case QRhi::Vulkan: { const QRhiVulkanNativeHandles *vkHandles = static_cast(rhiHandles); QVERIFY(vkHandles->physDev); QVERIFY(vkHandles->dev); QVERIFY(vkHandles->gfxQueue); QVERIFY(vkHandles->vmemAllocator); } break; #endif #ifdef TST_GL case QRhi::OpenGLES2: { const QRhiGles2NativeHandles *glHandles = static_cast(rhiHandles); QVERIFY(glHandles->context); QVERIFY(glHandles->context->isValid()); glHandles->context->doneCurrent(); QVERIFY(!QOpenGLContext::currentContext()); rhi->makeThreadLocalNativeContextCurrent(); QVERIFY(QOpenGLContext::currentContext() == glHandles->context); } break; #endif #ifdef TST_D3D11 case QRhi::D3D11: { const QRhiD3D11NativeHandles *d3dHandles = static_cast(rhiHandles); QVERIFY(d3dHandles->dev); QVERIFY(d3dHandles->context); QVERIFY(d3dHandles->featureLevel > 0); QVERIFY(d3dHandles->adapterLuidLow || d3dHandles->adapterLuidHigh); } break; #endif #ifdef TST_MTL case QRhi::Metal: { const QRhiMetalNativeHandles *mtlHandles = static_cast(rhiHandles); QVERIFY(mtlHandles->dev); QVERIFY(mtlHandles->cmdQueue); } break; #endif default: Q_ASSERT(false); } } // QRhiCommandBuffer::nativeHandles() { QRhiCommandBuffer *cb = nullptr; QRhi::FrameOpResult result = rhi->beginOffscreenFrame(&cb); QVERIFY(result == QRhi::FrameOpSuccess); QVERIFY(cb); Q_DECL_UNUSED const QRhiNativeHandles *cbHandles = cb->nativeHandles(); // no null check here, backends where not applicable will return null switch (impl) { case QRhi::Null: break; #ifdef TST_VK case QRhi::Vulkan: { const QRhiVulkanCommandBufferNativeHandles *vkHandles = static_cast(cbHandles); QVERIFY(vkHandles); QVERIFY(vkHandles->commandBuffer); } break; #endif #ifdef TST_GL case QRhi::OpenGLES2: break; #endif #ifdef TST_D3D11 case QRhi::D3D11: break; #endif #ifdef TST_MTL case QRhi::Metal: { const QRhiMetalCommandBufferNativeHandles *mtlHandles = static_cast(cbHandles); QVERIFY(mtlHandles); QVERIFY(mtlHandles->commandBuffer); QVERIFY(!mtlHandles->encoder); QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); QVERIFY(rpDesc); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); cb->beginPass(rt.data(), Qt::red, { 1.0f, 0 }); QVERIFY(static_cast(cb->nativeHandles())->encoder); cb->endPass(); } break; #endif default: Q_ASSERT(false); } rhi->endOffscreenFrame(); } // QRhiRenderPassDescriptor::nativeHandles() { QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); QVERIFY(rpDesc); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); switch (impl) { case QRhi::Null: break; #ifdef TST_VK case QRhi::Vulkan: { const QRhiNativeHandles *rpHandles = rpDesc->nativeHandles(); const QRhiVulkanRenderPassNativeHandles *vkHandles = static_cast(rpHandles); QVERIFY(vkHandles); QVERIFY(vkHandles->renderPass); } break; #endif #ifdef TST_GL case QRhi::OpenGLES2: break; #endif #ifdef TST_D3D11 case QRhi::D3D11: break; #endif #ifdef TST_MTL case QRhi::Metal: break; #endif default: Q_ASSERT(false); } } } void tst_QRhi::nativeHandlesImportVulkan() { #ifdef TST_VK // VkDevice and everything else. For simplicity we'll get QRhi to create one, and then use that with another QRhi. { QScopedPointer rhi(QRhi::create(QRhi::Vulkan, &initParams.vk, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("Skipping native Vulkan test"); const QRhiVulkanNativeHandles *nativeHandles = static_cast(rhi->nativeHandles()); QRhiVulkanNativeHandles h = *nativeHandles; // do not pass the rarely used fields, this is useful to test if it creates its own as expected h.vmemAllocator = nullptr; QScopedPointer adoptingRhi(QRhi::create(QRhi::Vulkan, &initParams.vk, QRhi::Flags(), &h)); QVERIFY(adoptingRhi); const QRhiVulkanNativeHandles *newNativeHandles = static_cast(adoptingRhi->nativeHandles()); QCOMPARE(newNativeHandles->physDev, nativeHandles->physDev); QCOMPARE(newNativeHandles->dev, nativeHandles->dev); QCOMPARE(newNativeHandles->gfxQueueFamilyIdx, nativeHandles->gfxQueueFamilyIdx); QCOMPARE(newNativeHandles->gfxQueueIdx, nativeHandles->gfxQueueIdx); QVERIFY(newNativeHandles->vmemAllocator != nativeHandles->vmemAllocator); } // Physical device only { uint32_t physDevCount = 0; QVulkanFunctions *f = vulkanInstance.functions(); f->vkEnumeratePhysicalDevices(vulkanInstance.vkInstance(), &physDevCount, nullptr); if (physDevCount < 1) QSKIP("No Vulkan physical devices, skip"); QVarLengthArray physDevs(physDevCount); f->vkEnumeratePhysicalDevices(vulkanInstance.vkInstance(), &physDevCount, physDevs.data()); for (uint32_t i = 0; i < physDevCount; ++i) { QRhiVulkanNativeHandles h; h.physDev = physDevs[i]; QScopedPointer rhi(QRhi::create(QRhi::Vulkan, &initParams.vk, QRhi::Flags(), &h)); // ok if fails, what we want to know is that if it succeeds, it must use that given phys.dev. if (!rhi) { qWarning("Skipping native Vulkan handle test for physical device %u", i); continue; } const QRhiVulkanNativeHandles *actualNativeHandles = static_cast(rhi->nativeHandles()); QCOMPARE(actualNativeHandles->physDev, physDevs[i]); } } #else QSKIP("Skipping Vulkan-specific test"); #endif } void tst_QRhi::nativeHandlesImportD3D11() { #ifdef TST_D3D11 QScopedPointer rhi(QRhi::create(QRhi::D3D11, &initParams.d3d, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing D3D11 native handle import"); const QRhiD3D11NativeHandles *nativeHandles = static_cast(rhi->nativeHandles()); // Case 1: device and context { QRhiD3D11NativeHandles h = *nativeHandles; h.featureLevel = 0; // see if these are queried as expected, even when not provided h.adapterLuidLow = 0; h.adapterLuidHigh = 0; QScopedPointer adoptingRhi(QRhi::create(QRhi::D3D11, &initParams.d3d, QRhi::Flags(), &h)); QVERIFY(adoptingRhi); const QRhiD3D11NativeHandles *newNativeHandles = static_cast(adoptingRhi->nativeHandles()); QCOMPARE(newNativeHandles->dev, nativeHandles->dev); QCOMPARE(newNativeHandles->context, nativeHandles->context); QCOMPARE(newNativeHandles->featureLevel, nativeHandles->featureLevel); QCOMPARE(newNativeHandles->adapterLuidLow, nativeHandles->adapterLuidLow); QCOMPARE(newNativeHandles->adapterLuidHigh, nativeHandles->adapterLuidHigh); } // Case 2: adapter and feature level only (hello OpenXR) { QRhiD3D11NativeHandles h = *nativeHandles; h.dev = nullptr; h.context = nullptr; QScopedPointer adoptingRhi(QRhi::create(QRhi::D3D11, &initParams.d3d, QRhi::Flags(), &h)); QVERIFY(adoptingRhi); const QRhiD3D11NativeHandles *newNativeHandles = static_cast(adoptingRhi->nativeHandles()); QVERIFY(newNativeHandles->dev != nativeHandles->dev); QVERIFY(newNativeHandles->context != nativeHandles->context); QCOMPARE(newNativeHandles->featureLevel, nativeHandles->featureLevel); QCOMPARE(newNativeHandles->adapterLuidLow, nativeHandles->adapterLuidLow); QCOMPARE(newNativeHandles->adapterLuidHigh, nativeHandles->adapterLuidHigh); } #else QSKIP("Skipping D3D11-specific test"); #endif } void tst_QRhi::nativeHandlesImportOpenGL() { #ifdef TST_GL QRhiGles2NativeHandles h; QScopedPointer ctx(new QOpenGLContext); if (!ctx->create()) QSKIP("No OpenGL context, skipping OpenGL-specific test"); h.context = ctx.data(); QScopedPointer rhi(QRhi::create(QRhi::OpenGLES2, &initParams.gl, QRhi::Flags(), &h)); if (!rhi) QSKIP("QRhi could not be created, skipping testing OpenGL native handle import"); const QRhiGles2NativeHandles *actualNativeHandles = static_cast(rhi->nativeHandles()); QCOMPARE(actualNativeHandles->context, ctx.data()); rhi->makeThreadLocalNativeContextCurrent(); QCOMPARE(QOpenGLContext::currentContext(), ctx.data()); #else QSKIP("Skipping OpenGL-specific test"); #endif } void tst_QRhi::nativeTexture_data() { rhiTestData(); } void tst_QRhi::nativeTexture() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing native texture"); QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 256))); QVERIFY(tex->create()); const QRhiTexture::NativeTexture nativeTex = tex->nativeTexture(); switch (impl) { case QRhi::Null: break; #ifdef TST_VK case QRhi::Vulkan: { auto image = VkImage(nativeTex.object); QVERIFY(image); QVERIFY(nativeTex.layout >= 1); // VK_IMAGE_LAYOUT_GENERAL QVERIFY(nativeTex.layout <= 8); // VK_IMAGE_LAYOUT_PREINITIALIZED } break; #endif #ifdef TST_GL case QRhi::OpenGLES2: { auto textureId = uint(nativeTex.object); QVERIFY(textureId); } break; #endif #ifdef TST_D3D11 case QRhi::D3D11: { auto *texture = reinterpret_cast(nativeTex.object); QVERIFY(texture); } break; #endif #ifdef TST_MTL case QRhi::Metal: { auto texture = (void *)nativeTex.object; QVERIFY(texture); } break; #endif default: Q_ASSERT(false); } } void tst_QRhi::nativeBuffer_data() { rhiTestData(); } void tst_QRhi::nativeBuffer() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing native buffer query"); const QRhiBuffer::Type types[3] = { QRhiBuffer::Immutable, QRhiBuffer::Static, QRhiBuffer::Dynamic }; const QRhiBuffer::UsageFlags usages[3] = { QRhiBuffer::VertexBuffer, QRhiBuffer::IndexBuffer, QRhiBuffer::UniformBuffer }; for (int typeUsageIdx = 0; typeUsageIdx < 3; ++typeUsageIdx) { QScopedPointer buf(rhi->newBuffer(types[typeUsageIdx], usages[typeUsageIdx], 256)); QVERIFY(buf->create()); const QRhiBuffer::NativeBuffer nativeBuf = buf->nativeBuffer(); QVERIFY(nativeBuf.slotCount <= rhi->resourceLimit(QRhi::FramesInFlight)); switch (impl) { case QRhi::Null: break; #ifdef TST_VK case QRhi::Vulkan: { QVERIFY(nativeBuf.slotCount >= 1); // always backed by native buffers for (int i = 0; i < nativeBuf.slotCount; ++i) { auto *buffer = static_cast(nativeBuf.objects[i]); QVERIFY(buffer); QVERIFY(*buffer); } } break; #endif #ifdef TST_GL case QRhi::OpenGLES2: { QVERIFY(nativeBuf.slotCount >= 0); // UniformBuffers are not backed by native buffers, so 0 is perfectly valid for (int i = 0; i < nativeBuf.slotCount; ++i) { auto *bufferId = static_cast(nativeBuf.objects[i]); QVERIFY(bufferId); QVERIFY(*bufferId); } } break; #endif #ifdef TST_D3D11 case QRhi::D3D11: { QVERIFY(nativeBuf.slotCount >= 1); // always backed by native buffers for (int i = 0; i < nativeBuf.slotCount; ++i) { auto *buffer = static_cast(nativeBuf.objects[i]); QVERIFY(buffer); QVERIFY(*buffer); } } break; #endif #ifdef TST_MTL case QRhi::Metal: { QVERIFY(nativeBuf.slotCount >= 1); // always backed by native buffers for (int i = 0; i < nativeBuf.slotCount; ++i) { void * const * buffer = (void * const *) nativeBuf.objects[i]; QVERIFY(buffer); QVERIFY(*buffer); } } break; #endif default: Q_ASSERT(false); } } } static bool submitResourceUpdates(QRhi *rhi, QRhiResourceUpdateBatch *batch) { QRhiCommandBuffer *cb = nullptr; QRhi::FrameOpResult result = rhi->beginOffscreenFrame(&cb); if (result != QRhi::FrameOpSuccess) { qWarning("beginOffscreenFrame returned %d", result); return false; } if (!cb) { qWarning("No command buffer from beginOffscreenFrame"); return false; } cb->resourceUpdate(batch); rhi->endOffscreenFrame(); return true; } void tst_QRhi::resourceUpdateBatchBuffer_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchBuffer() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing buffer resource updates"); const int bufferSize = 23; const QByteArray a(bufferSize, 'A'); const QByteArray b(bufferSize, 'B'); // dynamic buffer, updates, readback { QScopedPointer dynamicBuffer(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, bufferSize)); QVERIFY(dynamicBuffer->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); batch->updateDynamicBuffer(dynamicBuffer.data(), 10, bufferSize - 10, a.constData()); batch->updateDynamicBuffer(dynamicBuffer.data(), 0, 12, b.constData()); QRhiBufferReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackBuffer(dynamicBuffer.data(), 5, 10, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); // Offscreen frames are synchronous, so the readback must have // completed at this point. With swapchain frames this would not be the // case. QVERIFY(readCompleted); QVERIFY(readResult.data.size() == 10); QCOMPARE(readResult.data.left(7), QByteArrayLiteral("BBBBBBB")); QCOMPARE(readResult.data.mid(7), QByteArrayLiteral("AAA")); } // static buffer, updates, readback { QScopedPointer dynamicBuffer(rhi->newBuffer(QRhiBuffer::Static, QRhiBuffer::VertexBuffer, bufferSize)); QVERIFY(dynamicBuffer->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); batch->uploadStaticBuffer(dynamicBuffer.data(), 10, bufferSize - 10, a.constData()); batch->uploadStaticBuffer(dynamicBuffer.data(), 0, 12, b.constData()); QRhiBufferReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; if (rhi->isFeatureSupported(QRhi::ReadBackNonUniformBuffer)) batch->readBackBuffer(dynamicBuffer.data(), 5, 10, &readResult); else qDebug("Skipping verification of buffer data as ReadBackNonUniformBuffer is not supported"); QVERIFY(submitResourceUpdates(rhi.data(), batch)); if (rhi->isFeatureSupported(QRhi::ReadBackNonUniformBuffer)) { QVERIFY(readCompleted); QVERIFY(readResult.data.size() == 10); QCOMPARE(readResult.data.left(7), QByteArrayLiteral("BBBBBBB")); QCOMPARE(readResult.data.mid(7), QByteArrayLiteral("AAA")); } else { qDebug("Skipping verifying buffer contents because readback is not supported"); } } } inline bool imageRGBAEquals(const QImage &a, const QImage &b, int maxFuzz = 1) { if (a.size() != b.size()) return false; const QImage image0 = a.convertToFormat(QImage::Format_RGBA8888_Premultiplied); const QImage image1 = b.convertToFormat(QImage::Format_RGBA8888_Premultiplied); const int width = image0.width(); const int height = image0.height(); for (int y = 0; y < height; ++y) { const quint32 *p0 = reinterpret_cast(image0.constScanLine(y)); const quint32 *p1 = reinterpret_cast(image1.constScanLine(y)); int x = width - 1; while (x-- >= 0) { const QRgb c0(*p0++); const QRgb c1(*p1++); const int red = qAbs(qRed(c0) - qRed(c1)); const int green = qAbs(qGreen(c0) - qGreen(c1)); const int blue = qAbs(qBlue(c0) - qBlue(c1)); const int alpha = qAbs(qAlpha(c0) - qAlpha(c1)); if (red > maxFuzz || green > maxFuzz || blue > maxFuzz || alpha > maxFuzz) return false; } } return true; } void tst_QRhi::resourceUpdateBatchRGBATextureUpload_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchRGBATextureUpload() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing texture resource updates"); QImage image(234, 123, QImage::Format_RGBA8888_Premultiplied); image.fill(Qt::red); QPainter painter; const QPoint greenRectPos(35, 50); const QSize greenRectSize(100, 50); painter.begin(&image); painter.fillRect(QRect(greenRectPos, greenRectSize), Qt::green); painter.end(); // simple image upload; uploading and reading back RGBA8 is supported by the Null backend even { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, image.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); batch->uploadTexture(texture.data(), image); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); // like with buffers, the readback is now complete due to endOffscreenFrame() QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, image.size()); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), image.format()); QVERIFY(imageRGBAEquals(image, wrapperImage)); } // the same with raw data { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, image.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QRhiTextureUploadEntry upload(0, 0, { image.constBits(), quint32(image.sizeInBytes()) }); QRhiTextureUploadDescription uploadDesc(upload); batch->uploadTexture(texture.data(), uploadDesc); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, image.size()); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), image.format()); QVERIFY(imageRGBAEquals(image, wrapperImage)); } // partial image upload at a non-zero destination position { const QSize copySize(30, 40); const int gap = 10; const QSize fullSize(copySize.width() + gap, copySize.height() + gap); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, fullSize, 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QImage clearImage(fullSize, image.format()); clearImage.fill(Qt::black); batch->uploadTexture(texture.data(), clearImage); // copy green pixels of copySize to (gap, gap), leaving a black bar of // gap pixels on the left and top QRhiTextureSubresourceUploadDescription desc; desc.setImage(image); desc.setSourceSize(copySize); desc.setDestinationTopLeft(QPoint(gap, gap)); desc.setSourceTopLeft(greenRectPos); batch->uploadTexture(texture.data(), QRhiTextureUploadDescription({ 0, 0, desc })); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, clearImage.size()); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), image.format()); QVERIFY(!imageRGBAEquals(clearImage, wrapperImage)); QImage expectedImage = clearImage; QPainter painter(&expectedImage); painter.fillRect(QRect(QPoint(gap, gap), QSize(copySize)), Qt::green); painter.end(); QVERIFY(imageRGBAEquals(expectedImage, wrapperImage)); } // the same (partial upload) with raw data as source { const QSize copySize(30, 40); const int gap = 10; const QSize fullSize(copySize.width() + gap, copySize.height() + gap); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, fullSize, 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QImage clearImage(fullSize, image.format()); clearImage.fill(Qt::black); batch->uploadTexture(texture.data(), clearImage); // SourceTopLeft is not supported for non-QImage-based uploads. const QImage im = image.copy(QRect(greenRectPos, copySize)); QRhiTextureSubresourceUploadDescription desc; desc.setData(QByteArray::fromRawData(reinterpret_cast(im.constBits()), im.sizeInBytes())); desc.setSourceSize(copySize); desc.setDestinationTopLeft(QPoint(gap, gap)); batch->uploadTexture(texture.data(), QRhiTextureUploadDescription({ 0, 0, desc })); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, clearImage.size()); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), image.format()); QVERIFY(!imageRGBAEquals(clearImage, wrapperImage)); QImage expectedImage = clearImage; QPainter painter(&expectedImage); painter.fillRect(QRect(QPoint(gap, gap), QSize(copySize)), Qt::green); painter.end(); QVERIFY(imageRGBAEquals(expectedImage, wrapperImage)); } // now a QImage from an actual file { QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); inputImage = std::move(inputImage).convertToFormat(image.format()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); batch->uploadTexture(texture.data(), inputImage); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), inputImage.format()); QVERIFY(imageRGBAEquals(inputImage, wrapperImage)); } } void tst_QRhi::resourceUpdateBatchRGBATextureCopy_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchRGBATextureCopy() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing texture resource updates"); QImage red(256, 256, QImage::Format_RGBA8888_Premultiplied); red.fill(Qt::red); QImage green(35, 73, red.format()); green.fill(Qt::green); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QScopedPointer redTexture(rhi->newTexture(QRhiTexture::RGBA8, red.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(redTexture->create()); batch->uploadTexture(redTexture.data(), red); QScopedPointer greenTexture(rhi->newTexture(QRhiTexture::RGBA8, green.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(greenTexture->create()); batch->uploadTexture(greenTexture.data(), green); // 1. simple copy red -> texture; 2. subimage copy green -> texture; 3. partial subimage copy green -> texture { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, red.size(), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); // 1. batch->copyTexture(texture.data(), redTexture.data()); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), red.format()); QVERIFY(imageRGBAEquals(red, wrapperImage)); batch = rhi->nextResourceUpdateBatch(); readCompleted = false; // 2. QRhiTextureCopyDescription copyDesc; copyDesc.setDestinationTopLeft(QPoint(15, 23)); batch->copyTexture(texture.data(), greenTexture.data(), copyDesc); batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); wrapperImage = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), red.format()); QImage expectedImage = red; QPainter painter(&expectedImage); painter.drawImage(copyDesc.destinationTopLeft(), green); painter.end(); QVERIFY(imageRGBAEquals(expectedImage, wrapperImage)); batch = rhi->nextResourceUpdateBatch(); readCompleted = false; // 3. copyDesc.setDestinationTopLeft(QPoint(125, 89)); copyDesc.setSourceTopLeft(QPoint(5, 5)); copyDesc.setPixelSize(QSize(26, 45)); batch->copyTexture(texture.data(), greenTexture.data(), copyDesc); batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); wrapperImage = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), red.format()); painter.begin(&expectedImage); painter.drawImage(copyDesc.destinationTopLeft(), green, QRect(copyDesc.sourceTopLeft(), copyDesc.pixelSize())); painter.end(); QVERIFY(imageRGBAEquals(expectedImage, wrapperImage)); } } void tst_QRhi::resourceUpdateBatchRGBATextureMip_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchRGBATextureMip() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing texture resource updates"); QImage red(512, 512, QImage::Format_RGBA8888_Premultiplied); red.fill(Qt::red); const QRhiTexture::Flags textureFlags = QRhiTexture::UsedAsTransferSource | QRhiTexture::MipMapped | QRhiTexture::UsedWithGenerateMips; QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, red.size(), 1, textureFlags)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); batch->uploadTexture(texture.data(), red); batch->generateMips(texture.data()); QVERIFY(submitResourceUpdates(rhi.data(), batch)); const int levelCount = rhi->mipLevelsForSize(red.size()); QCOMPARE(levelCount, 10); for (int level = 0; level < levelCount; ++level) { batch = rhi->nextResourceUpdateBatch(); QRhiReadbackDescription readDesc(texture.data()); readDesc.setLevel(level); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(readDesc, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); const QSize expectedSize = rhi->sizeForMipLevel(level, texture->pixelSize()); QCOMPARE(readResult.pixelSize, expectedSize); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), red.format()); QImage expectedImage; if (level == 0 || rhi->isFeatureSupported(QRhi::ReadBackNonBaseMipLevel)) { // Compare to a scaled version; we can do this safely only because we // only have plain red pixels in the source image. expectedImage = red.scaled(expectedSize); } else { qDebug("Expecting all-zero image for level %d because reading back a level other than 0 is not supported", level); expectedImage = QImage(readResult.pixelSize, red.format()); expectedImage.fill(0); } QVERIFY(imageRGBAEquals(expectedImage, wrapperImage)); } } void tst_QRhi::resourceUpdateBatchTextureRawDataStride_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchTextureRawDataStride() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing texture resource updates"); const int WIDTH = 150; const int DATA_WIDTH = 180; const int HEIGHT = 50; QByteArray image; image.resize(DATA_WIDTH * HEIGHT * 4); for (int y = 0; y < HEIGHT; ++y) { char *p = image.data() + y * DATA_WIDTH * 4; memset(p, y, DATA_WIDTH * 4); } { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(WIDTH, HEIGHT), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QRhiTextureSubresourceUploadDescription subresDesc(image.constData(), image.size()); subresDesc.setDataStride(DATA_WIDTH * 4); QRhiTextureUploadEntry upload(0, 0, subresDesc); QRhiTextureUploadDescription uploadDesc(upload); batch->uploadTexture(texture.data(), uploadDesc); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; batch->readBackTexture(texture.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, QSize(WIDTH, HEIGHT)); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); // wrap the original data, note the bytesPerLine argument QImage originalWrapperImage(reinterpret_cast(image.constData()), WIDTH, HEIGHT, DATA_WIDTH * 4, QImage::Format_RGBA8888_Premultiplied); QVERIFY(imageRGBAEquals(wrapperImage, originalWrapperImage)); } } void tst_QRhi::resourceUpdateBatchLotsOfResources_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchLotsOfResources() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing resource updates"); QImage image(128, 128, QImage::Format_RGBA8888_Premultiplied); image.fill(Qt::red); static const float bufferData[64] = {}; QRhiResourceUpdateBatch *b = rhi->nextResourceUpdateBatch(); std::vector> textures; std::vector> buffers; // QTBUG-96619 static const int TEXTURE_COUNT = 3 * QRhiResourceUpdateBatchPrivate::TEXTURE_OPS_STATIC_ALLOC; static const int BUFFER_COUNT = 3 * QRhiResourceUpdateBatchPrivate::BUFFER_OPS_STATIC_ALLOC; for (int i = 0; i < TEXTURE_COUNT; ++i) { std::unique_ptr texture(rhi->newTexture(QRhiTexture::RGBA8, image.size())); QVERIFY(texture->create()); b->uploadTexture(texture.get(), image); textures.push_back(std::move(texture)); } for (int i = 0; i < BUFFER_COUNT; ++i) { std::unique_ptr buffer(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, 256)); QVERIFY(buffer->create()); b->uploadStaticBuffer(buffer.get(), bufferData); buffers.push_back(std::move(buffer)); } submitResourceUpdates(rhi.data(), b); } static QShader loadShader(const char *name) { QFile f(QString::fromUtf8(name)); if (f.open(QIODevice::ReadOnly)) { const QByteArray contents = f.readAll(); return QShader::fromSerialized(contents); } return QShader(); } void tst_QRhi::invalidPipeline_data() { rhiTestData(); } void tst_QRhi::invalidPipeline() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing empty shader"); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(256, 256), 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 2 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 } }); // no stages QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(!pipeline->create()); QShader vs; QShader fs; // no shaders in the stages pipeline.reset(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(!pipeline->create()); vs = loadShader(":/data/simple.vert.qsb"); QVERIFY(vs.isValid()); fs = loadShader(":/data/simple.frag.qsb"); QVERIFY(fs.isValid()); // no vertex stage pipeline.reset(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(!pipeline->create()); // no renderpass descriptor pipeline.reset(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); QVERIFY(!pipeline->create()); // no shader resource bindings pipeline.reset(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(!pipeline->create()); // correct pipeline.reset(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setRenderPassDescriptor(rpDesc.data()); pipeline->setShaderResourceBindings(srb.data()); QVERIFY(pipeline->create()); } void tst_QRhi::renderToTextureSimple_data() { rhiTestData(); } static QRhiGraphicsPipeline *createSimplePipeline(QRhi *rhi, QRhiShaderResourceBindings *srb, QRhiRenderPassDescriptor *rpDesc) { std::unique_ptr pipeline(rhi->newGraphicsPipeline()); QShader vs = loadShader(":/data/simple.vert.qsb"); if (!vs.isValid()) return nullptr; QShader fs = loadShader(":/data/simple.frag.qsb"); if (!fs.isValid()) return nullptr; pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 2 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb); pipeline->setRenderPassDescriptor(rpDesc); return pipeline->create() ? pipeline.release() : nullptr; } static const float triangleVertices[] = { -1.0f, -1.0f, 1.0f, -1.0f, 0.0f, 1.0f }; void tst_QRhi::renderToTextureSimple() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); const QSize outputSize(1920, 1080); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); // non-owning, no copy needed because readResult outlives result }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); // Offscreen frames are synchronous, so the readback is guaranteed to // complete at this point. This would not be the case with swapchain-based // frames. QCOMPARE(result.size(), texture->pixelSize()); if (impl == QRhi::Null) return; // Now we have a red rectangle on blue background. const int y = 100; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, texture->pixelSize().width()); // The triangle is "pointing up" in the resulting image with OpenGL // (because Y is up both in normalized device coordinates and in images) // and Vulkan (because Y is down in both and the vertex data was specified // with Y up in mind), but "pointing down" with D3D (because Y is up in NDC // but down in images). if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) QVERIFY(redCount < blueCount); else QVERIFY(redCount > blueCount); } void tst_QRhi::renderToTextureMip_data() { rhiTestData(); } void tst_QRhi::renderToTextureMip() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); if (!rhi->isFeatureSupported(QRhi::RenderToNonBaseMipLevel)) QSKIP("Rendering to non-base mip levels is not supported on this platform, skipping test"); const QSize baseLevelSize(1024, 1024); const int LEVEL = 3; // render into mip #3 (128x128) QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, baseLevelSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource | QRhiTexture::MipMapped)); QVERIFY(texture->create()); QRhiColorAttachment colorAtt(texture.data()); colorAtt.setLevel(LEVEL); QRhiTextureRenderTargetDescription rtDesc(colorAtt); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QCOMPARE(rt->pixelSize(), rhi->sizeForMipLevel(LEVEL, baseLevelSize)); const QSize mipSize(baseLevelSize.width() >> LEVEL, baseLevelSize.height() >> LEVEL); QCOMPARE(rt->pixelSize(), mipSize); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(rt->pixelSize().width()), float(rt->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLevel(LEVEL); readbackBatch->readBackTexture(readbackDescription, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); if (!rhi->isFeatureSupported(QRhi::ReadBackNonBaseMipLevel)) QSKIP("Reading back non-base mip levels is not supported on this platform, skipping readback"); QCOMPARE(result.size(), mipSize); if (impl == QRhi::Null) return; const int y = 100; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, mipSize.width()); if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) QVERIFY(redCount > blueCount); // 100, 28 else QVERIFY(redCount < blueCount); // 28, 100 } void tst_QRhi::renderToTextureCubemapFace_data() { rhiTestData(); } void tst_QRhi::renderToTextureCubemapFace() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); const QSize outputSize(512, 512); // width must be same as height QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource | QRhiTexture::CubeMap)); // will be a cubemap, so 6 layers QVERIFY(texture->create()); const int LAYER = 1; // render into the layer for face -X const int BAD_LAYER = 2; // +Y QRhiColorAttachment colorAtt(texture.data()); colorAtt.setLayer(LAYER); QRhiTextureRenderTargetDescription rtDesc(colorAtt); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QCOMPARE(rt->pixelSize(), texture->pixelSize()); QCOMPARE(rt->pixelSize(), outputSize); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(rt->pixelSize().width()), float(rt->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLayer(LAYER); readbackBatch->readBackTexture(readbackDescription, &readResult); // also read back a layer we did not render into QRhiReadbackResult readResult2; QImage result2; readResult2.completed = [&readResult2, &result2] { result2 = QImage(reinterpret_cast(readResult2.data.constData()), readResult2.pixelSize.width(), readResult2.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiReadbackDescription readbackDescription2(texture.data()); readbackDescription2.setLayer(BAD_LAYER); readbackBatch->readBackTexture(readbackDescription2, &readResult2); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QCOMPARE(result.size(), outputSize); QCOMPARE(result2.size(), outputSize); if (impl == QRhi::Null) return; // just want to ensure that we did not read the same thing back twice, i.e. // that the 'layer' parameter was not ignored QVERIFY(result != result2); const int y = 100; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QVERIFY(redCount > 0 && blueCount > 0); QCOMPARE(redCount + blueCount, outputSize.width()); if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) QVERIFY(redCount < blueCount); // 100, 412 else QVERIFY(redCount > blueCount); // 412, 100 } void tst_QRhi::renderToTextureTextureArray_data() { rhiTestData(); } void tst_QRhi::renderToTextureTextureArray() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); if (!rhi->isFeatureSupported(QRhi::TextureArrays)) QSKIP("TextureArrays is not supported with this backend, skipping test"); const QSize outputSize(512, 256); const int ARRAY_SIZE = 8; QScopedPointer texture(rhi->newTextureArray(QRhiTexture::RGBA8, ARRAY_SIZE, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); const int LAYER = 5; // render into element #5 QRhiColorAttachment colorAtt(texture.data()); colorAtt.setLayer(LAYER); QRhiTextureRenderTargetDescription rtDesc(colorAtt); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QCOMPARE(rt->pixelSize(), texture->pixelSize()); QCOMPARE(rt->pixelSize(), outputSize); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(rt->pixelSize().width()), float(rt->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLayer(LAYER); readbackBatch->readBackTexture(readbackDescription, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QCOMPARE(result.size(), outputSize); if (impl == QRhi::Null) return; const int y = 100; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QVERIFY(redCount > 0 && blueCount > 0); QCOMPARE(redCount + blueCount, outputSize.width()); if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) QVERIFY(redCount < blueCount); // 100, 412 else QVERIFY(redCount > blueCount); // 412, 100 } static const float quadVerticesUvs[] = { -1.0f, -1.0f, 0.0f, 0.0f, 1.0f, -1.0f, 1.0f, 0.0f, -1.0f, 1.0f, 0.0f, 1.0f, 1.0f, 1.0f, 1.0f, 1.0f }; void tst_QRhi::renderToTextureTexturedQuad_data() { rhiTestData(); } void tst_QRhi::renderToTextureTexturedQuad() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer srb(rhi->newShaderResourceBindings()); srb->setBindings({ QRhiShaderResourceBinding::sampledTexture(0, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/simpletextured.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/simpletextured.frag.qsb"); QVERIFY(fs.isValid()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; // Flip with D3D and Metal because these have Y down in images. Vulkan does // not need this because there Y is down both in images and in NDC, which // just happens to give correct results with our OpenGL-targeted vertex and // UV data. if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // check a few points that are expected to match regardless of the implementation QRgb white = qRgba(255, 255, 255, 255); QCOMPARE(result.pixel(79, 77), white); QCOMPARE(result.pixel(124, 81), white); QCOMPARE(result.pixel(128, 149), white); QCOMPARE(result.pixel(120, 189), white); QCOMPARE(result.pixel(116, 185), white); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result.pixel(11, 45), empty); QCOMPARE(result.pixel(246, 202), empty); QCOMPARE(result.pixel(130, 18), empty); QCOMPARE(result.pixel(4, 227), empty); QVERIFY(qGreen(result.pixel(32, 52)) > 2 * qRed(result.pixel(32, 52))); QVERIFY(qGreen(result.pixel(32, 52)) > 2 * qBlue(result.pixel(32, 52))); QVERIFY(qGreen(result.pixel(214, 191)) > 2 * qRed(result.pixel(214, 191))); QVERIFY(qGreen(result.pixel(214, 191)) > 2 * qBlue(result.pixel(214, 191))); } void tst_QRhi::renderToTextureSampleWithSeparateTextureAndSampler_data() { rhiTestData(); } void tst_QRhi::renderToTextureSampleWithSeparateTextureAndSampler() { // Same as renderToTextureTexturedQuad but the fragment shader uses a // separate image and sampler. For Vulkan/Metal/D3D11 these are natively // supported. For OpenGL this exercises the auto-generated combined sampler // in the GLSL code and the mapping table that gets applied at run time by // the backend. QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer srb(rhi->newShaderResourceBindings()); srb->setBindings({ QRhiShaderResourceBinding::texture(3, QRhiShaderResourceBinding::FragmentStage, inputTexture.data()), QRhiShaderResourceBinding::sampler(5, QRhiShaderResourceBinding::FragmentStage, sampler.data()) }); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/simpletextured.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/simpletextured_separate.frag.qsb"); QVERIFY(fs.isValid()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); QRgb white = qRgba(255, 255, 255, 255); QCOMPARE(result.pixel(79, 77), white); QCOMPARE(result.pixel(124, 81), white); QCOMPARE(result.pixel(128, 149), white); QCOMPARE(result.pixel(120, 189), white); QCOMPARE(result.pixel(116, 185), white); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result.pixel(11, 45), empty); QCOMPARE(result.pixel(246, 202), empty); QCOMPARE(result.pixel(130, 18), empty); QCOMPARE(result.pixel(4, 227), empty); QVERIFY(qGreen(result.pixel(32, 52)) > 2 * qRed(result.pixel(32, 52))); QVERIFY(qGreen(result.pixel(32, 52)) > 2 * qBlue(result.pixel(32, 52))); QVERIFY(qGreen(result.pixel(214, 191)) > 2 * qRed(result.pixel(214, 191))); QVERIFY(qGreen(result.pixel(214, 191)) > 2 * qBlue(result.pixel(214, 191))); } void tst_QRhi::renderToTextureArrayOfTexturedQuad_data() { rhiTestData(); } void tst_QRhi::renderToTextureArrayOfTexturedQuad() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); // In this test we pass 3 textures (and samplers) to the fragment shader in // form of an array of combined image samplers. QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QImage redImage(inputImage.size(), QImage::Format_RGBA8888); redImage.fill(Qt::red); QScopedPointer redTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(redTexture->create()); updates->uploadTexture(redTexture.data(), redImage); QImage greenImage(inputImage.size(), QImage::Format_RGBA8888); greenImage.fill(Qt::green); QScopedPointer greenTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(greenTexture->create()); updates->uploadTexture(greenTexture.data(), greenImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer srb(rhi->newShaderResourceBindings()); QRhiShaderResourceBinding::TextureAndSampler texSamplers[3] = { { inputTexture.data(), sampler.data() }, { redTexture.data(), sampler.data() }, { greenTexture.data(), sampler.data() } }; srb->setBindings({ QRhiShaderResourceBinding::sampledTextures(0, QRhiShaderResourceBinding::FragmentStage, 3, texSamplers) }); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/simpletextured.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/simpletextured_array.frag.qsb"); QVERIFY(fs.isValid()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; // Flip with D3D and Metal because these have Y down in images. Vulkan does // not need this because there Y is down both in images and in NDC, which // just happens to give correct results with our OpenGL-targeted vertex and // UV data. if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // we added the input image + red + green together, so red and green must be all 1 for (int y = 0; y < result.height(); ++y) { for (int x = 0; x < result.width(); ++x) { const QRgb pixel = result.pixel(x, y); QCOMPARE(qRed(pixel), 255); QCOMPARE(qGreen(pixel), 255); } } } void tst_QRhi::renderToTextureTexturedQuadAndUniformBuffer_data() { rhiTestData(); } void tst_QRhi::renderToTextureTexturedQuadAndUniformBuffer() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); // There will be two renderpasses. One renders with no transformation and // an opacity of 0.5, the second has a rotation. Bake the uniform data for // both into a single buffer. const int UNIFORM_BLOCK_SIZE = 64 + 4; // matrix + opacity const int secondUbufOffset = rhi->ubufAligned(UNIFORM_BLOCK_SIZE); const int UBUF_SIZE = secondUbufOffset + UNIFORM_BLOCK_SIZE; QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, UBUF_SIZE)); QVERIFY(ubuf->create()); QMatrix4x4 matrix; updates->updateDynamicBuffer(ubuf.data(), 0, 64, matrix.constData()); float opacity = 0.5f; updates->updateDynamicBuffer(ubuf.data(), 64, 4, &opacity); // rotation by 45 degrees around the Z axis matrix.rotate(45, 0, 0, 1); updates->updateDynamicBuffer(ubuf.data(), secondUbufOffset, 64, matrix.constData()); updates->updateDynamicBuffer(ubuf.data(), secondUbufOffset + 64, 4, &opacity); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); const QRhiShaderResourceBinding::StageFlags commonVisibility = QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage; QScopedPointer srb0(rhi->newShaderResourceBindings()); srb0->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data(), 0, UNIFORM_BLOCK_SIZE), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb0->create()); QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data(), secondUbufOffset, UNIFORM_BLOCK_SIZE), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb1->create()); QVERIFY(srb1->isLayoutCompatible(srb0.data())); // hence no need for a second pipeline QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/textured.vert.qsb"); QVERIFY(vs.isValid()); QShaderDescription shaderDesc = vs.description(); QVERIFY(!shaderDesc.uniformBlocks().isEmpty()); QCOMPARE(shaderDesc.uniformBlocks().first().size, UNIFORM_BLOCK_SIZE); QShader fs = loadShader(":/data/textured.frag.qsb"); QVERIFY(fs.isValid()); shaderDesc = fs.description(); QVERIFY(!shaderDesc.uniformBlocks().isEmpty()); QCOMPARE(shaderDesc.uniformBlocks().first().size, UNIFORM_BLOCK_SIZE); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb0.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult0; QImage result0; readResult0.completed = [&readResult0, &result0] { result0 = QImage(reinterpret_cast(readResult0.data.constData()), readResult0.pixelSize.width(), readResult0.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult0); cb->endPass(readbackBatch); // second pass (rotated) cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(srb1.data()); // sources data from a different offset in ubuf cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult1; QImage result1; readResult1.completed = [&readResult1, &result1] { result1 = QImage(reinterpret_cast(readResult1.data.constData()), readResult1.pixelSize.width(), readResult1.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult1); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result0.isNull()); QVERIFY(!result1.isNull()); if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) { result0 = std::move(result0).mirrored(); result1 = std::move(result1).mirrored(); } if (impl == QRhi::Null) return; // opacity 0.5 (premultiplied) static const auto checkSemiWhite = [](const QRgb &c) { QRgb semiWhite127 = qPremultiply(qRgba(255, 255, 255, 127)); QRgb semiWhite128 = qPremultiply(qRgba(255, 255, 255, 128)); return c == semiWhite127 || c == semiWhite128; }; QVERIFY(checkSemiWhite(result0.pixel(79, 77))); QVERIFY(checkSemiWhite(result0.pixel(124, 81))); QVERIFY(checkSemiWhite(result0.pixel(128, 149))); QVERIFY(checkSemiWhite(result0.pixel(120, 189))); QVERIFY(checkSemiWhite(result0.pixel(116, 185))); QVERIFY(checkSemiWhite(result0.pixel(191, 172))); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result0.pixel(11, 45), empty); QCOMPARE(result0.pixel(246, 202), empty); QCOMPARE(result0.pixel(130, 18), empty); QCOMPARE(result0.pixel(4, 227), empty); // also rotated 45 degrees around Z QRgb black = qRgba(0, 0, 0, 255); QCOMPARE(result1.pixel(20, 23), black); QCOMPARE(result1.pixel(47, 5), black); QCOMPARE(result1.pixel(238, 22), black); QCOMPARE(result1.pixel(250, 203), black); QCOMPARE(result1.pixel(224, 237), black); QCOMPARE(result1.pixel(12, 221), black); QVERIFY(checkSemiWhite(result1.pixel(142, 67))); QVERIFY(checkSemiWhite(result1.pixel(81, 79))); QVERIFY(checkSemiWhite(result1.pixel(79, 168))); QVERIFY(checkSemiWhite(result1.pixel(146, 204))); QVERIFY(checkSemiWhite(result1.pixel(186, 156))); QCOMPARE(result1.pixel(204, 45), empty); QCOMPARE(result1.pixel(28, 178), empty); } void tst_QRhi::renderToTextureTexturedQuadAllDynamicBuffers_data() { rhiTestData(); } void tst_QRhi::renderToTextureTexturedQuadAllDynamicBuffers() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); // Do like renderToTextureTexturedQuadAndUniformBuffer but only use Dynamic // buffers, and do updates with the direct beginFullDynamicBufferUpdate // function. (for some backend this is different for UniformBuffer and // others, hence useful exercising it also on a VertexBuffer) QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); char *p = vbuf->beginFullDynamicBufferUpdateForCurrentFrame(); QVERIFY(p); memcpy(p, quadVerticesUvs, sizeof(quadVerticesUvs)); vbuf->endFullDynamicBufferUpdateForCurrentFrame(); const int UNIFORM_BLOCK_SIZE = 64 + 4; // matrix + opacity const int secondUbufOffset = rhi->ubufAligned(UNIFORM_BLOCK_SIZE); const int UBUF_SIZE = secondUbufOffset + UNIFORM_BLOCK_SIZE; QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, UBUF_SIZE)); QVERIFY(ubuf->create()); p = ubuf->beginFullDynamicBufferUpdateForCurrentFrame(); QVERIFY(p); QMatrix4x4 matrix; memcpy(p, matrix.constData(), 64); float opacity = 0.5f; memcpy(p + 64, &opacity, 4); // rotation by 45 degrees around the Z axis matrix.rotate(45, 0, 0, 1); memcpy(p + secondUbufOffset, matrix.constData(), 64); memcpy(p + secondUbufOffset + 64, &opacity, 4); ubuf->endFullDynamicBufferUpdateForCurrentFrame(); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); cb->resourceUpdate(updates); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); const QRhiShaderResourceBinding::StageFlags commonVisibility = QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage; QScopedPointer srb0(rhi->newShaderResourceBindings()); srb0->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data(), 0, UNIFORM_BLOCK_SIZE), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb0->create()); QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data(), secondUbufOffset, UNIFORM_BLOCK_SIZE), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb1->create()); QVERIFY(srb1->isLayoutCompatible(srb0.data())); // hence no need for a second pipeline QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/textured.vert.qsb"); QVERIFY(vs.isValid()); QShaderDescription shaderDesc = vs.description(); QVERIFY(!shaderDesc.uniformBlocks().isEmpty()); QCOMPARE(shaderDesc.uniformBlocks().first().size, UNIFORM_BLOCK_SIZE); QShader fs = loadShader(":/data/textured.frag.qsb"); QVERIFY(fs.isValid()); shaderDesc = fs.description(); QVERIFY(!shaderDesc.uniformBlocks().isEmpty()); QCOMPARE(shaderDesc.uniformBlocks().first().size, UNIFORM_BLOCK_SIZE); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb0.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult0; QImage result0; readResult0.completed = [&readResult0, &result0] { result0 = QImage(reinterpret_cast(readResult0.data.constData()), readResult0.pixelSize.width(), readResult0.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult0); cb->endPass(readbackBatch); // second pass (rotated) cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(srb1.data()); // sources data from a different offset in ubuf cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult1; QImage result1; readResult1.completed = [&readResult1, &result1] { result1 = QImage(reinterpret_cast(readResult1.data.constData()), readResult1.pixelSize.width(), readResult1.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult1); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result0.isNull()); QVERIFY(!result1.isNull()); if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) { result0 = std::move(result0).mirrored(); result1 = std::move(result1).mirrored(); } if (impl == QRhi::Null) return; // opacity 0.5 (premultiplied) static const auto checkSemiWhite = [](const QRgb &c) { QRgb semiWhite127 = qPremultiply(qRgba(255, 255, 255, 127)); QRgb semiWhite128 = qPremultiply(qRgba(255, 255, 255, 128)); return c == semiWhite127 || c == semiWhite128; }; QVERIFY(checkSemiWhite(result0.pixel(79, 77))); QVERIFY(checkSemiWhite(result0.pixel(124, 81))); QVERIFY(checkSemiWhite(result0.pixel(128, 149))); QVERIFY(checkSemiWhite(result0.pixel(120, 189))); QVERIFY(checkSemiWhite(result0.pixel(116, 185))); QVERIFY(checkSemiWhite(result0.pixel(191, 172))); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result0.pixel(11, 45), empty); QCOMPARE(result0.pixel(246, 202), empty); QCOMPARE(result0.pixel(130, 18), empty); QCOMPARE(result0.pixel(4, 227), empty); // also rotated 45 degrees around Z QRgb black = qRgba(0, 0, 0, 255); QCOMPARE(result1.pixel(20, 23), black); QCOMPARE(result1.pixel(47, 5), black); QCOMPARE(result1.pixel(238, 22), black); QCOMPARE(result1.pixel(250, 203), black); QCOMPARE(result1.pixel(224, 237), black); QCOMPARE(result1.pixel(12, 221), black); QVERIFY(checkSemiWhite(result1.pixel(142, 67))); QVERIFY(checkSemiWhite(result1.pixel(81, 79))); QVERIFY(checkSemiWhite(result1.pixel(79, 168))); QVERIFY(checkSemiWhite(result1.pixel(146, 204))); QVERIFY(checkSemiWhite(result1.pixel(186, 156))); QCOMPARE(result1.pixel(204, 45), empty); QCOMPARE(result1.pixel(28, 178), empty); } void tst_QRhi::renderToTextureDeferredSrb_data() { rhiTestData(); } void tst_QRhi::renderToTextureDeferredSrb() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 64 + 4)); QVERIFY(ubuf->create()); QMatrix4x4 matrix; updates->updateDynamicBuffer(ubuf.data(), 0, 64, matrix.constData()); float opacity = 0.5f; updates->updateDynamicBuffer(ubuf.data(), 64, 4, &opacity); // this is the specific thing to test here: an srb with null resources const QRhiShaderResourceBinding::StageFlags commonVisibility = QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage; QScopedPointer layoutOnlySrb(rhi->newShaderResourceBindings()); layoutOnlySrb->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, nullptr), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, nullptr, nullptr) }); QVERIFY(layoutOnlySrb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/textured.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/textured.frag.qsb"); QVERIFY(fs.isValid()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(layoutOnlySrb.data()); // no resources needed yet pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); // another, layout compatible, srb with the actual resources QScopedPointer layoutCompatibleSrbWithResources(rhi->newShaderResourceBindings()); layoutCompatibleSrbWithResources->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(layoutCompatibleSrbWithResources->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setShaderResources(layoutCompatibleSrbWithResources.data()); // here we must use the srb referencing the resources cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // opacity 0.5 (premultiplied) static const auto checkSemiWhite = [](const QRgb &c) { QRgb semiWhite127 = qPremultiply(qRgba(255, 255, 255, 127)); QRgb semiWhite128 = qPremultiply(qRgba(255, 255, 255, 128)); return c == semiWhite127 || c == semiWhite128; }; QVERIFY(checkSemiWhite(result.pixel(79, 77))); QVERIFY(checkSemiWhite(result.pixel(124, 81))); QVERIFY(checkSemiWhite(result.pixel(128, 149))); QVERIFY(checkSemiWhite(result.pixel(120, 189))); QVERIFY(checkSemiWhite(result.pixel(116, 185))); QVERIFY(checkSemiWhite(result.pixel(191, 172))); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result.pixel(11, 45), empty); QCOMPARE(result.pixel(246, 202), empty); QCOMPARE(result.pixel(130, 18), empty); QCOMPARE(result.pixel(4, 227), empty); } void tst_QRhi::renderToTextureMultipleUniformBuffersAndDynamicOffset_data() { rhiTestData(); } void tst_QRhi::renderToTextureMultipleUniformBuffersAndDynamicOffset() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); const int MATRIX_COUNT = 4; // put 4 mat4s into the buffer, will only use one const int ubufElemSize = rhi->ubufAligned(64); QVERIFY(ubufElemSize >= 64); QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, MATRIX_COUNT * ubufElemSize)); QVERIFY(ubuf->create()); float zeroes[16]; memset(zeroes, 0, sizeof(zeroes)); updates->updateDynamicBuffer(ubuf.data(), 0, 64, zeroes); updates->updateDynamicBuffer(ubuf.data(), ubufElemSize, 64, zeroes); // the only correct matrix is the third one QMatrix4x4 matrix; updates->updateDynamicBuffer(ubuf.data(), ubufElemSize * 2, 64, matrix.constData()); updates->updateDynamicBuffer(ubuf.data(), ubufElemSize * 3, 64, zeroes); const int OPACITY_COUNT = 6; // put 6 floats into the buffer, will only use one const int ubuf2ElemSize = rhi->ubufAligned(4); QVERIFY(ubuf2ElemSize >= 4); QScopedPointer ubuf2(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, OPACITY_COUNT * ubuf2ElemSize)); QVERIFY(ubuf2->create()); updates->updateDynamicBuffer(ubuf2.data(), 0, 4, &zeroes[0]); updates->updateDynamicBuffer(ubuf2.data(), ubuf2ElemSize, 4, &zeroes[0]); updates->updateDynamicBuffer(ubuf2.data(), ubuf2ElemSize * 2, 4, &zeroes[0]); // the only correct opacity value is the fourth one float opacity = 0.5f; updates->updateDynamicBuffer(ubuf2.data(), ubuf2ElemSize * 3, 4, &opacity); updates->updateDynamicBuffer(ubuf2.data(), ubuf2ElemSize * 4, 4, &zeroes[0]); updates->updateDynamicBuffer(ubuf2.data(), ubuf2ElemSize * 5, 4, &zeroes[0]); const QRhiShaderResourceBinding::StageFlags commonVisibility = QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage; QScopedPointer srb(rhi->newShaderResourceBindings()); srb->setBindings({ QRhiShaderResourceBinding::uniformBufferWithDynamicOffset(0, commonVisibility, ubuf.data(), 64), QRhiShaderResourceBinding::uniformBufferWithDynamicOffset(1, commonVisibility, ubuf2.data(), 4), QRhiShaderResourceBinding::sampledTexture(2, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/textured_multiubuf.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/textured_multiubuf.frag.qsb"); QVERIFY(fs.isValid()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); // Now the magic, expose the 3rd matrix and 4th opacity value to the shader. // If the handling of dynamic offsets were broken, the shaders would likely // "see" an all zero matrix and zero opacity, thus leading to different // rendering output. This way we can verify if using dynamic offsets, and // more than one at the same time, is functional. QVarLengthArray, 2> dynamicOffset = { { 0, quint32(ubufElemSize * 2) }, { 1, quint32(ubuf2ElemSize * 3) }, }; cb->setShaderResources(srb.data(), 2, dynamicOffset.constData()); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // opacity 0.5 (premultiplied) static const auto checkSemiWhite = [](const QRgb &c) { QRgb semiWhite127 = qPremultiply(qRgba(255, 255, 255, 127)); QRgb semiWhite128 = qPremultiply(qRgba(255, 255, 255, 128)); return c == semiWhite127 || c == semiWhite128; }; QVERIFY(checkSemiWhite(result.pixel(79, 77))); QVERIFY(checkSemiWhite(result.pixel(124, 81))); QVERIFY(checkSemiWhite(result.pixel(128, 149))); QVERIFY(checkSemiWhite(result.pixel(120, 189))); QVERIFY(checkSemiWhite(result.pixel(116, 185))); QVERIFY(checkSemiWhite(result.pixel(191, 172))); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result.pixel(11, 45), empty); QCOMPARE(result.pixel(246, 202), empty); QCOMPARE(result.pixel(130, 18), empty); QCOMPARE(result.pixel(4, 227), empty); } void tst_QRhi::renderToTextureSrbReuse_data() { rhiTestData(); } void tst_QRhi::renderToTextureSrbReuse() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); // Draw a textured quad with opacity 0.5. The difference to the simple tests // of the same kind is that there are two (configuration-wise identical) // pipeline objects that are bound after each other, with the same one srb, // on the command buffer. This exercises, in particular for the OpenGL // backend, that the uniforms are set for the pipelines' underlying shader // program correctly. (with OpenGL we may not use real uniform buffers, // which presents extra pipeline-srb tracking work for the backend) QImage inputImage; inputImage.load(QLatin1String(":/data/qt256.png")); QVERIFY(!inputImage.isNull()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size(), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(quadVerticesUvs))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), quadVerticesUvs); QScopedPointer inputTexture(rhi->newTexture(QRhiTexture::RGBA8, inputImage.size())); QVERIFY(inputTexture->create()); updates->uploadTexture(inputTexture.data(), inputImage); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 64 + 4)); QVERIFY(ubuf->create()); QMatrix4x4 matrix; updates->updateDynamicBuffer(ubuf.data(), 0, 64, matrix.constData()); float opacity = 0.5f; updates->updateDynamicBuffer(ubuf.data(), 64, 4, &opacity); const QRhiShaderResourceBinding::StageFlags commonVisibility = QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage; QScopedPointer srb(rhi->newShaderResourceBindings()); srb->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, commonVisibility, ubuf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, inputTexture.data(), sampler.data()) }); QVERIFY(srb->create()); QScopedPointer pipeline1(rhi->newGraphicsPipeline()); pipeline1->setTopology(QRhiGraphicsPipeline::TriangleStrip); QShader vs = loadShader(":/data/textured.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/textured.frag.qsb"); QVERIFY(fs.isValid()); pipeline1->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 4 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 }, { 0, 1, QRhiVertexInputAttribute::Float2, 2 * sizeof(float) } }); pipeline1->setVertexInputLayout(inputLayout); pipeline1->setShaderResourceBindings(srb.data()); pipeline1->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline1->create()); QScopedPointer pipeline2(rhi->newGraphicsPipeline()); pipeline2->setTopology(QRhiGraphicsPipeline::TriangleStrip); pipeline2->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline2->setVertexInputLayout(inputLayout); pipeline2->setShaderResourceBindings(srb.data()); pipeline2->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline2->create()); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, updates); // The key step in this test: set the 1st pipeline, then the 2nd, the // srb is the same. This should lead to identical results to just // binding one of them. cb->setGraphicsPipeline(pipeline1.data()); cb->setShaderResources(srb.data()); cb->setGraphicsPipeline(pipeline2.data()); cb->setShaderResources(srb.data()); cb->setViewport({ 0, 0, float(texture->pixelSize().width()), float(texture->pixelSize().height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(4); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QVERIFY(!result.isNull()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // opacity 0.5 (premultiplied) static const auto checkSemiWhite = [](const QRgb &c) { QRgb semiWhite127 = qPremultiply(qRgba(255, 255, 255, 127)); QRgb semiWhite128 = qPremultiply(qRgba(255, 255, 255, 128)); return c == semiWhite127 || c == semiWhite128; }; QVERIFY(checkSemiWhite(result.pixel(79, 77))); QVERIFY(checkSemiWhite(result.pixel(124, 81))); QVERIFY(checkSemiWhite(result.pixel(128, 149))); QVERIFY(checkSemiWhite(result.pixel(120, 189))); QVERIFY(checkSemiWhite(result.pixel(116, 185))); QVERIFY(checkSemiWhite(result.pixel(191, 172))); QRgb empty = qRgba(0, 0, 0, 0); QCOMPARE(result.pixel(11, 45), empty); QCOMPARE(result.pixel(246, 202), empty); QCOMPARE(result.pixel(130, 18), empty); QCOMPARE(result.pixel(4, 227), empty); } void tst_QRhi::setWindowType(QWindow *window, QRhi::Implementation impl) { switch (impl) { #ifdef TST_GL case QRhi::OpenGLES2: window->setSurfaceType(QSurface::OpenGLSurface); break; #endif case QRhi::D3D11: window->setSurfaceType(QSurface::Direct3DSurface); break; case QRhi::Metal: window->setSurfaceType(QSurface::MetalSurface); break; #ifdef TST_VK case QRhi::Vulkan: window->setSurfaceType(QSurface::VulkanSurface); window->setVulkanInstance(&vulkanInstance); break; #endif default: break; } } void tst_QRhi::renderToTextureIndexedDraw_data() { rhiTestData(); } void tst_QRhi::renderToTextureIndexedDraw() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); const QSize outputSize(1920, 1080); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); static const quint16 indices[] = { 0, 1, 2 }; QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer ibuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::IndexBuffer, sizeof(indices))); QVERIFY(ibuf->create()); updates->uploadStaticBuffer(ibuf.data(), indices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); // Do three render passes, even though all render the same thing. This is done to // verify that QTBUG-89765 is fixed. One of them specifies ExternalContent which // triggers special behavior with some backends (uses a secondary command buffer with // Vulkan for example). This way we can see that optimizations, such as keeping track // of what index buffer is active, are handled correctly across pass boundaries in the // QRhi backends. Without the fix for QTBUG-89765 this test would show validation // warnings and even crash when run with Vulkan. cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); cb->setVertexInput(0, 1, &vbindings, ibuf.data(), 0, QRhiCommandBuffer::IndexUInt16); cb->drawIndexed(3); cb->endPass(); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, nullptr, QRhiCommandBuffer::ExternalContent); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); cb->setVertexInput(0, 1, &vbindings, ibuf.data(), 0, QRhiCommandBuffer::IndexUInt16); cb->drawIndexed(3); cb->endPass(); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, nullptr); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); cb->setVertexInput(0, 1, &vbindings, ibuf.data(), 0, QRhiCommandBuffer::IndexUInt16); cb->drawIndexed(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QCOMPARE(result.size(), texture->pixelSize()); if (impl == QRhi::Null) return; // Now we have a red rectangle on blue background. const int y = 100; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, texture->pixelSize().width()); if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) QVERIFY(redCount < blueCount); else QVERIFY(redCount > blueCount); } void tst_QRhi::renderToWindowSimple_data() { rhiTestData(); } void tst_QRhi::renderToWindowSimple() { if (QGuiApplication::platformName().startsWith(QLatin1String("offscreen"), Qt::CaseInsensitive)) QSKIP("Offscreen: This fails."); QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QScopedPointer window(new QWindow); setWindowType(window.data(), impl); window->setGeometry(0, 0, 640, 480); window->show(); QVERIFY(QTest::qWaitForWindowExposed(window.data())); QScopedPointer swapChain(rhi->newSwapChain()); swapChain->setWindow(window.data()); swapChain->setFlags(QRhiSwapChain::UsedAsTransferSource); QScopedPointer rpDesc(swapChain->newCompatibleRenderPassDescriptor()); swapChain->setRenderPassDescriptor(rpDesc.data()); QVERIFY(swapChain->createOrResize()); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); const int asyncReadbackFrames = rhi->resourceLimit(QRhi::MaxAsyncReadbackFrames); // one frame issues the readback, then we do MaxAsyncReadbackFrames more to ensure the readback completes const int FRAME_COUNT = asyncReadbackFrames + 1; bool readCompleted = false; QRhiReadbackResult readResult; QImage result; int readbackWidth = 0; for (int frameNo = 0; frameNo < FRAME_COUNT; ++frameNo) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); QRhiRenderTarget *rt = swapChain->currentFrameRenderTarget(); QCOMPARE(rt->resourceType(), QRhiResource::SwapChainRenderTarget); QVERIFY(rt->renderPassDescriptor()); QCOMPARE(static_cast(rt)->swapChain(), swapChain.data()); const QSize outputSize = swapChain->currentPixelSize(); QCOMPARE(rt->pixelSize(), outputSize); QRhiViewport viewport(0, 0, float(outputSize.width()), float(outputSize.height())); cb->beginPass(rt, Qt::blue, { 1.0f, 0 }, updates); updates = nullptr; cb->setGraphicsPipeline(pipeline.data()); cb->setViewport(viewport); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); if (frameNo == 0) { readResult.completed = [&readCompleted, &readResult, &result, &rhi] { readCompleted = true; QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_ARGB32_Premultiplied); if (readResult.format == QRhiTexture::RGBA8) wrapperImage = wrapperImage.rgbSwapped(); if (rhi->isYUpInFramebuffer() == rhi->isYUpInNDC()) result = wrapperImage.mirrored(); else result = wrapperImage.copy(); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({}, &readResult); // read back the current backbuffer readbackWidth = outputSize.width(); cb->endPass(readbackBatch); } else { cb->endPass(); } rhi->endFrame(swapChain.data()); } // The readback is asynchronous here. However it is guaranteed that it // finished at latest after rendering QRhi::MaxAsyncReadbackFrames frames // after the one that enqueues the readback. QVERIFY(readCompleted); QVERIFY(readbackWidth > 0); if (impl == QRhi::Null) return; // Now we have a red rectangle on blue background. const int y = 50; const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; while (x-- >= 0) { const QRgb c(*p++); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, readbackWidth); QVERIFY(redCount < blueCount); } void tst_QRhi::finishWithinSwapchainFrame_data() { rhiTestData(); } void tst_QRhi::finishWithinSwapchainFrame() { if (QGuiApplication::platformName().startsWith(QLatin1String("offscreen"), Qt::CaseInsensitive)) QSKIP("Offscreen: This fails."); QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); QScopedPointer window(new QWindow); setWindowType(window.data(), impl); window->setGeometry(0, 0, 640, 480); window->show(); QVERIFY(QTest::qWaitForWindowExposed(window.data())); QScopedPointer swapChain(rhi->newSwapChain()); swapChain->setWindow(window.data()); swapChain->setFlags(QRhiSwapChain::UsedAsTransferSource); QScopedPointer rpDesc(swapChain->newCompatibleRenderPassDescriptor()); swapChain->setRenderPassDescriptor(rpDesc.data()); QVERIFY(swapChain->createOrResize()); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); // exercise begin/endExternal() just a little bit, note ExternalContent for beginPass() QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); QRhiRenderTarget *rt = swapChain->currentFrameRenderTarget(); const QSize outputSize = swapChain->currentPixelSize(); // repeat a sequence of upload, renderpass, readback, finish a number of // times within the same frame for (int i = 0; i < 5; ++i) { QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); cb->beginPass(rt, Qt::blue, { 1.0f, 0 }, updates, QRhiCommandBuffer::ExternalContent); // just have some commands, do not bother with draw calls cb->setGraphicsPipeline(pipeline.data()); QRhiViewport viewport(0, 0, float(outputSize.width()), float(outputSize.height())); cb->setViewport(viewport); // do a dummy begin/endExternal round: interesting for Vulkan because // there this may start end then submit a secondary command buffer cb->beginExternal(); cb->endExternal(); cb->endPass(); QRhiReadbackResult readResult; bool ok = false; readResult.completed = [&readResult, &ok, impl] { QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_ARGB32_Premultiplied); if (readResult.format == QRhiTexture::RGBA8) wrapperImage = wrapperImage.rgbSwapped(); if (impl != QRhi::Null) ok = qBlue(wrapperImage.pixel(43, 89)) > 250; else ok = true; // the Null backend does not actually render }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({}, &readResult); // read back the current backbuffer cb->resourceUpdate(readbackBatch); // force submit what we have so far, wait for the queue, and then start // a new primary command buffer rhi->finish(); QVERIFY(ok); } rhi->endFrame(swapChain.data()); } void tst_QRhi::resourceUpdateBatchBufferTextureWithSwapchainFrames_data() { rhiTestData(); } void tst_QRhi::resourceUpdateBatchBufferTextureWithSwapchainFrames() { if (QGuiApplication::platformName().startsWith(QLatin1String("offscreen"), Qt::CaseInsensitive)) QSKIP("Offscreen: Skipping onscreen test"); QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing buffer resource updates"); QScopedPointer window(new QWindow); setWindowType(window.data(), impl); window->setGeometry(0, 0, 640, 480); window->show(); QVERIFY(QTest::qWaitForWindowExposed(window.data())); QScopedPointer swapChain(rhi->newSwapChain()); swapChain->setWindow(window.data()); swapChain->setFlags(QRhiSwapChain::UsedAsTransferSource); QScopedPointer rpDesc(swapChain->newCompatibleRenderPassDescriptor()); swapChain->setRenderPassDescriptor(rpDesc.data()); QVERIFY(swapChain->createOrResize()); const int bufferSize = 18; const char *a = "123456789"; const char *b = "abcdefghi"; bool readCompleted = false; QRhiBufferReadbackResult readResult; readResult.completed = [&readCompleted] { readCompleted = true; }; QRhiReadbackResult texReadResult; texReadResult.completed = [&readCompleted] { readCompleted = true; }; { QScopedPointer dynamicBuffer(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, bufferSize)); QVERIFY(dynamicBuffer->create()); for (int i = 0; i < bufferSize; ++i) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); // One byte every 16.66 ms should be enough for everyone: fill up // the buffer with "123456789abcdefghi", one byte in each frame. if (i >= bufferSize / 2) batch->updateDynamicBuffer(dynamicBuffer.data(), i, 1, b + (i - bufferSize / 2)); else batch->updateDynamicBuffer(dynamicBuffer.data(), i, 1, a + i); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); // just clear to black, but submit the resource update cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); } { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); readCompleted = false; batch->readBackBuffer(dynamicBuffer.data(), 0, bufferSize, &readResult); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); // This is a proper, typically at least double buffered renderer (as // a real swapchain is involved). readCompleted may only become true // in a future frame. while (!readCompleted) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); rhi->endFrame(swapChain.data()); } QVERIFY(readResult.data.size() == bufferSize); QCOMPARE(readResult.data.left(bufferSize / 2), QByteArray(a)); QCOMPARE(readResult.data.mid(bufferSize / 2), QByteArray(b)); } } // Repeat for types Immutable and Static, declare Vertex usage. // This may not be readable on GLES 2.0 so skip the verification then. for (QRhiBuffer::Type type : { QRhiBuffer::Immutable, QRhiBuffer::Static }) { QScopedPointer buffer(rhi->newBuffer(type, QRhiBuffer::VertexBuffer, bufferSize)); QVERIFY(buffer->create()); for (int i = 0; i < bufferSize; ++i) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); if (i >= bufferSize / 2) batch->uploadStaticBuffer(buffer.data(), i, 1, b + (i - bufferSize / 2)); else batch->uploadStaticBuffer(buffer.data(), i, 1, a + i); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); } if (rhi->isFeatureSupported(QRhi::ReadBackNonUniformBuffer)) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); readCompleted = false; batch->readBackBuffer(buffer.data(), 0, bufferSize, &readResult); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); while (!readCompleted) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); rhi->endFrame(swapChain.data()); } QVERIFY(readResult.data.size() == bufferSize); QCOMPARE(readResult.data.left(bufferSize / 2), QByteArray(a)); QCOMPARE(readResult.data.mid(bufferSize / 2), QByteArray(b)); } else { qDebug("Skipping verification of buffer data as ReadBackNonUniformBuffer is not supported"); } } // Now exercise a texture. Internally this is expected (with low level APIs // at least) to be similar to what happens with a staic buffer: copy to host // visible staging buffer, enqueue buffer-to-buffer (or here // buffer-to-image) copy. { const int w = 234; const int h = 8; // use a small height because vsync throttling is active const QColor colors[] = { Qt::red, Qt::green, Qt::blue, Qt::gray, Qt::yellow, Qt::black, Qt::white, Qt::magenta }; QImage image(w, h, QImage::Format_RGBA8888); for (int i = 0; i < h; ++i) { QRgb c = colors[i].rgb(); uchar *p = image.scanLine(i); int x = w; while (x--) { *p++ = qRed(c); *p++ = qGreen(c); *p++ = qBlue(c); *p++ = qAlpha(c); } } QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(w, h), 1, QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); // fill a texture from the image, two lines at a time for (int i = 0; i < h / 2; ++i) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QRhiTextureSubresourceUploadDescription subresDesc(image); subresDesc.setSourceSize(QSize(w, 2)); subresDesc.setSourceTopLeft(QPoint(0, i * 2)); subresDesc.setDestinationTopLeft(QPoint(0, i * 2)); QRhiTextureUploadDescription uploadDesc(QRhiTextureUploadEntry(0, 0, subresDesc)); batch->uploadTexture(texture.data(), uploadDesc); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); } { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); readCompleted = false; batch->readBackTexture(texture.data(), &texReadResult); QRhiCommandBuffer *cb = swapChain->currentFrameCommandBuffer(); cb->beginPass(swapChain->currentFrameRenderTarget(), Qt::black, { 1.0f, 0 }, batch); cb->endPass(); rhi->endFrame(swapChain.data()); while (!readCompleted) { QVERIFY(rhi->beginFrame(swapChain.data()) == QRhi::FrameOpSuccess); rhi->endFrame(swapChain.data()); } QCOMPARE(texReadResult.pixelSize, image.size()); QImage wrapperImage(reinterpret_cast(texReadResult.data.constData()), texReadResult.pixelSize.width(), texReadResult.pixelSize.height(), image.format()); QVERIFY(imageRGBAEquals(image, wrapperImage)); } } } void tst_QRhi::textureRenderTargetAutoRebuild_data() { rhiTestData(); } void tst_QRhi::textureRenderTargetAutoRebuild() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); // case 1: beginPass's implicit create() { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ { texture.data() } })); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); cb->beginPass(rt.data(), Qt::red, { 1.0f, 0 }); cb->endPass(); rhi->endOffscreenFrame(); texture->setPixelSize(QSize(256, 256)); QVERIFY(texture->create()); QCOMPARE(texture->pixelSize(), QSize(256, 256)); QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); // no rt->create() but beginPass() does it implicitly for us cb->beginPass(rt.data(), Qt::red, { 1.0f, 0 }); QCOMPARE(rt->pixelSize(), QSize(256, 256)); cb->endPass(); rhi->endOffscreenFrame(); } // case 2: pixelSize's implicit create() { QSize sz(512, 512); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, sz, 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ { texture.data() } })); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); QCOMPARE(rt->pixelSize(), sz); sz = QSize(256, 256); texture->setPixelSize(sz); QVERIFY(texture->create()); QCOMPARE(rt->pixelSize(), sz); } } void tst_QRhi::srbLayoutCompatibility_data() { rhiTestData(); } void tst_QRhi::srbLayoutCompatibility() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing texture resource updates"); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512))); QVERIFY(texture->create()); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer otherSampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(otherSampler->create()); QScopedPointer buf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 1024)); QVERIFY(buf->create()); QScopedPointer otherBuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 256)); QVERIFY(otherBuf->create()); // empty (compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); QVERIFY(srb2->create()); QVERIFY(srb1->isLayoutCompatible(srb2.data())); QVERIFY(srb2->isLayoutCompatible(srb1.data())); QCOMPARE(srb1->serializedLayoutDescription(), srb2->serializedLayoutDescription()); QVERIFY(srb1->serializedLayoutDescription().size() == 0); } // different count (not compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::sampledTexture(0, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb2->create()); QVERIFY(!srb1->isLayoutCompatible(srb2.data())); QVERIFY(!srb2->isLayoutCompatible(srb1.data())); QVERIFY(srb1->serializedLayoutDescription() != srb2->serializedLayoutDescription()); QVERIFY(srb1->serializedLayoutDescription().size() == 0); QVERIFY(srb2->serializedLayoutDescription().size() == 1 * QRhiShaderResourceBinding::LAYOUT_DESC_ENTRIES_PER_BINDING); } // full match (compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb2->create()); QVERIFY(srb1->isLayoutCompatible(srb2.data())); QVERIFY(srb2->isLayoutCompatible(srb1.data())); QVERIFY(!srb1->serializedLayoutDescription().isEmpty()); QVERIFY(!srb2->serializedLayoutDescription().isEmpty()); QCOMPARE(srb1->serializedLayoutDescription(), srb2->serializedLayoutDescription()); QVERIFY(srb1->serializedLayoutDescription().size() == 2 * QRhiShaderResourceBinding::LAYOUT_DESC_ENTRIES_PER_BINDING); // see what we would get if a binding list got serialized "manually", without pulling it out from the srb after building // (the results should be identical) QVector layoutDesc1; QRhiShaderResourceBinding::serializeLayoutDescription(srb1->cbeginBindings(), srb1->cendBindings(), std::back_inserter(layoutDesc1)); QCOMPARE(layoutDesc1, srb1->serializedLayoutDescription()); QVector layoutDesc2; QRhiShaderResourceBinding::serializeLayoutDescription(srb2->cbeginBindings(), srb2->cendBindings(), std::back_inserter(layoutDesc2)); QCOMPARE(layoutDesc2, srb2->serializedLayoutDescription()); // exercise with an "output iterator" different from back_inserter quint32 layoutDesc3[2 * QRhiShaderResourceBinding::LAYOUT_DESC_ENTRIES_PER_BINDING]; QRhiShaderResourceBinding::serializeLayoutDescription(srb1->cbeginBindings(), srb1->cendBindings(), layoutDesc3); QVERIFY(!memcmp(layoutDesc3, layoutDesc1.constData(), sizeof(quint32) * 2 * QRhiShaderResourceBinding::LAYOUT_DESC_ENTRIES_PER_BINDING)); } // different visibility (not compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage | QRhiShaderResourceBinding::FragmentStage, buf.data()), }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), }); QVERIFY(srb2->create()); QVERIFY(!srb1->isLayoutCompatible(srb2.data())); QVERIFY(!srb2->isLayoutCompatible(srb1.data())); QVERIFY(srb1->serializedLayoutDescription() != srb2->serializedLayoutDescription()); } // different binding points (not compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(1, QRhiShaderResourceBinding::VertexStage, buf.data()), }); QVERIFY(srb2->create()); QVERIFY(!srb1->isLayoutCompatible(srb2.data())); QVERIFY(!srb2->isLayoutCompatible(srb1.data())); QVERIFY(srb1->serializedLayoutDescription() != srb2->serializedLayoutDescription()); } // different buffer region offset and size (compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data(), rhi->ubufAligned(1), 128), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb2->create()); QVERIFY(srb1->isLayoutCompatible(srb2.data())); QVERIFY(srb2->isLayoutCompatible(srb1.data())); QCOMPARE(srb1->serializedLayoutDescription(), srb2->serializedLayoutDescription()); } // different resources (compatible) { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, otherBuf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), otherSampler.data()) }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb2->create()); QVERIFY(srb1->isLayoutCompatible(srb2.data())); QVERIFY(srb2->isLayoutCompatible(srb1.data())); QCOMPARE(srb1->serializedLayoutDescription(), srb2->serializedLayoutDescription()); } } void tst_QRhi::srbWithNoResource_data() { rhiTestData(); } void tst_QRhi::srbWithNoResource() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing srb"); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512))); QVERIFY(texture->create()); QScopedPointer sampler(rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest, QRhiSampler::None, QRhiSampler::ClampToEdge, QRhiSampler::ClampToEdge)); QVERIFY(sampler->create()); QScopedPointer buf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 1024)); QVERIFY(buf->create()); { QScopedPointer srb1(rhi->newShaderResourceBindings()); srb1->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, nullptr), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, nullptr, nullptr) }); QVERIFY(srb1->create()); QScopedPointer srb2(rhi->newShaderResourceBindings()); srb2->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::VertexStage, buf.data()), QRhiShaderResourceBinding::sampledTexture(1, QRhiShaderResourceBinding::FragmentStage, texture.data(), sampler.data()) }); QVERIFY(srb2->create()); QVERIFY(srb1->isLayoutCompatible(srb2.data())); QVERIFY(srb2->isLayoutCompatible(srb1.data())); } } void tst_QRhi::renderPassDescriptorCompatibility_data() { rhiTestData(); } void tst_QRhi::renderPassDescriptorCompatibility() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing renderpass descriptors"); // Note that checking compatibility is only relevant with backends where // there is a concept of renderpass descriptions (Vulkan, and partially // Metal). It is perfectly fine for isCompatible() to always return true // when that is not the case (D3D11, OpenGL). Hence the 'if (Vulkan or // Metal)' for all the negative tests. Also note "partial" for Metal: // resolve textures for examples have no effect on compatibility with Metal. // tex and tex2 have the same format QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex->create()); QScopedPointer tex2(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex2->create()); QScopedPointer ds(rhi->newRenderBuffer(QRhiRenderBuffer::DepthStencil, QSize(512, 512))); QVERIFY(ds->create()); // two texture rendertargets with tex and tex2 as color0 (compatible) { QScopedPointer rt(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer rt2(rhi->newTextureRenderTarget({ tex2.data() })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); QVERIFY(rpDesc->isCompatible(rpDesc2.data())); QVERIFY(rpDesc2->isCompatible(rpDesc.data())); QCOMPARE(rpDesc->serializedFormat(), rpDesc2->serializedFormat()); } // two texture rendertargets with tex and tex2 as color0, and a depth-stencil attachment as well (compatible) { QRhiTextureRenderTargetDescription desc({ tex.data() }, ds.data()); QScopedPointer rt(rhi->newTextureRenderTarget(desc)); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer rt2(rhi->newTextureRenderTarget(desc)); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); QVERIFY(rpDesc->isCompatible(rpDesc2.data())); QVERIFY(rpDesc2->isCompatible(rpDesc.data())); QCOMPARE(rpDesc->serializedFormat(), rpDesc2->serializedFormat()); } // now one of them does not have the ds attachment (not compatible) { QScopedPointer rt(rhi->newTextureRenderTarget({ { tex.data() }, ds.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer rt2(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); // these backends have a real concept of rp compatibility, with those we // know that incompatibility must be reported; verify this if (impl == QRhi::Vulkan || impl == QRhi::Metal) { QVERIFY(!rpDesc->isCompatible(rpDesc2.data())); QVERIFY(!rpDesc2->isCompatible(rpDesc.data())); QVERIFY(!rpDesc->serializedFormat().isEmpty()); QVERIFY(rpDesc->serializedFormat() != rpDesc2->serializedFormat()); } } if (rhi->isFeatureSupported(QRhi::MultisampleRenderBuffer)) { // resolve attachments (compatible) { QScopedPointer msaaRenderBuffer(rhi->newRenderBuffer(QRhiRenderBuffer::Color, QSize(512, 512), 4)); QVERIFY(msaaRenderBuffer->create()); QScopedPointer msaaRenderBuffer2(rhi->newRenderBuffer(QRhiRenderBuffer::Color, QSize(512, 512), 4)); QVERIFY(msaaRenderBuffer2->create()); QRhiColorAttachment colorAtt(msaaRenderBuffer.data()); // color0, multisample colorAtt.setResolveTexture(tex.data()); // resolved into a non-msaa texture QScopedPointer rt(rhi->newTextureRenderTarget({ colorAtt })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiColorAttachment colorAtt2(msaaRenderBuffer2.data()); // color0, multisample colorAtt2.setResolveTexture(tex2.data()); // resolved into a non-msaa texture QScopedPointer rt2(rhi->newTextureRenderTarget({ colorAtt2 })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); QVERIFY(rpDesc->isCompatible(rpDesc2.data())); QVERIFY(rpDesc2->isCompatible(rpDesc.data())); QCOMPARE(rpDesc->serializedFormat(), rpDesc2->serializedFormat()); } // missing resolve for one of them (not compatible) { QScopedPointer msaaRenderBuffer(rhi->newRenderBuffer(QRhiRenderBuffer::Color, QSize(512, 512), 4)); QVERIFY(msaaRenderBuffer->create()); QScopedPointer msaaRenderBuffer2(rhi->newRenderBuffer(QRhiRenderBuffer::Color, QSize(512, 512), 4)); QVERIFY(msaaRenderBuffer2->create()); QRhiColorAttachment colorAtt(msaaRenderBuffer.data()); // color0, multisample colorAtt.setResolveTexture(tex.data()); // resolved into a non-msaa texture QScopedPointer rt(rhi->newTextureRenderTarget({ colorAtt })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiColorAttachment colorAtt2(msaaRenderBuffer2.data()); // color0, multisample QScopedPointer rt2(rhi->newTextureRenderTarget({ colorAtt2 })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); if (impl == QRhi::Vulkan) { // no Metal here QVERIFY(!rpDesc->isCompatible(rpDesc2.data())); QVERIFY(!rpDesc2->isCompatible(rpDesc.data())); QVERIFY(!rpDesc->serializedFormat().isEmpty()); QVERIFY(rpDesc->serializedFormat() != rpDesc2->serializedFormat()); } } } else { qDebug("Skipping multisample renderbuffer dependent tests"); } if (rhi->isTextureFormatSupported(QRhiTexture::RGBA32F)) { QScopedPointer tex3(rhi->newTexture(QRhiTexture::RGBA32F, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex3->create()); // different texture formats (not compatible) { QScopedPointer rt(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer rt2(rhi->newTextureRenderTarget({ tex3.data() })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); if (impl == QRhi::Vulkan || impl == QRhi::Metal) { QVERIFY(!rpDesc->isCompatible(rpDesc2.data())); QVERIFY(!rpDesc2->isCompatible(rpDesc.data())); QVERIFY(!rpDesc->serializedFormat().isEmpty()); QVERIFY(rpDesc->serializedFormat() != rpDesc2->serializedFormat()); } } } else { qDebug("Skipping texture format dependent tests"); } } void tst_QRhi::renderPassDescriptorClone_data() { rhiTestData(); } void tst_QRhi::renderPassDescriptorClone() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing renderpass descriptors"); // tex and tex2 have the same format QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex->create()); QScopedPointer tex2(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(tex2->create()); QScopedPointer ds(rhi->newRenderBuffer(QRhiRenderBuffer::DepthStencil, QSize(512, 512))); QVERIFY(ds->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ tex.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer rpDescClone(rpDesc->newCompatibleRenderPassDescriptor()); QVERIFY(rpDescClone); QVERIFY(rpDesc->isCompatible(rpDescClone.data())); // rt and rt2 have the same set of attachments QScopedPointer rt2(rhi->newTextureRenderTarget({ tex2.data() })); QScopedPointer rpDesc2(rt2->newCompatibleRenderPassDescriptor()); rt2->setRenderPassDescriptor(rpDesc2.data()); QVERIFY(rt2->create()); QVERIFY(rpDesc2->isCompatible(rpDescClone.data())); } void tst_QRhi::pipelineCache_data() { rhiTestData(); } void tst_QRhi::pipelineCache() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QByteArray pcd; QShader vs = loadShader(":/data/simple.vert.qsb"); QVERIFY(vs.isValid()); QShader fs = loadShader(":/data/simple.frag.qsb"); QVERIFY(fs.isValid()); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 2 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float2, 0 } }); { QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::EnablePipelineCacheDataSave)); if (!rhi) QSKIP("QRhi could not be created, skipping testing (set)pipelineCacheData()"); if (!rhi->isFeatureSupported(QRhi::PipelineCacheDataLoadSave)) QSKIP("PipelineCacheDataLoadSave is not supported with this backend, skipping test"); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(256, 256), 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); // This cannot be more than a basic smoketest: ensure that passing // in the data we retrieve still gives us successful pipeline // creation. What happens internally we cannot check. pcd = rhi->pipelineCacheData(); rhi->setPipelineCacheData(pcd); QVERIFY(pipeline->create()); } { // Now from scratch, with seeding the cache right from the start, // presumably leading to a cache hit when creating the pipeline. QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::EnablePipelineCacheDataSave)); QVERIFY(rhi); rhi->setPipelineCacheData(pcd); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(256, 256), 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, vs }, { QRhiShaderStage::Fragment, fs } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); } } void tst_QRhi::textureImportOpenGL() { #ifdef TST_GL if (!QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::OpenGL)) QSKIP("Skipping OpenGL-dependent test"); QScopedPointer rhi(QRhi::create(QRhi::OpenGLES2, &initParams.gl, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing native texture"); QVERIFY(rhi->makeThreadLocalNativeContextCurrent()); QOpenGLContext *ctx = QOpenGLContext::currentContext(); QVERIFY(ctx); QOpenGLFunctions *f = ctx->functions(); QImage image(320, 200, QImage::Format_RGBA8888_Premultiplied); image.fill(Qt::red); GLuint t = 0; f->glGenTextures(1, &t); f->glBindTexture(GL_TEXTURE_2D, t); f->glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, image.width(), image.height(), 0, GL_RGBA, GL_UNSIGNED_BYTE, image.constBits()); QScopedPointer tex(rhi->newTexture(QRhiTexture::RGBA8, image.size())); QRhiTexture::NativeTexture nativeTex = { t, 0 }; QVERIFY(tex->createFrom(nativeTex)); QCOMPARE(tex->nativeTexture().object, nativeTex.object); QRhiReadbackResult readResult; bool readCompleted = false; readResult.completed = [&readCompleted] { readCompleted = true; }; QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); batch->readBackTexture(tex.data(), &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(readCompleted); QCOMPARE(readResult.format, QRhiTexture::RGBA8); QCOMPARE(readResult.pixelSize, image.size()); QImage wrapperImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), image.format()); QVERIFY(imageRGBAEquals(image, wrapperImage)); f->glDeleteTextures(1, &t); #endif } void tst_QRhi::renderbufferImportOpenGL() { #ifdef TST_GL if (!QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::OpenGL)) QSKIP("Skipping OpenGL-dependent test"); QScopedPointer rhi(QRhi::create(QRhi::OpenGLES2, &initParams.gl, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing native texture"); QVERIFY(rhi->makeThreadLocalNativeContextCurrent()); QOpenGLContext *ctx = QOpenGLContext::currentContext(); QVERIFY(ctx); QOpenGLFunctions *f = ctx->functions(); const QSize size(320, 200); GLuint b = 0; f->glGenRenderbuffers(1, &b); f->glBindRenderbuffer(GL_RENDERBUFFER, b); // in a real world use case this would be some extension, e.g. glEGLImageTargetRenderbufferStorageOES instead f->glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA4, size.width(), size.height()); f->glBindRenderbuffer(GL_RENDERBUFFER, 0); QScopedPointer rb(rhi->newRenderBuffer(QRhiRenderBuffer::Color, size)); QVERIFY(rb->createFrom({ b })); QScopedPointer depthStencil(rhi->newRenderBuffer(QRhiRenderBuffer::DepthStencil, size)); QVERIFY(depthStencil->create()); QRhiColorAttachment att(rb.data()); QRhiTextureRenderTargetDescription rtDesc(att); rtDesc.setDepthStencilBuffer(depthStencil.data()); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); cb->beginPass(rt.data(), Qt::red, { 1.0f, 0 }, nullptr, QRhiCommandBuffer::ExternalContent); cb->beginExternal(); QByteArray tmpBuf; tmpBuf.resize(size.width() * size.height() * 4); f->glReadPixels(0, 0, size.width(), size.height(), GL_RGBA, GL_UNSIGNED_BYTE, tmpBuf.data()); cb->endExternal(); cb->endPass(); rhi->endOffscreenFrame(); f->glDeleteRenderbuffers(1, &b); QImage wrapperImage(reinterpret_cast(tmpBuf.constData()), size.width(), size.height(), QImage::Format_RGBA8888_Premultiplied); QImage image(320, 200, QImage::Format_RGBA8888_Premultiplied); image.fill(Qt::red); QVERIFY(imageRGBAEquals(image, wrapperImage)); #endif } void tst_QRhi::threeDimTexture_data() { rhiTestData(); } void tst_QRhi::threeDimTexture() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams)); if (!rhi) QSKIP("QRhi could not be created, skipping testing 3D textures"); if (!rhi->isFeatureSupported(QRhi::ThreeDimensionalTextures)) QSKIP("Skipping testing 3D textures because they are reported as unsupported"); const int WIDTH = 512; const int HEIGHT = 256; const int DEPTH = 128; { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, WIDTH, HEIGHT, DEPTH)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < DEPTH; ++i) { QImage img(WIDTH, HEIGHT, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry sliceUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), sliceUpload); } QVERIFY(submitResourceUpdates(rhi.data(), batch)); } // mipmaps { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, WIDTH, HEIGHT, DEPTH, 1, QRhiTexture::MipMapped | QRhiTexture::UsedWithGenerateMips)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < DEPTH; ++i) { QImage img(WIDTH, HEIGHT, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry sliceUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), sliceUpload); } batch->generateMips(texture.data()); QVERIFY(submitResourceUpdates(rhi.data(), batch)); // read back slice 63 of level 1 (256x128, almost red) batch = rhi->nextResourceUpdateBatch(); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLevel(1); readbackDescription.setLayer(63); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH / 2, HEIGHT / 2, result.format()); referenceImage.fill(QColor::fromRgb(253, 0, 0)); // Now restrict the test a bit. The Null QRhi backend has broken support for // mipmap generation of 3D textures (it ignores the depth, effectively behaving as // if the 3D texture was a 2D array which is incorrect wrt mipmapping) // Some software-based OpenGL implementations, such as Mesa llvmpipe builds that are // used both in Qt CI and are shipped with the official Qt binaries also seem to have // problems with this. if (impl != QRhi::Null && impl != QRhi::OpenGLES2) QVERIFY(imageRGBAEquals(result, referenceImage, 2)); } // render target (one slice) // NB with Vulkan we require Vulkan 1.1 for this to work. { const int SLICE = 23; QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, WIDTH, HEIGHT, DEPTH, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiColorAttachment att(texture.data()); att.setLayer(SLICE); QRhiTextureRenderTargetDescription rtDesc(att); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); // render to slice 23 QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }); // slice 23 is now blue cb->endPass(); rhi->endOffscreenFrame(); // Fill all other slices with some color. We should be free to do this // step *before* the "render to slice 23" block above as well. However, // as QTBUG-111772 shows, some Vulkan implementations have problems // then. (or it could be QRhi is doing something wrong, but there is no // evidence of that yet) For now, keep the order of first rendering to // a slice and then uploading data for the rest. QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < DEPTH; ++i) { if (i != SLICE) { QImage img(WIDTH, HEIGHT, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry sliceUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), sliceUpload); } } QVERIFY(submitResourceUpdates(rhi.data(), batch)); // read back slice 23 (blue) batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLayer(23); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, HEIGHT, result.format()); referenceImage.fill(QColor::fromRgbF(0.0f, 0.0f, 1.0f)); // the Null backend does not render so skip the verification for that if (impl != QRhi::Null) QVERIFY(imageRGBAEquals(result, referenceImage)); // read back slice 0 (black) batch = rhi->nextResourceUpdateBatch(); result = QImage(); readbackDescription.setLayer(0); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); referenceImage.fill(QColor::fromRgbF(0.0f, 0.0f, 0.0f)); QVERIFY(imageRGBAEquals(result, referenceImage)); // read back slice 127 (almost red) batch = rhi->nextResourceUpdateBatch(); result = QImage(); readbackDescription.setLayer(127); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); referenceImage.fill(QColor::fromRgb(254, 0, 0)); QVERIFY(imageRGBAEquals(result, referenceImage)); } } void tst_QRhi::oneDimTexture_data() { rhiTestData(); } void tst_QRhi::oneDimTexture() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams)); if (!rhi) QSKIP("QRhi could not be created, skipping testing 1D textures"); if (!rhi->isFeatureSupported(QRhi::OneDimensionalTextures)) QSKIP("Skipping testing 1D textures because they are reported as unsupported"); const int WIDTH = 512; const int LAYERS = 128; { QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, WIDTH, 0, 0)); QVERIFY(texture->create()); QVERIFY(texture->flags().testFlag(QRhiTexture::Flag::OneDimensional)); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(255, 0, 0)); QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), upload); QVERIFY(submitResourceUpdates(rhi.data(), batch)); } { QScopedPointer texture( rhi->newTextureArray(QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0))); QVERIFY(texture->create()); QVERIFY(texture->flags().testFlag(QRhiTexture::Flag::OneDimensional)); QVERIFY(texture->flags().testFlag(QRhiTexture::Flag::TextureArray)); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < LAYERS; ++i) { QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry layerUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), layerUpload); } QVERIFY(submitResourceUpdates(rhi.data(), batch)); } // Copy from 2D texture to 1D texture { const int WIDTH = 256; const int HEIGHT = 256; QScopedPointer srcTexture(rhi->newTexture( QRhiTexture::RGBA8, WIDTH, HEIGHT, 0, 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(srcTexture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, HEIGHT, QImage::Format_RGBA8888); for (int x = 0; x < WIDTH; ++x) { for (int y = 0; y < HEIGHT; ++y) { img.setPixelColor(x, y, QColor::fromRgb(x, y, 0)); } } QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(srcTexture.data(), upload); QScopedPointer dstTexture(rhi->newTexture( QRhiTexture::RGBA8, WIDTH, 0, 0, 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(dstTexture->create()); QRhiTextureCopyDescription copy; copy.setPixelSize(QSize(WIDTH / 2, 1)); copy.setDestinationTopLeft(QPoint(WIDTH / 2, 0)); copy.setSourceTopLeft(QPoint(33, 67)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); copy.setDestinationTopLeft(QPoint(0, 0)); copy.setSourceTopLeft(QPoint(99, 12)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(dstTexture.data()); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); for (int i = 0; i < WIDTH / 2; ++i) { referenceImage.setPixelColor(i, 0, img.pixelColor(99 + i, 12)); referenceImage.setPixelColor(WIDTH / 2 + i, 0, img.pixelColor(33 + i, 67)); } QVERIFY(imageRGBAEquals(result, referenceImage)); } // Copy from 2D texture to 1D texture array { const int WIDTH = 256; const int HEIGHT = 256; const int LAYERS = 64; QScopedPointer srcTexture(rhi->newTexture( QRhiTexture::RGBA8, WIDTH, HEIGHT, 0, 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(srcTexture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, HEIGHT, QImage::Format_RGBA8888); for (int x = 0; x < WIDTH; ++x) { for (int y = 0; y < HEIGHT; ++y) { img.setPixelColor(x, y, QColor::fromRgb(x, y, 0)); } } QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(srcTexture.data(), upload); QScopedPointer dstTexture( rhi->newTextureArray(QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0), 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(dstTexture->create()); QRhiTextureCopyDescription copy; copy.setPixelSize(QSize(WIDTH / 2, 1)); copy.setDestinationTopLeft(QPoint(WIDTH / 2, 0)); copy.setSourceTopLeft(QPoint(33, 67)); copy.setDestinationLayer(12); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); copy.setDestinationTopLeft(QPoint(0, 0)); copy.setSourceTopLeft(QPoint(99, 12)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(dstTexture.data()); readbackDescription.setLayer(12); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); for (int i = 0; i < WIDTH / 2; ++i) { referenceImage.setPixelColor(i, 0, img.pixelColor(99 + i, 12)); referenceImage.setPixelColor(WIDTH / 2 + i, 0, img.pixelColor(33 + i, 67)); } QVERIFY(imageRGBAEquals(result, referenceImage)); } // Copy from 1D texture array to 1D texture { const int WIDTH = 256; const int LAYERS = 256; QScopedPointer srcTexture( rhi->newTextureArray(QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0), 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(srcTexture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int y = 0; y < LAYERS; ++y) { QImage img(WIDTH, 1, QImage::Format_RGBA8888); for (int x = 0; x < WIDTH; ++x) { img.setPixelColor(x, 0, QColor::fromRgb(x, y, 0)); } QRhiTextureUploadEntry upload(y, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(srcTexture.data(), upload); } QScopedPointer dstTexture(rhi->newTexture( QRhiTexture::RGBA8, WIDTH, 0, 0, 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(dstTexture->create()); QRhiTextureCopyDescription copy; copy.setPixelSize(QSize(WIDTH / 2, 1)); copy.setDestinationTopLeft(QPoint(WIDTH / 2, 0)); copy.setSourceLayer(67); copy.setSourceTopLeft(QPoint(33, 0)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); copy.setDestinationTopLeft(QPoint(0, 0)); copy.setSourceLayer(12); copy.setSourceTopLeft(QPoint(99, 0)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(dstTexture.data()); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); for (int i = 0; i < WIDTH / 2; ++i) { referenceImage.setPixelColor(i, 0, QColor::fromRgb(99 + i, 12, 0)); referenceImage.setPixelColor(WIDTH / 2 + i, 0, QColor::fromRgb(33 + i, 67, 0)); } QVERIFY(imageRGBAEquals(result, referenceImage)); } // Copy from 1D texture to 1D texture array { const int WIDTH = 256; const int LAYERS = 256; QScopedPointer srcTexture(rhi->newTexture( QRhiTexture::RGBA8, WIDTH, 0, 0, 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(srcTexture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, 1, QImage::Format_RGBA8888); for (int x = 0; x < WIDTH; ++x) { img.setPixelColor(x, 0, QColor::fromRgb(x, 0, 0)); } QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(srcTexture.data(), upload); QScopedPointer dstTexture( rhi->newTextureArray(QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0), 1, QRhiTexture::Flag::UsedAsTransferSource)); QVERIFY(dstTexture->create()); QRhiTextureCopyDescription copy; copy.setPixelSize(QSize(WIDTH / 2, 1)); copy.setDestinationTopLeft(QPoint(WIDTH / 2, 0)); copy.setDestinationLayer(67); copy.setSourceTopLeft(QPoint(33, 0)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); copy.setDestinationTopLeft(QPoint(0, 0)); copy.setSourceTopLeft(QPoint(99, 0)); batch->copyTexture(dstTexture.data(), srcTexture.data(), copy); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(dstTexture.data()); readbackDescription.setLayer(67); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); for (int i = 0; i < WIDTH / 2; ++i) { referenceImage.setPixelColor(i, 0, QColor::fromRgb(99 + i, 0, 0)); referenceImage.setPixelColor(WIDTH / 2 + i, 0, QColor::fromRgb(33 + i, 0, 0)); } QVERIFY(imageRGBAEquals(result, referenceImage)); } // mipmaps and 1D render target if (!rhi->isFeatureSupported(QRhi::OneDimensionalTextureMipmaps)) QSKIP("Skipping testing 1D texture mipmaps and 1D render target because they are reported " "as unsupported"); { QScopedPointer texture( rhi->newTexture(QRhiTexture::RGBA8, WIDTH, 0, 0, 1, QRhiTexture::MipMapped | QRhiTexture::UsedWithGenerateMips)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(128, 0, 0)); QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), upload); batch->generateMips(texture.data()); QVERIFY(submitResourceUpdates(rhi.data(), batch)); // read back level 1 (256x1, #800000ff) batch = rhi->nextResourceUpdateBatch(); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLevel(1); readbackDescription.setLayer(0); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH / 2, 1, result.format()); referenceImage.fill(QColor::fromRgb(128, 0, 0)); QVERIFY(imageRGBAEquals(result, referenceImage, 2)); } { QScopedPointer texture( rhi->newTextureArray(QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0), 1, QRhiTexture::MipMapped | QRhiTexture::UsedWithGenerateMips)); QVERIFY(texture->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < LAYERS; ++i) { QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry sliceUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), sliceUpload); } batch->generateMips(texture.data()); QVERIFY(submitResourceUpdates(rhi.data(), batch)); // read back slice 63 of level 1 (256x1, #7E0000FF) batch = rhi->nextResourceUpdateBatch(); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLevel(1); readbackDescription.setLayer(63); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH / 2, 1, result.format()); referenceImage.fill(QColor::fromRgb(126, 0, 0)); // Now restrict the test a bit. The Null QRhi backend has broken support for // mipmap generation of 1D texture arrays. if (impl != QRhi::Null) QVERIFY(imageRGBAEquals(result, referenceImage, 2)); } // 1D texture render target // NB with Vulkan we require Vulkan 1.1 for this to work. // Metal does not allow 1D texture render targets { QScopedPointer texture( rhi->newTexture(QRhiTexture::RGBA8, WIDTH, 0, 0, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiColorAttachment att(texture.data()); QRhiTextureRenderTargetDescription rtDesc(att); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(128, 0, 0)); QRhiTextureUploadEntry upload(0, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), upload); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, batch); // texture is now blue cb->endPass(); rhi->endOffscreenFrame(); // read back texture (blue) batch = rhi->nextResourceUpdateBatch(); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); referenceImage.fill(QColor::fromRgbF(0.0f, 0.0f, 1.0f)); // the Null backend does not render so skip the verification for that if (impl != QRhi::Null) QVERIFY(imageRGBAEquals(result, referenceImage)); } // 1D array texture render target (one slice) // NB with Vulkan we require Vulkan 1.1 for this to work. // Metal does not allow 1D texture render targets { const int SLICE = 23; QScopedPointer texture(rhi->newTextureArray( QRhiTexture::RGBA8, LAYERS, QSize(WIDTH, 0), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QRhiColorAttachment att(texture.data()); att.setLayer(SLICE); QRhiTextureRenderTargetDescription rtDesc(att); QScopedPointer rt(rhi->newTextureRenderTarget(rtDesc)); QScopedPointer rp(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rp.data()); QVERIFY(rt->create()); QRhiResourceUpdateBatch *batch = rhi->nextResourceUpdateBatch(); QVERIFY(batch); for (int i = 0; i < LAYERS; ++i) { QImage img(WIDTH, 1, QImage::Format_RGBA8888); img.fill(QColor::fromRgb(i * 2, 0, 0)); QRhiTextureUploadEntry sliceUpload(i, 0, QRhiTextureSubresourceUploadDescription(img)); batch->uploadTexture(texture.data(), sliceUpload); } QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, batch); // slice 23 is now blue cb->endPass(); rhi->endOffscreenFrame(); // read back slice 23 (blue) batch = rhi->nextResourceUpdateBatch(); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiReadbackDescription readbackDescription(texture.data()); readbackDescription.setLayer(23); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); QImage referenceImage(WIDTH, 1, result.format()); referenceImage.fill(QColor::fromRgbF(0.0f, 0.0f, 1.0f)); // the Null backend does not render so skip the verification for that if (impl != QRhi::Null) QVERIFY(imageRGBAEquals(result, referenceImage)); // read back slice 0 (black) batch = rhi->nextResourceUpdateBatch(); result = QImage(); readbackDescription.setLayer(0); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); referenceImage.fill(QColor::fromRgbF(0.0f, 0.0f, 0.0f)); QVERIFY(imageRGBAEquals(result, referenceImage)); // read back slice 127 (almost red) batch = rhi->nextResourceUpdateBatch(); result = QImage(); readbackDescription.setLayer(127); batch->readBackTexture(readbackDescription, &readResult); QVERIFY(submitResourceUpdates(rhi.data(), batch)); QVERIFY(!result.isNull()); referenceImage.fill(QColor::fromRgb(254, 0, 0)); QVERIFY(imageRGBAEquals(result, referenceImage)); } } void tst_QRhi::leakedResourceDestroy_data() { rhiTestData(); } void tst_QRhi::leakedResourceDestroy() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams)); if (!rhi) QSKIP("QRhi could not be created, skipping"); // Incorrectly destroy the QRhi before the resources created from it. Attempting to // destroy the resources afterwards is pointless, the native resources are leaked. // Nonetheless, it should not crash, which is what we are testing here. // // We do not however have control over other, native and 3rd party components: a // validation or debug layer, or a memory allocator may warn, assert, or abort when // not releasing all native resources correctly. #ifndef QT_NO_DEBUG // don't want asserts from vkmemalloc, skip the test in debug builds if (impl == QRhi::Vulkan) QSKIP("Skipping leaked resource destroy test due to Vulkan and debug build"); #endif QScopedPointer buffer(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, 256)); QVERIFY(buffer->create()); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(512, 512), 1, QRhiTexture::RenderTarget)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); QVERIFY(rpDesc); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiRenderBuffer *rb = rhi->newRenderBuffer(QRhiRenderBuffer::DepthStencil, QSize(512, 512)); QVERIFY(rb->create()); QRhiShaderResourceBindings *srb = rhi->newShaderResourceBindings(); QVERIFY(srb->create()); if (impl == QRhi::Vulkan) qDebug("Vulkan validation layer warnings may be printed below - this is expected"); qDebug("QRhi resource leak check warnings may be printed below - this is expected"); // make the QRhi go away early rhi.reset(); // see if the internal rhi backpointer got nulled out QVERIFY(buffer->rhi() == nullptr); QVERIFY(texture->rhi() == nullptr); QVERIFY(rt->rhi() == nullptr); QVERIFY(rpDesc->rhi() == nullptr); QVERIFY(rb->rhi() == nullptr); QVERIFY(srb->rhi() == nullptr); // test out deleteLater on some of the resources rb->deleteLater(); srb->deleteLater(); // let the scoped ptr do its job with the rest } void tst_QRhi::renderToFloatTexture_data() { rhiTestData(); } void tst_QRhi::renderToFloatTexture() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); if (!rhi->isTextureFormatSupported(QRhiTexture::RGBA16F)) QSKIP("RGBA16F is not supported, skipping test"); const QSize outputSize(1920, 1080); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA16F, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA16FPx4); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QCOMPARE(result.size(), texture->pixelSize()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // Now we have a red rectangle on blue background. const int y = 100; const QRgbaFloat16 *p = reinterpret_cast(result.constScanLine(y)); int redCount = 0; int blueCount = 0; int x = result.width() - 1; while (x-- >= 0) { QRgbaFloat16 c = *p++; if (c.red() >= 0.95f && qFuzzyIsNull(c.green()) && qFuzzyIsNull(c.blue())) ++redCount; else if (qFuzzyIsNull(c.red()) && qFuzzyIsNull(c.green()) && c.blue() >= 0.95f) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, texture->pixelSize().width()); QVERIFY(redCount > blueCount); // 1742 > 178 } void tst_QRhi::renderToRgb10Texture_data() { rhiTestData(); } void tst_QRhi::renderToRgb10Texture() { QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); if (!rhi->isTextureFormatSupported(QRhiTexture::RGB10A2)) QSKIP("RGB10A2 is not supported, skipping test"); const QSize outputSize(1920, 1080); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGB10A2, outputSize, 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); QRhiCommandBuffer *cb = nullptr; QVERIFY(rhi->beginOffscreenFrame(&cb) == QRhi::FrameOpSuccess); QVERIFY(cb); QRhiResourceUpdateBatch *updates = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); updates->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer srb(rhi->newShaderResourceBindings()); QVERIFY(srb->create()); QScopedPointer pipeline(createSimplePipeline(rhi.data(), srb.data(), rpDesc.data())); QVERIFY(pipeline); cb->beginPass(rt.data(), Qt::blue, { 1.0f, 0 }, updates); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(outputSize.width()), float(outputSize.height()) }); QRhiCommandBuffer::VertexInput vbindings(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbindings); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_A2BGR30_Premultiplied); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); QCOMPARE(result.size(), texture->pixelSize()); if (impl == QRhi::Null) return; if (rhi->isYUpInFramebuffer() != rhi->isYUpInNDC()) result = std::move(result).mirrored(); // Now we have a red rectangle on blue background. const int y = 100; int redCount = 0; int blueCount = 0; const int maxFuzz = 1; for (int x = 0; x < result.width(); ++x) { QRgb c = result.pixel(x, y); if (qRed(c) >= (255 - maxFuzz) && qGreen(c) == 0 && qBlue(c) == 0) ++redCount; else if (qRed(c) == 0 && qGreen(c) == 0 && qBlue(c) >= (255 - maxFuzz)) ++blueCount; else QFAIL("Encountered a pixel that is neither red or blue"); } QCOMPARE(redCount + blueCount, texture->pixelSize().width()); QVERIFY(redCount > blueCount); // 1742 > 178 } void tst_QRhi::tessellation_data() { rhiTestData(); } void tst_QRhi::tessellation() { #ifdef Q_OS_ANDROID if (QNativeInterface::QAndroidApplication::sdkVersion() >= 31) QSKIP("Fails on Android 12 (QTBUG-108844)"); #endif QFETCH(QRhi::Implementation, impl); QFETCH(QRhiInitParams *, initParams); QScopedPointer rhi(QRhi::create(impl, initParams, QRhi::Flags(), nullptr)); if (!rhi) QSKIP("QRhi could not be created, skipping testing rendering"); if (!rhi->isFeatureSupported(QRhi::Tessellation)) { // From a Vulkan or Metal implementation we expect tessellation to work, // even though it is optional (as per spec) for Vulkan. QVERIFY(rhi->backend() != QRhi::Vulkan); QVERIFY(rhi->backend() != QRhi::Metal); QSKIP("Tessellation is not supported with this graphics API, skipping test"); } if (rhi->backend() == QRhi::D3D11) QSKIP("Skipping tessellation test on D3D for now, test assets not prepared for HLSL yet"); QScopedPointer texture(rhi->newTexture(QRhiTexture::RGBA8, QSize(1280, 720), 1, QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource)); QVERIFY(texture->create()); QScopedPointer rt(rhi->newTextureRenderTarget({ texture.data() })); QScopedPointer rpDesc(rt->newCompatibleRenderPassDescriptor()); rt->setRenderPassDescriptor(rpDesc.data()); QVERIFY(rt->create()); static const float triangleVertices[] = { 0.0f, 0.5f, 0.0f, 0.0f, 0.0f, 1.0f, -0.5f, -0.5f, 0.0f, 1.0f, 0.0f, 0.0f, 0.5f, -0.5f, 0.0f, 0.0f, 1.0f, 0.0f, }; QRhiResourceUpdateBatch *u = rhi->nextResourceUpdateBatch(); QScopedPointer vbuf(rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer, sizeof(triangleVertices))); QVERIFY(vbuf->create()); u->uploadStaticBuffer(vbuf.data(), triangleVertices); QScopedPointer ubuf(rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer, 64)); QVERIFY(ubuf->create()); // Use the 3D API specific correction matrix that flips Y, so we can use // the OpenGL-targeted vertex data and the tessellation winding order of // counter-clockwise to get uniform results. QMatrix4x4 mvp = rhi->clipSpaceCorrMatrix(); u->updateDynamicBuffer(ubuf.data(), 0, 64, mvp.constData()); QScopedPointer srb(rhi->newShaderResourceBindings()); srb->setBindings({ QRhiShaderResourceBinding::uniformBuffer(0, QRhiShaderResourceBinding::TessellationEvaluationStage, ubuf.data()), }); QVERIFY(srb->create()); QScopedPointer pipeline(rhi->newGraphicsPipeline()); pipeline->setTopology(QRhiGraphicsPipeline::Patches); pipeline->setPatchControlPointCount(3); pipeline->setShaderStages({ { QRhiShaderStage::Vertex, loadShader(":/data/simpletess.vert.qsb") }, { QRhiShaderStage::TessellationControl, loadShader(":/data/simpletess.tesc.qsb") }, { QRhiShaderStage::TessellationEvaluation, loadShader(":/data/simpletess.tese.qsb") }, { QRhiShaderStage::Fragment, loadShader(":/data/simpletess.frag.qsb") } }); pipeline->setCullMode(QRhiGraphicsPipeline::Back); // to ensure the winding order is correct // won't get the wireframe with OpenGL ES if (rhi->isFeatureSupported(QRhi::NonFillPolygonMode)) pipeline->setPolygonMode(QRhiGraphicsPipeline::Line); QRhiVertexInputLayout inputLayout; inputLayout.setBindings({ { 6 * sizeof(float) } }); inputLayout.setAttributes({ { 0, 0, QRhiVertexInputAttribute::Float3, 0 }, { 0, 1, QRhiVertexInputAttribute::Float3, 3 * sizeof(float) } }); pipeline->setVertexInputLayout(inputLayout); pipeline->setShaderResourceBindings(srb.data()); pipeline->setRenderPassDescriptor(rpDesc.data()); QVERIFY(pipeline->create()); QRhiCommandBuffer *cb = nullptr; QCOMPARE(rhi->beginOffscreenFrame(&cb), QRhi::FrameOpSuccess); cb->beginPass(rt.data(), Qt::black, { 1.0f, 0 }, u); cb->setGraphicsPipeline(pipeline.data()); cb->setViewport({ 0, 0, float(rt->pixelSize().width()), float(rt->pixelSize().height()) }); cb->setShaderResources(); QRhiCommandBuffer::VertexInput vbufBinding(vbuf.data(), 0); cb->setVertexInput(0, 1, &vbufBinding); cb->draw(3); QRhiReadbackResult readResult; QImage result; readResult.completed = [&readResult, &result] { result = QImage(reinterpret_cast(readResult.data.constData()), readResult.pixelSize.width(), readResult.pixelSize.height(), QImage::Format_RGBA8888); }; QRhiResourceUpdateBatch *readbackBatch = rhi->nextResourceUpdateBatch(); readbackBatch->readBackTexture({ texture.data() }, &readResult); cb->endPass(readbackBatch); rhi->endOffscreenFrame(); if (rhi->isYUpInFramebuffer()) // we used clipSpaceCorrMatrix so this is different from many other tests result = std::move(result).mirrored(); QCOMPARE(result.size(), rt->pixelSize()); // cannot check rendering results with Null, because there is no rendering there if (impl == QRhi::Null) return; int redCount = 0, greenCount = 0, blueCount = 0; for (int y = 0; y < result.height(); ++y) { const quint32 *p = reinterpret_cast(result.constScanLine(y)); int x = result.width() - 1; while (x-- >= 0) { const QRgb c(*p++); const int red = qRed(c); const int green = qGreen(c); const int blue = qBlue(c); // just count the color components that are above a certain threshold if (red > 240) ++redCount; if (green > 240) ++greenCount; if (blue > 240) ++blueCount; } } // Line drawing can be different between the 3D APIs. What we will check if // the number of strong-enough r/g/b components above a certain threshold. // That is good enough to ensure that something got rendered, i.e. that // tessellation is not completely broken. // // For the record the actual values are something like: // OpenGL (NVIDIA, Windows) 59 82 82 // Metal (Intel, macOS 12.5) 59 79 79 // Vulkan (NVIDIA, Windows) 71 85 85 QVERIFY(redCount > 50); QVERIFY(blueCount > 50); QVERIFY(greenCount > 50); } #include QTEST_MAIN(tst_QRhi)