blob: cdbe654edec2e9e2bb3f76f4d383592cb7180d57 [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"
joshualittb0a8a372014-09-23 09:50:21 -070012#include "GrProcessor.h"
13#include "gl/GrGLProcessor.h"
14#include "GrTBackendProcessorFactory.h"
sugoi24dcac22014-07-07 15:09:48 -070015
16namespace {
17
joshualittb0a8a372014-09-23 09:50:21 -070018class YUVtoRGBEffect : public GrFragmentProcessor {
sugoi24dcac22014-07-07 15:09:48 -070019public:
joshualittb0a8a372014-09-23 09:50:21 -070020 static GrFragmentProcessor* Create(GrTexture* yTexture, GrTexture* uTexture,
21 GrTexture* vTexture, SkYUVColorSpace colorSpace) {
rileyaabaef862014-09-12 17:45:58 -070022 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
joshualittb0a8a372014-09-23 09:50:21 -070027 virtual const GrBackendFragmentProcessorFactory& getFactory() const SK_OVERRIDE {
28 return GrTBackendFragmentProcessorFactory<YUVtoRGBEffect>::getInstance();
sugoi24dcac22014-07-07 15:09:48 -070029 }
30
rileyaabaef862014-09-12 17:45:58 -070031 SkYUVColorSpace getColorSpace() const {
32 return fColorSpace;
33 }
34
joshualittb0a8a372014-09-23 09:50:21 -070035 class GLProcessor : public GrGLFragmentProcessor {
sugoi24dcac22014-07-07 15:09:48 -070036 public:
rileyaabaef862014-09-12 17:45:58 -070037 static const GrGLfloat kJPEGConversionMatrix[16];
38 static const GrGLfloat kRec601ConversionMatrix[16];
39
sugoi24dcac22014-07-07 15:09:48 -070040 // this class always generates the same code.
joshualittb0a8a372014-09-23 09:50:21 -070041 static void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*) {}
sugoi24dcac22014-07-07 15:09:48 -070042
joshualittb0a8a372014-09-23 09:50:21 -070043 GLProcessor(const GrBackendProcessorFactory& factory,
44 const GrProcessor&)
sugoi24dcac22014-07-07 15:09:48 -070045 : INHERITED(factory) {
46 }
47
joshualitt30ba4362014-08-21 20:18:45 -070048 virtual void emitCode(GrGLProgramBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -070049 const GrFragmentProcessor&,
50 const GrProcessorKey&,
sugoi24dcac22014-07-07 15:09:48 -070051 const char* outputColor,
52 const char* inputColor,
53 const TransformedCoordsArray& coords,
54 const TextureSamplerArray& samplers) SK_OVERRIDE {
joshualitt30ba4362014-08-21 20:18:45 -070055 GrGLFragmentShaderBuilder* fsBuilder = builder->getFragmentShaderBuilder();
rileyaabaef862014-09-12 17:45:58 -070056
57 const char* yuvMatrix = NULL;
58 fMatrixUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
59 kMat44f_GrSLType, "YUVMatrix",
60 &yuvMatrix);
joshualitt30ba4362014-08-21 20:18:45 -070061 fsBuilder->codeAppendf("\t%s = vec4(\n\t\t", outputColor);
joshualitt23e280d2014-09-18 12:26:38 -070062 fsBuilder->appendTextureLookup(samplers[0], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070063 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070064 fsBuilder->appendTextureLookup(samplers[1], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070065 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070066 fsBuilder->appendTextureLookup(samplers[2], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070067 fsBuilder->codeAppendf(".r,\n\t\t1.0) * %s;\n", yuvMatrix);
sugoi24dcac22014-07-07 15:09:48 -070068 }
69
rileyaabaef862014-09-12 17:45:58 -070070 virtual void setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -070071 const GrProcessor& processor) SK_OVERRIDE {
72 const YUVtoRGBEffect& yuvEffect = processor.cast<YUVtoRGBEffect>();
rileyaabaef862014-09-12 17:45:58 -070073 switch (yuvEffect.getColorSpace()) {
74 case kJPEG_SkYUVColorSpace:
75 pdman.setMatrix4f(fMatrixUni, kJPEGConversionMatrix);
76 break;
77 case kRec601_SkYUVColorSpace:
78 pdman.setMatrix4f(fMatrixUni, kRec601ConversionMatrix);
79 break;
80 }
81 }
82
83 private:
84 GrGLProgramDataManager::UniformHandle fMatrixUni;
85
joshualittb0a8a372014-09-23 09:50:21 -070086 typedef GrGLFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -070087 };
88
89private:
rileyaabaef862014-09-12 17:45:58 -070090 YUVtoRGBEffect(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
91 SkYUVColorSpace colorSpace)
92 : fCoordTransform(kLocal_GrCoordSet, GrCoordTransform::MakeDivByTextureWHMatrix(yTexture),
93 yTexture)
sugoi24dcac22014-07-07 15:09:48 -070094 , fYAccess(yTexture)
95 , fUAccess(uTexture)
rileyaabaef862014-09-12 17:45:58 -070096 , fVAccess(vTexture)
97 , fColorSpace(colorSpace) {
sugoi24dcac22014-07-07 15:09:48 -070098 this->addCoordTransform(&fCoordTransform);
99 this->addTextureAccess(&fYAccess);
100 this->addTextureAccess(&fUAccess);
101 this->addTextureAccess(&fVAccess);
102 this->setWillNotUseInputColor();
103 }
104
joshualittb0a8a372014-09-23 09:50:21 -0700105 virtual bool onIsEqual(const GrProcessor& sBase) const {
joshualitt49586be2014-09-16 08:21:41 -0700106 const YUVtoRGBEffect& s = sBase.cast<YUVtoRGBEffect>();
sugoi24dcac22014-07-07 15:09:48 -0700107 return fYAccess.getTexture() == s.fYAccess.getTexture() &&
108 fUAccess.getTexture() == s.fUAccess.getTexture() &&
rileyaabaef862014-09-12 17:45:58 -0700109 fVAccess.getTexture() == s.fVAccess.getTexture() &&
110 fColorSpace == s.getColorSpace();
sugoi24dcac22014-07-07 15:09:48 -0700111 }
112
egdaniel3b8af072014-10-02 09:57:48 -0700113 virtual void onComputeInvariantOutput(InvariantOutput* inout) const SK_OVERRIDE {
114 // YUV is opaque
115 inout->fColor = 0xFF;
116 inout->fValidFlags = kA_GrColorComponentFlag;
117 inout->fIsSingleComponent = false;
118 }
119
sugoi24dcac22014-07-07 15:09:48 -0700120 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
joshualittb0a8a372014-09-23 09:50:21 -0700126 typedef GrFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -0700127};
128
joshualittb0a8a372014-09-23 09:50:21 -0700129const GrGLfloat YUVtoRGBEffect::GLProcessor::kJPEGConversionMatrix[16] = {
rileyaabaef862014-09-12 17:45:58 -0700130 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};
joshualittb0a8a372014-09-23 09:50:21 -0700134const GrGLfloat YUVtoRGBEffect::GLProcessor::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
joshualittb0a8a372014-09-23 09:50:21 -0700143GrFragmentProcessor*
144GrYUVtoRGBEffect::Create(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
145 SkYUVColorSpace colorSpace) {
rileyaabaef862014-09-12 17:45:58 -0700146 return YUVtoRGBEffect::Create(yTexture, uTexture, vTexture, colorSpace);
sugoi24dcac22014-07-07 15:09:48 -0700147}