xref: /aosp_15_r20/frameworks/wilhelm/src/android/CallbackProtector.cpp (revision bebae9c0e76121f8312ccb50385c080b3a0b023c)
1*bebae9c0SAndroid Build Coastguard Worker /*
2*bebae9c0SAndroid Build Coastguard Worker  * Copyright (C) 2011 The Android Open Source Project
3*bebae9c0SAndroid Build Coastguard Worker  *
4*bebae9c0SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*bebae9c0SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*bebae9c0SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*bebae9c0SAndroid Build Coastguard Worker  *
8*bebae9c0SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*bebae9c0SAndroid Build Coastguard Worker  *
10*bebae9c0SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*bebae9c0SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*bebae9c0SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*bebae9c0SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*bebae9c0SAndroid Build Coastguard Worker  * limitations under the License.
15*bebae9c0SAndroid Build Coastguard Worker  */
16*bebae9c0SAndroid Build Coastguard Worker 
17*bebae9c0SAndroid Build Coastguard Worker #include "CallbackProtector.h"
18*bebae9c0SAndroid Build Coastguard Worker #include "sllog.h"
19*bebae9c0SAndroid Build Coastguard Worker 
20*bebae9c0SAndroid Build Coastguard Worker #include <media/stagefright/foundation/ADebug.h>
21*bebae9c0SAndroid Build Coastguard Worker 
22*bebae9c0SAndroid Build Coastguard Worker //--------------------------------------------------------------------------------------------------
23*bebae9c0SAndroid Build Coastguard Worker namespace android {
24*bebae9c0SAndroid Build Coastguard Worker 
25*bebae9c0SAndroid Build Coastguard Worker 
CallbackProtector()26*bebae9c0SAndroid Build Coastguard Worker CallbackProtector::CallbackProtector() : RefBase(),
27*bebae9c0SAndroid Build Coastguard Worker         mSafeToEnterCb(true),
28*bebae9c0SAndroid Build Coastguard Worker         mCbCount(0)
29*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
30*bebae9c0SAndroid Build Coastguard Worker         , mCallbackThread((pthread_t) NULL),
31*bebae9c0SAndroid Build Coastguard Worker         mCallbackTid(0),
32*bebae9c0SAndroid Build Coastguard Worker         mRequesterThread((pthread_t) NULL),
33*bebae9c0SAndroid Build Coastguard Worker         mRequesterTid(0)
34*bebae9c0SAndroid Build Coastguard Worker #endif
35*bebae9c0SAndroid Build Coastguard Worker {
36*bebae9c0SAndroid Build Coastguard Worker }
37*bebae9c0SAndroid Build Coastguard Worker 
38*bebae9c0SAndroid Build Coastguard Worker 
~CallbackProtector()39*bebae9c0SAndroid Build Coastguard Worker CallbackProtector::~CallbackProtector() {
40*bebae9c0SAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
41*bebae9c0SAndroid Build Coastguard Worker     if (mCbCount) {
42*bebae9c0SAndroid Build Coastguard Worker         SL_LOGE("Callback protector detected an active callback after destroy");
43*bebae9c0SAndroid Build Coastguard Worker     }
44*bebae9c0SAndroid Build Coastguard Worker 
45*bebae9c0SAndroid Build Coastguard Worker }
46*bebae9c0SAndroid Build Coastguard Worker 
47*bebae9c0SAndroid Build Coastguard Worker 
48*bebae9c0SAndroid Build Coastguard Worker // static
enterCbIfOk(const sp<CallbackProtector> & protector)49*bebae9c0SAndroid Build Coastguard Worker bool CallbackProtector::enterCbIfOk(const sp<CallbackProtector> &protector) {
50*bebae9c0SAndroid Build Coastguard Worker     if (protector != 0) {
51*bebae9c0SAndroid Build Coastguard Worker         return protector->enterCb();
52*bebae9c0SAndroid Build Coastguard Worker     } else {
53*bebae9c0SAndroid Build Coastguard Worker         SL_LOGE("Callback protector is missing");
54*bebae9c0SAndroid Build Coastguard Worker         return false;
55*bebae9c0SAndroid Build Coastguard Worker     }
56*bebae9c0SAndroid Build Coastguard Worker }
57*bebae9c0SAndroid Build Coastguard Worker 
58*bebae9c0SAndroid Build Coastguard Worker 
enterCb()59*bebae9c0SAndroid Build Coastguard Worker bool CallbackProtector::enterCb() {
60*bebae9c0SAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
61*bebae9c0SAndroid Build Coastguard Worker     if (mSafeToEnterCb) {
62*bebae9c0SAndroid Build Coastguard Worker         mCbCount++;
63*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
64*bebae9c0SAndroid Build Coastguard Worker         if (mCbCount > 1) {
65*bebae9c0SAndroid Build Coastguard Worker             SL_LOGV("Callback protector allowed multiple or nested callback entry: %u", mCbCount);
66*bebae9c0SAndroid Build Coastguard Worker         } else {
67*bebae9c0SAndroid Build Coastguard Worker             mCallbackThread = pthread_self();
68*bebae9c0SAndroid Build Coastguard Worker             mCallbackTid = gettid();
69*bebae9c0SAndroid Build Coastguard Worker         }
70*bebae9c0SAndroid Build Coastguard Worker #endif
71*bebae9c0SAndroid Build Coastguard Worker     } else {
72*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
73*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector denied callback entry by thread %p tid %d during destroy"
74*bebae9c0SAndroid Build Coastguard Worker                 " requested by thread %p tid %d",
75*bebae9c0SAndroid Build Coastguard Worker                 (void *) pthread_self(), gettid(),
76*bebae9c0SAndroid Build Coastguard Worker                 (void *) mRequesterThread, mRequesterTid);
77*bebae9c0SAndroid Build Coastguard Worker #else
78*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector denied callback entry during destroy");
79*bebae9c0SAndroid Build Coastguard Worker #endif
80*bebae9c0SAndroid Build Coastguard Worker     }
81*bebae9c0SAndroid Build Coastguard Worker     return mSafeToEnterCb;
82*bebae9c0SAndroid Build Coastguard Worker }
83*bebae9c0SAndroid Build Coastguard Worker 
84*bebae9c0SAndroid Build Coastguard Worker 
exitCb()85*bebae9c0SAndroid Build Coastguard Worker void CallbackProtector::exitCb() {
86*bebae9c0SAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
87*bebae9c0SAndroid Build Coastguard Worker 
88*bebae9c0SAndroid Build Coastguard Worker     CHECK(mCbCount > 0);
89*bebae9c0SAndroid Build Coastguard Worker     mCbCount--;
90*bebae9c0SAndroid Build Coastguard Worker 
91*bebae9c0SAndroid Build Coastguard Worker     if (mCbCount == 0) {
92*bebae9c0SAndroid Build Coastguard Worker         if (!mSafeToEnterCb) {
93*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
94*bebae9c0SAndroid Build Coastguard Worker             SL_LOGV("Callback protector detected return from callback by thread %p tid %d during"
95*bebae9c0SAndroid Build Coastguard Worker                     " destroy requested by thread %p tid %d",
96*bebae9c0SAndroid Build Coastguard Worker                     (void *) mCallbackThread, mCallbackTid,
97*bebae9c0SAndroid Build Coastguard Worker                     (void *) mRequesterThread, mRequesterTid);
98*bebae9c0SAndroid Build Coastguard Worker #else
99*bebae9c0SAndroid Build Coastguard Worker             SL_LOGV("Callback protector detected return from callback during destroy");
100*bebae9c0SAndroid Build Coastguard Worker #endif
101*bebae9c0SAndroid Build Coastguard Worker             mCbExitedCondition.broadcast();
102*bebae9c0SAndroid Build Coastguard Worker         }
103*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
104*bebae9c0SAndroid Build Coastguard Worker         mCallbackThread = (pthread_t) NULL;
105*bebae9c0SAndroid Build Coastguard Worker         mCallbackTid = 0;
106*bebae9c0SAndroid Build Coastguard Worker #endif
107*bebae9c0SAndroid Build Coastguard Worker     }
108*bebae9c0SAndroid Build Coastguard Worker }
109*bebae9c0SAndroid Build Coastguard Worker 
110*bebae9c0SAndroid Build Coastguard Worker 
requestCbExitAndWait()111*bebae9c0SAndroid Build Coastguard Worker void CallbackProtector::requestCbExitAndWait() {
112*bebae9c0SAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
113*bebae9c0SAndroid Build Coastguard Worker     mSafeToEnterCb = false;
114*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
115*bebae9c0SAndroid Build Coastguard Worker     mRequesterThread = pthread_self();
116*bebae9c0SAndroid Build Coastguard Worker     mRequesterTid = gettid();
117*bebae9c0SAndroid Build Coastguard Worker #endif
118*bebae9c0SAndroid Build Coastguard Worker     while (mCbCount) {
119*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
120*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector detected in-progress callback by thread %p tid %d during"
121*bebae9c0SAndroid Build Coastguard Worker                 " blocking destroy requested by thread %p tid %d",
122*bebae9c0SAndroid Build Coastguard Worker                 (void *) mCallbackThread, mCallbackTid,
123*bebae9c0SAndroid Build Coastguard Worker                 (void *) pthread_self(), gettid());
124*bebae9c0SAndroid Build Coastguard Worker #else
125*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector detected in-progress callback during blocking destroy");
126*bebae9c0SAndroid Build Coastguard Worker #endif
127*bebae9c0SAndroid Build Coastguard Worker         mCbExitedCondition.wait(mLock);
128*bebae9c0SAndroid Build Coastguard Worker     }
129*bebae9c0SAndroid Build Coastguard Worker }
130*bebae9c0SAndroid Build Coastguard Worker 
131*bebae9c0SAndroid Build Coastguard Worker 
requestCbExit()132*bebae9c0SAndroid Build Coastguard Worker void CallbackProtector::requestCbExit() {
133*bebae9c0SAndroid Build Coastguard Worker     Mutex::Autolock _l(mLock);
134*bebae9c0SAndroid Build Coastguard Worker     mSafeToEnterCb = false;
135*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
136*bebae9c0SAndroid Build Coastguard Worker     mRequesterThread = pthread_self();
137*bebae9c0SAndroid Build Coastguard Worker     mRequesterTid = gettid();
138*bebae9c0SAndroid Build Coastguard Worker #endif
139*bebae9c0SAndroid Build Coastguard Worker     if (mCbCount) {
140*bebae9c0SAndroid Build Coastguard Worker #ifdef USE_DEBUG
141*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector detected in-progress callback by thread %p tid %d during"
142*bebae9c0SAndroid Build Coastguard Worker                 " non-blocking destroy requested by thread %p tid %d",
143*bebae9c0SAndroid Build Coastguard Worker                 (void *) mCallbackThread, mCallbackTid,
144*bebae9c0SAndroid Build Coastguard Worker                 (void *) pthread_self(), gettid());
145*bebae9c0SAndroid Build Coastguard Worker #else
146*bebae9c0SAndroid Build Coastguard Worker         SL_LOGV("Callback protector detected in-progress callback during non-blocking destroy");
147*bebae9c0SAndroid Build Coastguard Worker #endif
148*bebae9c0SAndroid Build Coastguard Worker     }
149*bebae9c0SAndroid Build Coastguard Worker }
150*bebae9c0SAndroid Build Coastguard Worker 
151*bebae9c0SAndroid Build Coastguard Worker } // namespace android
152