rhi: gl: Support reading back 1 byte texture formats
The GL texture readback is limited due to the underspecified glReadPixels, especially on GLES. To preserve our sanity, we just do a GL_RGBA readback always. This only worked for 4 byte formats, but now we extend it to handle the 1 byte (R8 and RED_OR_ALPHA8) formats. Note that this relies on the fact that the GL implementation is able to do a GL_RGBA readback for a GL_R8 or GL_ALPHA texture. Change-Id: I8286dca42964f0cbc6645355e105bbd81ec685ca Reviewed-by: Christian Strømme <christian.stromme@qt.io> Reviewed-by: Eirik Aavitsland <eirik.aavitsland@qt.io>
This commit is contained in:
parent
03dfd4199d
commit
9d0a15b7a3
@ -2396,13 +2396,33 @@ void QRhiGles2::executeCommandBuffer(QRhiCommandBuffer *cb)
|
|||||||
result->format = QRhiTexture::RGBA8;
|
result->format = QRhiTexture::RGBA8;
|
||||||
// readPixels handles multisample resolving implicitly
|
// readPixels handles multisample resolving implicitly
|
||||||
}
|
}
|
||||||
result->data.resize(result->pixelSize.width() * result->pixelSize.height() * 4);
|
const int w = result->pixelSize.width();
|
||||||
|
const int h = result->pixelSize.height();
|
||||||
if (mipLevel == 0 || caps.nonBaseLevelFramebufferTexture) {
|
if (mipLevel == 0 || caps.nonBaseLevelFramebufferTexture) {
|
||||||
// With GLES (2.0?) GL_RGBA is the only mandated readback format, so stick with it.
|
// With GLES, GL_RGBA is the only mandated readback format, so stick with it.
|
||||||
f->glReadPixels(0, 0, result->pixelSize.width(), result->pixelSize.height(),
|
if (result->format == QRhiTexture::R8 || result->format == QRhiTexture::RED_OR_ALPHA8) {
|
||||||
GL_RGBA, GL_UNSIGNED_BYTE,
|
result->data.resize(w * h);
|
||||||
result->data.data());
|
QByteArray tmpBuf;
|
||||||
|
tmpBuf.resize(w * h * 4);
|
||||||
|
f->glReadPixels(0, 0, w, h, GL_RGBA, GL_UNSIGNED_BYTE, tmpBuf.data());
|
||||||
|
const quint8 *srcBase = reinterpret_cast<const quint8 *>(tmpBuf.constData());
|
||||||
|
quint8 *dstBase = reinterpret_cast<quint8 *>(result->data.data());
|
||||||
|
const int componentIndex = isFeatureSupported(QRhi::RedOrAlpha8IsRed) ? 0 : 3;
|
||||||
|
for (int y = 0; y < h; ++y) {
|
||||||
|
const quint8 *src = srcBase + y * w * 4;
|
||||||
|
quint8 *dst = dstBase + y * w;
|
||||||
|
int count = w;
|
||||||
|
while (count-- > 0) {
|
||||||
|
*dst++ = src[componentIndex];
|
||||||
|
src += 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
|
result->data.resize(w * h * 4);
|
||||||
|
f->glReadPixels(0, 0, w, h, GL_RGBA, GL_UNSIGNED_BYTE, result->data.data());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
result->data.resize(w * h * 4);
|
||||||
result->data.fill('\0');
|
result->data.fill('\0');
|
||||||
}
|
}
|
||||||
if (fbo) {
|
if (fbo) {
|
||||||
|
Loading…
Reference in New Issue
Block a user