/* * Copyright 2013 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrBitmapTextGeoProc.h" #include "GrAtlasedShaderHelpers.h" #include "GrCaps.h" #include "GrShaderCaps.h" #include "GrTexture.h" #include "glsl/GrGLSLFragmentShaderBuilder.h" #include "glsl/GrGLSLGeometryProcessor.h" #include "glsl/GrGLSLProgramDataManager.h" #include "glsl/GrGLSLUniformHandler.h" #include "glsl/GrGLSLVarying.h" #include "glsl/GrGLSLVertexGeoBuilder.h" class GrGLBitmapTextGeoProc : public GrGLSLGeometryProcessor { public: GrGLBitmapTextGeoProc() : fColor(SK_PMColor4fILLEGAL), fAtlasSize({0,0}) {} void onEmitCode(EmitArgs& args, GrGPArgs* gpArgs) override { const GrBitmapTextGeoProc& btgp = args.fGP.cast(); GrGLSLVertexBuilder* vertBuilder = args.fVertBuilder; GrGLSLVaryingHandler* varyingHandler = args.fVaryingHandler; GrGLSLUniformHandler* uniformHandler = args.fUniformHandler; // emit attributes varyingHandler->emitAttributes(btgp); const char* atlasSizeInvName; fAtlasSizeInvUniform = uniformHandler->addUniform(kVertex_GrShaderFlag, kFloat2_GrSLType, "AtlasSizeInv", &atlasSizeInvName); GrGLSLVarying uv(kFloat2_GrSLType); GrSLType texIdxType = args.fShaderCaps->integerSupport() ? kInt_GrSLType : kFloat_GrSLType; GrGLSLVarying texIdx(texIdxType); append_index_uv_varyings(args, btgp.inTextureCoords().name(), atlasSizeInvName, &uv, &texIdx, nullptr); GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder; // Setup pass through color if (btgp.hasVertexColor()) { varyingHandler->addPassThroughAttribute(btgp.inColor(), args.fOutputColor); } else { this->setupUniformColor(fragBuilder, uniformHandler, args.fOutputColor, &fColorUniform); } // Setup position gpArgs->fPositionVar = btgp.inPosition().asShaderVar(); // emit transforms this->emitTransforms(vertBuilder, varyingHandler, uniformHandler, btgp.inPosition().asShaderVar(), btgp.localMatrix(), args.fFPCoordTransformHandler); fragBuilder->codeAppend("half4 texColor;"); append_multitexture_lookup(args, btgp.numTextureSamplers(), texIdx, uv.fsIn(), "texColor"); if (btgp.maskFormat() == kARGB_GrMaskFormat) { // modulate by color fragBuilder->codeAppendf("%s = %s * texColor;", args.fOutputColor, args.fOutputColor); fragBuilder->codeAppendf("%s = half4(1);", args.fOutputCoverage); } else { fragBuilder->codeAppendf("%s = texColor;", args.fOutputCoverage); } } void setData(const GrGLSLProgramDataManager& pdman, const GrPrimitiveProcessor& gp, FPCoordTransformIter&& transformIter) override { const GrBitmapTextGeoProc& btgp = gp.cast(); if (btgp.color() != fColor && !btgp.hasVertexColor()) { pdman.set4fv(fColorUniform, 1, btgp.color().vec()); fColor = btgp.color(); } const SkISize& atlasSize = btgp.atlasSize(); SkASSERT(SkIsPow2(atlasSize.fWidth) && SkIsPow2(atlasSize.fHeight)); if (fAtlasSize != atlasSize) { pdman.set2f(fAtlasSizeInvUniform, 1.0f / atlasSize.fWidth, 1.0f / atlasSize.fHeight); fAtlasSize = atlasSize; } this->setTransformDataHelper(btgp.localMatrix(), pdman, &transformIter); } static inline void GenKey(const GrGeometryProcessor& proc, const GrShaderCaps&, GrProcessorKeyBuilder* b) { const GrBitmapTextGeoProc& btgp = proc.cast(); uint32_t key = 0; key |= btgp.usesW() ? 0x1 : 0x0; key |= btgp.maskFormat() << 1; b->add32(key); b->add32(btgp.numTextureSamplers()); } private: SkPMColor4f fColor; UniformHandle fColorUniform; SkISize fAtlasSize; UniformHandle fAtlasSizeInvUniform; typedef GrGLSLGeometryProcessor INHERITED; }; /////////////////////////////////////////////////////////////////////////////// GrBitmapTextGeoProc::GrBitmapTextGeoProc(const GrShaderCaps& caps, const SkPMColor4f& color, bool wideColor, const sk_sp* proxies, int numActiveProxies, const GrSamplerState& params, GrMaskFormat format, const SkMatrix& localMatrix, bool usesW) : INHERITED(kGrBitmapTextGeoProc_ClassID) , fColor(color) , fLocalMatrix(localMatrix) , fUsesW(usesW) , fMaskFormat(format) { SkASSERT(numActiveProxies <= kMaxTextures); if (usesW) { fInPosition = {"inPosition", kFloat3_GrVertexAttribType, kFloat3_GrSLType}; } else { fInPosition = {"inPosition", kFloat2_GrVertexAttribType, kFloat2_GrSLType}; } bool hasVertexColor = kA8_GrMaskFormat == fMaskFormat || kA565_GrMaskFormat == fMaskFormat; if (hasVertexColor) { fInColor = MakeColorAttribute("inColor", wideColor); } fInTextureCoords = {"inTextureCoords", kUShort2_GrVertexAttribType, caps.integerSupport() ? kUShort2_GrSLType : kFloat2_GrSLType}; this->setVertexAttributes(&fInPosition, 3); if (numActiveProxies) { fAtlasSize = proxies[0]->isize(); } for (int i = 0; i < numActiveProxies; ++i) { SkASSERT(proxies[i]); SkASSERT(proxies[i]->isize() == fAtlasSize); fTextureSamplers[i].reset(proxies[i]->textureType(), proxies[i]->config(), params); } this->setTextureSamplerCnt(numActiveProxies); } void GrBitmapTextGeoProc::addNewProxies(const sk_sp* proxies, int numActiveProxies, const GrSamplerState& params) { SkASSERT(numActiveProxies <= kMaxTextures); if (!fTextureSamplers[0].isInitialized()) { fAtlasSize = proxies[0]->isize(); } for (int i = 0; i < numActiveProxies; ++i) { SkASSERT(proxies[i]); SkASSERT(proxies[i]->isize() == fAtlasSize); if (!fTextureSamplers[i].isInitialized()) { fTextureSamplers[i].reset(proxies[i]->textureType(), proxies[i]->config(), params); } } this->setTextureSamplerCnt(numActiveProxies); } void GrBitmapTextGeoProc::getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const { GrGLBitmapTextGeoProc::GenKey(*this, caps, b); } GrGLSLPrimitiveProcessor* GrBitmapTextGeoProc::createGLSLInstance(const GrShaderCaps& caps) const { return new GrGLBitmapTextGeoProc(); } /////////////////////////////////////////////////////////////////////////////// GR_DEFINE_GEOMETRY_PROCESSOR_TEST(GrBitmapTextGeoProc); #if GR_TEST_UTILS sk_sp GrBitmapTextGeoProc::TestCreate(GrProcessorTestData* d) { int texIdx = d->fRandom->nextBool() ? GrProcessorUnitTest::kSkiaPMTextureIdx : GrProcessorUnitTest::kAlphaTextureIdx; sk_sp proxies[kMaxTextures] = { d->textureProxy(texIdx), nullptr, nullptr, nullptr }; GrSamplerState::WrapMode wrapModes[2]; GrTest::TestWrapModes(d->fRandom, wrapModes); GrSamplerState samplerState(wrapModes, d->fRandom->nextBool() ? GrSamplerState::Filter::kBilerp : GrSamplerState::Filter::kNearest); GrMaskFormat format = kARGB_GrMaskFormat; // init to avoid warning switch (d->fRandom->nextULessThan(3)) { case 0: format = kA8_GrMaskFormat; break; case 1: format = kA565_GrMaskFormat; break; case 2: format = kARGB_GrMaskFormat; break; } return GrBitmapTextGeoProc::Make(*d->caps()->shaderCaps(), SkPMColor4f::FromBytes_RGBA(GrRandomColor(d->fRandom)), d->fRandom->nextBool(), proxies, 1, samplerState, format, GrTest::TestMatrix(d->fRandom), d->fRandom->nextBool()); } #endif