blob: 4ffc345a2cdc755acc8834d7c24103af8e7c5963 [file] [log] [blame]
sugoi24dcac22014-07-07 15:09:48 -07001/*
2 * Copyright 2014 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
joshualitt30ba4362014-08-21 20:18:45 -07008#include "gl/builders/GrGLProgramBuilder.h"
sugoi24dcac22014-07-07 15:09:48 -07009#include "GrYUVtoRGBEffect.h"
10
11#include "GrCoordTransform.h"
12#include "GrEffect.h"
13#include "gl/GrGLEffect.h"
14#include "GrTBackendEffectFactory.h"
15
16namespace {
17
18class YUVtoRGBEffect : public GrEffect {
19public:
rileyaabaef862014-09-12 17:45:58 -070020 static GrEffect* Create(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
21 SkYUVColorSpace colorSpace) {
22 return SkNEW_ARGS(YUVtoRGBEffect, (yTexture, uTexture, vTexture, colorSpace));
sugoi24dcac22014-07-07 15:09:48 -070023 }
24
25 static const char* Name() { return "YUV to RGB"; }
26
27 virtual const GrBackendEffectFactory& getFactory() const SK_OVERRIDE {
28 return GrTBackendEffectFactory<YUVtoRGBEffect>::getInstance();
29 }
30
31 virtual void getConstantColorComponents(GrColor* color,
32 uint32_t* validFlags) const SK_OVERRIDE {
33 // YUV is opaque
34 *color = 0xFF;
35 *validFlags = kA_GrColorComponentFlag;
36 }
37
rileyaabaef862014-09-12 17:45:58 -070038 SkYUVColorSpace getColorSpace() const {
39 return fColorSpace;
40 }
41
sugoi24dcac22014-07-07 15:09:48 -070042 class GLEffect : public GrGLEffect {
43 public:
rileyaabaef862014-09-12 17:45:58 -070044 static const GrGLfloat kJPEGConversionMatrix[16];
45 static const GrGLfloat kRec601ConversionMatrix[16];
46
sugoi24dcac22014-07-07 15:09:48 -070047 // this class always generates the same code.
joshualitt08da4f22014-09-16 07:17:28 -070048 static void GenKey(const GrDrawEffect&, const GrGLCaps&, GrEffectKeyBuilder*) {}
sugoi24dcac22014-07-07 15:09:48 -070049
50 GLEffect(const GrBackendEffectFactory& factory,
joshualitt08da4f22014-09-16 07:17:28 -070051 const GrDrawEffect&)
sugoi24dcac22014-07-07 15:09:48 -070052 : INHERITED(factory) {
53 }
54
joshualitt30ba4362014-08-21 20:18:45 -070055 virtual void emitCode(GrGLProgramBuilder* builder,
joshualitt08da4f22014-09-16 07:17:28 -070056 const GrDrawEffect& drawEffect,
bsalomon63e99f72014-07-21 08:03:14 -070057 const GrEffectKey&,
sugoi24dcac22014-07-07 15:09:48 -070058 const char* outputColor,
59 const char* inputColor,
60 const TransformedCoordsArray& coords,
61 const TextureSamplerArray& samplers) SK_OVERRIDE {
joshualitt30ba4362014-08-21 20:18:45 -070062 GrGLFragmentShaderBuilder* fsBuilder = builder->getFragmentShaderBuilder();
rileyaabaef862014-09-12 17:45:58 -070063
64 const char* yuvMatrix = NULL;
65 fMatrixUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
66 kMat44f_GrSLType, "YUVMatrix",
67 &yuvMatrix);
joshualitt30ba4362014-08-21 20:18:45 -070068 fsBuilder->codeAppendf("\t%s = vec4(\n\t\t", outputColor);
69 fsBuilder->appendTextureLookup(samplers[0], coords[0].c_str(), coords[0].type());
70 fsBuilder->codeAppend(".r,\n\t\t");
71 fsBuilder->appendTextureLookup(samplers[1], coords[0].c_str(), coords[0].type());
72 fsBuilder->codeAppend(".r,\n\t\t");
73 fsBuilder->appendTextureLookup(samplers[2], coords[0].c_str(), coords[0].type());
74 fsBuilder->codeAppendf(".r,\n\t\t1.0) * %s;\n", yuvMatrix);
sugoi24dcac22014-07-07 15:09:48 -070075 }
76
rileyaabaef862014-09-12 17:45:58 -070077 virtual void setData(const GrGLProgramDataManager& pdman,
joshualitt08da4f22014-09-16 07:17:28 -070078 const GrDrawEffect& drawEffect) SK_OVERRIDE {
79 const YUVtoRGBEffect& yuvEffect = drawEffect.castEffect<YUVtoRGBEffect>();
rileyaabaef862014-09-12 17:45:58 -070080 switch (yuvEffect.getColorSpace()) {
81 case kJPEG_SkYUVColorSpace:
82 pdman.setMatrix4f(fMatrixUni, kJPEGConversionMatrix);
83 break;
84 case kRec601_SkYUVColorSpace:
85 pdman.setMatrix4f(fMatrixUni, kRec601ConversionMatrix);
86 break;
87 }
88 }
89
90 private:
91 GrGLProgramDataManager::UniformHandle fMatrixUni;
92
sugoi24dcac22014-07-07 15:09:48 -070093 typedef GrGLEffect INHERITED;
94 };
95
96private:
rileyaabaef862014-09-12 17:45:58 -070097 YUVtoRGBEffect(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
98 SkYUVColorSpace colorSpace)
99 : fCoordTransform(kLocal_GrCoordSet, GrCoordTransform::MakeDivByTextureWHMatrix(yTexture),
100 yTexture)
sugoi24dcac22014-07-07 15:09:48 -0700101 , fYAccess(yTexture)
102 , fUAccess(uTexture)
rileyaabaef862014-09-12 17:45:58 -0700103 , fVAccess(vTexture)
104 , fColorSpace(colorSpace) {
sugoi24dcac22014-07-07 15:09:48 -0700105 this->addCoordTransform(&fCoordTransform);
106 this->addTextureAccess(&fYAccess);
107 this->addTextureAccess(&fUAccess);
108 this->addTextureAccess(&fVAccess);
109 this->setWillNotUseInputColor();
110 }
111
112 virtual bool onIsEqual(const GrEffect& sBase) const {
joshualitt08da4f22014-09-16 07:17:28 -0700113 const YUVtoRGBEffect& s = CastEffect<YUVtoRGBEffect>(sBase);
sugoi24dcac22014-07-07 15:09:48 -0700114 return fYAccess.getTexture() == s.fYAccess.getTexture() &&
115 fUAccess.getTexture() == s.fUAccess.getTexture() &&
rileyaabaef862014-09-12 17:45:58 -0700116 fVAccess.getTexture() == s.fVAccess.getTexture() &&
117 fColorSpace == s.getColorSpace();
sugoi24dcac22014-07-07 15:09:48 -0700118 }
119
120 GrCoordTransform fCoordTransform;
121 GrTextureAccess fYAccess;
122 GrTextureAccess fUAccess;
123 GrTextureAccess fVAccess;
rileyaabaef862014-09-12 17:45:58 -0700124 SkYUVColorSpace fColorSpace;
sugoi24dcac22014-07-07 15:09:48 -0700125
126 typedef GrEffect INHERITED;
127};
128
rileyaabaef862014-09-12 17:45:58 -0700129const GrGLfloat YUVtoRGBEffect::GLEffect::kJPEGConversionMatrix[16] = {
130 1.0f, 0.0f, 1.402f, -0.701f,
131 1.0f, -0.34414f, -0.71414f, 0.529f,
132 1.0f, 1.772f, 0.0f, -0.886f,
133 0.0f, 0.0f, 0.0f, 1.0};
134const GrGLfloat YUVtoRGBEffect::GLEffect::kRec601ConversionMatrix[16] = {
rileya223ba622014-09-16 06:23:50 -0700135 1.164f, 0.0f, 1.596f, -0.87075f,
136 1.164f, -0.391f, -0.813f, 0.52925f,
rileyaabaef862014-09-12 17:45:58 -0700137 1.164f, 2.018f, 0.0f, -1.08175f,
138 0.0f, 0.0f, 0.0f, 1.0};
sugoi24dcac22014-07-07 15:09:48 -0700139}
140
141//////////////////////////////////////////////////////////////////////////////
142
rileyaabaef862014-09-12 17:45:58 -0700143GrEffect* GrYUVtoRGBEffect::Create(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
144 SkYUVColorSpace colorSpace) {
145 return YUVtoRGBEffect::Create(yTexture, uTexture, vTexture, colorSpace);
sugoi24dcac22014-07-07 15:09:48 -0700146}