blob: f668dbae17658716a68aa3e5932d588c0a45c8ff [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"
egdaniel605dd0f2014-11-12 08:35:25 -080012#include "GrInvariantOutput.h"
joshualittb0a8a372014-09-23 09:50:21 -070013#include "GrProcessor.h"
14#include "gl/GrGLProcessor.h"
15#include "GrTBackendProcessorFactory.h"
sugoi24dcac22014-07-07 15:09:48 -070016
17namespace {
18
joshualittb0a8a372014-09-23 09:50:21 -070019class YUVtoRGBEffect : public GrFragmentProcessor {
sugoi24dcac22014-07-07 15:09:48 -070020public:
joshualittb0a8a372014-09-23 09:50:21 -070021 static GrFragmentProcessor* Create(GrTexture* yTexture, GrTexture* uTexture,
22 GrTexture* vTexture, SkYUVColorSpace colorSpace) {
rileyaabaef862014-09-12 17:45:58 -070023 return SkNEW_ARGS(YUVtoRGBEffect, (yTexture, uTexture, vTexture, colorSpace));
sugoi24dcac22014-07-07 15:09:48 -070024 }
25
26 static const char* Name() { return "YUV to RGB"; }
27
joshualittb0a8a372014-09-23 09:50:21 -070028 virtual const GrBackendFragmentProcessorFactory& getFactory() const SK_OVERRIDE {
29 return GrTBackendFragmentProcessorFactory<YUVtoRGBEffect>::getInstance();
sugoi24dcac22014-07-07 15:09:48 -070030 }
31
rileyaabaef862014-09-12 17:45:58 -070032 SkYUVColorSpace getColorSpace() const {
33 return fColorSpace;
34 }
35
joshualittb0a8a372014-09-23 09:50:21 -070036 class GLProcessor : public GrGLFragmentProcessor {
sugoi24dcac22014-07-07 15:09:48 -070037 public:
rileyaabaef862014-09-12 17:45:58 -070038 static const GrGLfloat kJPEGConversionMatrix[16];
39 static const GrGLfloat kRec601ConversionMatrix[16];
40
sugoi24dcac22014-07-07 15:09:48 -070041 // this class always generates the same code.
joshualittb0a8a372014-09-23 09:50:21 -070042 static void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*) {}
sugoi24dcac22014-07-07 15:09:48 -070043
joshualittb0a8a372014-09-23 09:50:21 -070044 GLProcessor(const GrBackendProcessorFactory& factory,
45 const GrProcessor&)
sugoi24dcac22014-07-07 15:09:48 -070046 : INHERITED(factory) {
47 }
48
joshualitt15988992014-10-09 15:04:05 -070049 virtual void emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -070050 const GrFragmentProcessor&,
51 const GrProcessorKey&,
sugoi24dcac22014-07-07 15:09:48 -070052 const char* outputColor,
53 const char* inputColor,
54 const TransformedCoordsArray& coords,
55 const TextureSamplerArray& samplers) SK_OVERRIDE {
joshualitt15988992014-10-09 15:04:05 -070056 GrGLFPFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
rileyaabaef862014-09-12 17:45:58 -070057
58 const char* yuvMatrix = NULL;
59 fMatrixUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
60 kMat44f_GrSLType, "YUVMatrix",
61 &yuvMatrix);
joshualitt30ba4362014-08-21 20:18:45 -070062 fsBuilder->codeAppendf("\t%s = vec4(\n\t\t", outputColor);
joshualitt23e280d2014-09-18 12:26:38 -070063 fsBuilder->appendTextureLookup(samplers[0], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070064 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070065 fsBuilder->appendTextureLookup(samplers[1], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070066 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070067 fsBuilder->appendTextureLookup(samplers[2], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070068 fsBuilder->codeAppendf(".r,\n\t\t1.0) * %s;\n", yuvMatrix);
sugoi24dcac22014-07-07 15:09:48 -070069 }
70
rileyaabaef862014-09-12 17:45:58 -070071 virtual void setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -070072 const GrProcessor& processor) SK_OVERRIDE {
73 const YUVtoRGBEffect& yuvEffect = processor.cast<YUVtoRGBEffect>();
rileyaabaef862014-09-12 17:45:58 -070074 switch (yuvEffect.getColorSpace()) {
75 case kJPEG_SkYUVColorSpace:
76 pdman.setMatrix4f(fMatrixUni, kJPEGConversionMatrix);
77 break;
78 case kRec601_SkYUVColorSpace:
79 pdman.setMatrix4f(fMatrixUni, kRec601ConversionMatrix);
80 break;
81 }
82 }
83
84 private:
85 GrGLProgramDataManager::UniformHandle fMatrixUni;
86
joshualittb0a8a372014-09-23 09:50:21 -070087 typedef GrGLFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -070088 };
89
90private:
rileyaabaef862014-09-12 17:45:58 -070091 YUVtoRGBEffect(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
92 SkYUVColorSpace colorSpace)
93 : fCoordTransform(kLocal_GrCoordSet, GrCoordTransform::MakeDivByTextureWHMatrix(yTexture),
94 yTexture)
sugoi24dcac22014-07-07 15:09:48 -070095 , fYAccess(yTexture)
96 , fUAccess(uTexture)
rileyaabaef862014-09-12 17:45:58 -070097 , fVAccess(vTexture)
98 , fColorSpace(colorSpace) {
sugoi24dcac22014-07-07 15:09:48 -070099 this->addCoordTransform(&fCoordTransform);
100 this->addTextureAccess(&fYAccess);
101 this->addTextureAccess(&fUAccess);
102 this->addTextureAccess(&fVAccess);
sugoi24dcac22014-07-07 15:09:48 -0700103 }
104
bsalomon0e08fc12014-10-15 08:19:04 -0700105 virtual bool onIsEqual(const GrFragmentProcessor& sBase) const {
joshualitt49586be2014-09-16 08:21:41 -0700106 const YUVtoRGBEffect& s = sBase.cast<YUVtoRGBEffect>();
bsalomon420d7e92014-10-16 09:18:09 -0700107 return fColorSpace == s.getColorSpace();
sugoi24dcac22014-07-07 15:09:48 -0700108 }
109
egdaniel605dd0f2014-11-12 08:35:25 -0800110 virtual void onComputeInvariantOutput(GrInvariantOutput* inout) const SK_OVERRIDE {
egdaniel1a8ecdf2014-10-03 06:24:12 -0700111 // YUV is opaque
egdaniel9e4d6d12014-10-15 13:49:02 -0700112 inout->setToOther(kA_GrColorComponentFlag, 0xFF << GrColor_SHIFT_A,
egdaniel605dd0f2014-11-12 08:35:25 -0800113 GrInvariantOutput::kWillNot_ReadInput);
egdaniel1a8ecdf2014-10-03 06:24:12 -0700114 }
115
sugoi24dcac22014-07-07 15:09:48 -0700116 GrCoordTransform fCoordTransform;
117 GrTextureAccess fYAccess;
118 GrTextureAccess fUAccess;
119 GrTextureAccess fVAccess;
rileyaabaef862014-09-12 17:45:58 -0700120 SkYUVColorSpace fColorSpace;
sugoi24dcac22014-07-07 15:09:48 -0700121
joshualittb0a8a372014-09-23 09:50:21 -0700122 typedef GrFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -0700123};
124
joshualittb0a8a372014-09-23 09:50:21 -0700125const GrGLfloat YUVtoRGBEffect::GLProcessor::kJPEGConversionMatrix[16] = {
rileyaabaef862014-09-12 17:45:58 -0700126 1.0f, 0.0f, 1.402f, -0.701f,
127 1.0f, -0.34414f, -0.71414f, 0.529f,
128 1.0f, 1.772f, 0.0f, -0.886f,
129 0.0f, 0.0f, 0.0f, 1.0};
joshualittb0a8a372014-09-23 09:50:21 -0700130const GrGLfloat YUVtoRGBEffect::GLProcessor::kRec601ConversionMatrix[16] = {
rileya223ba622014-09-16 06:23:50 -0700131 1.164f, 0.0f, 1.596f, -0.87075f,
132 1.164f, -0.391f, -0.813f, 0.52925f,
rileyaabaef862014-09-12 17:45:58 -0700133 1.164f, 2.018f, 0.0f, -1.08175f,
134 0.0f, 0.0f, 0.0f, 1.0};
sugoi24dcac22014-07-07 15:09:48 -0700135}
136
137//////////////////////////////////////////////////////////////////////////////
138
joshualittb0a8a372014-09-23 09:50:21 -0700139GrFragmentProcessor*
140GrYUVtoRGBEffect::Create(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
141 SkYUVColorSpace colorSpace) {
rileyaabaef862014-09-12 17:45:58 -0700142 return YUVtoRGBEffect::Create(yTexture, uTexture, vTexture, colorSpace);
sugoi24dcac22014-07-07 15:09:48 -0700143}