|  | /* | 
|  | * Copyright (C) 2018 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. | 
|  | */ | 
|  |  | 
|  | #define LOG_TAG "IPCThreadStateBase" | 
|  |  | 
|  | #include <binderthreadstate/IPCThreadStateBase.h> | 
|  | #include <android-base/macros.h> | 
|  |  | 
|  | #include <utils/Log.h> | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <inttypes.h> | 
|  | #include <pthread.h> | 
|  |  | 
|  | namespace android { | 
|  |  | 
|  | static pthread_mutex_t gTLSMutex = PTHREAD_MUTEX_INITIALIZER; | 
|  | static bool gHaveTLS = false; | 
|  | static pthread_key_t gTLS = 0; | 
|  |  | 
|  | IPCThreadStateBase::IPCThreadStateBase() { | 
|  | pthread_setspecific(gTLS, this); | 
|  | } | 
|  |  | 
|  | IPCThreadStateBase* IPCThreadStateBase::self() | 
|  | { | 
|  | if (gHaveTLS) { | 
|  | restart: | 
|  | const pthread_key_t k = gTLS; | 
|  | IPCThreadStateBase* st = (IPCThreadStateBase*)pthread_getspecific(k); | 
|  | if (st) return st; | 
|  | return new IPCThreadStateBase; | 
|  | } | 
|  |  | 
|  | pthread_mutex_lock(&gTLSMutex); | 
|  | if (!gHaveTLS) { | 
|  | int key_create_value = pthread_key_create(&gTLS, threadDestructor); | 
|  | if (key_create_value != 0) { | 
|  | pthread_mutex_unlock(&gTLSMutex); | 
|  | ALOGW("IPCThreadStateBase::self() unable to create TLS key, expect a crash: %s\n", | 
|  | strerror(key_create_value)); | 
|  | return nullptr; | 
|  | } | 
|  | gHaveTLS = true; | 
|  | } | 
|  | pthread_mutex_unlock(&gTLSMutex); | 
|  | goto restart; | 
|  | } | 
|  |  | 
|  | void IPCThreadStateBase::pushCurrentState(CallState callState) { | 
|  | mCallStateStack.emplace(callState); | 
|  | } | 
|  |  | 
|  | IPCThreadStateBase::CallState IPCThreadStateBase::popCurrentState() { | 
|  | ALOG_ASSERT(mCallStateStack.size > 0); | 
|  | CallState val = mCallStateStack.top(); | 
|  | mCallStateStack.pop(); | 
|  | return val; | 
|  | } | 
|  |  | 
|  | IPCThreadStateBase::CallState IPCThreadStateBase::getCurrentBinderCallState() { | 
|  | if (mCallStateStack.size() > 0) { | 
|  | return mCallStateStack.top(); | 
|  | } | 
|  | return CallState::NONE; | 
|  | } | 
|  |  | 
|  | void IPCThreadStateBase::threadDestructor(void *st) | 
|  | { | 
|  | IPCThreadStateBase* const self = static_cast<IPCThreadStateBase*>(st); | 
|  | if (self) { | 
|  | delete self; | 
|  | } | 
|  | } | 
|  |  | 
|  | }; // namespace android |