| /* |
| * Copyright 2018 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "src/gpu/effects/GrYUVtoRGBEffect.h" |
| |
| #include "include/gpu/GrTexture.h" |
| #include "src/core/SkYUVMath.h" |
| #include "src/gpu/glsl/GrGLSLFragmentProcessor.h" |
| #include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h" |
| #include "src/gpu/glsl/GrGLSLProgramBuilder.h" |
| #include "src/sksl/SkSLCPP.h" |
| #include "src/sksl/SkSLUtil.h" |
| |
| std::unique_ptr<GrFragmentProcessor> GrYUVtoRGBEffect::Make(const sk_sp<GrTextureProxy> proxies[], |
| const SkYUVAIndex yuvaIndices[4], |
| SkYUVColorSpace yuvColorSpace, |
| GrSamplerState::Filter filterMode, |
| const GrCaps& caps, |
| const SkMatrix& localMatrix, |
| const SkRect* domain) { |
| int numPlanes; |
| SkAssertResult(SkYUVAIndex::AreValidIndices(yuvaIndices, &numPlanes)); |
| |
| const SkISize YDimensions = proxies[yuvaIndices[SkYUVAIndex::kY_Index].fIndex]->dimensions(); |
| |
| // This promotion of nearest to bilinear for UV planes exists to mimic libjpeg[-turbo]'s |
| // do_fancy_upsampling option. However, skbug.com/9693. |
| GrSamplerState::Filter subsampledPlaneFilterMode = GrSamplerState::Filter::kMipMap == filterMode |
| ? GrSamplerState::Filter::kMipMap |
| : GrSamplerState::Filter::kBilerp; |
| std::unique_ptr<GrFragmentProcessor> planeFPs[4]; |
| for (int i = 0; i < numPlanes; ++i) { |
| SkISize dimensions = proxies[i]->dimensions(); |
| SkTCopyOnFirstWrite<SkMatrix> planeMatrix(&localMatrix); |
| GrSamplerState::Filter planeFilter = filterMode; |
| SkRect planeDomain; |
| if (dimensions != YDimensions) { |
| // JPEG chroma subsampling of odd dimensions produces U and V planes with the ceiling of |
| // the image size divided by the subsampling factor (2). Our API for creating YUVA |
| // doesn't capture the intended subsampling (and we should fix that). This fixes up 2x |
| // subsampling for images with odd widths/heights (e.g. JPEG 420 or 422). |
| float sx = (float)dimensions.width() / YDimensions.width(); |
| float sy = (float)dimensions.height() / YDimensions.height(); |
| if ((YDimensions.width() & 0b1) && dimensions.width() == YDimensions.width() / 2 + 1) { |
| sx = 0.5f; |
| } |
| if ((YDimensions.height() & 0b1) && |
| dimensions.height() == YDimensions.height() / 2 + 1) { |
| sy = 0.5f; |
| } |
| *planeMatrix.writable() = SkMatrix::MakeScale(sx, sy); |
| planeMatrix.writable()->preConcat(localMatrix); |
| planeFilter = subsampledPlaneFilterMode; |
| if (domain) { |
| planeDomain = {domain->fLeft * sx, |
| domain->fTop * sy, |
| domain->fRight * sx, |
| domain->fBottom * sy}; |
| } |
| } else if (domain) { |
| planeDomain = *domain; |
| } |
| |
| GrSurfaceOrigin origin = proxies[i]->origin(); |
| GrSwizzle swizzle = proxies[i]->textureSwizzle(); |
| GrSurfaceProxyView view(proxies[i], origin, swizzle); |
| if (domain) { |
| SkASSERT(planeFilter != GrSamplerState::Filter::kMipMap); |
| if (planeFilter != GrSamplerState::Filter::kNearest) { |
| // Inset by half a pixel for bilerp, after scaling to the size of the plane |
| planeDomain.inset(0.5f, 0.5f); |
| } |
| planeFPs[i] = GrTextureEffect::MakeSubset(std::move(view), kUnknown_SkAlphaType, |
| *planeMatrix, planeFilter, planeDomain, caps); |
| } else { |
| planeFPs[i] = GrTextureEffect::Make(std::move(view), kUnknown_SkAlphaType, *planeMatrix, |
| planeFilter); |
| } |
| } |
| |
| return std::unique_ptr<GrFragmentProcessor>( |
| new GrYUVtoRGBEffect(planeFPs, numPlanes, yuvaIndices, yuvColorSpace)); |
| } |
| |
| static SkAlphaType alpha_type(const SkYUVAIndex yuvaIndices[4]) { |
| return yuvaIndices[3].fIndex >= 0 ? kPremul_SkAlphaType : kOpaque_SkAlphaType; |
| } |
| |
| GrYUVtoRGBEffect::GrYUVtoRGBEffect(std::unique_ptr<GrFragmentProcessor> planeFPs[4], int numPlanes, |
| const SkYUVAIndex yuvaIndices[4], SkYUVColorSpace yuvColorSpace) |
| : GrFragmentProcessor(kGrYUVtoRGBEffect_ClassID, |
| ModulateForClampedSamplerOptFlags(alpha_type(yuvaIndices))) |
| , fYUVColorSpace(yuvColorSpace) { |
| for (int i = 0; i < numPlanes; ++i) { |
| this->registerChildProcessor(std::move(planeFPs[i])); |
| } |
| std::copy_n(yuvaIndices, 4, fYUVAIndices); |
| } |
| |
| #ifdef SK_DEBUG |
| SkString GrYUVtoRGBEffect::dumpInfo() const { |
| SkString str; |
| for (int i = 0; i < this->numTextureSamplers(); ++i) { |
| str.appendf("%d: %d %d ", i, |
| this->textureSampler(i).view().proxy()->uniqueID().asUInt(), |
| this->textureSampler(i).view().proxy()->underlyingUniqueID().asUInt()); |
| } |
| str.appendf("\n"); |
| |
| return str; |
| } |
| #endif |
| |
| GrGLSLFragmentProcessor* GrYUVtoRGBEffect::onCreateGLSLInstance() const { |
| class GrGLSLYUVtoRGBEffect : public GrGLSLFragmentProcessor { |
| public: |
| GrGLSLYUVtoRGBEffect() {} |
| |
| void emitCode(EmitArgs& args) override { |
| GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder; |
| const GrYUVtoRGBEffect& yuvEffect = args.fFp.cast<GrYUVtoRGBEffect>(); |
| |
| int numPlanes = yuvEffect.numChildProcessors(); |
| |
| SkString coords[4]; |
| fragBuilder->codeAppendf("half4 planes[%d];", numPlanes); |
| for (int i = 0; i < numPlanes; ++i) { |
| SkString tempVar = this->invokeChild(i, args); |
| fragBuilder->codeAppendf("planes[%d] = %s;", i, tempVar.c_str()); |
| } |
| |
| bool hasAlpha = yuvEffect.fYUVAIndices[3].fIndex >= 0; |
| SkString rgba[4]; |
| rgba[3] = "1"; |
| for (int i = 0; i < (hasAlpha ? 4 : 3); ++i) { |
| auto info = yuvEffect.fYUVAIndices[i]; |
| auto letter = "rgba"[static_cast<int>(info.fChannel)]; |
| rgba[i].printf("planes[%d].%c", info.fIndex, letter); |
| } |
| |
| fragBuilder->codeAppendf("half4 color = half4(%s, %s, %s, %s);", |
| rgba[0].c_str(), rgba[1].c_str(), rgba[2].c_str(), rgba[3].c_str()); |
| |
| if (kIdentity_SkYUVColorSpace != yuvEffect.fYUVColorSpace) { |
| fColorSpaceMatrixVar = args.fUniformHandler->addUniform( |
| kFragment_GrShaderFlag, kHalf3x3_GrSLType, "colorSpaceMatrix"); |
| fColorSpaceTranslateVar = args.fUniformHandler->addUniform( |
| kFragment_GrShaderFlag, kHalf3_GrSLType, "colorSpaceTranslate"); |
| fragBuilder->codeAppendf( |
| "color.rgb = saturate(color.rgb * %s + %s);", |
| args.fUniformHandler->getUniformCStr(fColorSpaceMatrixVar), |
| args.fUniformHandler->getUniformCStr(fColorSpaceTranslateVar)); |
| } |
| |
| if (hasAlpha) { |
| // premultiply alpha |
| fragBuilder->codeAppendf("color.rgb *= color.a;"); |
| } |
| fragBuilder->codeAppendf("%s = color;", args.fOutputColor); |
| } |
| |
| private: |
| void onSetData(const GrGLSLProgramDataManager& pdman, |
| const GrFragmentProcessor& proc) override { |
| const GrYUVtoRGBEffect& yuvEffect = proc.cast<GrYUVtoRGBEffect>(); |
| |
| if (yuvEffect.fYUVColorSpace != kIdentity_SkYUVColorSpace) { |
| SkASSERT(fColorSpaceMatrixVar.isValid()); |
| float yuvM[20]; |
| SkColorMatrix_YUV2RGB(yuvEffect.fYUVColorSpace, yuvM); |
| // We drop the fourth column entirely since the transformation |
| // should not depend on alpha. The fifth column is sent as a separate |
| // vector. The fourth row is also dropped entirely because alpha should |
| // never be modified. |
| SkASSERT(yuvM[3] == 0 && yuvM[8] == 0 && yuvM[13] == 0 && yuvM[18] == 1); |
| SkASSERT(yuvM[15] == 0 && yuvM[16] == 0 && yuvM[17] == 0 && yuvM[19] == 0); |
| float mtx[9] = { |
| yuvM[ 0], yuvM[ 1], yuvM[ 2], |
| yuvM[ 5], yuvM[ 6], yuvM[ 7], |
| yuvM[10], yuvM[11], yuvM[12], |
| }; |
| float v[3] = {yuvM[4], yuvM[9], yuvM[14]}; |
| pdman.setMatrix3f(fColorSpaceMatrixVar, mtx); |
| pdman.set3fv(fColorSpaceTranslateVar, 1, v); |
| } |
| } |
| |
| UniformHandle fColorSpaceMatrixVar; |
| UniformHandle fColorSpaceTranslateVar; |
| }; |
| |
| return new GrGLSLYUVtoRGBEffect; |
| } |
| void GrYUVtoRGBEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps, |
| GrProcessorKeyBuilder* b) const { |
| uint32_t packed = 0; |
| for (int i = 0; i < 4; ++i) { |
| if (fYUVAIndices[i].fIndex < 0) { |
| continue; |
| } |
| |
| uint8_t index = fYUVAIndices[i].fIndex; |
| uint8_t chann = static_cast<int>(fYUVAIndices[i].fChannel); |
| |
| SkASSERT(index < 4 && chann < 4); |
| |
| packed |= (index | (chann << 2)) << (i * 4); |
| } |
| if (fYUVColorSpace == kIdentity_SkYUVColorSpace) { |
| packed |= 0x1 << 16; |
| } |
| b->add32(packed); |
| } |
| |
| bool GrYUVtoRGBEffect::onIsEqual(const GrFragmentProcessor& other) const { |
| const GrYUVtoRGBEffect& that = other.cast<GrYUVtoRGBEffect>(); |
| |
| for (int i = 0; i < 4; ++i) { |
| if (fYUVAIndices[i] != that.fYUVAIndices[i]) { |
| return false; |
| } |
| } |
| |
| if (fYUVColorSpace != that.fYUVColorSpace) { |
| return false; |
| } |
| |
| return true; |
| } |
| |
| GrYUVtoRGBEffect::GrYUVtoRGBEffect(const GrYUVtoRGBEffect& src) |
| : GrFragmentProcessor(kGrYUVtoRGBEffect_ClassID, src.optimizationFlags()) |
| , fYUVColorSpace(src.fYUVColorSpace) { |
| int numPlanes = src.numChildProcessors(); |
| for (int i = 0; i < numPlanes; ++i) { |
| this->registerChildProcessor(this->childProcessor(i).clone()); |
| } |
| std::copy_n(src.fYUVAIndices, this->numChildProcessors(), fYUVAIndices); |
| } |
| |
| std::unique_ptr<GrFragmentProcessor> GrYUVtoRGBEffect::clone() const { |
| return std::unique_ptr<GrFragmentProcessor>(new GrYUVtoRGBEffect(*this)); |
| } |