Use temporary FBO for GrGLGpu::onReadPixels.

Allow configs to be supported as FBO attachments for copies/readbacks without being "renderable" elsewhere in Ganesh.

The motivation for this is to add support for int textures as srcs but not as dsts (at least initially) but to still be able to read them back. This means we don't pay for a unneeded GPU copy when reading back a GrSurface that is a GrTexture and not a GrRenderTarget.

GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=4345

Change-Id: I824a73c6b8c1b9634206d76ccf0848d5f9b64441
Reviewed-on: https://skia-review.googlesource.com/4345
Commit-Queue: Brian Salomon <bsalomon@google.com>
Reviewed-by: Robert Phillips <robertphillips@google.com>
This commit is contained in:
Brian Salomon 2016-11-03 13:42:00 -04:00 committed by Skia Commit-Bot
parent 89eda8a5e3
commit 71d9d84d6c
5 changed files with 101 additions and 87 deletions

View File

@ -852,19 +852,19 @@ bool GrGLCaps::hasPathRenderingSupport(const GrGLContextInfo& ctxInfo, const GrG
return true; return true;
} }
bool GrGLCaps::readPixelsSupported(GrPixelConfig rtConfig, bool GrGLCaps::readPixelsSupported(GrPixelConfig surfaceConfig,
GrPixelConfig readConfig, GrPixelConfig readConfig,
std::function<void (GrGLenum, GrGLint*)> getIntegerv, std::function<void (GrGLenum, GrGLint*)> getIntegerv,
std::function<bool ()> bindRenderTarget) const { std::function<bool ()> bindRenderTarget) const {
// If it's not possible to even have a render target of rtConfig then read pixels is // If it's not possible to even have a color attachment of surfaceConfig then read pixels is
// not supported regardless of readConfig. // not supported regardless of readConfig.
if (!this->isConfigRenderable(rtConfig, false)) { if (!this->canConfigBeFBOColorAttachment(surfaceConfig)) {
return false; return false;
} }
GrGLenum readFormat; GrGLenum readFormat;
GrGLenum readType; GrGLenum readType;
if (!this->getReadPixelsFormat(rtConfig, readConfig, &readFormat, &readType)) { if (!this->getReadPixelsFormat(surfaceConfig, readConfig, &readFormat, &readType)) {
return false; return false;
} }
@ -890,20 +890,20 @@ bool GrGLCaps::readPixelsSupported(GrPixelConfig rtConfig,
// See Section 16.1.2 in the ES 3.2 specification. // See Section 16.1.2 in the ES 3.2 specification.
if (kNormalizedFixedPoint_FormatType == fConfigTable[rtConfig].fFormatType) { if (kNormalizedFixedPoint_FormatType == fConfigTable[surfaceConfig].fFormatType) {
if (GR_GL_RGBA == readFormat && GR_GL_UNSIGNED_BYTE == readType) { if (GR_GL_RGBA == readFormat && GR_GL_UNSIGNED_BYTE == readType) {
return true; return true;
} }
} else { } else {
SkASSERT(kFloat_FormatType == fConfigTable[rtConfig].fFormatType); SkASSERT(kFloat_FormatType == fConfigTable[surfaceConfig].fFormatType);
if (GR_GL_RGBA == readFormat && GR_GL_FLOAT == readType) { if (GR_GL_RGBA == readFormat && GR_GL_FLOAT == readType) {
return true; return true;
} }
} }
if (0 == fConfigTable[rtConfig].fSecondReadPixelsFormat.fFormat) { if (0 == fConfigTable[surfaceConfig].fSecondReadPixelsFormat.fFormat) {
ReadPixelsFormat* rpFormat = ReadPixelsFormat* rpFormat =
const_cast<ReadPixelsFormat*>(&fConfigTable[rtConfig].fSecondReadPixelsFormat); const_cast<ReadPixelsFormat*>(&fConfigTable[surfaceConfig].fSecondReadPixelsFormat);
GrGLint format = 0, type = 0; GrGLint format = 0, type = 0;
if (!bindRenderTarget()) { if (!bindRenderTarget()) {
return false; return false;
@ -914,8 +914,8 @@ bool GrGLCaps::readPixelsSupported(GrPixelConfig rtConfig,
rpFormat->fType = type; rpFormat->fType = type;
} }
return fConfigTable[rtConfig].fSecondReadPixelsFormat.fFormat == readFormat && return fConfigTable[surfaceConfig].fSecondReadPixelsFormat.fFormat == readFormat &&
fConfigTable[rtConfig].fSecondReadPixelsFormat.fType == readType; fConfigTable[surfaceConfig].fSecondReadPixelsFormat.fType == readType;
} }
void GrGLCaps::initFSAASupport(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli) { void GrGLCaps::initFSAASupport(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli) {
@ -1400,11 +1400,12 @@ void GrGLCaps::initConfigTable(const GrGLContextInfo& ctxInfo, const GrGLInterfa
ES 3.2 ES 3.2
Adds R16F, RG16F, RGBA16F, R32F, RG32F, RGBA32F, R11F_G11F_B10F. Adds R16F, RG16F, RGBA16F, R32F, RG32F, RGBA32F, R11F_G11F_B10F.
*/ */
uint32_t allRenderFlags = ConfigInfo::kRenderable_Flag; uint32_t nonMSAARenderFlags = ConfigInfo::kRenderable_Flag |
ConfigInfo::kFBOColorAttachment_Flag;
uint32_t allRenderFlags = nonMSAARenderFlags;
if (kNone_MSFBOType != fMSFBOType) { if (kNone_MSFBOType != fMSFBOType) {
allRenderFlags |= ConfigInfo::kRenderableWithMSAA_Flag; allRenderFlags |= ConfigInfo::kRenderableWithMSAA_Flag;
} }
GrGLStandard standard = ctxInfo.standard(); GrGLStandard standard = ctxInfo.standard();
GrGLVersion version = ctxInfo.version(); GrGLVersion version = ctxInfo.version();
@ -1487,7 +1488,7 @@ void GrGLCaps::initConfigTable(const GrGLContextInfo& ctxInfo, const GrGLInterfa
} }
} else if (ctxInfo.hasExtension("GL_EXT_texture_format_BGRA8888")) { } else if (ctxInfo.hasExtension("GL_EXT_texture_format_BGRA8888")) {
fConfigTable[kBGRA_8888_GrPixelConfig].fFlags = ConfigInfo::kTextureable_Flag | fConfigTable[kBGRA_8888_GrPixelConfig].fFlags = ConfigInfo::kTextureable_Flag |
ConfigInfo::kRenderable_Flag; nonMSAARenderFlags;
if (ctxInfo.hasExtension("GL_CHROMIUM_renderbuffer_format_BGRA8888") && if (ctxInfo.hasExtension("GL_CHROMIUM_renderbuffer_format_BGRA8888") &&
(this->usesMSAARenderBuffers() || this->fMSFBOType == kMixedSamples_MSFBOType)) { (this->usesMSAARenderBuffers() || this->fMSFBOType == kMixedSamples_MSFBOType)) {
fConfigTable[kBGRA_8888_GrPixelConfig].fFlags |= fConfigTable[kBGRA_8888_GrPixelConfig].fFlags |=
@ -1652,8 +1653,7 @@ void GrGLCaps::initConfigTable(const GrGLContextInfo& ctxInfo, const GrGLInterfa
bool hasFPTextures = false; bool hasFPTextures = false;
bool hasHalfFPTextures = false; bool hasHalfFPTextures = false;
// for now we don't support floating point MSAA on ES // for now we don't support floating point MSAA on ES
uint32_t fpRenderFlags = (kGL_GrGLStandard == standard) ? uint32_t fpRenderFlags = (kGL_GrGLStandard == standard) ? allRenderFlags : nonMSAARenderFlags;
allRenderFlags : (uint32_t)ConfigInfo::kRenderable_Flag;
if (kGL_GrGLStandard == standard) { if (kGL_GrGLStandard == standard) {
if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_float")) { if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_float")) {
@ -1946,6 +1946,10 @@ void GrGLCaps::initConfigTable(const GrGLContextInfo& ctxInfo, const GrGLInterfa
// Make sure we initialized everything. // Make sure we initialized everything.
ConfigInfo defaultEntry; ConfigInfo defaultEntry;
for (int i = 0; i < kGrPixelConfigCnt; ++i) { for (int i = 0; i < kGrPixelConfigCnt; ++i) {
// Make sure we didn't set renderable and not blittable or renderable with msaa and not
// renderable.
SkASSERT(!((ConfigInfo::kRenderable_Flag) && !(ConfigInfo::kFBOColorAttachment_Flag)));
SkASSERT(!((ConfigInfo::kRenderableWithMSAA_Flag) && !(ConfigInfo::kRenderable_Flag)));
SkASSERT(defaultEntry.fFormats.fBaseInternalFormat != SkASSERT(defaultEntry.fFormats.fBaseInternalFormat !=
fConfigTable[i].fFormats.fBaseInternalFormat); fConfigTable[i].fFormats.fBaseInternalFormat);
SkASSERT(defaultEntry.fFormats.fSizedInternalFormat != SkASSERT(defaultEntry.fFormats.fSizedInternalFormat !=

View File

@ -127,6 +127,10 @@ public:
} }
} }
bool canConfigBeFBOColorAttachment(GrPixelConfig config) const {
return SkToBool(fConfigTable[config].fFlags & ConfigInfo::kFBOColorAttachment_Flag);
}
bool isConfigTexSupportEnabled(GrPixelConfig config) const { bool isConfigTexSupportEnabled(GrPixelConfig config) const {
return SkToBool(fConfigTable[config].fFlags & ConfigInfo::kCanUseTexStorage_Flag); return SkToBool(fConfigTable[config].fFlags & ConfigInfo::kCanUseTexStorage_Flag);
} }
@ -312,8 +316,8 @@ public:
/// Use indices or vertices in CPU arrays rather than VBOs for dynamic content. /// Use indices or vertices in CPU arrays rather than VBOs for dynamic content.
bool useNonVBOVertexAndIndexDynamicData() const { return fUseNonVBOVertexAndIndexDynamicData; } bool useNonVBOVertexAndIndexDynamicData() const { return fUseNonVBOVertexAndIndexDynamicData; }
/// Does ReadPixels support reading readConfig pixels from a FBO that is renderTargetConfig? /// Does ReadPixels support reading readConfig pixels from a FBO that is surfaceConfig?
bool readPixelsSupported(GrPixelConfig renderTargetConfig, bool readPixelsSupported(GrPixelConfig surfaceConfig,
GrPixelConfig readConfig, GrPixelConfig readConfig,
std::function<void (GrGLenum, GrGLint*)> getIntegerv, std::function<void (GrGLenum, GrGLint*)> getIntegerv,
std::function<bool ()> bindRenderTarget) const; std::function<bool ()> bindRenderTarget) const;
@ -478,8 +482,11 @@ private:
kTextureable_Flag = 0x2, kTextureable_Flag = 0x2,
kRenderable_Flag = 0x4, kRenderable_Flag = 0x4,
kRenderableWithMSAA_Flag = 0x8, kRenderableWithMSAA_Flag = 0x8,
kCanUseTexStorage_Flag = 0x10, /** kFBOColorAttachment means that even if the config cannot be a GrRenderTarget, we can
kCanUseWithTexelBuffer_Flag = 0x20, still attach it to a FBO for blitting or reading pixels. */
kFBOColorAttachment_Flag = 0x10,
kCanUseTexStorage_Flag = 0x20,
kCanUseWithTexelBuffer_Flag = 0x40,
}; };
uint32_t fFlags; uint32_t fFlags;

View File

@ -2261,11 +2261,11 @@ void GrGLGpu::clearStencilClip(const GrFixedClip& clip,
fHWStencilSettings.invalidate(); fHWStencilSettings.invalidate();
} }
static bool read_pixels_pays_for_y_flip(GrRenderTarget* renderTarget, const GrGLCaps& caps, static bool read_pixels_pays_for_y_flip(GrSurfaceOrigin origin, const GrGLCaps& caps,
int width, int height, GrPixelConfig config, int width, int height, GrPixelConfig config,
size_t rowBytes) { size_t rowBytes) {
// If this render target is already TopLeft, we don't need to flip. // If the surface is already TopLeft, we don't need to flip.
if (kTopLeft_GrSurfaceOrigin == renderTarget->origin()) { if (kTopLeft_GrSurfaceOrigin == origin) {
return false; return false;
} }
@ -2372,12 +2372,6 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
return true; return true;
} }
GrRenderTarget* srcAsRT = srcSurface->asRenderTarget();
if (!srcAsRT) {
// For now keep assuming the draw is not a format transformation, just a draw to get to a
// RT. We may add additional transformations below.
ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference);
}
if (this->glCaps().rgba8888PixelsOpsAreSlow() && kRGBA_8888_GrPixelConfig == readConfig && if (this->glCaps().rgba8888PixelsOpsAreSlow() && kRGBA_8888_GrPixelConfig == readConfig &&
this->readPixelsSupported(kBGRA_8888_GrPixelConfig, kBGRA_8888_GrPixelConfig)) { this->readPixelsSupported(kBGRA_8888_GrPixelConfig, kBGRA_8888_GrPixelConfig)) {
tempDrawInfo->fTempSurfaceDesc.fConfig = kBGRA_8888_GrPixelConfig; tempDrawInfo->fTempSurfaceDesc.fConfig = kBGRA_8888_GrPixelConfig;
@ -2396,7 +2390,7 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
ElevateDrawPreference(drawPreference, kGpuPrefersDraw_DrawPreference); ElevateDrawPreference(drawPreference, kGpuPrefersDraw_DrawPreference);
} else if (!this->readPixelsSupported(srcSurface, readConfig)) { } else if (!this->readPixelsSupported(srcSurface, readConfig)) {
if (readConfig == kBGRA_8888_GrPixelConfig && if (readConfig == kBGRA_8888_GrPixelConfig &&
this->glCaps().isConfigRenderable(kRGBA_8888_GrPixelConfig, false) && this->glCaps().canConfigBeFBOColorAttachment(kRGBA_8888_GrPixelConfig) &&
this->readPixelsSupported(kRGBA_8888_GrPixelConfig, kRGBA_8888_GrPixelConfig)) { this->readPixelsSupported(kRGBA_8888_GrPixelConfig, kRGBA_8888_GrPixelConfig)) {
// We're trying to read BGRA but it's not supported. If RGBA is renderable and // We're trying to read BGRA but it's not supported. If RGBA is renderable and
// we can read it back, then do a swizzling draw to a RGBA and read it back (which // we can read it back, then do a swizzling draw to a RGBA and read it back (which
@ -2406,7 +2400,7 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
tempDrawInfo->fReadConfig = kRGBA_8888_GrPixelConfig; tempDrawInfo->fReadConfig = kRGBA_8888_GrPixelConfig;
ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference); ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference);
} else if (readConfig == kSBGRA_8888_GrPixelConfig && } else if (readConfig == kSBGRA_8888_GrPixelConfig &&
this->glCaps().isConfigRenderable(kSRGBA_8888_GrPixelConfig, false) && this->glCaps().canConfigBeFBOColorAttachment(kSRGBA_8888_GrPixelConfig) &&
this->readPixelsSupported(kSRGBA_8888_GrPixelConfig, kSRGBA_8888_GrPixelConfig)) { this->readPixelsSupported(kSRGBA_8888_GrPixelConfig, kSRGBA_8888_GrPixelConfig)) {
// We're trying to read sBGRA but it's not supported. If sRGBA is renderable and // We're trying to read sBGRA but it's not supported. If sRGBA is renderable and
// we can read it back, then do a swizzling draw to a sRGBA and read it back (which // we can read it back, then do a swizzling draw to a sRGBA and read it back (which
@ -2426,7 +2420,7 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
if (!this->readPixelsSupported(srcSurface, cpuTempConfig)) { if (!this->readPixelsSupported(srcSurface, cpuTempConfig)) {
// If we can't read RGBA from the src try to draw to a kRGBA_8888 (or kSRGBA_8888) // If we can't read RGBA from the src try to draw to a kRGBA_8888 (or kSRGBA_8888)
// first and then onReadPixels will read that to a 32bit temporary buffer. // first and then onReadPixels will read that to a 32bit temporary buffer.
if (this->caps()->isConfigRenderable(cpuTempConfig, false)) { if (this->glCaps().canConfigBeFBOColorAttachment(cpuTempConfig)) {
ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference); ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference);
tempDrawInfo->fTempSurfaceDesc.fConfig = cpuTempConfig; tempDrawInfo->fTempSurfaceDesc.fConfig = cpuTempConfig;
tempDrawInfo->fReadConfig = kAlpha_8_GrPixelConfig; tempDrawInfo->fReadConfig = kAlpha_8_GrPixelConfig;
@ -2437,7 +2431,7 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
SkASSERT(tempDrawInfo->fTempSurfaceDesc.fConfig == srcConfig); SkASSERT(tempDrawInfo->fTempSurfaceDesc.fConfig == srcConfig);
SkASSERT(tempDrawInfo->fReadConfig == kAlpha_8_GrPixelConfig); SkASSERT(tempDrawInfo->fReadConfig == kAlpha_8_GrPixelConfig);
} }
} else if (this->caps()->isConfigRenderable(readConfig, false) && } else if (this->glCaps().canConfigBeFBOColorAttachment(readConfig) &&
this->readPixelsSupported(readConfig, readConfig)) { this->readPixelsSupported(readConfig, readConfig)) {
// Do a draw to convert from the src config to the read config. // Do a draw to convert from the src config to the read config.
ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference); ElevateDrawPreference(drawPreference, kRequireDraw_DrawPreference);
@ -2448,8 +2442,9 @@ bool GrGLGpu::onGetReadPixelsInfo(GrSurface* srcSurface, int width, int height,
} }
} }
if (srcAsRT && if ((srcSurface->asRenderTarget() || this->glCaps().canConfigBeFBOColorAttachment(srcConfig)) &&
read_pixels_pays_for_y_flip(srcAsRT, this->glCaps(), width, height, readConfig, rowBytes)) { read_pixels_pays_for_y_flip(srcSurface->origin(), this->glCaps(), width, height, readConfig,
rowBytes)) {
ElevateDrawPreference(drawPreference, kGpuPrefersDraw_DrawPreference); ElevateDrawPreference(drawPreference, kGpuPrefersDraw_DrawPreference);
} }
@ -2465,7 +2460,7 @@ bool GrGLGpu::onReadPixels(GrSurface* surface,
SkASSERT(surface); SkASSERT(surface);
GrGLRenderTarget* renderTarget = static_cast<GrGLRenderTarget*>(surface->asRenderTarget()); GrGLRenderTarget* renderTarget = static_cast<GrGLRenderTarget*>(surface->asRenderTarget());
if (!renderTarget) { if (!renderTarget && !this->glCaps().canConfigBeFBOColorAttachment(surface->config())) {
return false; return false;
} }
@ -2476,16 +2471,16 @@ bool GrGLGpu::onReadPixels(GrSurface* surface,
// We have a special case fallback for reading eight bit alpha. We will read back all four 8 // We have a special case fallback for reading eight bit alpha. We will read back all four 8
// bit channels as RGBA and then extract A. // bit channels as RGBA and then extract A.
if (!this->readPixelsSupported(renderTarget, config)) { if (!this->readPixelsSupported(surface, config)) {
// Don't attempt to do any srgb conversions since we only care about alpha. // Don't attempt to do any srgb conversions since we only care about alpha.
GrPixelConfig tempConfig = kRGBA_8888_GrPixelConfig; GrPixelConfig tempConfig = kRGBA_8888_GrPixelConfig;
if (GrPixelConfigIsSRGB(renderTarget->config())) { if (GrPixelConfigIsSRGB(surface->config())) {
tempConfig = kSRGBA_8888_GrPixelConfig; tempConfig = kSRGBA_8888_GrPixelConfig;
} }
if (kAlpha_8_GrPixelConfig == config && if (kAlpha_8_GrPixelConfig == config &&
this->readPixelsSupported(renderTarget, tempConfig)) { this->readPixelsSupported(surface, tempConfig)) {
std::unique_ptr<uint32_t[]> temp(new uint32_t[width * height * 4]); std::unique_ptr<uint32_t[]> temp(new uint32_t[width * height * 4]);
if (this->onReadPixels(renderTarget, left, top, width, height, tempConfig, temp.get(), if (this->onReadPixels(surface, left, top, width, height, tempConfig, temp.get(),
width*4)) { width*4)) {
uint8_t* dst = reinterpret_cast<uint8_t*>(buffer); uint8_t* dst = reinterpret_cast<uint8_t*>(buffer);
for (int j = 0; j < height; ++j) { for (int j = 0; j < height; ++j) {
@ -2501,34 +2496,40 @@ bool GrGLGpu::onReadPixels(GrSurface* surface,
GrGLenum externalFormat; GrGLenum externalFormat;
GrGLenum externalType; GrGLenum externalType;
if (!this->glCaps().getReadPixelsFormat(renderTarget->config(), config, &externalFormat, if (!this->glCaps().getReadPixelsFormat(surface->config(), config, &externalFormat,
&externalType)) { &externalType)) {
return false; return false;
} }
bool flipY = kBottomLeft_GrSurfaceOrigin == surface->origin(); bool flipY = kBottomLeft_GrSurfaceOrigin == surface->origin();
// resolve the render target if necessary GrGLIRect glvp;
switch (renderTarget->getResolveType()) { if (renderTarget) {
case GrGLRenderTarget::kCantResolve_ResolveType: // resolve the render target if necessary
return false; switch (renderTarget->getResolveType()) {
case GrGLRenderTarget::kAutoResolves_ResolveType: case GrGLRenderTarget::kCantResolve_ResolveType:
this->flushRenderTarget(renderTarget, &SkIRect::EmptyIRect()); return false;
break; case GrGLRenderTarget::kAutoResolves_ResolveType:
case GrGLRenderTarget::kCanResolve_ResolveType: this->flushRenderTarget(renderTarget, &SkIRect::EmptyIRect());
this->onResolveRenderTarget(renderTarget); break;
// we don't track the state of the READ FBO ID. case GrGLRenderTarget::kCanResolve_ResolveType:
fStats.incRenderTargetBinds(); this->onResolveRenderTarget(renderTarget);
GL_CALL(BindFramebuffer(GR_GL_READ_FRAMEBUFFER, renderTarget->textureFBOID())); // we don't track the state of the READ FBO ID.
break; fStats.incRenderTargetBinds();
default: GL_CALL(BindFramebuffer(GR_GL_READ_FRAMEBUFFER, renderTarget->textureFBOID()));
SkFAIL("Unknown resolve type"); break;
default:
SkFAIL("Unknown resolve type");
}
glvp = renderTarget->getViewport();
} else {
// Use a temporary FBO.
this->bindSurfaceFBOForPixelOps(surface, GR_GL_FRAMEBUFFER, &glvp, kSrc_TempFBOTarget);
fHWBoundRenderTargetUniqueID = SK_InvalidUniqueID;
} }
const GrGLIRect& glvp = renderTarget->getViewport();
// the read rect is viewport-relative // the read rect is viewport-relative
GrGLIRect readRect; GrGLIRect readRect;
readRect.setRelativeTo(glvp, left, top, width, height, renderTarget->origin()); readRect.setRelativeTo(glvp, left, top, width, height, surface->origin());
size_t bytesPerPixel = GrBytesPerPixel(config); size_t bytesPerPixel = GrBytesPerPixel(config);
size_t tightRowBytes = bytesPerPixel * width; size_t tightRowBytes = bytesPerPixel * width;
@ -2606,6 +2607,9 @@ bool GrGLGpu::onReadPixels(GrSurface* surface,
} }
} }
} }
if (!renderTarget) {
this->unbindTextureFBOForPixelOps(GR_GL_FRAMEBUFFER, surface);
}
return true; return true;
} }
@ -3459,8 +3463,8 @@ static inline bool can_blit_framebuffer_for_copy_surface(const GrSurface* dst,
const SkIPoint& dstPoint, const SkIPoint& dstPoint,
const GrGLGpu* gpu) { const GrGLGpu* gpu) {
auto blitFramebufferFlags = gpu->glCaps().blitFramebufferSupportFlags(); auto blitFramebufferFlags = gpu->glCaps().blitFramebufferSupportFlags();
if (!gpu->glCaps().isConfigRenderable(dst->config(), dst->desc().fSampleCnt > 0) || if (!gpu->glCaps().canConfigBeFBOColorAttachment(dst->config()) ||
!gpu->glCaps().isConfigRenderable(src->config(), src->desc().fSampleCnt > 0)) { !gpu->glCaps().canConfigBeFBOColorAttachment(src->config())) {
return false; return false;
} }
const GrGLTexture* dstTex = static_cast<const GrGLTexture*>(dst->asTexture()); const GrGLTexture* dstTex = static_cast<const GrGLTexture*>(dst->asTexture());
@ -3546,10 +3550,9 @@ static inline bool can_copy_texsubimage(const GrSurface* dst,
// Check that we could wrap the source in an FBO, that the dst is TEXTURE_2D, that no mirroring // Check that we could wrap the source in an FBO, that the dst is TEXTURE_2D, that no mirroring
// is required. // is required.
if (gpu->glCaps().isConfigRenderable(src->config(), src->desc().fSampleCnt > 0) && if (gpu->glCaps().canConfigBeFBOColorAttachment(src->config()) &&
!GrPixelConfigIsCompressed(src->config()) && !GrPixelConfigIsCompressed(src->config()) &&
(!srcTex || srcTex->target() == GR_GL_TEXTURE_2D) && (!srcTex || srcTex->target() == GR_GL_TEXTURE_2D) && dstTex->target() == GR_GL_TEXTURE_2D &&
dstTex->target() == GR_GL_TEXTURE_2D &&
dst->origin() == src->origin()) { dst->origin() == src->origin()) {
return true; return true;
} else { } else {
@ -3559,8 +3562,8 @@ static inline bool can_copy_texsubimage(const GrSurface* dst,
// If a temporary FBO was created, its non-zero ID is returned. The viewport that the copy rect is // If a temporary FBO was created, its non-zero ID is returned. The viewport that the copy rect is
// relative to is output. // relative to is output.
void GrGLGpu::bindSurfaceFBOForCopy(GrSurface* surface, GrGLenum fboTarget, GrGLIRect* viewport, void GrGLGpu::bindSurfaceFBOForPixelOps(GrSurface* surface, GrGLenum fboTarget, GrGLIRect* viewport,
TempFBOTarget tempFBOTarget) { TempFBOTarget tempFBOTarget) {
GrGLRenderTarget* rt = static_cast<GrGLRenderTarget*>(surface->asRenderTarget()); GrGLRenderTarget* rt = static_cast<GrGLRenderTarget*>(surface->asRenderTarget());
if (!rt) { if (!rt) {
SkASSERT(surface->asTexture()); SkASSERT(surface->asTexture());
@ -3591,8 +3594,8 @@ void GrGLGpu::bindSurfaceFBOForCopy(GrSurface* surface, GrGLenum fboTarget, GrGL
} }
} }
void GrGLGpu::unbindTextureFBOForCopy(GrGLenum fboTarget, GrSurface* surface) { void GrGLGpu::unbindTextureFBOForPixelOps(GrGLenum fboTarget, GrSurface* surface) {
// bindSurfaceFBOForCopy temporarily binds textures that are not render targets to // bindSurfaceFBOForPixelOps temporarily binds textures that are not render targets to
if (!surface->asRenderTarget()) { if (!surface->asRenderTarget()) {
SkASSERT(surface->asTexture()); SkASSERT(surface->asTexture());
GrGLenum textureTarget = static_cast<GrGLTexture*>(surface->asTexture())->target(); GrGLenum textureTarget = static_cast<GrGLTexture*>(surface->asTexture())->target();
@ -3636,9 +3639,8 @@ bool GrGLGpu::initDescForDstCopy(const GrRenderTarget* src, GrSurfaceDesc* desc)
kBGRA_8888_GrPixelConfig == src->config()) { kBGRA_8888_GrPixelConfig == src->config()) {
// glCopyTexSubImage2D doesn't work with this config. If the bgra can be used with fbo blit // glCopyTexSubImage2D doesn't work with this config. If the bgra can be used with fbo blit
// then we set up for that, otherwise fail. // then we set up for that, otherwise fail.
if (this->caps()->isConfigRenderable(kBGRA_8888_GrPixelConfig, false)) { if (this->glCaps().canConfigBeFBOColorAttachment(kBGRA_8888_GrPixelConfig)) {
desc->fOrigin = originForBlitFramebuffer; desc->fOrigin = originForBlitFramebuffer;
desc->fFlags = kRenderTarget_GrSurfaceFlag;
desc->fConfig = kBGRA_8888_GrPixelConfig; desc->fConfig = kBGRA_8888_GrPixelConfig;
return true; return true;
} }
@ -3649,9 +3651,8 @@ bool GrGLGpu::initDescForDstCopy(const GrRenderTarget* src, GrSurfaceDesc* desc)
if (srcRT->renderFBOID() != srcRT->textureFBOID()) { if (srcRT->renderFBOID() != srcRT->textureFBOID()) {
// It's illegal to call CopyTexSubImage2D on a MSAA renderbuffer. Set up for FBO blit or // It's illegal to call CopyTexSubImage2D on a MSAA renderbuffer. Set up for FBO blit or
// fail. // fail.
if (this->caps()->isConfigRenderable(src->config(), false)) { if (this->glCaps().canConfigBeFBOColorAttachment(src->config())) {
desc->fOrigin = originForBlitFramebuffer; desc->fOrigin = originForBlitFramebuffer;
desc->fFlags = kRenderTarget_GrSurfaceFlag;
desc->fConfig = src->config(); desc->fConfig = src->config();
return true; return true;
} }
@ -4164,7 +4165,7 @@ bool GrGLGpu::copySurfaceAsDraw(GrSurface* dst,
this->bindTexture(0, params, true, srcTex); this->bindTexture(0, params, true, srcTex);
GrGLIRect dstVP; GrGLIRect dstVP;
this->bindSurfaceFBOForCopy(dst, GR_GL_FRAMEBUFFER, &dstVP, kDst_TempFBOTarget); this->bindSurfaceFBOForPixelOps(dst, GR_GL_FRAMEBUFFER, &dstVP, kDst_TempFBOTarget);
this->flushViewport(dstVP); this->flushViewport(dstVP);
fHWBoundRenderTargetUniqueID = SK_InvalidUniqueID; fHWBoundRenderTargetUniqueID = SK_InvalidUniqueID;
@ -4227,7 +4228,7 @@ bool GrGLGpu::copySurfaceAsDraw(GrSurface* dst,
this->disableStencil(); this->disableStencil();
GL_CALL(DrawArrays(GR_GL_TRIANGLE_STRIP, 0, 4)); GL_CALL(DrawArrays(GR_GL_TRIANGLE_STRIP, 0, 4));
this->unbindTextureFBOForCopy(GR_GL_FRAMEBUFFER, dst); this->unbindTextureFBOForPixelOps(GR_GL_FRAMEBUFFER, dst);
this->didWriteToSurface(dst, &dstRect); this->didWriteToSurface(dst, &dstRect);
return true; return true;
@ -4239,7 +4240,7 @@ void GrGLGpu::copySurfaceAsCopyTexSubImage(GrSurface* dst,
const SkIPoint& dstPoint) { const SkIPoint& dstPoint) {
SkASSERT(can_copy_texsubimage(dst, src, this)); SkASSERT(can_copy_texsubimage(dst, src, this));
GrGLIRect srcVP; GrGLIRect srcVP;
this->bindSurfaceFBOForCopy(src, GR_GL_FRAMEBUFFER, &srcVP, kSrc_TempFBOTarget); this->bindSurfaceFBOForPixelOps(src, GR_GL_FRAMEBUFFER, &srcVP, kSrc_TempFBOTarget);
GrGLTexture* dstTex = static_cast<GrGLTexture *>(dst->asTexture()); GrGLTexture* dstTex = static_cast<GrGLTexture *>(dst->asTexture());
SkASSERT(dstTex); SkASSERT(dstTex);
// We modified the bound FBO // We modified the bound FBO
@ -4264,7 +4265,7 @@ void GrGLGpu::copySurfaceAsCopyTexSubImage(GrSurface* dst,
dstPoint.fX, dstY, dstPoint.fX, dstY,
srcGLRect.fLeft, srcGLRect.fBottom, srcGLRect.fLeft, srcGLRect.fBottom,
srcGLRect.fWidth, srcGLRect.fHeight)); srcGLRect.fWidth, srcGLRect.fHeight));
this->unbindTextureFBOForCopy(GR_GL_FRAMEBUFFER, src); this->unbindTextureFBOForPixelOps(GR_GL_FRAMEBUFFER, src);
SkIRect dstRect = SkIRect::MakeXYWH(dstPoint.fX, dstPoint.fY, SkIRect dstRect = SkIRect::MakeXYWH(dstPoint.fX, dstPoint.fY,
srcRect.width(), srcRect.height()); srcRect.width(), srcRect.height());
this->didWriteToSurface(dst, &dstRect); this->didWriteToSurface(dst, &dstRect);
@ -4285,8 +4286,8 @@ bool GrGLGpu::copySurfaceAsBlitFramebuffer(GrSurface* dst,
GrGLIRect dstVP; GrGLIRect dstVP;
GrGLIRect srcVP; GrGLIRect srcVP;
this->bindSurfaceFBOForCopy(dst, GR_GL_DRAW_FRAMEBUFFER, &dstVP, kDst_TempFBOTarget); this->bindSurfaceFBOForPixelOps(dst, GR_GL_DRAW_FRAMEBUFFER, &dstVP, kDst_TempFBOTarget);
this->bindSurfaceFBOForCopy(src, GR_GL_READ_FRAMEBUFFER, &srcVP, kSrc_TempFBOTarget); this->bindSurfaceFBOForPixelOps(src, GR_GL_READ_FRAMEBUFFER, &srcVP, kSrc_TempFBOTarget);
// We modified the bound FBO // We modified the bound FBO
fHWBoundRenderTargetUniqueID = SK_InvalidUniqueID; fHWBoundRenderTargetUniqueID = SK_InvalidUniqueID;
GrGLIRect srcGLRect; GrGLIRect srcGLRect;
@ -4327,8 +4328,8 @@ bool GrGLGpu::copySurfaceAsBlitFramebuffer(GrSurface* dst,
dstGLRect.fLeft + dstGLRect.fWidth, dstGLRect.fLeft + dstGLRect.fWidth,
dstGLRect.fBottom + dstGLRect.fHeight, dstGLRect.fBottom + dstGLRect.fHeight,
GR_GL_COLOR_BUFFER_BIT, GR_GL_NEAREST)); GR_GL_COLOR_BUFFER_BIT, GR_GL_NEAREST));
this->unbindTextureFBOForCopy(GR_GL_DRAW_FRAMEBUFFER, dst); this->unbindTextureFBOForPixelOps(GR_GL_DRAW_FRAMEBUFFER, dst);
this->unbindTextureFBOForCopy(GR_GL_READ_FRAMEBUFFER, src); this->unbindTextureFBOForPixelOps(GR_GL_READ_FRAMEBUFFER, src);
this->didWriteToSurface(dst, &dstRect); this->didWriteToSurface(dst, &dstRect);
return true; return true;
} }
@ -4348,7 +4349,7 @@ bool GrGLGpu::generateMipmap(GrGLTexture* texture, bool gammaCorrect) {
} }
// We need to be able to render to the texture for this to work: // We need to be able to render to the texture for this to work:
if (!this->caps()->isConfigRenderable(texture->config(), false)) { if (!this->glCaps().canConfigBeFBOColorAttachment(texture->config())) {
return false; return false;
} }

View File

@ -384,14 +384,14 @@ private:
kDst_TempFBOTarget kDst_TempFBOTarget
}; };
// Binds a surface as a FBO for a copy operation. If the surface already owns an FBO ID then // Binds a surface as a FBO for copying or reading. If the surface already owns an FBO ID then
// that ID is bound. If not the surface is temporarily bound to a FBO and that FBO is bound. // that ID is bound. If not the surface is temporarily bound to a FBO and that FBO is bound.
// This must be paired with a call to unbindSurfaceFBOForCopy(). // This must be paired with a call to unbindSurfaceFBOForPixelOps().
void bindSurfaceFBOForCopy(GrSurface* surface, GrGLenum fboTarget, GrGLIRect* viewport, void bindSurfaceFBOForPixelOps(GrSurface* surface, GrGLenum fboTarget, GrGLIRect* viewport,
TempFBOTarget tempFBOTarget); TempFBOTarget tempFBOTarget);
// Must be called if bindSurfaceFBOForCopy was used to bind a surface for copying. // Must be called if bindSurfaceFBOForPixelOps was used to bind a surface for copying.
void unbindTextureFBOForCopy(GrGLenum fboTarget, GrSurface* surface); void unbindTextureFBOForPixelOps(GrGLenum fboTarget, GrSurface* surface);
SkAutoTUnref<GrGLContext> fGLContext; SkAutoTUnref<GrGLContext> fGLContext;

View File

@ -191,7 +191,9 @@ static bool check_write(skiatest::Reporter* reporter, SkCanvas* canvas, const Sk
// At some point this will be unsupported, as we won't allow accessBitmap() to magically call // At some point this will be unsupported, as we won't allow accessBitmap() to magically call
// readPixels for the client. // readPixels for the client.
SkBitmap secretDevBitmap; SkBitmap secretDevBitmap;
canvas->readPixels(canvasInfo.bounds(), &secretDevBitmap); if (!canvas->readPixels(canvasInfo.bounds(), &secretDevBitmap)) {
return false;
}
SkAutoLockPixels alp(secretDevBitmap); SkAutoLockPixels alp(secretDevBitmap);
canvasRowBytes = secretDevBitmap.rowBytes(); canvasRowBytes = secretDevBitmap.rowBytes();