| /* |
| * Copyright (C) 2009 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include "rsObjectBase.h" |
| |
| #ifndef ANDROID_RS_BUILD_FOR_HOST |
| #include "rsContext.h" |
| #else |
| #include "rsContextHostStub.h" |
| #endif |
| |
| using namespace android; |
| using namespace android::renderscript; |
| |
| ObjectBase::ObjectBase(Context *rsc) |
| { |
| mUserRefCount = 0; |
| mSysRefCount = 0; |
| mRSC = NULL; |
| mNext = NULL; |
| mPrev = NULL; |
| mAllocFile = __FILE__; |
| mAllocLine = __LINE__; |
| setContext(rsc); |
| } |
| |
| ObjectBase::~ObjectBase() |
| { |
| //LOGV("~ObjectBase %p ref %i,%i", this, mUserRefCount, mSysRefCount); |
| rsAssert(!mUserRefCount); |
| rsAssert(!mSysRefCount); |
| remove(); |
| } |
| |
| void ObjectBase::dumpLOGV(const char *op) const |
| { |
| if (mName.size()) { |
| LOGV("%s RSobj %p, name %s, refs %i,%i from %s,%i links %p,%p,%p", |
| op, this, mName.string(), mUserRefCount, mSysRefCount, mAllocFile, mAllocLine, mNext, mPrev, mRSC); |
| } else { |
| LOGV("%s RSobj %p, no-name, refs %i,%i from %s,%i links %p,%p,%p", |
| op, this, mUserRefCount, mSysRefCount, mAllocFile, mAllocLine, mNext, mPrev, mRSC); |
| } |
| } |
| |
| void ObjectBase::setContext(Context *rsc) |
| { |
| if (mRSC) { |
| remove(); |
| } |
| rsAssert(rsc); |
| mRSC = rsc; |
| if (rsc) { |
| add(); |
| } |
| } |
| |
| void ObjectBase::incUserRef() const |
| { |
| mUserRefCount ++; |
| //LOGV("ObjectBase %p inc ref %i", this, mRefCount); |
| } |
| |
| void ObjectBase::incSysRef() const |
| { |
| mSysRefCount ++; |
| //LOGV("ObjectBase %p inc ref %i", this, mRefCount); |
| } |
| |
| bool ObjectBase::checkDelete() const |
| { |
| if (!(mSysRefCount | mUserRefCount)) { |
| if (mRSC && mRSC->props.mLogObjects) { |
| dumpLOGV("checkDelete"); |
| } |
| delete this; |
| return true; |
| } |
| return false; |
| } |
| |
| bool ObjectBase::decUserRef() const |
| { |
| rsAssert(mUserRefCount > 0); |
| mUserRefCount --; |
| //dumpObj("decUserRef"); |
| return checkDelete(); |
| } |
| |
| bool ObjectBase::zeroUserRef() const |
| { |
| mUserRefCount = 0; |
| //dumpObj("zeroUserRef"); |
| return checkDelete(); |
| } |
| |
| bool ObjectBase::decSysRef() const |
| { |
| rsAssert(mSysRefCount > 0); |
| mSysRefCount --; |
| //dumpObj("decSysRef"); |
| return checkDelete(); |
| } |
| |
| void ObjectBase::setName(const char *name) |
| { |
| mName.setTo(name); |
| } |
| |
| void ObjectBase::setName(const char *name, uint32_t len) |
| { |
| mName.setTo(name, len); |
| } |
| |
| void ObjectBase::add() const |
| { |
| rsAssert(!mNext); |
| rsAssert(!mPrev); |
| //LOGV("calling add rsc %p", mRSC); |
| mNext = mRSC->mObjHead; |
| if (mRSC->mObjHead) { |
| mRSC->mObjHead->mPrev = this; |
| } |
| mRSC->mObjHead = this; |
| } |
| |
| void ObjectBase::remove() const |
| { |
| //LOGV("calling remove rsc %p", mRSC); |
| if (!mRSC) { |
| rsAssert(!mPrev); |
| rsAssert(!mNext); |
| return; |
| } |
| if (mRSC->mObjHead == this) { |
| mRSC->mObjHead = mNext; |
| } |
| if (mPrev) { |
| mPrev->mNext = mNext; |
| } |
| if (mNext) { |
| mNext->mPrev = mPrev; |
| } |
| mPrev = NULL; |
| mNext = NULL; |
| } |
| |
| void ObjectBase::zeroAllUserRef(Context *rsc) |
| { |
| if (rsc->props.mLogObjects) { |
| LOGV("Forcing release of all outstanding user refs."); |
| } |
| |
| // This operation can be slow, only to be called during context cleanup. |
| const ObjectBase * o = rsc->mObjHead; |
| while (o) { |
| //LOGE("o %p", o); |
| if (o->zeroUserRef()) { |
| // deleted the object and possibly others, restart from head. |
| o = rsc->mObjHead; |
| //LOGE("o head %p", o); |
| } else { |
| o = o->mNext; |
| //LOGE("o next %p", o); |
| } |
| } |
| |
| if (rsc->props.mLogObjects) { |
| LOGV("Objects remaining."); |
| dumpAll(rsc); |
| } |
| } |
| |
| void ObjectBase::dumpAll(Context *rsc) |
| { |
| LOGV("Dumping all objects"); |
| const ObjectBase * o = rsc->mObjHead; |
| while (o) { |
| LOGV(" Object %p", o); |
| o->dumpLOGV(" "); |
| o = o->mNext; |
| } |
| } |
| |
| bool ObjectBase::isValid(const Context *rsc, const ObjectBase *obj) |
| { |
| const ObjectBase * o = rsc->mObjHead; |
| while (o) { |
| if (o == obj) { |
| return true; |
| } |
| o = o->mNext; |
| } |
| return false; |
| } |
| |