1 /* 2 * Copyright 2011 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 9 #ifndef GrGLProgram_DEFINED 10 #define GrGLProgram_DEFINED 11 12 #include "GrGLProgramDataManager.h" 13 #include "glsl/GrGLSLProgramDataManager.h" 14 #include "glsl/GrGLSLUniformHandler.h" 15 16 class GrGLSLFragmentProcessor; 17 class GrGLSLPrimitiveProcessor; 18 class GrGLSLXferProcessor; 19 class GrPipeline; 20 class GrPrimitiveProcessor; 21 class GrRenderTarget; 22 class GrTextureProxy; 23 24 /** 25 * This class manages a GPU program and records per-program information. It also records the vertex 26 * and instance attribute layouts that are to be used with the program. 27 */ 28 class GrGLProgram : public SkRefCnt { 29 public: 30 /** 31 * This class has its own Attribute representation as it does not need the name and we don't 32 * want to worry about copying the name string to memory with life time of GrGLProgram. 33 * Additionally, these store the attribute location. 34 */ 35 struct Attribute { 36 GrVertexAttribType fCPUType; 37 GrSLType fGPUType; 38 size_t fOffset; 39 GrGLint fLocation; 40 }; 41 42 using UniformHandle = GrGLSLProgramDataManager::UniformHandle; 43 using UniformInfoArray = GrGLProgramDataManager::UniformInfoArray; 44 using VaryingInfoArray = GrGLProgramDataManager::VaryingInfoArray; 45 46 /** 47 * The attribute array consists of vertexAttributeCnt + instanceAttributeCnt elements with 48 * the vertex attributes preceding the instance attributes. 49 */ 50 GrGLProgram(GrGLGpu*, 51 const GrGLSLBuiltinUniformHandles&, 52 GrGLuint programID, 53 const UniformInfoArray& uniforms, 54 const UniformInfoArray& textureSamplers, 55 const VaryingInfoArray&, // used for NVPR only currently 56 std::unique_ptr<GrGLSLPrimitiveProcessor> geometryProcessor, 57 std::unique_ptr<GrGLSLXferProcessor> xferProcessor, 58 std::unique_ptr<std::unique_ptr<GrGLSLFragmentProcessor>[]> fragmentProcessors, 59 int fragmentProcessorCnt, 60 std::unique_ptr<Attribute[]>, 61 int vertexAttributeCnt, 62 int instanceAttributeCnt, 63 int vertexStride, 64 int instanceStride); 65 66 ~GrGLProgram(); 67 68 /** 69 * Call to abandon GL objects owned by this program. 70 */ 71 void abandon(); 72 73 /** 74 * Gets the GL program ID for this program. 75 */ programID()76 GrGLuint programID() const { return fProgramID; } 77 78 /** 79 * We use the RT's size and origin to adjust from Skia device space to OpenGL normalized device 80 * space and to make device space positions have the correct origin for processors that require 81 * them. 82 */ 83 struct RenderTargetState { 84 SkISize fRenderTargetSize; 85 GrSurfaceOrigin fRenderTargetOrigin; 86 RenderTargetStateRenderTargetState87 RenderTargetState() { this->invalidate(); } invalidateRenderTargetState88 void invalidate() { 89 fRenderTargetSize.fWidth = -1; 90 fRenderTargetSize.fHeight = -1; 91 fRenderTargetOrigin = (GrSurfaceOrigin) -1; 92 } 93 94 /** 95 * Gets a float4 that adjusts the position from Skia device coords to GL's normalized device 96 * coords. Assuming the transformed position, pos, is a homogeneous float3, the vec, v, is 97 * applied as such: 98 * pos.x = dot(v.xy, pos.xz) 99 * pos.y = dot(v.zw, pos.yz) 100 */ getRTAdjustmentVecRenderTargetState101 void getRTAdjustmentVec(float* destVec) { 102 destVec[0] = 2.f / fRenderTargetSize.fWidth; 103 destVec[1] = -1.f; 104 if (kBottomLeft_GrSurfaceOrigin == fRenderTargetOrigin) { 105 destVec[2] = -2.f / fRenderTargetSize.fHeight; 106 destVec[3] = 1.f; 107 } else { 108 destVec[2] = 2.f / fRenderTargetSize.fHeight; 109 destVec[3] = -1.f; 110 } 111 } 112 }; 113 114 /** 115 * This function uploads uniforms, calls each GrGLSL*Processor's setData. It binds all fragment 116 * processor textures. Primitive process textures can be bound using this function or by 117 * calling updatePrimitiveProcessorTextureBindings. 118 * 119 * It is the caller's responsibility to ensure the program is bound before calling. 120 */ 121 void updateUniformsAndTextureBindings(const GrRenderTarget*, GrSurfaceOrigin, 122 const GrPrimitiveProcessor&, const GrPipeline&, 123 const GrTextureProxy* const primitiveProcessorTextures[]); 124 125 void updatePrimitiveProcessorTextureBindings(const GrPrimitiveProcessor&, 126 const GrTextureProxy* const[]); 127 vertexStride()128 int vertexStride() const { return fVertexStride; } instanceStride()129 int instanceStride() const { return fInstanceStride; } 130 numVertexAttributes()131 int numVertexAttributes() const { return fVertexAttributeCnt; } vertexAttribute(int i)132 const Attribute& vertexAttribute(int i) const { 133 SkASSERT(i >= 0 && i < fVertexAttributeCnt); 134 return fAttributes[i]; 135 } 136 numInstanceAttributes()137 int numInstanceAttributes() const { return fInstanceAttributeCnt; } instanceAttribute(int i)138 const Attribute& instanceAttribute(int i) const { 139 SkASSERT(i >= 0 && i < fInstanceAttributeCnt); 140 return fAttributes[i + fVertexAttributeCnt]; 141 } 142 143 private: 144 // A helper to loop over effects, set the transforms (via subclass) and bind textures 145 void setFragmentData(const GrPipeline&, int* nextTexSamplerIdx); 146 147 // Helper for setData() that sets the view matrix and loads the render target height uniform 148 void setRenderTargetState(const GrRenderTarget*, GrSurfaceOrigin, const GrPrimitiveProcessor&); 149 150 // these reflect the current values of uniforms (GL uniform values travel with program) 151 RenderTargetState fRenderTargetState; 152 GrGLSLBuiltinUniformHandles fBuiltinUniformHandles; 153 GrGLuint fProgramID; 154 155 // the installed effects 156 std::unique_ptr<GrGLSLPrimitiveProcessor> fPrimitiveProcessor; 157 std::unique_ptr<GrGLSLXferProcessor> fXferProcessor; 158 std::unique_ptr<std::unique_ptr<GrGLSLFragmentProcessor>[]> fFragmentProcessors; 159 int fFragmentProcessorCnt; 160 161 std::unique_ptr<Attribute[]> fAttributes; 162 int fVertexAttributeCnt; 163 int fInstanceAttributeCnt; 164 int fVertexStride; 165 int fInstanceStride; 166 167 GrGLGpu* fGpu; 168 GrGLProgramDataManager fProgramDataManager; 169 170 int fNumTextureSamplers; 171 172 typedef SkRefCnt INHERITED; 173 }; 174 175 #endif 176