blob: 736135282c2dbc7d1a51e5703542f3c070abe83e [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
8#include "GrYUVtoRGBEffect.h"
9
10#include "GrCoordTransform.h"
egdaniel605dd0f2014-11-12 08:35:25 -080011#include "GrInvariantOutput.h"
joshualittb0a8a372014-09-23 09:50:21 -070012#include "GrProcessor.h"
13#include "gl/GrGLProcessor.h"
joshualitteb2a6762014-12-04 11:35:33 -080014#include "gl/builders/GrGLProgramBuilder.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
mtklein72c9faa2015-01-09 10:06:39 -080025 const char* name() const SK_OVERRIDE { return "YUV to RGB"; }
sugoi24dcac22014-07-07 15:09:48 -070026
rileyaabaef862014-09-12 17:45:58 -070027 SkYUVColorSpace getColorSpace() const {
28 return fColorSpace;
29 }
30
joshualittb0a8a372014-09-23 09:50:21 -070031 class GLProcessor : public GrGLFragmentProcessor {
sugoi24dcac22014-07-07 15:09:48 -070032 public:
rileyaabaef862014-09-12 17:45:58 -070033 static const GrGLfloat kJPEGConversionMatrix[16];
34 static const GrGLfloat kRec601ConversionMatrix[16];
35
sugoi24dcac22014-07-07 15:09:48 -070036 // this class always generates the same code.
joshualittb0a8a372014-09-23 09:50:21 -070037 static void GenKey(const GrProcessor&, const GrGLCaps&, GrProcessorKeyBuilder*) {}
sugoi24dcac22014-07-07 15:09:48 -070038
joshualitteb2a6762014-12-04 11:35:33 -080039 GLProcessor(const GrProcessor&) {}
sugoi24dcac22014-07-07 15:09:48 -070040
joshualitt15988992014-10-09 15:04:05 -070041 virtual void emitCode(GrGLFPBuilder* builder,
joshualittb0a8a372014-09-23 09:50:21 -070042 const GrFragmentProcessor&,
sugoi24dcac22014-07-07 15:09:48 -070043 const char* outputColor,
44 const char* inputColor,
45 const TransformedCoordsArray& coords,
46 const TextureSamplerArray& samplers) SK_OVERRIDE {
joshualitt15988992014-10-09 15:04:05 -070047 GrGLFPFragmentBuilder* fsBuilder = builder->getFragmentShaderBuilder();
rileyaabaef862014-09-12 17:45:58 -070048
49 const char* yuvMatrix = NULL;
50 fMatrixUni = builder->addUniform(GrGLProgramBuilder::kFragment_Visibility,
bsalomon422f56f2014-12-09 10:18:12 -080051 kMat44f_GrSLType, kDefault_GrSLPrecision,
52 "YUVMatrix", &yuvMatrix);
joshualitt30ba4362014-08-21 20:18:45 -070053 fsBuilder->codeAppendf("\t%s = vec4(\n\t\t", outputColor);
joshualitt23e280d2014-09-18 12:26:38 -070054 fsBuilder->appendTextureLookup(samplers[0], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070055 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070056 fsBuilder->appendTextureLookup(samplers[1], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070057 fsBuilder->codeAppend(".r,\n\t\t");
joshualitt23e280d2014-09-18 12:26:38 -070058 fsBuilder->appendTextureLookup(samplers[2], coords[0].c_str(), coords[0].getType());
joshualitt30ba4362014-08-21 20:18:45 -070059 fsBuilder->codeAppendf(".r,\n\t\t1.0) * %s;\n", yuvMatrix);
sugoi24dcac22014-07-07 15:09:48 -070060 }
61
rileyaabaef862014-09-12 17:45:58 -070062 virtual void setData(const GrGLProgramDataManager& pdman,
joshualittb0a8a372014-09-23 09:50:21 -070063 const GrProcessor& processor) SK_OVERRIDE {
64 const YUVtoRGBEffect& yuvEffect = processor.cast<YUVtoRGBEffect>();
rileyaabaef862014-09-12 17:45:58 -070065 switch (yuvEffect.getColorSpace()) {
66 case kJPEG_SkYUVColorSpace:
67 pdman.setMatrix4f(fMatrixUni, kJPEGConversionMatrix);
68 break;
69 case kRec601_SkYUVColorSpace:
70 pdman.setMatrix4f(fMatrixUni, kRec601ConversionMatrix);
71 break;
72 }
73 }
74
75 private:
76 GrGLProgramDataManager::UniformHandle fMatrixUni;
77
joshualittb0a8a372014-09-23 09:50:21 -070078 typedef GrGLFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -070079 };
80
joshualitteb2a6762014-12-04 11:35:33 -080081 virtual void getGLProcessorKey(const GrGLCaps& caps,
82 GrProcessorKeyBuilder* b) const SK_OVERRIDE {
83 GLProcessor::GenKey(*this, caps, b);
84 }
85
mtklein72c9faa2015-01-09 10:06:39 -080086 GrGLFragmentProcessor* createGLInstance() const SK_OVERRIDE {
joshualitteb2a6762014-12-04 11:35:33 -080087 return SkNEW_ARGS(GLProcessor, (*this));
88 }
89
sugoi24dcac22014-07-07 15:09:48 -070090private:
rileyaabaef862014-09-12 17:45:58 -070091 YUVtoRGBEffect(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
92 SkYUVColorSpace colorSpace)
bsalomon9f876a32014-12-09 10:51:07 -080093 : fCoordTransform(kLocal_GrCoordSet,
94 GrCoordTransform::MakeDivByTextureWHMatrix(yTexture),
95 yTexture, GrTextureParams::kNone_FilterMode)
sugoi24dcac22014-07-07 15:09:48 -070096 , fYAccess(yTexture)
97 , fUAccess(uTexture)
rileyaabaef862014-09-12 17:45:58 -070098 , fVAccess(vTexture)
99 , fColorSpace(colorSpace) {
joshualitteb2a6762014-12-04 11:35:33 -0800100 this->initClassID<YUVtoRGBEffect>();
sugoi24dcac22014-07-07 15:09:48 -0700101 this->addCoordTransform(&fCoordTransform);
102 this->addTextureAccess(&fYAccess);
103 this->addTextureAccess(&fUAccess);
104 this->addTextureAccess(&fVAccess);
sugoi24dcac22014-07-07 15:09:48 -0700105 }
106
mtklein72c9faa2015-01-09 10:06:39 -0800107 bool onIsEqual(const GrFragmentProcessor& sBase) const SK_OVERRIDE {
joshualitt49586be2014-09-16 08:21:41 -0700108 const YUVtoRGBEffect& s = sBase.cast<YUVtoRGBEffect>();
bsalomon420d7e92014-10-16 09:18:09 -0700109 return fColorSpace == s.getColorSpace();
sugoi24dcac22014-07-07 15:09:48 -0700110 }
111
mtklein72c9faa2015-01-09 10:06:39 -0800112 void onComputeInvariantOutput(GrInvariantOutput* inout) const SK_OVERRIDE {
egdaniel1a8ecdf2014-10-03 06:24:12 -0700113 // YUV is opaque
egdaniel9e4d6d12014-10-15 13:49:02 -0700114 inout->setToOther(kA_GrColorComponentFlag, 0xFF << GrColor_SHIFT_A,
egdaniel605dd0f2014-11-12 08:35:25 -0800115 GrInvariantOutput::kWillNot_ReadInput);
egdaniel1a8ecdf2014-10-03 06:24:12 -0700116 }
117
sugoi24dcac22014-07-07 15:09:48 -0700118 GrCoordTransform fCoordTransform;
119 GrTextureAccess fYAccess;
120 GrTextureAccess fUAccess;
121 GrTextureAccess fVAccess;
rileyaabaef862014-09-12 17:45:58 -0700122 SkYUVColorSpace fColorSpace;
sugoi24dcac22014-07-07 15:09:48 -0700123
joshualittb0a8a372014-09-23 09:50:21 -0700124 typedef GrFragmentProcessor INHERITED;
sugoi24dcac22014-07-07 15:09:48 -0700125};
126
joshualittb0a8a372014-09-23 09:50:21 -0700127const GrGLfloat YUVtoRGBEffect::GLProcessor::kJPEGConversionMatrix[16] = {
rileyaabaef862014-09-12 17:45:58 -0700128 1.0f, 0.0f, 1.402f, -0.701f,
129 1.0f, -0.34414f, -0.71414f, 0.529f,
130 1.0f, 1.772f, 0.0f, -0.886f,
131 0.0f, 0.0f, 0.0f, 1.0};
joshualittb0a8a372014-09-23 09:50:21 -0700132const GrGLfloat YUVtoRGBEffect::GLProcessor::kRec601ConversionMatrix[16] = {
rileya223ba622014-09-16 06:23:50 -0700133 1.164f, 0.0f, 1.596f, -0.87075f,
134 1.164f, -0.391f, -0.813f, 0.52925f,
rileyaabaef862014-09-12 17:45:58 -0700135 1.164f, 2.018f, 0.0f, -1.08175f,
136 0.0f, 0.0f, 0.0f, 1.0};
sugoi24dcac22014-07-07 15:09:48 -0700137}
138
139//////////////////////////////////////////////////////////////////////////////
140
joshualittb0a8a372014-09-23 09:50:21 -0700141GrFragmentProcessor*
142GrYUVtoRGBEffect::Create(GrTexture* yTexture, GrTexture* uTexture, GrTexture* vTexture,
143 SkYUVColorSpace colorSpace) {
rileyaabaef862014-09-12 17:45:58 -0700144 return YUVtoRGBEffect::Create(yTexture, uTexture, vTexture, colorSpace);
sugoi24dcac22014-07-07 15:09:48 -0700145}