59 std::weak_ptr<QRhi> producerRhi, QPointer<QObject> producer,
60 QPointer<QOpenGLContext> producerContext)
73 if (m_mapMode != QVideoFrame::NotMapped || mode != QVideoFrame::ReadOnly)
75 m_mapMode = QVideoFrame::ReadOnly;
77 m_image = readbackOnProducerThread();
78 if (m_image.isNull()) {
79 m_mapMode = QVideoFrame::NotMapped;
83 data.bytesPerLine[0] = m_image.bytesPerLine();
84 data.dataSize[0] =
static_cast<
int>(m_image.sizeInBytes());
85 data.data[0] = m_image.bits();
92 m_mapMode = QVideoFrame::NotMapped;
102 if (!isCompatibleRhi(rhi))
105 m_tex->nativeTexture().object);
109 bool isCompatibleRhi(QRhi &rhi)
const
111 if (rhi.backend() != QRhi::OpenGLES2)
113 if (!m_producerContext)
115 const auto *handles =
116 static_cast<
const QRhiGles2NativeHandles *>(rhi.nativeHandles());
117 if (!handles || !handles->context)
119 return handles->context->shareGroup() == m_producerContext->shareGroup();
122 QImage readbackOnProducerThread()
const
124 auto producerRhi = m_producerRhi.lock();
125 if (!producerRhi || !m_producer)
128 QRhi *rhi = producerRhi.get();
129 QRhiTexture *tex = m_tex.get();
130 QMetaObject::invokeMethod(
133 QRhiReadbackResult result;
135 result.completed = [&done] { done =
true; };
136 QRhiCommandBuffer *cb =
nullptr;
137 if (rhi->beginOffscreenFrame(&cb) != QRhi::FrameOpSuccess)
139 QRhiResourceUpdateBatch *rub = rhi->nextResourceUpdateBatch();
140 rub->readBackTexture({ tex }, &result);
141 cb->resourceUpdate(rub);
142 rhi->endOffscreenFrame();
143 if (!done || result.data.isEmpty())
145 QImage img(
reinterpret_cast<
const uchar *>(result.data.constData()),
146 result.pixelSize.width(), result.pixelSize.height(),
147 result.pixelSize.width() * 4, QImage::Format_RGBA8888);
150 Qt::BlockingQueuedConnection);
155 std::unique_ptr<QRhiTexture> m_tex;
158 std::weak_ptr<QRhi> m_producerRhi;
159 QPointer<QObject> m_producer;
160 QPointer<QOpenGLContext> m_producerContext;
168 m_vertexBuffer.reset(m_rhi->newBuffer(QRhiBuffer::Immutable, QRhiBuffer::VertexBuffer,
170 m_vertexBuffer->create();
172 m_uniformBuffer.reset(m_rhi->newBuffer(QRhiBuffer::Dynamic, QRhiBuffer::UniformBuffer,
174 m_uniformBuffer->create();
176 m_sampler.reset(m_rhi->newSampler(QRhiSampler::Nearest, QRhiSampler::Nearest,
177 QRhiSampler::None, QRhiSampler::ClampToEdge,
178 QRhiSampler::ClampToEdge));
181 m_srb.reset(m_rhi->newShaderResourceBindings());
183 QRhiShaderResourceBinding::uniformBuffer(
185 QRhiShaderResourceBinding::VertexStage
186 | QRhiShaderResourceBinding::FragmentStage,
187 m_uniformBuffer.get()),
188 QRhiShaderResourceBinding::sampledTexture(
189 1, QRhiShaderResourceBinding::FragmentStage, externalTex, m_sampler.get())
193 m_vertexShader = loadShader(
194 QStringLiteral(
":/qt-project.org/multimedia/shaders/externalsampler.vert.qsb"));
195 m_fragmentShader = loadShader(
196 QStringLiteral(
":/qt-project.org/multimedia/shaders/externalsampler.frag.qsb"));
202 static QShader loadShader(
const QString &name)
205 if (f.open(QIODevice::ReadOnly))
206 return QShader::fromSerialized(f.readAll());
210 QRhi *m_rhi{
nullptr };
211 std::unique_ptr<QRhiBuffer> m_vertexBuffer;
212 std::unique_ptr<QRhiBuffer> m_uniformBuffer;
213 std::unique_ptr<QRhiSampler> m_sampler;
214 std::unique_ptr<QRhiShaderResourceBindings> m_srb;
215 QShader m_vertexShader;
216 QShader m_fragmentShader;
220 QRhiShaderResourceBindings *srb,
221 QRhiRenderPassDescriptor *rpd,
222 QShader vs, QShader fs)
224 std::unique_ptr<QRhiGraphicsPipeline> gp(rhi->newGraphicsPipeline());
225 gp->setTopology(QRhiGraphicsPipeline::TriangleFan);
226 gp->setShaderStages({
227 { QRhiShaderStage::Vertex, vs },
228 { QRhiShaderStage::Fragment, fs }
231 layout.setBindings({ { 4 *
sizeof(
float) } });
232 layout.setAttributes({
233 { 0, 0, QRhiVertexInputAttribute::Float2, 0 },
234 { 0, 1, QRhiVertexInputAttribute::Float2, 2 *
sizeof(
float) }
236 gp->setVertexInputLayout(layout);
237 gp->setShaderResourceBindings(srb);
238 gp->setRenderPassDescriptor(rpd);
246 std::unique_ptr<QRhiTexture> tex(
247 m_rhi->newTexture(QRhiTexture::RGBA8, size, 1, QRhiTexture::RenderTarget));
248 if (!tex->create()) {
249 qCWarning(qLcOhosMediaPlugin) <<
"Failed to create frame texture";
253 std::unique_ptr<QRhiTextureRenderTarget> renderTarget(
254 m_rhi->newTextureRenderTarget({ { tex.get() } }));
255 std::unique_ptr<QRhiRenderPassDescriptor> rpd(
256 renderTarget->newCompatibleRenderPassDescriptor());
257 renderTarget->setRenderPassDescriptor(rpd.get());
258 renderTarget->create();
260 QRhiResourceUpdateBatch *rub = m_rhi->nextResourceUpdateBatch();
261 rub->uploadStaticBuffer(m_vertexBuffer.get(),
g_quad);
263 const QMatrix4x4 identity;
264 char *p = m_uniformBuffer->beginFullDynamicBufferUpdateForCurrentFrame();
265 memcpy(p, identity.constData(), 64);
266 memcpy(p + 64, externalTexMatrix.constData(), 64);
267 const float opacity = 1.0f;
268 memcpy(p + 64 + 64, &opacity, 4);
269 m_uniformBuffer->endFullDynamicBufferUpdateForCurrentFrame();
271 auto pipeline = newGraphicsPipeline(m_rhi, m_srb.get(), rpd.get(), m_vertexShader,
274 const QRhiCommandBuffer::VertexInput vbufBinding(m_vertexBuffer.get(), 0);
275 QRhiCommandBuffer *cb =
nullptr;
276 if (m_rhi->beginOffscreenFrame(&cb) != QRhi::FrameOpSuccess)
279 cb->beginPass(renderTarget.get(), Qt::transparent, { 1.0f, 0 }, rub);
280 cb->setGraphicsPipeline(pipeline.get());
281 cb->setViewport({ 0, 0,
float(size.width()),
float(size.height()) });
282 cb->setShaderResources(m_srb.get());
283 cb->setVertexInput(0, 1, &vbufBinding);
286 m_rhi->endOffscreenFrame();
359 0.0f, -1.0f, 0.0f, 1.0f,
360 0.0f, 0.0f, 1.0f, 0.0f,
361 0.0f, 0.0f, 0.0f, 1.0f);
383 OHNativeWindow *ensureSurface(QRhi *rhi)
388 const bool reuse = m_surfaceImage && m_surfaceImage->isValid()
389 && (rhi ? m_rhi.get() == rhi : m_isHeadless);
391 return m_surfaceImage->nativeWindow();
397 QRhiGles2InitParams params;
398 const auto *nativeHandles =
399 rhi ?
static_cast<
const QRhiGles2NativeHandles *>(rhi->nativeHandles())
401 params.shareContext = nativeHandles ? nativeHandles->context :
nullptr;
402 params.fallbackSurface = QRhiGles2InitParams::newFallbackSurface();
403 m_isHeadless = (rhi ==
nullptr);
404 m_rhi.reset(QRhi::create(QRhi::OpenGLES2, ¶ms));
406 qCWarning(qLcOhosMediaPlugin) <<
"Failed to create offscreen GLES2 RHI";
410 m_externalTexture.reset(
411 m_rhi->newTexture(QRhiTexture::RGBA8, m_size.isEmpty() ? QSize{ 1, 1 } : m_size, 1,
412 QRhiTexture::ExternalOES));
413 if (!m_externalTexture->create()) {
414 qCWarning(qLcOhosMediaPlugin) <<
"External OES texture create failed";
415 m_externalTexture.reset();
420 const auto nativeTex = m_externalTexture->nativeTexture();
421 m_surfaceImage = std::make_unique<QOhosSurfaceImage>(uint32_t(nativeTex.object));
422 if (!m_surfaceImage->isValid()) {
423 m_surfaceImage.reset();
424 m_externalTexture.reset();
429 const quint64 index = m_surfaceImage->index();
430 connect(m_surfaceImage.get(), &QOhosSurfaceImage::frameAvailable,
this,
431 [
this, index]() { onFrameAvailable(index); }, Qt::QueuedConnection);
433 m_textureCopy = std::make_unique<TextureCopy>(m_rhi.get(), m_externalTexture.get());
434 return m_surfaceImage->nativeWindow();
437 std::shared_ptr<QRhi> m_rhi;
438 std::unique_ptr<QRhiTexture> m_externalTexture;
439 std::unique_ptr<QOhosSurfaceImage> m_surfaceImage;
440 std::unique_ptr<TextureCopy> m_textureCopy;
441 QSize m_size{ 1, 1 };
442 bool m_isHeadless{
false };
446 : QObject(parent), m_sink(sink)
448 m_textureThread = std::make_shared<QOhosTextureThread>();
449 connect(m_textureThread.get(), &QOhosTextureThread::newFrame,
this,
450 &QOhosVideoOutput::onNewFrame, Qt::QueuedConnection);
451 m_textureThread->launch();
453 if (
auto *p = sink ? sink->platformVideoSink() :
nullptr) {
454 connect(p, &QPlatformVideoSink::rhiChanged,
this, &
QOhosVideoOutput::onRhiChanged);
478 auto *rhi = m_sink ? m_sink->rhi() :
nullptr;
480 m_surfaceCreatedWithoutRhi =
true;
481 }
else if (m_surfaceCreatedWithoutRhi) {
482 QMetaObject::invokeMethod(m_textureThread.get(), &QOhosTextureThread::tearDown,
483 Qt::BlockingQueuedConnection);
484 m_surfaceCreatedWithoutRhi =
false;
486 return m_textureThread->nativeWindowBlocking(rhi);
495 auto *rhi = m_sink ? m_sink->rhi() :
nullptr;
497 m_surfaceCreatedWithoutRhi =
true;
498 else if (m_surfaceCreatedWithoutRhi) {
499 QMetaObject::invokeMethod(m_textureThread.get(), &QOhosTextureThread::tearDown,
500 Qt::BlockingQueuedConnection);
501 m_surfaceCreatedWithoutRhi =
false;
503 return m_textureThread->surfaceIdBlocking(rhi);
QOhosTextureVideoBuffer(std::unique_ptr< QRhiTexture > tex, const QSize &size, std::weak_ptr< QRhi > producerRhi, QPointer< QObject > producer, QPointer< QOpenGLContext > producerContext)
void unmap() override
Releases the memory mapped by the map() function.
QVideoFrameTexturesUPtr mapTextures(QRhi &rhi, QVideoFrameTexturesUPtr &) override
MapData map(QVideoFrame::MapMode mode) override
Maps the planes of a video buffer to memory.
QRhiTexture * texture(uint plane) const override
QOhosVideoFrameTextures(QRhi *rhi, QSize size, quint64 handle)
std::unique_ptr< QRhiTexture > copyExternalTexture(QSize size, const QMatrix4x4 &externalTexMatrix)
TextureCopy(QRhi *rhi, QRhiTexture *externalTex)