blob: 0516a82f9573a8fedfd9b55e0885db18cbd0a4ce [file] [log] [blame]
Ethan Nicholas7461a4a2017-12-21 14:18:01 -05001/*
Ethan Nicholas130fb3f2018-02-01 12:14:34 -05002 * Copyright 2018 Google Inc.
Ethan Nicholas7461a4a2017-12-21 14:18:01 -05003 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/effects/GrYUVtoRGBEffect.h"
Ethan Nicholas7461a4a2017-12-21 14:18:01 -05009
Brian Osman904cac82019-10-30 16:57:14 -040010#include "src/core/SkYUVMath.h"
Greg Daniel456f9b52020-03-05 19:14:18 +000011#include "src/gpu/GrTexture.h"
Brian Salomonb9b13732020-06-25 11:13:49 -040012#include "src/gpu/effects/GrMatrixEffect.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "src/gpu/glsl/GrGLSLFragmentProcessor.h"
14#include "src/gpu/glsl/GrGLSLFragmentShaderBuilder.h"
15#include "src/gpu/glsl/GrGLSLProgramBuilder.h"
16#include "src/sksl/SkSLCPP.h"
17#include "src/sksl/SkSLUtil.h"
Michael Ludwiga6a84002019-04-12 15:03:02 -040018
Brian Salomond71548a2020-02-29 19:43:30 -050019static void border_colors(SkYUVColorSpace cs,
20 const SkYUVAIndex yuvaIndices[4],
21 float planeBorders[4][4]) {
22 float m[20];
23 SkColorMatrix_RGB2YUV(cs, m);
24 for (int i = 0; i < 4; ++i) {
25 if (yuvaIndices[i].fIndex == -1) {
26 return;
27 }
28 auto c = static_cast<int>(yuvaIndices[i].fChannel);
29 planeBorders[yuvaIndices[i].fIndex][c] = m[i*5 + 4];
30 }
31}
32
Greg Danielc7672092020-02-06 14:32:54 -050033std::unique_ptr<GrFragmentProcessor> GrYUVtoRGBEffect::Make(GrSurfaceProxyView views[],
Robert Phillips94ade752018-10-09 12:32:31 -040034 const SkYUVAIndex yuvaIndices[4],
Jim Van Verth30e0d7f2018-11-02 13:36:42 -040035 SkYUVColorSpace yuvColorSpace,
Brian Salomond71548a2020-02-29 19:43:30 -050036 GrSamplerState samplerState,
Brian Salomonca6b2f42020-01-24 11:31:21 -050037 const GrCaps& caps,
Michael Ludwiga6a84002019-04-12 15:03:02 -040038 const SkMatrix& localMatrix,
Brian Salomon0ea33072020-07-14 10:43:42 -040039 const SkRect* subset,
40 const SkRect* domain) {
Robert Phillips94ade752018-10-09 12:32:31 -040041 int numPlanes;
Jim Van Verthf00b1622018-10-10 13:03:23 -040042 SkAssertResult(SkYUVAIndex::AreValidIndices(yuvaIndices, &numPlanes));
Robert Phillips94ade752018-10-09 12:32:31 -040043
Brian Salomond71548a2020-02-29 19:43:30 -050044 const SkISize yDimensions =
45 views[yuvaIndices[SkYUVAIndex::kY_Index].fIndex].proxy()->dimensions();
Robert Phillips94ade752018-10-09 12:32:31 -040046
Brian Salomond71548a2020-02-29 19:43:30 -050047 bool usesBorder = samplerState.wrapModeX() == GrSamplerState::WrapMode::kClampToBorder ||
48 samplerState.wrapModeY() == GrSamplerState::WrapMode::kClampToBorder;
49 float planeBorders[4][4] = {};
50 if (usesBorder) {
51 border_colors(yuvColorSpace, yuvaIndices, planeBorders);
52 }
53
Brian Salomonb9b13732020-06-25 11:13:49 -040054 bool snap[2] = {false, false};
Brian Salomonfa9232c2019-12-04 16:09:04 -050055 std::unique_ptr<GrFragmentProcessor> planeFPs[4];
Robert Phillips94ade752018-10-09 12:32:31 -040056 for (int i = 0; i < numPlanes; ++i) {
Greg Danielc7672092020-02-06 14:32:54 -050057 SkISize dimensions = views[i].proxy()->dimensions();
Brian Salomonb9b13732020-06-25 11:13:49 -040058 SkTCopyOnFirstWrite<SkMatrix> planeMatrix(&SkMatrix::I());
59 SkRect planeSubset;
Brian Salomon0ea33072020-07-14 10:43:42 -040060 SkRect planeDomain;
Brian Salomona3b02f52020-07-15 16:02:01 -040061 bool makeLinearWithSnap = false;
Brian Salomonb9b13732020-06-25 11:13:49 -040062 float sx = 1.f,
63 sy = 1.f;
Brian Salomond71548a2020-02-29 19:43:30 -050064 if (dimensions != yDimensions) {
Brian Salomonfa9232c2019-12-04 16:09:04 -050065 // JPEG chroma subsampling of odd dimensions produces U and V planes with the ceiling of
66 // the image size divided by the subsampling factor (2). Our API for creating YUVA
67 // doesn't capture the intended subsampling (and we should fix that). This fixes up 2x
68 // subsampling for images with odd widths/heights (e.g. JPEG 420 or 422).
Brian Salomonb9b13732020-06-25 11:13:49 -040069 sx = (float)dimensions.width() / yDimensions.width();
70 sy = (float)dimensions.height() / yDimensions.height();
Brian Salomond71548a2020-02-29 19:43:30 -050071 if ((yDimensions.width() & 0b1) && dimensions.width() == yDimensions.width() / 2 + 1) {
Brian Salomonfa9232c2019-12-04 16:09:04 -050072 sx = 0.5f;
73 }
Brian Salomond71548a2020-02-29 19:43:30 -050074 if ((yDimensions.height() & 0b1) &&
75 dimensions.height() == yDimensions.height() / 2 + 1) {
Brian Salomonfa9232c2019-12-04 16:09:04 -050076 sy = 0.5f;
77 }
Brian Salomon0ea33072020-07-14 10:43:42 -040078 *planeMatrix.writable() = SkMatrix::Scale(sx, sy);
79 if (subset) {
80 planeSubset = {subset->fLeft * sx,
81 subset->fTop * sy,
82 subset->fRight * sx,
83 subset->fBottom * sy};
84 }
85 if (domain) {
86 planeDomain = {domain->fLeft * sx,
87 domain->fTop * sy,
88 domain->fRight * sx,
89 domain->fBottom * sy};
90 }
Brian Salomona3b02f52020-07-15 16:02:01 -040091 // This promotion of nearest to linear filtering for UV planes exists to mimic
92 // libjpeg[-turbo]'s do_fancy_upsampling option. We will filter the subsampled plane,
93 // however we want to filter at a fixed point for each logical image pixel to simulate
94 // nearest neighbor.
Brian Salomonb9b13732020-06-25 11:13:49 -040095 if (samplerState.filter() == GrSamplerState::Filter::kNearest) {
96 bool snapX = (sx != 1.f),
97 snapY = (sy != 1.f);
Brian Salomona3b02f52020-07-15 16:02:01 -040098 makeLinearWithSnap = snapX || snapY;
Brian Salomonb9b13732020-06-25 11:13:49 -040099 snap[0] |= snapX;
100 snap[1] |= snapY;
Brian Salomon0ea33072020-07-14 10:43:42 -0400101 if (domain) {
102 // The outer YUVToRGB effect will ensure sampling happens at pixel centers
103 // within this plane.
104 planeDomain = {std::floor(planeDomain.fLeft) + 0.5f,
105 std::floor(planeDomain.fTop) + 0.5f,
106 std::floor(planeDomain.fRight) + 0.5f,
107 std::floor(planeDomain.fBottom) + 0.5f};
108 }
Brian Salomonb9b13732020-06-25 11:13:49 -0400109 }
Brian Salomon0ea33072020-07-14 10:43:42 -0400110 } else {
Brian Salomond71548a2020-02-29 19:43:30 -0500111 if (subset) {
Brian Salomon0ea33072020-07-14 10:43:42 -0400112 planeSubset = *subset;
Brian Salomonfa9232c2019-12-04 16:09:04 -0500113 }
Brian Salomon0ea33072020-07-14 10:43:42 -0400114 if (domain) {
115 planeDomain = *domain;
116 }
Brian Salomon8a99a412019-12-04 11:05:35 -0500117 }
Brian Salomond71548a2020-02-29 19:43:30 -0500118 if (subset) {
Brian Salomone69b9ef2020-07-22 11:18:06 -0400119 SkASSERT(samplerState.mipmapped() == GrMipmapped::kNo);
Brian Salomona3b02f52020-07-15 16:02:01 -0400120 if (makeLinearWithSnap) {
Brian Salomonb9b13732020-06-25 11:13:49 -0400121 // The plane is subsampled and we have an overall subset on the image. We're
Brian Salomona3b02f52020-07-15 16:02:01 -0400122 // emulating do_fancy_upsampling using linear filtering but snapping look ups to the
123 // y-plane pixel centers. Consider a logical image pixel at the edge of the subset.
124 // When computing the logical pixel color value we should use a 50/50 blend of two
125 // values from the subsampled plane. Depending on where the subset edge falls in
126 // actual subsampled plane, one of those values may come from outside the subset.
127 // Hence, we use this custom inset factory which applies the wrap mode to
128 // planeSubset but allows linear filtering to read pixels from the plane that are
129 // just outside planeSubset.
Brian Salomon0ea33072020-07-14 10:43:42 -0400130 SkRect* domainRect = domain ? &planeDomain : nullptr;
Brian Salomona3b02f52020-07-15 16:02:01 -0400131 planeFPs[i] = GrTextureEffect::MakeCustomLinearFilterInset(
Brian Salomonb9b13732020-06-25 11:13:49 -0400132 views[i], kUnknown_SkAlphaType, *planeMatrix, samplerState.wrapModeX(),
Brian Salomona3b02f52020-07-15 16:02:01 -0400133 samplerState.wrapModeY(), planeSubset, domainRect, {sx / 2.f, sy / 2.f},
134 caps, planeBorders[i]);
Brian Salomon0ea33072020-07-14 10:43:42 -0400135 } else if (domain) {
136 planeFPs[i] = GrTextureEffect::MakeSubset(views[i], kUnknown_SkAlphaType,
137 *planeMatrix, samplerState, planeSubset,
138 planeDomain, caps, planeBorders[i]);
Brian Salomonb9b13732020-06-25 11:13:49 -0400139 } else {
Brian Salomonb9b13732020-06-25 11:13:49 -0400140 planeFPs[i] = GrTextureEffect::MakeSubset(views[i], kUnknown_SkAlphaType,
141 *planeMatrix, samplerState, planeSubset,
142 caps, planeBorders[i]);
143 }
Brian Salomonca6b2f42020-01-24 11:31:21 -0500144 } else {
Brian Salomonb9b13732020-06-25 11:13:49 -0400145 GrSamplerState planeSampler = samplerState;
Brian Salomona3b02f52020-07-15 16:02:01 -0400146 if (makeLinearWithSnap) {
147 planeSampler.setFilterMode(GrSamplerState::Filter::kLinear);
Brian Salomonb9b13732020-06-25 11:13:49 -0400148 }
Greg Danielc7672092020-02-06 14:32:54 -0500149 planeFPs[i] = GrTextureEffect::Make(views[i], kUnknown_SkAlphaType, *planeMatrix,
Brian Salomonb9b13732020-06-25 11:13:49 -0400150 planeSampler, caps, planeBorders[i]);
Brian Salomon8a99a412019-12-04 11:05:35 -0500151 }
Robert Phillips94ade752018-10-09 12:32:31 -0400152 }
Brian Salomonb9b13732020-06-25 11:13:49 -0400153 auto fp = std::unique_ptr<GrFragmentProcessor>(
154 new GrYUVtoRGBEffect(planeFPs, numPlanes, yuvaIndices, snap, yuvColorSpace));
155 return GrMatrixEffect::Make(localMatrix, std::move(fp));
Brian Salomonfa9232c2019-12-04 16:09:04 -0500156}
157
158static SkAlphaType alpha_type(const SkYUVAIndex yuvaIndices[4]) {
159 return yuvaIndices[3].fIndex >= 0 ? kPremul_SkAlphaType : kOpaque_SkAlphaType;
160}
161
Brian Salomonb9b13732020-06-25 11:13:49 -0400162GrYUVtoRGBEffect::GrYUVtoRGBEffect(std::unique_ptr<GrFragmentProcessor> planeFPs[4],
163 int numPlanes,
164 const SkYUVAIndex yuvaIndices[4],
165 const bool snap[2],
166 SkYUVColorSpace yuvColorSpace)
Brian Salomonfa9232c2019-12-04 16:09:04 -0500167 : GrFragmentProcessor(kGrYUVtoRGBEffect_ClassID,
168 ModulateForClampedSamplerOptFlags(alpha_type(yuvaIndices)))
169 , fYUVColorSpace(yuvColorSpace) {
Brian Salomonfa9232c2019-12-04 16:09:04 -0500170 std::copy_n(yuvaIndices, 4, fYUVAIndices);
Brian Salomonb9b13732020-06-25 11:13:49 -0400171 std::copy_n(snap, 2, fSnap);
172
173 if (fSnap[0] || fSnap[1]) {
174 // Need this so that we can access coords in SKSL to perform snapping.
Michael Ludwigfbe28592020-06-26 16:02:15 -0400175 this->setUsesSampleCoordsDirectly();
Brian Salomonb9b13732020-06-25 11:13:49 -0400176 for (int i = 0; i < numPlanes; ++i) {
Brian Osman1298bc42020-06-30 13:39:35 -0400177 this->registerChild(std::move(planeFPs[i]), SkSL::SampleUsage::Explicit());
Brian Salomonb9b13732020-06-25 11:13:49 -0400178 }
179 } else {
180 for (int i = 0; i < numPlanes; ++i) {
181 this->registerChild(std::move(planeFPs[i]));
182 }
183 }
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500184}
Robert Phillipsba5c4392018-07-25 12:37:14 -0400185
Brian Osman9a390ac2018-11-12 09:47:48 -0500186#ifdef SK_DEBUG
Robert Phillipsba5c4392018-07-25 12:37:14 -0400187SkString GrYUVtoRGBEffect::dumpInfo() const {
John Stilesba1879d2020-08-11 13:58:32 -0400188 SkString str("YUVtoRGBEffect(");
Brian Salomond90b3d32020-07-09 12:04:31 -0400189 for (int i = 0; i < 4; ++i) {
John Stilesba1879d2020-08-11 13:58:32 -0400190 str.appendf("YUVAIndices[%d]=%d %d, ",
191 i, fYUVAIndices[i].fIndex, static_cast<int>(fYUVAIndices[i].fChannel));
Robert Phillips94ade752018-10-09 12:32:31 -0400192 }
John Stilesba1879d2020-08-11 13:58:32 -0400193 str.appendf("YUVColorSpace=%d, snap=(%d, %d))",
194 static_cast<int>(fYUVColorSpace), fSnap[0], fSnap[1]);
Robert Phillipsba5c4392018-07-25 12:37:14 -0400195 return str;
196}
Brian Osman9a390ac2018-11-12 09:47:48 -0500197#endif
198
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500199GrGLSLFragmentProcessor* GrYUVtoRGBEffect::onCreateGLSLInstance() const {
Michael Ludwiga6a84002019-04-12 15:03:02 -0400200 class GrGLSLYUVtoRGBEffect : public GrGLSLFragmentProcessor {
201 public:
202 GrGLSLYUVtoRGBEffect() {}
203
204 void emitCode(EmitArgs& args) override {
205 GrGLSLFPFragmentBuilder* fragBuilder = args.fFragBuilder;
Brian Salomonfa9232c2019-12-04 16:09:04 -0500206 const GrYUVtoRGBEffect& yuvEffect = args.fFp.cast<GrYUVtoRGBEffect>();
Michael Ludwiga6a84002019-04-12 15:03:02 -0400207
Brian Salomonfa9232c2019-12-04 16:09:04 -0500208 int numPlanes = yuvEffect.numChildProcessors();
Michael Ludwiga6a84002019-04-12 15:03:02 -0400209
Brian Salomonb9b13732020-06-25 11:13:49 -0400210 const char* sampleCoords = "";
211 if (yuvEffect.fSnap[0] || yuvEffect.fSnap[1]) {
212 fragBuilder->codeAppendf("float2 snappedCoords = %s;", args.fSampleCoord);
213 if (yuvEffect.fSnap[0]) {
214 fragBuilder->codeAppend("snappedCoords.x = floor(snappedCoords.x) + 0.5;");
215 }
216 if (yuvEffect.fSnap[1]) {
217 fragBuilder->codeAppend("snappedCoords.y = floor(snappedCoords.y) + 0.5;");
218 }
219 sampleCoords = "snappedCoords";
220 }
221
Brian Salomonfa9232c2019-12-04 16:09:04 -0500222 fragBuilder->codeAppendf("half4 planes[%d];", numPlanes);
223 for (int i = 0; i < numPlanes; ++i) {
Brian Salomonb9b13732020-06-25 11:13:49 -0400224 SkString tempVar = this->invokeChild(i, args, sampleCoords);
Brian Salomonfa9232c2019-12-04 16:09:04 -0500225 fragBuilder->codeAppendf("planes[%d] = %s;", i, tempVar.c_str());
Michael Ludwiga6a84002019-04-12 15:03:02 -0400226 }
227
Brian Salomonfa9232c2019-12-04 16:09:04 -0500228 bool hasAlpha = yuvEffect.fYUVAIndices[3].fIndex >= 0;
229 SkString rgba[4];
230 rgba[3] = "1";
231 for (int i = 0; i < (hasAlpha ? 4 : 3); ++i) {
232 auto info = yuvEffect.fYUVAIndices[i];
233 auto letter = "rgba"[static_cast<int>(info.fChannel)];
234 rgba[i].printf("planes[%d].%c", info.fIndex, letter);
Michael Ludwiga6a84002019-04-12 15:03:02 -0400235 }
236
Brian Salomonfa9232c2019-12-04 16:09:04 -0500237 fragBuilder->codeAppendf("half4 color = half4(%s, %s, %s, %s);",
238 rgba[0].c_str(), rgba[1].c_str(), rgba[2].c_str(), rgba[3].c_str());
Michael Ludwiga6a84002019-04-12 15:03:02 -0400239
Brian Salomonfa9232c2019-12-04 16:09:04 -0500240 if (kIdentity_SkYUVColorSpace != yuvEffect.fYUVColorSpace) {
Ethan Nicholas16464c32020-04-06 13:53:05 -0400241 fColorSpaceMatrixVar = args.fUniformHandler->addUniform(&yuvEffect,
Brian Salomonfa9232c2019-12-04 16:09:04 -0500242 kFragment_GrShaderFlag, kHalf3x3_GrSLType, "colorSpaceMatrix");
Ethan Nicholas16464c32020-04-06 13:53:05 -0400243 fColorSpaceTranslateVar = args.fUniformHandler->addUniform(&yuvEffect,
Brian Salomonfa9232c2019-12-04 16:09:04 -0500244 kFragment_GrShaderFlag, kHalf3_GrSLType, "colorSpaceTranslate");
Michael Ludwiga6a84002019-04-12 15:03:02 -0400245 fragBuilder->codeAppendf(
Brian Salomonfa9232c2019-12-04 16:09:04 -0500246 "color.rgb = saturate(color.rgb * %s + %s);",
247 args.fUniformHandler->getUniformCStr(fColorSpaceMatrixVar),
248 args.fUniformHandler->getUniformCStr(fColorSpaceTranslateVar));
Michael Ludwiga6a84002019-04-12 15:03:02 -0400249 }
250
Brian Salomonfa9232c2019-12-04 16:09:04 -0500251 if (hasAlpha) {
Michael Ludwiga6a84002019-04-12 15:03:02 -0400252 // premultiply alpha
Brian Salomonfa9232c2019-12-04 16:09:04 -0500253 fragBuilder->codeAppendf("color.rgb *= color.a;");
Michael Ludwiga6a84002019-04-12 15:03:02 -0400254 }
Brian Salomonfa9232c2019-12-04 16:09:04 -0500255 fragBuilder->codeAppendf("%s = color;", args.fOutputColor);
Michael Ludwiga6a84002019-04-12 15:03:02 -0400256 }
257
258 private:
259 void onSetData(const GrGLSLProgramDataManager& pdman,
Brian Salomonfa9232c2019-12-04 16:09:04 -0500260 const GrFragmentProcessor& proc) override {
261 const GrYUVtoRGBEffect& yuvEffect = proc.cast<GrYUVtoRGBEffect>();
Michael Ludwiga6a84002019-04-12 15:03:02 -0400262
Brian Salomonfa9232c2019-12-04 16:09:04 -0500263 if (yuvEffect.fYUVColorSpace != kIdentity_SkYUVColorSpace) {
Brian Osman904cac82019-10-30 16:57:14 -0400264 SkASSERT(fColorSpaceMatrixVar.isValid());
265 float yuvM[20];
Brian Salomonfa9232c2019-12-04 16:09:04 -0500266 SkColorMatrix_YUV2RGB(yuvEffect.fYUVColorSpace, yuvM);
267 // We drop the fourth column entirely since the transformation
268 // should not depend on alpha. The fifth column is sent as a separate
269 // vector. The fourth row is also dropped entirely because alpha should
270 // never be modified.
271 SkASSERT(yuvM[3] == 0 && yuvM[8] == 0 && yuvM[13] == 0 && yuvM[18] == 1);
272 SkASSERT(yuvM[15] == 0 && yuvM[16] == 0 && yuvM[17] == 0 && yuvM[19] == 0);
273 float mtx[9] = {
274 yuvM[ 0], yuvM[ 1], yuvM[ 2],
275 yuvM[ 5], yuvM[ 6], yuvM[ 7],
276 yuvM[10], yuvM[11], yuvM[12],
Brian Osman904cac82019-10-30 16:57:14 -0400277 };
Brian Salomonfa9232c2019-12-04 16:09:04 -0500278 float v[3] = {yuvM[4], yuvM[9], yuvM[14]};
279 pdman.setMatrix3f(fColorSpaceMatrixVar, mtx);
280 pdman.set3fv(fColorSpaceTranslateVar, 1, v);
Michael Ludwiga6a84002019-04-12 15:03:02 -0400281 }
282 }
283
284 UniformHandle fColorSpaceMatrixVar;
Brian Salomonfa9232c2019-12-04 16:09:04 -0500285 UniformHandle fColorSpaceTranslateVar;
Michael Ludwiga6a84002019-04-12 15:03:02 -0400286 };
287
288 return new GrGLSLYUVtoRGBEffect;
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500289}
290void GrYUVtoRGBEffect::onGetGLSLProcessorKey(const GrShaderCaps& caps,
291 GrProcessorKeyBuilder* b) const {
Robert Phillips94ade752018-10-09 12:32:31 -0400292 uint32_t packed = 0;
293 for (int i = 0; i < 4; ++i) {
Brian Salomonfa9232c2019-12-04 16:09:04 -0500294 if (fYUVAIndices[i].fIndex < 0) {
Robert Phillips94ade752018-10-09 12:32:31 -0400295 continue;
296 }
297
Brian Salomonfa9232c2019-12-04 16:09:04 -0500298 uint8_t index = fYUVAIndices[i].fIndex;
299 uint8_t chann = static_cast<int>(fYUVAIndices[i].fChannel);
Robert Phillips6ba8c832018-10-10 11:43:01 -0400300
Robert Phillips94ade752018-10-09 12:32:31 -0400301 SkASSERT(index < 4 && chann < 4);
302
303 packed |= (index | (chann << 2)) << (i * 4);
304 }
Brian Salomonfa9232c2019-12-04 16:09:04 -0500305 if (fYUVColorSpace == kIdentity_SkYUVColorSpace) {
Brian Salomonb9b13732020-06-25 11:13:49 -0400306 packed |= 1 << 16;
307 }
308 if (fSnap[0]) {
309 packed |= 1 << 17;
310 }
311 if (fSnap[1]) {
312 packed |= 1 << 18;
Robert Phillipsb651aac2019-03-15 12:18:49 -0400313 }
Robert Phillips94ade752018-10-09 12:32:31 -0400314 b->add32(packed);
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500315}
Brian Salomonfa9232c2019-12-04 16:09:04 -0500316
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500317bool GrYUVtoRGBEffect::onIsEqual(const GrFragmentProcessor& other) const {
318 const GrYUVtoRGBEffect& that = other.cast<GrYUVtoRGBEffect>();
Robert Phillips6ba8c832018-10-10 11:43:01 -0400319
Brian Salomonb9b13732020-06-25 11:13:49 -0400320 return std::equal(fYUVAIndices, fYUVAIndices + 4, that.fYUVAIndices) &&
321 std::equal(fSnap, fSnap + 2, that.fSnap) &&
322 fYUVColorSpace == that.fYUVColorSpace;
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500323}
Brian Salomonfa9232c2019-12-04 16:09:04 -0500324
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500325GrYUVtoRGBEffect::GrYUVtoRGBEffect(const GrYUVtoRGBEffect& src)
Brian Salomonfa9232c2019-12-04 16:09:04 -0500326 : GrFragmentProcessor(kGrYUVtoRGBEffect_ClassID, src.optimizationFlags())
Robert Phillipsb651aac2019-03-15 12:18:49 -0400327 , fYUVColorSpace(src.fYUVColorSpace) {
Michael Ludwig9aba6252020-06-22 14:46:36 -0400328 this->cloneAndRegisterAllChildProcessors(src);
Michael Ludwigfbe28592020-06-26 16:02:15 -0400329 if (src.fSnap[0] || src.fSnap[1]) {
330 this->setUsesSampleCoordsDirectly();
331 }
Brian Salomonfa9232c2019-12-04 16:09:04 -0500332 std::copy_n(src.fYUVAIndices, this->numChildProcessors(), fYUVAIndices);
Brian Salomonb9b13732020-06-25 11:13:49 -0400333 std::copy_n(src.fSnap, 2, fSnap);
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500334}
Brian Salomonfa9232c2019-12-04 16:09:04 -0500335
Ethan Nicholas7461a4a2017-12-21 14:18:01 -0500336std::unique_ptr<GrFragmentProcessor> GrYUVtoRGBEffect::clone() const {
337 return std::unique_ptr<GrFragmentProcessor>(new GrYUVtoRGBEffect(*this));
338}