1 /*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 #include <GLcommon/ShareGroup.h>
17 #include <GLcommon/ObjectNameSpace.h>
18 #include <GLcommon/GLEScontext.h>
19
20 #include "base/Lock.h"
21 #include "base/Lookup.h"
22 #include "GLcommon/FramebufferData.h"
23
24 #include "host-common/logging.h"
25
26 #include <array>
27 #include <utility>
28
toIndex(NamedObjectType type)29 static constexpr int toIndex(NamedObjectType type) {
30 return static_cast<int>(type);
31 }
32
33 struct ShareGroup::ObjectDataAutoLock {
ObjectDataAutoLockShareGroup::ObjectDataAutoLock34 ObjectDataAutoLock(ShareGroup* self) : self(self) {
35 self->lockObjectData();
36 }
~ObjectDataAutoLockShareGroup::ObjectDataAutoLock37 ~ObjectDataAutoLock() {
38 self->unlockObjectData();
39 }
40
41 ShareGroup* self;
42 };
43
ShareGroup(GlobalNameSpace * globalNameSpace,uint64_t sharedGroupID,android::base::Stream * stream,const ObjectData::loadObject_t & loadObject)44 ShareGroup::ShareGroup(GlobalNameSpace *globalNameSpace,
45 uint64_t sharedGroupID,
46 android::base::Stream* stream,
47 const ObjectData::loadObject_t& loadObject) :
48 m_sharedGroupID(sharedGroupID) {
49 ObjectDataAutoLock lock(this);
50 for (int i = 0; i < toIndex(NamedObjectType::NUM_OBJECT_TYPES);
51 i++) {
52 m_nameSpace[i] = new NameSpace(static_cast<NamedObjectType>(i),
53 globalNameSpace, stream, loadObject);
54 }
55 if (stream) {
56 m_needLoadRestore = true;
57 int i = 0;
58 for (auto ns : m_nameSpace) {
59 GL_LOG("ShareGroup::%s: %p: start restore namespace for type %d\n", __func__, this, i);
60 ns->postLoad(
61 [this](NamedObjectType p_type, ObjectLocalName p_localName) {
62 return this->getObjectDataPtrNoLock(p_type, p_localName);
63 });
64 GL_LOG("ShareGroup::%s: %p: finish restore namespace for type %d\n", __func__, this, i);
65 ++i;
66 }
67 }
68 }
69
preSave(GlobalNameSpace * globalNameSpace)70 void ShareGroup::preSave(GlobalNameSpace *globalNameSpace) {
71 ObjectDataAutoLock lock(this);
72 if (m_saveStage == PreSaved) return;
73 assert(m_saveStage == Empty);
74 m_saveStage = PreSaved;
75 m_nameSpace[(int)NamedObjectType::TEXTURE]->preSave(globalNameSpace);
76 }
77
onSave(android::base::Stream * stream)78 void ShareGroup::onSave(android::base::Stream* stream) {
79 // we do not save m_nameSpace
80 ObjectDataAutoLock lock(this);
81 if (m_saveStage == Saved) return;
82 assert(m_saveStage == PreSaved);
83 m_saveStage = Saved;
84 int i = 0;
85 for (auto ns : m_nameSpace) {
86 GL_LOG("ShareGroup::%s: %p: start saving type %d\n", __func__, this, i);
87 ns->onSave(stream);
88 GL_LOG("ShareGroup::%s: %p: finish saving type %d\n", __func__, this, i);
89 ++i;
90 }
91 }
92
postSave(android::base::Stream * stream)93 void ShareGroup::postSave(android::base::Stream* stream) {
94 (void)stream;
95 m_saveStage = Empty;
96 // We need to mark the textures dirty, for those that has been bound to
97 // a potential render target.
98 NameSpace* renderbufferNs = m_nameSpace[(int)NamedObjectType::RENDERBUFFER];
99 for (ObjectDataMap::const_iterator it = renderbufferNs->objDataMapBegin();
100 it != renderbufferNs->objDataMapEnd();
101 it ++) {
102 RenderbufferData* rbData = (RenderbufferData*)it->second.get();
103 rbData->makeTextureDirty();
104 }
105 }
106
postLoadRestore()107 void ShareGroup::postLoadRestore() {
108 android::base::AutoLock lock(m_restoreLock);
109 if (m_needLoadRestore) {
110 int i = 0;
111 for (auto ns : m_nameSpace) {
112 GL_LOG("ShareGroup::%s: %p: start post load restore namespace for type %d\n", __func__, this, i);
113 ns->postLoadRestore([this](NamedObjectType p_type,
114 ObjectLocalName p_localName) {
115 return getGlobalName(p_type, p_localName);
116 });
117 GL_LOG("ShareGroup::%s: %p: end post load restore namespace for type %d\n", __func__, this, i);
118 ++i;
119 }
120 m_needLoadRestore = false;
121 }
122 }
123
needRestore()124 bool ShareGroup::needRestore() {
125 return m_needLoadRestore;
126 }
127
lockObjectData()128 void ShareGroup::lockObjectData() {
129 while (m_objectsDataLock.test_and_set(std::memory_order_acquire)) {
130 ;
131 }
132 }
133
unlockObjectData()134 void ShareGroup::unlockObjectData() {
135 m_objectsDataLock.clear(std::memory_order_release);
136 }
137
~ShareGroup()138 ShareGroup::~ShareGroup()
139 {
140 {
141 android::base::AutoLock lock(m_namespaceLock);
142 ObjectDataAutoLock objDataLock(this);
143 for (auto n : m_nameSpace) {
144 delete n;
145 }
146 }
147 }
148
149 ObjectLocalName
genName(GenNameInfo genNameInfo,ObjectLocalName p_localName,bool genLocal)150 ShareGroup::genName(GenNameInfo genNameInfo,
151 ObjectLocalName p_localName,
152 bool genLocal)
153 {
154 assert(genNameInfo.m_type != NamedObjectType::FRAMEBUFFER);
155 if (toIndex(genNameInfo.m_type) >=
156 toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
157 return 0;
158 }
159
160 android::base::AutoLock lock(m_namespaceLock);
161 ObjectLocalName localName =
162 m_nameSpace[toIndex(genNameInfo.m_type)]->genName(
163 genNameInfo,
164 p_localName, genLocal);
165 return localName;
166 }
167
genName(NamedObjectType namedObjectType,ObjectLocalName p_localName,bool genLocal)168 ObjectLocalName ShareGroup::genName(NamedObjectType namedObjectType,
169 ObjectLocalName p_localName,
170 bool genLocal) {
171 return genName(GenNameInfo(namedObjectType), p_localName, genLocal);
172 }
173
genName(ShaderProgramType shaderProgramType,ObjectLocalName p_localName,bool genLocal,GLuint existingGlobal)174 ObjectLocalName ShareGroup::genName(ShaderProgramType shaderProgramType,
175 ObjectLocalName p_localName,
176 bool genLocal,
177 GLuint existingGlobal) {
178 return genName(GenNameInfo(shaderProgramType, existingGlobal), p_localName, genLocal);
179 }
180
181 unsigned int
getGlobalName(NamedObjectType p_type,ObjectLocalName p_localName)182 ShareGroup::getGlobalName(NamedObjectType p_type,
183 ObjectLocalName p_localName)
184 {
185 assert(p_type != NamedObjectType::FRAMEBUFFER);
186 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
187 return 0;
188 }
189 android::base::AutoLock lock(m_namespaceLock);
190 return m_nameSpace[toIndex(p_type)]->getGlobalName(p_localName);
191 }
192
193 ObjectLocalName
getLocalName(NamedObjectType p_type,unsigned int p_globalName)194 ShareGroup::getLocalName(NamedObjectType p_type,
195 unsigned int p_globalName)
196 {
197 assert(p_type != NamedObjectType::FRAMEBUFFER);
198 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
199 return 0;
200 }
201
202 android::base::AutoLock lock(m_namespaceLock);
203 return m_nameSpace[toIndex(p_type)]->getLocalName(p_globalName);
204 }
205
getNamedObject(NamedObjectType p_type,ObjectLocalName p_localName)206 NamedObjectPtr ShareGroup::getNamedObject(NamedObjectType p_type,
207 ObjectLocalName p_localName) {
208 assert(p_type != NamedObjectType::FRAMEBUFFER);
209 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
210 return 0;
211 }
212
213 android::base::AutoLock lock(m_namespaceLock);
214 return m_nameSpace[toIndex(p_type)]->getNamedObject(p_localName);
215 }
216
217 void
deleteName(NamedObjectType p_type,ObjectLocalName p_localName)218 ShareGroup::deleteName(NamedObjectType p_type, ObjectLocalName p_localName)
219 {
220 assert(p_type != NamedObjectType::FRAMEBUFFER);
221 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
222 return;
223 }
224
225 android::base::AutoLock lock(m_namespaceLock);
226 ObjectDataAutoLock objDataLock(this);
227 m_nameSpace[toIndex(p_type)]->deleteName(p_localName);
228 }
229
230 bool
isObject(NamedObjectType p_type,ObjectLocalName p_localName)231 ShareGroup::isObject(NamedObjectType p_type, ObjectLocalName p_localName)
232 {
233 assert(p_type != NamedObjectType::FRAMEBUFFER);
234 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
235 return 0;
236 }
237
238 android::base::AutoLock lock(m_namespaceLock);
239 return m_nameSpace[toIndex(p_type)]->isObject(p_localName);
240 }
241
242 void
replaceGlobalObject(NamedObjectType p_type,ObjectLocalName p_localName,NamedObjectPtr p_globalObject)243 ShareGroup::replaceGlobalObject(NamedObjectType p_type,
244 ObjectLocalName p_localName,
245 NamedObjectPtr p_globalObject)
246 {
247 assert(p_type != NamedObjectType::FRAMEBUFFER);
248 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
249 return;
250 }
251
252 android::base::AutoLock lock(m_namespaceLock);
253 m_nameSpace[toIndex(p_type)]->replaceGlobalObject(p_localName,
254 p_globalObject);
255 }
256
257 void
setGlobalObject(NamedObjectType p_type,ObjectLocalName p_localName,NamedObjectPtr p_globalObject)258 ShareGroup::setGlobalObject(NamedObjectType p_type,
259 ObjectLocalName p_localName,
260 NamedObjectPtr p_globalObject)
261 {
262 assert(p_type != NamedObjectType::FRAMEBUFFER);
263 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
264 return;
265 }
266
267 android::base::AutoLock lock(m_namespaceLock);
268 m_nameSpace[toIndex(p_type)]->setGlobalObject(p_localName,
269 p_globalObject);
270 }
271
272 void
setObjectData(NamedObjectType p_type,ObjectLocalName p_localName,ObjectDataPtr data)273 ShareGroup::setObjectData(NamedObjectType p_type,
274 ObjectLocalName p_localName,
275 ObjectDataPtr data) {
276 ObjectDataAutoLock lock(this);
277 setObjectDataLocked(p_type, p_localName, std::move(data));
278 }
279
280 void
setObjectDataLocked(NamedObjectType p_type,ObjectLocalName p_localName,ObjectDataPtr && data)281 ShareGroup::setObjectDataLocked(NamedObjectType p_type,
282 ObjectLocalName p_localName,
283 ObjectDataPtr&& data)
284 {
285 assert(p_type != NamedObjectType::FRAMEBUFFER);
286 if (toIndex(p_type) >= toIndex(NamedObjectType::NUM_OBJECT_TYPES)) {
287 return;
288 }
289 m_nameSpace[toIndex(p_type)]->setObjectData(p_localName, data);
290 }
291
getObjectDataPtrNoLock(NamedObjectType p_type,ObjectLocalName p_localName)292 const ObjectDataPtr& ShareGroup::getObjectDataPtrNoLock(
293 NamedObjectType p_type, ObjectLocalName p_localName) {
294 assert(p_type != NamedObjectType::FRAMEBUFFER);
295 return m_nameSpace[toIndex(p_type)]->getObjectDataPtr(p_localName);
296 }
297
getObjectData(NamedObjectType p_type,ObjectLocalName p_localName)298 ObjectData* ShareGroup::getObjectData(NamedObjectType p_type,
299 ObjectLocalName p_localName) {
300 if (toIndex(p_type) >=
301 toIndex(NamedObjectType::NUM_OBJECT_TYPES))
302 return nullptr;
303
304 ObjectDataAutoLock lock(this);
305 return getObjectDataPtrNoLock(p_type, p_localName).get();
306 }
307
getObjectDataPtr(NamedObjectType p_type,ObjectLocalName p_localName)308 ObjectDataPtr ShareGroup::getObjectDataPtr(NamedObjectType p_type,
309 ObjectLocalName p_localName)
310 {
311 if (toIndex(p_type) >=
312 toIndex(NamedObjectType::NUM_OBJECT_TYPES))
313 return {};
314
315 ObjectDataAutoLock lock(this);
316 return getObjectDataPtrNoLock(p_type, p_localName);
317 }
318
319 #define CC_LIKELY( exp ) (__builtin_expect( !!(exp), true ))
320 #define CC_UNLIKELY( exp ) (__builtin_expect( !!(exp), false ))
321
ensureObjectOnBind(NamedObjectType p_type,ObjectLocalName p_localName)322 unsigned int ShareGroup::ensureObjectOnBind(NamedObjectType p_type, ObjectLocalName p_localName) {
323 android::base::AutoLock lock(m_namespaceLock);
324
325 auto ns = m_nameSpace[toIndex(p_type)];
326
327 bool isObj;
328 unsigned int globalName = ns->getGlobalName(p_localName, &isObj);
329
330 if (CC_LIKELY(isObj)) {
331 bool everBound = ns->everBound(p_localName);
332 if (CC_LIKELY(everBound)) return globalName;
333
334 auto ptr = ns->getObjectDataPtr(p_localName);
335
336 if (ptr) {
337 switch (p_type) {
338 case NamedObjectType::VERTEXBUFFER: {
339 auto vbo = ((GLESbuffer*)(ptr.get()));
340 vbo->setBinded();
341 break;
342 }
343 // TODO: Add other object types here
344 default:
345 fprintf(stderr, "%s: Warning: Unhandled object type 0x%x\n",
346 __func__, (uint32_t)p_type);
347 break;
348 }
349 }
350
351 ns->setBoundAtLeastOnce(p_localName);
352 return globalName;
353 }
354
355 // No such object, generate one and bind it
356 bool genLocal = false;
357 auto gi = GenNameInfo(p_type);
358 ns->genName( gi, p_localName, genLocal);
359
360 switch (p_type) {
361 case NamedObjectType::VERTEXBUFFER: {
362 auto vbo = new GLESbuffer;
363 vbo->setBinded();
364 ns->setObjectData(p_localName, ObjectDataPtr(vbo));
365 break;
366 }
367 // TODO: Add other object types here
368 default:
369 fprintf(stderr, "%s: Warning: Unhandled object type 0x%x\n",
370 __func__, (uint32_t)p_type);
371 break;
372 }
373
374 ns->setBoundAtLeastOnce(p_localName);
375 return (uint32_t)(ns->getGlobalName(p_localName));
376 }
377
ObjectNameManager(GlobalNameSpace * globalNameSpace)378 ObjectNameManager::ObjectNameManager(GlobalNameSpace *globalNameSpace) :
379 m_globalNameSpace(globalNameSpace) {}
380
381 ShareGroupPtr
createShareGroup(void * p_groupName,uint64_t sharedGroupID,android::base::Stream * stream,const ObjectData::loadObject_t & loadObject)382 ObjectNameManager::createShareGroup(void *p_groupName, uint64_t sharedGroupID,
383 android::base::Stream* stream, const ObjectData::loadObject_t& loadObject)
384 {
385 android::base::AutoLock lock(m_lock);
386
387 ShareGroupPtr& shareGroupReturn = m_groups[p_groupName];
388 if (!shareGroupReturn) {
389 if (!sharedGroupID) {
390 while (m_nextSharedGroupID == 0 ||
391 android::base::contains(m_usedSharedGroupIDs,
392 m_nextSharedGroupID)) {
393 m_nextSharedGroupID ++;
394 }
395 sharedGroupID = m_nextSharedGroupID;
396 m_usedSharedGroupIDs.insert(sharedGroupID);
397 ++m_nextSharedGroupID;
398 } else {
399 assert(!m_usedSharedGroupIDs.count(sharedGroupID));
400 m_usedSharedGroupIDs.insert(sharedGroupID);
401 }
402 shareGroupReturn.reset(new ShareGroup(m_globalNameSpace, sharedGroupID,
403 stream, loadObject));
404 } else {
405 assert(sharedGroupID == 0
406 || sharedGroupID == shareGroupReturn->getId());
407 }
408
409 return shareGroupReturn;
410 }
411
412 ShareGroupPtr
getShareGroup(void * p_groupName)413 ObjectNameManager::getShareGroup(void *p_groupName)
414 {
415 android::base::AutoLock lock(m_lock);
416
417 ShareGroupPtr shareGroupReturn;
418
419 ShareGroupsMap::iterator s( m_groups.find(p_groupName) );
420 if (s != m_groups.end()) {
421 shareGroupReturn = (*s).second;
422 }
423
424 return shareGroupReturn;
425 }
426
427 ShareGroupPtr
attachShareGroup(void * p_groupName,void * p_existingGroupName)428 ObjectNameManager::attachShareGroup(void *p_groupName,
429 void *p_existingGroupName)
430 {
431 android::base::AutoLock lock(m_lock);
432
433 ShareGroupsMap::iterator s( m_groups.find(p_existingGroupName) );
434 if (s == m_groups.end()) {
435 // ShareGroup is not found !!!
436 return ShareGroupPtr();
437 }
438
439 ShareGroupPtr shareGroupReturn((*s).second);
440 if (m_groups.find(p_groupName) == m_groups.end()) {
441 m_groups.emplace(p_groupName, shareGroupReturn);
442 m_usedSharedGroupIDs.insert(shareGroupReturn->getId());
443 }
444 return shareGroupReturn;
445 }
446
attachOrCreateShareGroup(void * p_groupName,uint64_t p_existingGroupID,android::base::Stream * stream,const ObjectData::loadObject_t & loadObject)447 ShareGroupPtr ObjectNameManager::attachOrCreateShareGroup(void *p_groupName,
448 uint64_t p_existingGroupID, android::base::Stream* stream,
449 const ObjectData::loadObject_t& loadObject) {
450 assert(m_groups.find(p_groupName) == m_groups.end());
451 ShareGroupsMap::iterator ite = p_existingGroupID ? m_groups.begin()
452 : m_groups.end();
453 while (ite != m_groups.end() && ite->second->getId() != p_existingGroupID) {
454 ++ite;
455 }
456 if (ite == m_groups.end()) {
457 return createShareGroup(p_groupName, p_existingGroupID, stream,
458 loadObject);
459 } else {
460 return attachShareGroup(p_groupName, ite->first);
461 }
462 }
463
464 void
deleteShareGroup(void * p_groupName)465 ObjectNameManager::deleteShareGroup(void *p_groupName)
466 {
467 android::base::AutoLock lock(m_lock);
468 auto sharedGroup = m_groups.find(p_groupName);
469 if (sharedGroup == m_groups.end()) return;
470 m_usedSharedGroupIDs.erase(sharedGroup->second->getId());
471 m_groups.erase(sharedGroup);
472 }
473
getGlobalContext()474 void *ObjectNameManager::getGlobalContext()
475 {
476 android::base::AutoLock lock(m_lock);
477 return m_groups.empty() ? nullptr : m_groups.begin()->first;
478 }
479
preSave()480 void ObjectNameManager::preSave() {
481 for (auto& shareGroup : m_groups) {
482 shareGroup.second->preSave(m_globalNameSpace);
483 }
484 }
485