OLD | NEW |
(Empty) | |
| 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 #include "gpu/command_buffer/service/program_cache.h" |
| 6 |
| 7 #include "base/memory/scoped_ptr.h" |
| 8 |
| 9 namespace gpu { |
| 10 namespace gles2 { |
| 11 |
| 12 ProgramCache::ProgramCache() {} |
| 13 ProgramCache::~ProgramCache() {} |
| 14 |
| 15 void ProgramCache::Clear() { |
| 16 shader_status_.clear(); |
| 17 link_status_.clear(); |
| 18 ClearBackend(); |
| 19 } |
| 20 |
| 21 ProgramCache::CompiledShaderStatus ProgramCache::GetShaderCompilationStatus( |
| 22 const std::string& shader_src) const { |
| 23 char sha[kHashLength]; |
| 24 ComputeShaderHash(shader_src, sha); |
| 25 const std::string sha_string(sha, kHashLength); |
| 26 |
| 27 CompileStatusMap::const_iterator found = shader_status_.find(sha_string); |
| 28 |
| 29 if (found == shader_status_.end()) { |
| 30 return ProgramCache::COMPILATION_UNKNOWN; |
| 31 } else { |
| 32 return found->second.status; |
| 33 } |
| 34 } |
| 35 |
| 36 void ProgramCache::ShaderCompilationSucceeded( |
| 37 const std::string& shader_src) { |
| 38 char sha[kHashLength]; |
| 39 ComputeShaderHash(shader_src, sha); |
| 40 const std::string sha_string(sha, kHashLength); |
| 41 |
| 42 CompileStatusMap::iterator it = shader_status_.find(sha_string); |
| 43 if (it == shader_status_.end()) { |
| 44 shader_status_[sha_string] = CompiledShaderInfo(COMPILATION_SUCCEEDED); |
| 45 } else { |
| 46 it->second.status = COMPILATION_SUCCEEDED; |
| 47 } |
| 48 } |
| 49 |
| 50 ProgramCache::LinkedProgramStatus ProgramCache::GetLinkedProgramStatus( |
| 51 const std::string& untranslated_a, |
| 52 const std::string& untranslated_b, |
| 53 const std::map<std::string, GLint>* bind_attrib_location_map) const { |
| 54 char a_sha[kHashLength]; |
| 55 char b_sha[kHashLength]; |
| 56 ComputeShaderHash(untranslated_a, a_sha); |
| 57 ComputeShaderHash(untranslated_b, b_sha); |
| 58 |
| 59 char sha[kHashLength]; |
| 60 ComputeProgramHash(a_sha, |
| 61 b_sha, |
| 62 bind_attrib_location_map, |
| 63 sha); |
| 64 const std::string sha_string(sha, kHashLength); |
| 65 |
| 66 LinkStatusMap::const_iterator found = link_status_.find(sha_string); |
| 67 if (found == link_status_.end()) { |
| 68 return ProgramCache::LINK_UNKNOWN; |
| 69 } else { |
| 70 return found->second; |
| 71 } |
| 72 } |
| 73 |
| 74 void ProgramCache::LinkedProgramCacheSuccess( |
| 75 const std::string& shader_a, |
| 76 const std::string& shader_b, |
| 77 const LocationMap* bind_attrib_location_map) { |
| 78 char a_sha[kHashLength]; |
| 79 char b_sha[kHashLength]; |
| 80 ComputeShaderHash(shader_a, a_sha); |
| 81 ComputeShaderHash(shader_b, b_sha); |
| 82 char sha[kHashLength]; |
| 83 ComputeProgramHash(a_sha, |
| 84 b_sha, |
| 85 bind_attrib_location_map, |
| 86 sha); |
| 87 const std::string sha_string(sha, kHashLength); |
| 88 |
| 89 LinkedProgramCacheSuccess(sha_string, |
| 90 std::string(a_sha, kHashLength), |
| 91 std::string(b_sha, kHashLength)); |
| 92 } |
| 93 |
| 94 void ProgramCache::LinkedProgramCacheSuccess(const std::string& program_hash, |
| 95 const std::string& shader_a_hash, |
| 96 const std::string& shader_b_hash) { |
| 97 link_status_[program_hash] = LINK_SUCCEEDED; |
| 98 shader_status_[shader_a_hash].ref_count++; |
| 99 shader_status_[shader_b_hash].ref_count++; |
| 100 } |
| 101 |
| 102 void ProgramCache::ComputeShaderHash(const std::string& str, |
| 103 char* result) const { |
| 104 base::SHA1HashBytes(reinterpret_cast<const unsigned char*>(str.c_str()), |
| 105 str.length(), reinterpret_cast<unsigned char*>(result)); |
| 106 } |
| 107 |
| 108 void ProgramCache::Evict(const std::string& program_hash, |
| 109 const std::string& shader_0_hash, |
| 110 const std::string& shader_1_hash) { |
| 111 CompileStatusMap::iterator info0 = shader_status_.find(shader_0_hash); |
| 112 CompileStatusMap::iterator info1 = shader_status_.find(shader_1_hash); |
| 113 DCHECK(info0 != shader_status_.end()); |
| 114 DCHECK(info1 != shader_status_.end()); |
| 115 DCHECK(info0->second.ref_count > 0); |
| 116 DCHECK(info1->second.ref_count > 0); |
| 117 if (--info0->second.ref_count <= 0) { |
| 118 shader_status_.erase(shader_0_hash); |
| 119 } |
| 120 if (--info1->second.ref_count <= 0) { |
| 121 shader_status_.erase(shader_1_hash); |
| 122 } |
| 123 link_status_.erase(program_hash); |
| 124 } |
| 125 |
| 126 namespace { |
| 127 size_t CalculateMapSize(const std::map<std::string, GLint>* map) { |
| 128 if (!map) { |
| 129 return 0; |
| 130 } |
| 131 std::map<std::string, GLint>::const_iterator it; |
| 132 size_t total = 0; |
| 133 for (it = map->begin(); it != map->end(); ++it) { |
| 134 total += 4 + it->first.length(); |
| 135 } |
| 136 return total; |
| 137 } |
| 138 } // anonymous namespace |
| 139 |
| 140 void ProgramCache::ComputeProgramHash( |
| 141 const char* hashed_shader_0, |
| 142 const char* hashed_shader_1, |
| 143 const std::map<std::string, GLint>* bind_attrib_location_map, |
| 144 char* result) const { |
| 145 const size_t shader0_size = kHashLength; |
| 146 const size_t shader1_size = kHashLength; |
| 147 const size_t map_size = CalculateMapSize(bind_attrib_location_map); |
| 148 const size_t total_size = shader0_size + shader1_size + map_size; |
| 149 |
| 150 scoped_array<unsigned char> buffer(new unsigned char[total_size]); |
| 151 memcpy(buffer.get(), hashed_shader_0, shader0_size); |
| 152 memcpy(&buffer[shader0_size], hashed_shader_1, shader1_size); |
| 153 if (map_size != 0) { |
| 154 // copy our map |
| 155 size_t current_pos = shader0_size + shader1_size; |
| 156 std::map<std::string, GLint>::const_iterator it; |
| 157 for (it = bind_attrib_location_map->begin(); |
| 158 it != bind_attrib_location_map->end(); |
| 159 ++it) { |
| 160 const size_t name_size = it->first.length(); |
| 161 memcpy(&buffer.get()[current_pos], it->first.c_str(), name_size); |
| 162 current_pos += name_size; |
| 163 const GLint value = it->second; |
| 164 buffer[current_pos++] = value >> 24; |
| 165 buffer[current_pos++] = value >> 16; |
| 166 buffer[current_pos++] = value >> 8; |
| 167 buffer[current_pos++] = value; |
| 168 } |
| 169 } |
| 170 base::SHA1HashBytes(buffer.get(), |
| 171 total_size, reinterpret_cast<unsigned char*>(result)); |
| 172 } |
| 173 |
| 174 } // namespace gles2 |
| 175 } // namespace gpu |
OLD | NEW |