commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2012 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
| 8 | #include "GrTextureDomain.h" |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 9 | |
Robert Phillips | b66b42f | 2017-03-14 08:53:02 -0400 | [diff] [blame] | 10 | #include "GrResourceProvider.h" |
Brian Salomon | 94efbf5 | 2016-11-29 13:43:05 -0500 | [diff] [blame] | 11 | #include "GrShaderCaps.h" |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 12 | #include "GrSimpleTextureEffect.h" |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 13 | #include "GrSurfaceProxyPriv.h" |
Robert Phillips | 646e429 | 2017-06-13 12:44:56 -0400 | [diff] [blame] | 14 | #include "GrTexture.h" |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 15 | #include "SkFloatingPoint.h" |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 16 | #include "glsl/GrGLSLColorSpaceXformHelper.h" |
egdaniel | 64c4728 | 2015-11-13 06:54:19 -0800 | [diff] [blame] | 17 | #include "glsl/GrGLSLFragmentProcessor.h" |
egdaniel | 7ea439b | 2015-12-03 09:20:44 -0800 | [diff] [blame] | 18 | #include "glsl/GrGLSLFragmentShaderBuilder.h" |
egdaniel | 018fb62 | 2015-10-28 07:26:40 -0700 | [diff] [blame] | 19 | #include "glsl/GrGLSLProgramDataManager.h" |
egdaniel | 7ea439b | 2015-12-03 09:20:44 -0800 | [diff] [blame] | 20 | #include "glsl/GrGLSLShaderBuilder.h" |
egdaniel | 7ea439b | 2015-12-03 09:20:44 -0800 | [diff] [blame] | 21 | #include "glsl/GrGLSLUniformHandler.h" |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 22 | |
Robert Phillips | b66b42f | 2017-03-14 08:53:02 -0400 | [diff] [blame] | 23 | static bool can_ignore_rect(GrTextureProxy* proxy, const SkRect& domain) { |
| 24 | if (GrResourceProvider::IsFunctionallyExact(proxy)) { |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 25 | const SkIRect kFullRect = SkIRect::MakeWH(proxy->width(), proxy->height()); |
| 26 | |
| 27 | return domain.contains(kFullRect); |
| 28 | } |
| 29 | |
| 30 | return false; |
| 31 | } |
| 32 | |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 33 | GrTextureDomain::GrTextureDomain(GrTextureProxy* proxy, const SkRect& domain, Mode mode, int index) |
| 34 | : fMode(mode) |
| 35 | , fIndex(index) { |
| 36 | |
| 37 | if (kIgnore_Mode == fMode) { |
| 38 | return; |
| 39 | } |
| 40 | |
| 41 | if (kClamp_Mode == mode && can_ignore_rect(proxy, domain)) { |
| 42 | fMode = kIgnore_Mode; |
| 43 | return; |
| 44 | } |
| 45 | |
| 46 | const SkRect kFullRect = SkRect::MakeIWH(proxy->width(), proxy->height()); |
| 47 | |
| 48 | // We don't currently handle domains that are empty or don't intersect the texture. |
| 49 | // It is OK if the domain rect is a line or point, but it should not be inverted. We do not |
| 50 | // handle rects that do not intersect the [0..1]x[0..1] rect. |
| 51 | SkASSERT(domain.fLeft <= domain.fRight); |
| 52 | SkASSERT(domain.fTop <= domain.fBottom); |
| 53 | fDomain.fLeft = SkScalarPin(domain.fLeft, 0.0f, kFullRect.fRight); |
| 54 | fDomain.fRight = SkScalarPin(domain.fRight, fDomain.fLeft, kFullRect.fRight); |
| 55 | fDomain.fTop = SkScalarPin(domain.fTop, 0.0f, kFullRect.fBottom); |
| 56 | fDomain.fBottom = SkScalarPin(domain.fBottom, fDomain.fTop, kFullRect.fBottom); |
| 57 | SkASSERT(fDomain.fLeft <= fDomain.fRight); |
| 58 | SkASSERT(fDomain.fTop <= fDomain.fBottom); |
| 59 | } |
| 60 | |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 61 | ////////////////////////////////////////////////////////////////////////////// |
| 62 | |
egdaniel | 2d721d3 | 2015-11-11 13:06:05 -0800 | [diff] [blame] | 63 | void GrTextureDomain::GLDomain::sampleTexture(GrGLSLShaderBuilder* builder, |
egdaniel | 7ea439b | 2015-12-03 09:20:44 -0800 | [diff] [blame] | 64 | GrGLSLUniformHandler* uniformHandler, |
Brian Salomon | 1edc5b9 | 2016-11-29 13:43:46 -0500 | [diff] [blame] | 65 | const GrShaderCaps* shaderCaps, |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 66 | const GrTextureDomain& textureDomain, |
| 67 | const char* outColor, |
| 68 | const SkString& inCoords, |
egdaniel | 09aa1fc | 2016-04-20 07:09:46 -0700 | [diff] [blame] | 69 | GrGLSLFragmentProcessor::SamplerHandle sampler, |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 70 | const char* inModulateColor, |
| 71 | GrGLSLColorSpaceXformHelper* colorXformHelper) { |
reed@google.com | d7b1af6 | 2013-12-09 20:31:50 +0000 | [diff] [blame] | 72 | SkASSERT((Mode)-1 == fMode || textureDomain.mode() == fMode); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 73 | SkDEBUGCODE(fMode = textureDomain.mode();) |
| 74 | |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 75 | if (textureDomain.mode() != kIgnore_Mode && !fDomainUni.isValid()) { |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 76 | const char* name; |
| 77 | SkString uniName("TexDom"); |
| 78 | if (textureDomain.fIndex >= 0) { |
| 79 | uniName.appendS32(textureDomain.fIndex); |
| 80 | } |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 81 | fDomainUni = uniformHandler->addUniform(kFragment_GrShaderFlag, |
| 82 | kVec4f_GrSLType, kDefault_GrSLPrecision, |
egdaniel | 7ea439b | 2015-12-03 09:20:44 -0800 | [diff] [blame] | 83 | uniName.c_str(), &name); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 84 | fDomainName = name; |
| 85 | } |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 86 | |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 87 | switch (textureDomain.mode()) { |
| 88 | case kIgnore_Mode: { |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 89 | builder->codeAppendf("%s = ", outColor); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 90 | builder->appendTextureLookupAndModulate(inModulateColor, sampler, inCoords.c_str(), |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 91 | kVec2f_GrSLType, colorXformHelper); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 92 | builder->codeAppend(";"); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 93 | break; |
| 94 | } |
| 95 | case kClamp_Mode: { |
| 96 | SkString clampedCoords; |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 97 | clampedCoords.appendf("clamp(%s, %s.xy, %s.zw)", |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 98 | inCoords.c_str(), fDomainName.c_str(), fDomainName.c_str()); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 99 | |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 100 | builder->codeAppendf("%s = ", outColor); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 101 | builder->appendTextureLookupAndModulate(inModulateColor, sampler, clampedCoords.c_str(), |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 102 | kVec2f_GrSLType, colorXformHelper); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 103 | builder->codeAppend(";"); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 104 | break; |
| 105 | } |
| 106 | case kDecal_Mode: { |
| 107 | // Add a block since we're going to declare variables. |
egdaniel | 2d721d3 | 2015-11-11 13:06:05 -0800 | [diff] [blame] | 108 | GrGLSLShaderBuilder::ShaderBlock block(builder); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 109 | |
| 110 | const char* domain = fDomainName.c_str(); |
Brian Salomon | 1edc5b9 | 2016-11-29 13:43:46 -0500 | [diff] [blame] | 111 | if (!shaderCaps->canUseAnyFunctionInShader()) { |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 112 | // On the NexusS and GalaxyNexus, the other path (with the 'any' |
| 113 | // call) causes the compilation error "Calls to any function that |
| 114 | // may require a gradient calculation inside a conditional block |
| 115 | // may return undefined results". This appears to be an issue with |
| 116 | // the 'any' call since even the simple "result=black; if (any()) |
| 117 | // result=white;" code fails to compile. |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 118 | builder->codeAppend("float4 outside = float4(0.0, 0.0, 0.0, 0.0);"); |
| 119 | builder->codeAppend("float4 inside = "); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 120 | builder->appendTextureLookupAndModulate(inModulateColor, sampler, inCoords.c_str(), |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 121 | kVec2f_GrSLType, colorXformHelper); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 122 | builder->codeAppend(";"); |
halcanary | 9d524f2 | 2016-03-29 09:03:52 -0700 | [diff] [blame] | 123 | |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 124 | builder->codeAppendf("highp float x = (%s).x;", inCoords.c_str()); |
| 125 | builder->codeAppendf("highp float y = (%s).y;", inCoords.c_str()); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 126 | |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 127 | builder->codeAppendf("x = abs(2.0*(x - %s.x)/(%s.z - %s.x) - 1.0);", |
| 128 | domain, domain, domain); |
| 129 | builder->codeAppendf("y = abs(2.0*(y - %s.y)/(%s.w - %s.y) - 1.0);", |
| 130 | domain, domain, domain); |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 131 | builder->codeAppend("float blend = step(1.0, max(x, y));"); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 132 | builder->codeAppendf("%s = mix(inside, outside, blend);", outColor); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 133 | } else { |
Ethan Nicholas | 5af9ea3 | 2017-07-28 15:19:46 -0400 | [diff] [blame] | 134 | builder->codeAppend("bool4 outside;\n"); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 135 | builder->codeAppendf("outside.xy = lessThan(%s, %s.xy);", inCoords.c_str(), |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 136 | domain); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 137 | builder->codeAppendf("outside.zw = greaterThan(%s, %s.zw);", inCoords.c_str(), |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 138 | domain); |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 139 | builder->codeAppendf("%s = any(outside) ? float4(0.0, 0.0, 0.0, 0.0) : ", |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 140 | outColor); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 141 | builder->appendTextureLookupAndModulate(inModulateColor, sampler, inCoords.c_str(), |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 142 | kVec2f_GrSLType, colorXformHelper); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 143 | builder->codeAppend(";"); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 144 | } |
| 145 | break; |
| 146 | } |
| 147 | case kRepeat_Mode: { |
| 148 | SkString clampedCoords; |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 149 | clampedCoords.printf("mod(%s - %s.xy, %s.zw - %s.xy) + %s.xy", |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 150 | inCoords.c_str(), fDomainName.c_str(), fDomainName.c_str(), |
| 151 | fDomainName.c_str(), fDomainName.c_str()); |
| 152 | |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 153 | builder->codeAppendf("%s = ", outColor); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 154 | builder->appendTextureLookupAndModulate(inModulateColor, sampler, clampedCoords.c_str(), |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 155 | kVec2f_GrSLType, colorXformHelper); |
jvanverth | 3fc6560 | 2015-07-22 08:41:51 -0700 | [diff] [blame] | 156 | builder->codeAppend(";"); |
joshualitt | 5ae5fc5 | 2014-07-29 12:59:27 -0700 | [diff] [blame] | 157 | break; |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 158 | } |
| 159 | } |
| 160 | } |
| 161 | |
egdaniel | 018fb62 | 2015-10-28 07:26:40 -0700 | [diff] [blame] | 162 | void GrTextureDomain::GLDomain::setData(const GrGLSLProgramDataManager& pdman, |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 163 | const GrTextureDomain& textureDomain, |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 164 | GrSurfaceProxy* proxy) { |
| 165 | GrTexture* tex = proxy->priv().peekTexture(); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 166 | SkASSERT(textureDomain.mode() == fMode); |
| 167 | if (kIgnore_Mode != textureDomain.mode()) { |
Robert Phillips | e98234f | 2017-01-09 14:23:59 -0500 | [diff] [blame] | 168 | SkScalar wInv = SK_Scalar1 / tex->width(); |
| 169 | SkScalar hInv = SK_Scalar1 / tex->height(); |
| 170 | |
egdaniel | 018fb62 | 2015-10-28 07:26:40 -0700 | [diff] [blame] | 171 | float values[kPrevDomainCount] = { |
Robert Phillips | e98234f | 2017-01-09 14:23:59 -0500 | [diff] [blame] | 172 | SkScalarToFloat(textureDomain.domain().fLeft * wInv), |
| 173 | SkScalarToFloat(textureDomain.domain().fTop * hInv), |
| 174 | SkScalarToFloat(textureDomain.domain().fRight * wInv), |
| 175 | SkScalarToFloat(textureDomain.domain().fBottom * hInv) |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 176 | }; |
Robert Phillips | e98234f | 2017-01-09 14:23:59 -0500 | [diff] [blame] | 177 | |
| 178 | SkASSERT(values[0] >= 0.0f && values[0] <= 1.0f); |
| 179 | SkASSERT(values[1] >= 0.0f && values[1] <= 1.0f); |
| 180 | SkASSERT(values[2] >= 0.0f && values[2] <= 1.0f); |
| 181 | SkASSERT(values[3] >= 0.0f && values[3] <= 1.0f); |
| 182 | |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 183 | // vertical flip if necessary |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 184 | if (kBottomLeft_GrSurfaceOrigin == proxy->origin()) { |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 185 | values[1] = 1.0f - values[1]; |
| 186 | values[3] = 1.0f - values[3]; |
| 187 | // The top and bottom were just flipped, so correct the ordering |
| 188 | // of elements so that values = (l, t, r, b). |
| 189 | SkTSwap(values[1], values[3]); |
| 190 | } |
egdaniel | 018fb62 | 2015-10-28 07:26:40 -0700 | [diff] [blame] | 191 | if (0 != memcmp(values, fPrevDomain, kPrevDomainCount * sizeof(float))) { |
kkinnunen | 7510b22 | 2014-07-30 00:04:16 -0700 | [diff] [blame] | 192 | pdman.set4fv(fDomainUni, 1, values); |
egdaniel | 018fb62 | 2015-10-28 07:26:40 -0700 | [diff] [blame] | 193 | memcpy(fPrevDomain, values, kPrevDomainCount * sizeof(float)); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 194 | } |
| 195 | } |
| 196 | } |
| 197 | |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 198 | /////////////////////////////////////////////////////////////////////////////// |
Brian Salomon | 587e08f | 2017-01-27 10:59:27 -0500 | [diff] [blame] | 199 | inline GrFragmentProcessor::OptimizationFlags GrTextureDomainEffect::OptFlags( |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 200 | GrPixelConfig config, GrTextureDomain::Mode mode) { |
| 201 | if (mode == GrTextureDomain::kDecal_Mode || !GrPixelConfigIsOpaque(config)) { |
Brian Salomon | f3b995b | 2017-02-15 10:22:23 -0500 | [diff] [blame] | 202 | return GrFragmentProcessor::kCompatibleWithCoverageAsAlpha_OptimizationFlag; |
Brian Salomon | 587e08f | 2017-01-27 10:59:27 -0500 | [diff] [blame] | 203 | } else { |
Brian Salomon | f3b995b | 2017-02-15 10:22:23 -0500 | [diff] [blame] | 204 | return GrFragmentProcessor::kCompatibleWithCoverageAsAlpha_OptimizationFlag | |
Brian Salomon | 587e08f | 2017-01-27 10:59:27 -0500 | [diff] [blame] | 205 | GrFragmentProcessor::kPreservesOpaqueInput_OptimizationFlag; |
| 206 | } |
| 207 | } |
| 208 | |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 209 | std::unique_ptr<GrFragmentProcessor> GrTextureDomainEffect::Make( |
| 210 | sk_sp<GrTextureProxy> proxy, |
| 211 | sk_sp<GrColorSpaceXform> colorSpaceXform, |
| 212 | const SkMatrix& matrix, |
| 213 | const SkRect& domain, |
| 214 | GrTextureDomain::Mode mode, |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 215 | GrSamplerState::Filter filterMode) { |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 216 | if (GrTextureDomain::kIgnore_Mode == mode || |
| 217 | (GrTextureDomain::kClamp_Mode == mode && can_ignore_rect(proxy.get(), domain))) { |
Robert Phillips | fbcef6e | 2017-06-15 12:07:18 -0400 | [diff] [blame] | 218 | return GrSimpleTextureEffect::Make(std::move(proxy), |
Robert Phillips | 296b1cc | 2017-03-15 10:42:12 -0400 | [diff] [blame] | 219 | std::move(colorSpaceXform), matrix, filterMode); |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 220 | } else { |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 221 | return std::unique_ptr<GrFragmentProcessor>(new GrTextureDomainEffect( |
| 222 | std::move(proxy), std::move(colorSpaceXform), matrix, domain, mode, filterMode)); |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 223 | } |
| 224 | } |
| 225 | |
Robert Phillips | fbcef6e | 2017-06-15 12:07:18 -0400 | [diff] [blame] | 226 | GrTextureDomainEffect::GrTextureDomainEffect(sk_sp<GrTextureProxy> proxy, |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 227 | sk_sp<GrColorSpaceXform> colorSpaceXform, |
| 228 | const SkMatrix& matrix, |
| 229 | const SkRect& domain, |
| 230 | GrTextureDomain::Mode mode, |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 231 | GrSamplerState::Filter filterMode) |
Brian Salomon | 6cd51b5 | 2017-07-26 19:07:15 -0400 | [diff] [blame] | 232 | : INHERITED(OptFlags(proxy->config(), mode)) |
| 233 | , fCoordTransform(matrix, proxy.get()) |
| 234 | , fTextureDomain(proxy.get(), domain, mode) |
| 235 | , fTextureSampler(std::move(proxy), filterMode) |
| 236 | , fColorSpaceXform(std::move(colorSpaceXform)) { |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 237 | SkASSERT(mode != GrTextureDomain::kRepeat_Mode || |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 238 | filterMode == GrSamplerState::Filter::kNearest); |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 239 | this->initClassID<GrTextureDomainEffect>(); |
Brian Salomon | 6cd51b5 | 2017-07-26 19:07:15 -0400 | [diff] [blame] | 240 | this->addCoordTransform(&fCoordTransform); |
| 241 | this->addTextureSampler(&fTextureSampler); |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 242 | } |
| 243 | |
Brian Salomon | 3f6f965 | 2017-07-28 07:34:05 -0400 | [diff] [blame] | 244 | GrTextureDomainEffect::GrTextureDomainEffect(const GrTextureDomainEffect& that) |
| 245 | : INHERITED(that.optimizationFlags()) |
| 246 | , fCoordTransform(that.fCoordTransform) |
| 247 | , fTextureDomain(that.fTextureDomain) |
| 248 | , fTextureSampler(that.fTextureSampler) |
| 249 | , fColorSpaceXform(that.fColorSpaceXform) { |
| 250 | this->initClassID<GrTextureDomainEffect>(); |
| 251 | this->addCoordTransform(&fCoordTransform); |
| 252 | this->addTextureSampler(&fTextureSampler); |
| 253 | } |
| 254 | |
Brian Salomon | 94efbf5 | 2016-11-29 13:43:05 -0500 | [diff] [blame] | 255 | void GrTextureDomainEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps, |
egdaniel | 57d3b03 | 2015-11-13 11:57:27 -0800 | [diff] [blame] | 256 | GrProcessorKeyBuilder* b) const { |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 257 | b->add32(GrTextureDomain::GLDomain::DomainKey(fTextureDomain)); |
Brian Osman | 70826d7 | 2016-12-20 15:06:18 -0500 | [diff] [blame] | 258 | b->add32(GrColorSpaceXform::XformKey(this->colorSpaceXform())); |
joshualitt | eb2a676 | 2014-12-04 11:35:33 -0800 | [diff] [blame] | 259 | } |
| 260 | |
egdaniel | 57d3b03 | 2015-11-13 11:57:27 -0800 | [diff] [blame] | 261 | GrGLSLFragmentProcessor* GrTextureDomainEffect::onCreateGLSLInstance() const { |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 262 | class GLSLProcessor : public GrGLSLFragmentProcessor { |
| 263 | public: |
| 264 | void emitCode(EmitArgs& args) override { |
| 265 | const GrTextureDomainEffect& tde = args.fFp.cast<GrTextureDomainEffect>(); |
| 266 | const GrTextureDomain& domain = tde.fTextureDomain; |
| 267 | |
| 268 | GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder; |
| 269 | SkString coords2D = fragBuilder->ensureCoords2D(args.fTransformedCoords[0]); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 270 | |
Brian Osman | c624d9d | 2017-03-08 11:42:02 -0500 | [diff] [blame] | 271 | fColorSpaceHelper.emitCode(args.fUniformHandler, tde.colorSpaceXform()); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 272 | fGLDomain.sampleTexture(fragBuilder, |
| 273 | args.fUniformHandler, |
Brian Salomon | 1edc5b9 | 2016-11-29 13:43:46 -0500 | [diff] [blame] | 274 | args.fShaderCaps, |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 275 | domain, |
| 276 | args.fOutputColor, |
| 277 | coords2D, |
| 278 | args.fTexSamplers[0], |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 279 | args.fInputColor, |
Brian Osman | c624d9d | 2017-03-08 11:42:02 -0500 | [diff] [blame] | 280 | &fColorSpaceHelper); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 281 | } |
| 282 | |
| 283 | protected: |
Brian Salomon | ab015ef | 2017-04-04 10:15:51 -0400 | [diff] [blame] | 284 | void onSetData(const GrGLSLProgramDataManager& pdman, |
| 285 | const GrFragmentProcessor& fp) override { |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 286 | const GrTextureDomainEffect& tde = fp.cast<GrTextureDomainEffect>(); |
| 287 | const GrTextureDomain& domain = tde.fTextureDomain; |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 288 | GrSurfaceProxy* proxy = tde.textureSampler(0).proxy(); |
Robert Phillips | 9bee2e5 | 2017-05-29 12:37:20 -0400 | [diff] [blame] | 289 | |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 290 | fGLDomain.setData(pdman, domain, proxy); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 291 | if (SkToBool(tde.colorSpaceXform())) { |
Brian Osman | c624d9d | 2017-03-08 11:42:02 -0500 | [diff] [blame] | 292 | fColorSpaceHelper.setData(pdman, tde.colorSpaceXform()); |
Brian Osman | c468963 | 2016-12-19 17:04:59 -0500 | [diff] [blame] | 293 | } |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 294 | } |
| 295 | |
| 296 | private: |
| 297 | GrTextureDomain::GLDomain fGLDomain; |
Brian Osman | c624d9d | 2017-03-08 11:42:02 -0500 | [diff] [blame] | 298 | GrGLSLColorSpaceXformHelper fColorSpaceHelper; |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 299 | }; |
| 300 | |
| 301 | return new GLSLProcessor; |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 302 | } |
| 303 | |
bsalomon | 0e08fc1 | 2014-10-15 08:19:04 -0700 | [diff] [blame] | 304 | bool GrTextureDomainEffect::onIsEqual(const GrFragmentProcessor& sBase) const { |
joshualitt | 49586be | 2014-09-16 08:21:41 -0700 | [diff] [blame] | 305 | const GrTextureDomainEffect& s = sBase.cast<GrTextureDomainEffect>(); |
Brian Salomon | a7c4c29 | 2016-11-17 12:47:06 -0500 | [diff] [blame] | 306 | return this->fTextureDomain == s.fTextureDomain; |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 307 | } |
| 308 | |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 309 | /////////////////////////////////////////////////////////////////////////////// |
| 310 | |
joshualitt | b0a8a37 | 2014-09-23 09:50:21 -0700 | [diff] [blame] | 311 | GR_DEFINE_FRAGMENT_PROCESSOR_TEST(GrTextureDomainEffect); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 312 | |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 313 | #if GR_TEST_UTILS |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 314 | std::unique_ptr<GrFragmentProcessor> GrTextureDomainEffect::TestCreate(GrProcessorTestData* d) { |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 315 | int texIdx = d->fRandom->nextBool() ? GrProcessorUnitTest::kSkiaPMTextureIdx |
| 316 | : GrProcessorUnitTest::kAlphaTextureIdx; |
| 317 | sk_sp<GrTextureProxy> proxy = d->textureProxy(texIdx); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 318 | SkRect domain; |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 319 | domain.fLeft = d->fRandom->nextRangeScalar(0, proxy->width()); |
| 320 | domain.fRight = d->fRandom->nextRangeScalar(domain.fLeft, proxy->width()); |
| 321 | domain.fTop = d->fRandom->nextRangeScalar(0, proxy->height()); |
| 322 | domain.fBottom = d->fRandom->nextRangeScalar(domain.fTop, proxy->height()); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 323 | GrTextureDomain::Mode mode = |
joshualitt | 0067ff5 | 2015-07-08 14:26:19 -0700 | [diff] [blame] | 324 | (GrTextureDomain::Mode) d->fRandom->nextULessThan(GrTextureDomain::kModeCount); |
| 325 | const SkMatrix& matrix = GrTest::TestMatrix(d->fRandom); |
| 326 | bool bilerp = mode != GrTextureDomain::kRepeat_Mode ? d->fRandom->nextBool() : false; |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 327 | sk_sp<GrColorSpaceXform> colorSpaceXform = GrTest::TestColorXform(d->fRandom); |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 328 | return GrTextureDomainEffect::Make( |
| 329 | std::move(proxy), |
| 330 | std::move(colorSpaceXform), |
| 331 | matrix, |
| 332 | domain, |
| 333 | mode, |
| 334 | bilerp ? GrSamplerState::Filter::kBilerp : GrSamplerState::Filter::kNearest); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 335 | } |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 336 | #endif |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 337 | |
| 338 | /////////////////////////////////////////////////////////////////////////////// |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 339 | std::unique_ptr<GrFragmentProcessor> GrDeviceSpaceTextureDecalFragmentProcessor::Make( |
| 340 | sk_sp<GrTextureProxy> proxy, const SkIRect& subset, const SkIPoint& deviceSpaceOffset) { |
| 341 | return std::unique_ptr<GrFragmentProcessor>(new GrDeviceSpaceTextureDecalFragmentProcessor( |
Robert Phillips | fbcef6e | 2017-06-15 12:07:18 -0400 | [diff] [blame] | 342 | std::move(proxy), subset, deviceSpaceOffset)); |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 343 | } |
| 344 | |
| 345 | GrDeviceSpaceTextureDecalFragmentProcessor::GrDeviceSpaceTextureDecalFragmentProcessor( |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 346 | sk_sp<GrTextureProxy> proxy, const SkIRect& subset, const SkIPoint& deviceSpaceOffset) |
Brian Salomon | f3b995b | 2017-02-15 10:22:23 -0500 | [diff] [blame] | 347 | : INHERITED(kCompatibleWithCoverageAsAlpha_OptimizationFlag) |
Brian Salomon | 2bbdcc4 | 2017-09-07 12:36:34 -0400 | [diff] [blame^] | 348 | , fTextureSampler(proxy, GrSamplerState::ClampNearest()) |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 349 | , fTextureDomain(proxy.get(), GrTextureDomain::MakeTexelDomain(subset), |
| 350 | GrTextureDomain::kDecal_Mode) { |
| 351 | this->addTextureSampler(&fTextureSampler); |
| 352 | fDeviceSpaceOffset.fX = deviceSpaceOffset.fX - subset.fLeft; |
| 353 | fDeviceSpaceOffset.fY = deviceSpaceOffset.fY - subset.fTop; |
| 354 | this->initClassID<GrDeviceSpaceTextureDecalFragmentProcessor>(); |
| 355 | } |
| 356 | |
Brian Salomon | 1a2a7ab | 2017-07-26 13:11:51 -0400 | [diff] [blame] | 357 | GrDeviceSpaceTextureDecalFragmentProcessor::GrDeviceSpaceTextureDecalFragmentProcessor( |
| 358 | const GrDeviceSpaceTextureDecalFragmentProcessor& that) |
| 359 | : INHERITED(kCompatibleWithCoverageAsAlpha_OptimizationFlag) |
| 360 | , fTextureSampler(that.fTextureSampler) |
| 361 | , fTextureDomain(that.fTextureDomain) |
| 362 | , fDeviceSpaceOffset(that.fDeviceSpaceOffset) { |
| 363 | this->initClassID<GrDeviceSpaceTextureDecalFragmentProcessor>(); |
| 364 | this->addTextureSampler(&fTextureSampler); |
| 365 | } |
| 366 | |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 367 | std::unique_ptr<GrFragmentProcessor> GrDeviceSpaceTextureDecalFragmentProcessor::clone() const { |
| 368 | return std::unique_ptr<GrFragmentProcessor>( |
| 369 | new GrDeviceSpaceTextureDecalFragmentProcessor(*this)); |
Brian Salomon | 1a2a7ab | 2017-07-26 13:11:51 -0400 | [diff] [blame] | 370 | } |
| 371 | |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 372 | GrGLSLFragmentProcessor* GrDeviceSpaceTextureDecalFragmentProcessor::onCreateGLSLInstance() const { |
| 373 | class GLSLProcessor : public GrGLSLFragmentProcessor { |
| 374 | public: |
| 375 | void emitCode(EmitArgs& args) override { |
| 376 | const GrDeviceSpaceTextureDecalFragmentProcessor& dstdfp = |
| 377 | args.fFp.cast<GrDeviceSpaceTextureDecalFragmentProcessor>(); |
| 378 | const char* scaleAndTranslateName; |
| 379 | fScaleAndTranslateUni = args.fUniformHandler->addUniform(kFragment_GrShaderFlag, |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 380 | kVec4f_GrSLType, |
| 381 | kDefault_GrSLPrecision, |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 382 | "scaleAndTranslate", |
| 383 | &scaleAndTranslateName); |
Brian Salomon | 1d816b9 | 2017-08-17 11:07:59 -0400 | [diff] [blame] | 384 | args.fFragBuilder->codeAppendf("float2 coords = sk_FragCoord.xy * %s.xy + %s.zw;", |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 385 | scaleAndTranslateName, scaleAndTranslateName); |
| 386 | fGLDomain.sampleTexture(args.fFragBuilder, |
| 387 | args.fUniformHandler, |
Brian Salomon | 1edc5b9 | 2016-11-29 13:43:46 -0500 | [diff] [blame] | 388 | args.fShaderCaps, |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 389 | dstdfp.fTextureDomain, |
| 390 | args.fOutputColor, |
| 391 | SkString("coords"), |
| 392 | args.fTexSamplers[0], |
| 393 | args.fInputColor); |
| 394 | } |
| 395 | |
| 396 | protected: |
Brian Salomon | ab015ef | 2017-04-04 10:15:51 -0400 | [diff] [blame] | 397 | void onSetData(const GrGLSLProgramDataManager& pdman, |
| 398 | const GrFragmentProcessor& fp) override { |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 399 | const GrDeviceSpaceTextureDecalFragmentProcessor& dstdfp = |
| 400 | fp.cast<GrDeviceSpaceTextureDecalFragmentProcessor>(); |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 401 | GrSurfaceProxy* proxy = dstdfp.textureSampler(0).proxy(); |
| 402 | GrTexture* texture = proxy->priv().peekTexture(); |
Robert Phillips | 9bee2e5 | 2017-05-29 12:37:20 -0400 | [diff] [blame] | 403 | |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 404 | fGLDomain.setData(pdman, dstdfp.fTextureDomain, proxy); |
Brian Salomon | 0bbecb2 | 2016-11-17 11:38:22 -0500 | [diff] [blame] | 405 | float iw = 1.f / texture->width(); |
| 406 | float ih = 1.f / texture->height(); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 407 | float scaleAndTransData[4] = { |
| 408 | iw, ih, |
| 409 | -dstdfp.fDeviceSpaceOffset.fX * iw, -dstdfp.fDeviceSpaceOffset.fY * ih |
| 410 | }; |
Robert Phillips | c686ce3 | 2017-07-21 14:12:29 -0400 | [diff] [blame] | 411 | if (proxy->origin() == kBottomLeft_GrSurfaceOrigin) { |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 412 | scaleAndTransData[1] = -scaleAndTransData[1]; |
| 413 | scaleAndTransData[3] = 1 - scaleAndTransData[3]; |
| 414 | } |
| 415 | pdman.set4fv(fScaleAndTranslateUni, 1, scaleAndTransData); |
| 416 | } |
| 417 | |
| 418 | private: |
| 419 | GrTextureDomain::GLDomain fGLDomain; |
| 420 | UniformHandle fScaleAndTranslateUni; |
| 421 | }; |
| 422 | |
| 423 | return new GLSLProcessor; |
| 424 | } |
| 425 | |
| 426 | bool GrDeviceSpaceTextureDecalFragmentProcessor::onIsEqual(const GrFragmentProcessor& fp) const { |
| 427 | const GrDeviceSpaceTextureDecalFragmentProcessor& dstdfp = |
| 428 | fp.cast<GrDeviceSpaceTextureDecalFragmentProcessor>(); |
Brian Salomon | 1a2a7ab | 2017-07-26 13:11:51 -0400 | [diff] [blame] | 429 | return dstdfp.fTextureSampler.proxy()->underlyingUniqueID() == |
| 430 | fTextureSampler.proxy()->underlyingUniqueID() && |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 431 | dstdfp.fDeviceSpaceOffset == fDeviceSpaceOffset && |
| 432 | dstdfp.fTextureDomain == fTextureDomain; |
| 433 | } |
| 434 | |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 435 | /////////////////////////////////////////////////////////////////////////////// |
| 436 | |
| 437 | GR_DEFINE_FRAGMENT_PROCESSOR_TEST(GrDeviceSpaceTextureDecalFragmentProcessor); |
| 438 | |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 439 | #if GR_TEST_UTILS |
Brian Salomon | aff329b | 2017-08-11 09:40:37 -0400 | [diff] [blame] | 440 | std::unique_ptr<GrFragmentProcessor> GrDeviceSpaceTextureDecalFragmentProcessor::TestCreate( |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 441 | GrProcessorTestData* d) { |
| 442 | int texIdx = d->fRandom->nextBool() ? GrProcessorUnitTest::kSkiaPMTextureIdx |
| 443 | : GrProcessorUnitTest::kAlphaTextureIdx; |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 444 | sk_sp<GrTextureProxy> proxy = d->textureProxy(texIdx); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 445 | SkIRect subset; |
Robert Phillips | 40fd7c9 | 2017-01-30 08:06:27 -0500 | [diff] [blame] | 446 | subset.fLeft = d->fRandom->nextULessThan(proxy->width() - 1); |
| 447 | subset.fRight = d->fRandom->nextRangeU(subset.fLeft, proxy->width()); |
| 448 | subset.fTop = d->fRandom->nextULessThan(proxy->height() - 1); |
| 449 | subset.fBottom = d->fRandom->nextRangeU(subset.fTop, proxy->height()); |
Brian Salomon | 2ebd0c8 | 2016-10-03 17:15:28 -0400 | [diff] [blame] | 450 | SkIPoint pt; |
| 451 | pt.fX = d->fRandom->nextULessThan(2048); |
| 452 | pt.fY = d->fRandom->nextULessThan(2048); |
Robert Phillips | fbcef6e | 2017-06-15 12:07:18 -0400 | [diff] [blame] | 453 | return GrDeviceSpaceTextureDecalFragmentProcessor::Make(std::move(proxy), subset, pt); |
commit-bot@chromium.org | 907fbd5 | 2013-12-09 17:03:02 +0000 | [diff] [blame] | 454 | } |
Hal Canary | 6f6961e | 2017-01-31 13:50:44 -0500 | [diff] [blame] | 455 | #endif |