1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #ifndef GPU_COMMAND_BUFFER_SERVICE_PROGRAM_MANAGER_H_ 6 #define GPU_COMMAND_BUFFER_SERVICE_PROGRAM_MANAGER_H_ 7 8 #include <map> 9 #include <string> 10 #include <vector> 11 #include "base/basictypes.h" 12 #include "base/logging.h" 13 #include "base/memory/ref_counted.h" 14 #include "gpu/command_buffer/service/common_decoder.h" 15 #include "gpu/command_buffer/service/gl_utils.h" 16 #include "gpu/command_buffer/service/gles2_cmd_decoder.h" 17 #include "gpu/command_buffer/service/shader_manager.h" 18 #include "gpu/gpu_export.h" 19 20 namespace gpu { 21 namespace gles2 { 22 23 class ProgramCache; 24 class ProgramManager; 25 class Shader; 26 class ShaderManager; 27 class ShaderTranslator; 28 29 // This is used to track which attributes a particular program needs 30 // so we can verify at glDrawXXX time that every attribute is either disabled 31 // or if enabled that it points to a valid source. 32 class GPU_EXPORT Program : public base::RefCounted<Program> { 33 public: 34 static const int kMaxAttachedShaders = 2; 35 36 enum VaryingsPackingOption { 37 kCountOnlyStaticallyUsed, 38 kCountAll 39 }; 40 41 enum UniformApiType { 42 kUniform1i = 1 << 0, 43 kUniform2i = 1 << 1, 44 kUniform3i = 1 << 2, 45 kUniform4i = 1 << 3, 46 kUniform1f = 1 << 4, 47 kUniform2f = 1 << 5, 48 kUniform3f = 1 << 6, 49 kUniform4f = 1 << 7, 50 kUniformMatrix2f = 1 << 8, 51 kUniformMatrix3f = 1 << 9, 52 kUniformMatrix4f = 1 << 10, 53 }; 54 55 struct UniformInfo { 56 UniformInfo(); 57 UniformInfo( 58 GLsizei _size, GLenum _type, GLint _fake_location_base, 59 const std::string& _name); 60 ~UniformInfo(); 61 IsValidUniformInfo62 bool IsValid() const { 63 return size != 0; 64 } 65 IsSamplerUniformInfo66 bool IsSampler() const { 67 return type == GL_SAMPLER_2D || type == GL_SAMPLER_2D_RECT_ARB || 68 type == GL_SAMPLER_CUBE || type == GL_SAMPLER_EXTERNAL_OES; 69 } 70 71 GLsizei size; 72 GLenum type; 73 uint32 accepts_api_type; 74 GLint fake_location_base; 75 bool is_array; 76 std::string name; 77 std::vector<GLint> element_locations; 78 std::vector<GLuint> texture_units; 79 }; 80 struct VertexAttrib { VertexAttribVertexAttrib81 VertexAttrib(GLsizei _size, GLenum _type, const std::string& _name, 82 GLint _location) 83 : size(_size), 84 type(_type), 85 location(_location), 86 name(_name) { 87 } 88 GLsizei size; 89 GLenum type; 90 GLint location; 91 std::string name; 92 }; 93 94 typedef std::vector<UniformInfo> UniformInfoVector; 95 typedef std::vector<VertexAttrib> AttribInfoVector; 96 typedef std::vector<int> SamplerIndices; 97 typedef std::map<std::string, GLint> LocationMap; 98 99 Program(ProgramManager* manager, GLuint service_id); 100 service_id()101 GLuint service_id() const { 102 return service_id_; 103 } 104 sampler_indices()105 const SamplerIndices& sampler_indices() { 106 return sampler_indices_; 107 } 108 GetAttribInfos()109 const AttribInfoVector& GetAttribInfos() const { 110 return attrib_infos_; 111 } 112 GetAttribInfo(GLint index)113 const VertexAttrib* GetAttribInfo(GLint index) const { 114 return (static_cast<size_t>(index) < attrib_infos_.size()) ? 115 &attrib_infos_[index] : NULL; 116 } 117 118 GLint GetAttribLocation(const std::string& name) const; 119 GetAttribInfoByLocation(GLuint location)120 const VertexAttrib* GetAttribInfoByLocation(GLuint location) const { 121 if (location < attrib_location_to_index_map_.size()) { 122 GLint index = attrib_location_to_index_map_[location]; 123 if (index >= 0) { 124 return &attrib_infos_[index]; 125 } 126 } 127 return NULL; 128 } 129 130 const UniformInfo* GetUniformInfo(GLint index) const; 131 132 // If the original name is not found, return NULL. 133 const std::string* GetAttribMappedName( 134 const std::string& original_name) const; 135 136 // If the hashed name is not found, return NULL. 137 const std::string* GetOriginalNameFromHashedName( 138 const std::string& hashed_name) const; 139 140 // Gets the fake location of a uniform by name. 141 GLint GetUniformFakeLocation(const std::string& name) const; 142 143 // Gets the UniformInfo of a uniform by location. 144 const UniformInfo* GetUniformInfoByFakeLocation( 145 GLint fake_location, GLint* real_location, GLint* array_index) const; 146 147 // Gets all the program info. 148 void GetProgramInfo( 149 ProgramManager* manager, CommonDecoder::Bucket* bucket) const; 150 151 // Sets the sampler values for a uniform. 152 // This is safe to call for any location. If the location is not 153 // a sampler uniform nothing will happen. 154 // Returns false if fake_location is a sampler and any value 155 // is >= num_texture_units. Returns true otherwise. 156 bool SetSamplers( 157 GLint num_texture_units, GLint fake_location, 158 GLsizei count, const GLint* value); 159 IsDeleted()160 bool IsDeleted() const { 161 return deleted_; 162 } 163 164 void GetProgramiv(GLenum pname, GLint* params); 165 IsValid()166 bool IsValid() const { 167 return valid_; 168 } 169 170 bool AttachShader(ShaderManager* manager, Shader* shader); 171 bool DetachShader(ShaderManager* manager, Shader* shader); 172 173 bool CanLink() const; 174 175 // Performs glLinkProgram and related activities. 176 bool Link(ShaderManager* manager, 177 ShaderTranslator* vertex_translator, 178 ShaderTranslator* fragment_shader, 179 VaryingsPackingOption varyings_packing_option, 180 const ShaderCacheCallback& shader_callback); 181 182 // Performs glValidateProgram and related activities. 183 void Validate(); 184 log_info()185 const std::string* log_info() const { 186 return log_info_.get(); 187 } 188 InUse()189 bool InUse() const { 190 DCHECK_GE(use_count_, 0); 191 return use_count_ != 0; 192 } 193 194 // Sets attribute-location binding from a glBindAttribLocation() call. SetAttribLocationBinding(const std::string & attrib,GLint location)195 void SetAttribLocationBinding(const std::string& attrib, GLint location) { 196 bind_attrib_location_map_[attrib] = location; 197 } 198 199 // Sets uniform-location binding from a glBindUniformLocationCHROMIUM call. 200 // returns false if error. 201 bool SetUniformLocationBinding(const std::string& name, GLint location); 202 203 // Detects if there are attribute location conflicts from 204 // glBindAttribLocation() calls. 205 // We only consider the declared attributes in the program. 206 bool DetectAttribLocationBindingConflicts() const; 207 208 // Detects if there are uniforms of the same name but different type 209 // or precision in vertex/fragment shaders. 210 // Return true and set the first found conflicting hashed name to 211 // conflicting_name if such cases are detected. 212 bool DetectUniformsMismatch(std::string* conflicting_name) const; 213 214 // Return true if a varying is statically used in fragment shader, but it 215 // is not declared in vertex shader. 216 bool DetectVaryingsMismatch(std::string* conflicting_name) const; 217 218 // Return true if an uniform and an attribute share the same name. 219 bool DetectGlobalNameConflicts(std::string* conflicting_name) const; 220 221 // Return false if varyings can't be packed into the max available 222 // varying registers. 223 bool CheckVaryingsPacking(VaryingsPackingOption option) const; 224 225 // Visible for testing bind_attrib_location_map()226 const LocationMap& bind_attrib_location_map() const { 227 return bind_attrib_location_map_; 228 } 229 230 private: 231 friend class base::RefCounted<Program>; 232 friend class ProgramManager; 233 234 ~Program(); 235 set_log_info(const char * str)236 void set_log_info(const char* str) { 237 log_info_.reset(str ? new std::string(str) : NULL); 238 } 239 ClearLinkStatus()240 void ClearLinkStatus() { 241 link_status_ = false; 242 } 243 IncUseCount()244 void IncUseCount() { 245 ++use_count_; 246 } 247 DecUseCount()248 void DecUseCount() { 249 --use_count_; 250 DCHECK_GE(use_count_, 0); 251 } 252 MarkAsDeleted()253 void MarkAsDeleted() { 254 DCHECK(!deleted_); 255 deleted_ = true; 256 } 257 258 // Resets the program. 259 void Reset(); 260 261 // Updates the program info after a successful link. 262 void Update(); 263 264 // Process the program log, replacing the hashed names with original names. 265 std::string ProcessLogInfo(const std::string& log); 266 267 // Updates the program log info from GL 268 void UpdateLogInfo(); 269 270 // Clears all the uniforms. 271 void ClearUniforms(std::vector<uint8>* zero_buffer); 272 273 // If long attribate names are mapped during shader translation, call 274 // glBindAttribLocation() again with the mapped names. 275 // This is called right before the glLink() call, but after shaders are 276 // translated. 277 void ExecuteBindAttribLocationCalls(); 278 279 bool AddUniformInfo( 280 GLsizei size, GLenum type, GLint location, GLint fake_base_location, 281 const std::string& name, const std::string& original_name, 282 size_t* next_available_index); 283 284 void GetCorrectedVariableInfo( 285 bool use_uniforms, const std::string& name, std::string* corrected_name, 286 std::string* original_name, GLsizei* size, GLenum* type) const; 287 288 void DetachShaders(ShaderManager* manager); 289 GetUniformInfoIndexFromFakeLocation(GLint fake_location)290 static inline GLint GetUniformInfoIndexFromFakeLocation( 291 GLint fake_location) { 292 return fake_location & 0xFFFF; 293 } 294 GetArrayElementIndexFromFakeLocation(GLint fake_location)295 static inline GLint GetArrayElementIndexFromFakeLocation( 296 GLint fake_location) { 297 return (fake_location >> 16) & 0xFFFF; 298 } 299 300 ProgramManager* manager_; 301 302 int use_count_; 303 304 GLsizei max_attrib_name_length_; 305 306 // Attrib by index. 307 AttribInfoVector attrib_infos_; 308 309 // Attrib by location to index. 310 std::vector<GLint> attrib_location_to_index_map_; 311 312 GLsizei max_uniform_name_length_; 313 314 // Uniform info by index. 315 UniformInfoVector uniform_infos_; 316 317 // The indices of the uniforms that are samplers. 318 SamplerIndices sampler_indices_; 319 320 // The program this Program is tracking. 321 GLuint service_id_; 322 323 // Shaders by type of shader. 324 scoped_refptr<Shader> 325 attached_shaders_[kMaxAttachedShaders]; 326 327 // True if this program is marked as deleted. 328 bool deleted_; 329 330 // This is true if glLinkProgram was successful at least once. 331 bool valid_; 332 333 // This is true if glLinkProgram was successful last time it was called. 334 bool link_status_; 335 336 // True if the uniforms have been cleared. 337 bool uniforms_cleared_; 338 339 // This is different than uniform_infos_.size() because 340 // that is a sparce array. 341 GLint num_uniforms_; 342 343 // Log info 344 scoped_ptr<std::string> log_info_; 345 346 // attribute-location binding map from glBindAttribLocation() calls. 347 LocationMap bind_attrib_location_map_; 348 349 // uniform-location binding map from glBindUniformLocationCHROMIUM() calls. 350 LocationMap bind_uniform_location_map_; 351 }; 352 353 // Tracks the Programs. 354 // 355 // NOTE: To support shared resources an instance of this class will 356 // need to be shared by multiple GLES2Decoders. 357 class GPU_EXPORT ProgramManager { 358 public: 359 explicit ProgramManager(ProgramCache* program_cache, 360 uint32 max_varying_vectors); 361 ~ProgramManager(); 362 363 // Must call before destruction. 364 void Destroy(bool have_context); 365 366 // Creates a new program. 367 Program* CreateProgram(GLuint client_id, GLuint service_id); 368 369 // Gets a program. 370 Program* GetProgram(GLuint client_id); 371 372 // Gets a client id for a given service id. 373 bool GetClientId(GLuint service_id, GLuint* client_id) const; 374 375 // Gets the shader cache 376 ProgramCache* program_cache() const; 377 378 // Marks a program as deleted. If it is not used the program will be deleted. 379 void MarkAsDeleted(ShaderManager* shader_manager, Program* program); 380 381 // Marks a program as used. 382 void UseProgram(Program* program); 383 384 // Makes a program as unused. If deleted the program will be removed. 385 void UnuseProgram(ShaderManager* shader_manager, Program* program); 386 387 // Clears the uniforms for this program. 388 void ClearUniforms(Program* program); 389 390 // Returns true if prefix is invalid for gl. 391 static bool IsInvalidPrefix(const char* name, size_t length); 392 393 // Check if a Program is owned by this ProgramManager. 394 bool IsOwned(Program* program); 395 396 static int32 MakeFakeLocation(int32 index, int32 element); 397 max_varying_vectors()398 uint32 max_varying_vectors() const { 399 return max_varying_vectors_; 400 } 401 402 private: 403 friend class Program; 404 405 void StartTracking(Program* program); 406 void StopTracking(Program* program); 407 408 void RemoveProgramInfoIfUnused( 409 ShaderManager* shader_manager, Program* program); 410 411 // Info for each "successfully linked" program by service side program Id. 412 // TODO(gman): Choose a faster container. 413 typedef std::map<GLuint, scoped_refptr<Program> > ProgramMap; 414 ProgramMap programs_; 415 416 // Counts the number of Program allocated with 'this' as its manager. 417 // Allows to check no Program will outlive this. 418 unsigned int program_count_; 419 420 bool have_context_; 421 422 // Used to clear uniforms. 423 std::vector<uint8> zero_; 424 425 ProgramCache* program_cache_; 426 427 uint32 max_varying_vectors_; 428 429 DISALLOW_COPY_AND_ASSIGN(ProgramManager); 430 }; 431 432 } // namespace gles2 433 } // namespace gpu 434 435 #endif // GPU_COMMAND_BUFFER_SERVICE_PROGRAM_MANAGER_H_ 436