/* * Copyright (C) 2017 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 "AAudioClientTracker" //#define LOG_NDEBUG 0 #include #include #include #include #include #include #include #include #include "utility/AAudioUtilities.h" #include "AAudioEndpointManager.h" #include "AAudioServiceEndpoint.h" #include "AAudioClientTracker.h" using namespace android; using namespace aaudio; ANDROID_SINGLETON_STATIC_INSTANCE(AAudioClientTracker); AAudioClientTracker::AAudioClientTracker() : Singleton() { } std::string AAudioClientTracker::dump() const NO_THREAD_SAFETY_ANALYSIS { std::stringstream result; const bool isLocked = AAudio_tryUntilTrue( [this]()->bool { return mLock.try_lock(); } /* f */, 50 /* times */, 20 /* sleepMs */); if (!isLocked) { result << "AAudioClientTracker may be deadlocked\n"; } result << "AAudioClientTracker:\n"; for (const auto& it : mNotificationClients) { result << it.second->dump(); } if (isLocked) { mLock.unlock(); } return result.str(); } // Create a tracker for the client. aaudio_result_t AAudioClientTracker::registerClient(pid_t pid, const sp& client) { ALOGV("registerClient(), calling pid = %d, getpid() = %d\n", pid, getpid()); if (client.get() == nullptr) { ALOGE("AAudioClientTracker::%s() client is NULL!", __func__); android_errorWriteLog(0x534e4554, "116230453"); return AAUDIO_ERROR_NULL; } const std::lock_guard lock(mLock); if (mNotificationClients.count(pid) == 0) { const sp binder = IInterface::asBinder(client); const sp notificationClient = new NotificationClient(pid, binder); mNotificationClients[pid] = notificationClient; const status_t status = binder->linkToDeath(notificationClient); ALOGW_IF(status != NO_ERROR, "registerClient() linkToDeath = %d\n", status); return AAudioConvert_androidToAAudioResult(status); } else { ALOGW("registerClient(%d) already registered!", pid); return AAUDIO_OK; // TODO should this be considered an error } } void AAudioClientTracker::unregisterClient(pid_t pid) { ALOGV("unregisterClient(), calling pid = %d, getpid() = %d\n", pid, getpid()); const std::lock_guard lock(mLock); mNotificationClients.erase(pid); } int32_t AAudioClientTracker::getStreamCount(pid_t pid) { const std::lock_guard lock(mLock); auto it = mNotificationClients.find(pid); if (it != mNotificationClients.end()) { return it->second->getStreamCount(); } else { return 0; // no existing client } } aaudio_result_t AAudioClientTracker::registerClientStream( pid_t pid, const sp& serviceStream) { ALOGV("registerClientStream(%d,)\n", pid); const std::lock_guard lock(mLock); return getNotificationClient_l(pid)->registerClientStream(serviceStream); } // Find the tracker for this process and remove it. aaudio_result_t AAudioClientTracker::unregisterClientStream(pid_t pid, const sp& serviceStream) { ALOGV("unregisterClientStream(%d,)\n", pid); const std::lock_guard lock(mLock); auto it = mNotificationClients.find(pid); if (it != mNotificationClients.end()) { ALOGV("unregisterClientStream(%d,) found NotificationClient\n", pid); it->second->unregisterClientStream(serviceStream); } else { ALOGE("unregisterClientStream(%d,) missing NotificationClient\n", pid); } return AAUDIO_OK; } void AAudioClientTracker::setExclusiveEnabled(pid_t pid, bool enabled) { ALOGD("%s(%d, %d)\n", __func__, pid, enabled); const std::lock_guard lock(mLock); getNotificationClient_l(pid)->setExclusiveEnabled(enabled); } bool AAudioClientTracker::isExclusiveEnabled(pid_t pid) { const std::lock_guard lock(mLock); return getNotificationClient_l(pid)->isExclusiveEnabled(); } sp AAudioClientTracker::getNotificationClient_l(pid_t pid) { sp notificationClient = mNotificationClients[pid]; if (notificationClient == nullptr) { // This will get called the first time the audio server uses this PID. ALOGV("%s(%d,) unrecognized PID\n", __func__, pid); notificationClient = new AAudioClientTracker::NotificationClient(pid, nullptr); mNotificationClients[pid] = notificationClient; } return notificationClient; } // ======================================= // AAudioClientTracker::NotificationClient // ======================================= AAudioClientTracker::NotificationClient::NotificationClient(pid_t pid, const sp& binder) : mProcessId(pid), mBinder(binder) { } int32_t AAudioClientTracker::NotificationClient::getStreamCount() { const std::lock_guard lock(mLock); return mStreams.size(); } aaudio_result_t AAudioClientTracker::NotificationClient::registerClientStream( const sp& serviceStream) { const std::lock_guard lock(mLock); mStreams.insert(serviceStream); return AAUDIO_OK; } aaudio_result_t AAudioClientTracker::NotificationClient::unregisterClientStream( const sp& serviceStream) { const std::lock_guard lock(mLock); mStreams.erase(serviceStream); return AAUDIO_OK; } // Close any open streams for the client. void AAudioClientTracker::NotificationClient::binderDied(const wp& who __unused) { AAudioService *aaudioService = AAudioClientTracker::getInstance().getAAudioService(); if (aaudioService != nullptr) { // Copy the current list of streams to another vector because closing them below // will cause unregisterClientStream() calls back to this object. std::set> streamsToClose; { const std::lock_guard lock(mLock); for (const auto& serviceStream : mStreams) { streamsToClose.insert(serviceStream); } } for (const auto& serviceStream : streamsToClose) { const aaudio_handle_t handle = serviceStream->getHandle(); ALOGW("binderDied() close abandoned stream 0x%08X\n", handle); AAudioHandleInfo handleInfo(DEFAULT_AAUDIO_SERVICE_ID, handle); aaudioService->asAAudioServiceInterface().closeStream(handleInfo); } // mStreams should be empty now } const sp keep(this); AAudioClientTracker::getInstance().unregisterClient(mProcessId); } std::string AAudioClientTracker::NotificationClient::dump() const NO_THREAD_SAFETY_ANALYSIS { std::stringstream result; const bool isLocked = AAudio_tryUntilTrue( [this]()->bool { return mLock.try_lock(); } /* f */, 50 /* times */, 20 /* sleepMs */); if (!isLocked) { result << "AAudioClientTracker::NotificationClient may be deadlocked\n"; } result << " client: pid = " << mProcessId << " has " << mStreams.size() << " streams\n"; for (const auto& serviceStream : mStreams) { result << " stream: 0x" << std::setfill('0') << std::setw(8) << std::hex << serviceStream->getHandle() << std::dec << std::setfill(' ') << "\n"; } if (isLocked) { mLock.unlock(); } return result.str(); }