/*
* Copyright (C) 2015 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 "APM::SessionRoute"
//#define LOG_NDEBUG 0
#include "SessionRoute.h"
#include "HwModule.h"
#include "AudioGain.h"
#include "DeviceDescriptor.h"
#include <utils/Log.h>
namespace android {
// --- SessionRoute class implementation
void SessionRoute::log(const char* prefix)
{
ALOGI("%s[SessionRoute strm:0x%X, src:%d, sess:0x%X, dev:0x%X refs:%d act:%d",
prefix, mStreamType, mSource, mSession,
mDeviceDescriptor != 0 ? mDeviceDescriptor->type() : AUDIO_DEVICE_NONE,
mRefCount, mActivityCount);
}
// --- SessionRouteMap class implementation
bool SessionRouteMap::hasRoute(audio_session_t session)
{
return indexOfKey(session) >= 0 && valueFor(session)->mDeviceDescriptor != 0;
}
bool SessionRouteMap::getAndClearRouteChanged(audio_session_t session)
{
if (indexOfKey(session) >= 0) {
if (valueFor(session)->mChanged) {
valueFor(session)->mChanged = false;
return true;
}
}
return false;
}
void SessionRouteMap::removeRoute(audio_session_t session)
{
sp<SessionRoute> route = indexOfKey(session) >= 0 ? valueFor(session) : 0;
if (route != 0) {
ALOG_ASSERT(route->mRefCount > 0);
--route->mRefCount;
if (route->mRefCount <= 0) {
removeItem(session);
}
}
}
int SessionRouteMap::incRouteActivity(audio_session_t session)
{
sp<SessionRoute> route = indexOfKey(session) >= 0 ? valueFor(session) : 0;
return route != 0 ? ++(route->mActivityCount) : -1;
}
int SessionRouteMap::decRouteActivity(audio_session_t session)
{
sp<SessionRoute> route = indexOfKey(session) >= 0 ? valueFor(session) : 0;
if (route != 0 && route->mActivityCount > 0) {
return --(route->mActivityCount);
} else {
return -1;
}
}
void SessionRouteMap::log(const char* caption)
{
ALOGI("%s ----", caption);
for (size_t index = 0; index < size(); index++) {
valueAt(index)->log(" ");
}
}
void SessionRouteMap::addRoute(audio_session_t session,
audio_stream_type_t streamType,
audio_source_t source,
const sp<DeviceDescriptor>& descriptor,
uid_t uid)
{
if (mMapType == MAPTYPE_INPUT && streamType != SessionRoute::STREAM_TYPE_NA) {
ALOGE("Adding Output Route to InputRouteMap");
return;
} else if (mMapType == MAPTYPE_OUTPUT && source != SessionRoute::SOURCE_TYPE_NA) {
ALOGE("Adding Input Route to OutputRouteMap");
return;
}
sp<SessionRoute> route = indexOfKey(session) >= 0 ? valueFor(session) : 0;
if (route != 0) {
if (descriptor != 0 || route->mDeviceDescriptor != 0) {
route->mChanged = true;
}
route->mRefCount++;
route->mDeviceDescriptor = descriptor;
} else {
route = new SessionRoute(session, streamType, source, descriptor, uid);
route->mRefCount++;
if (descriptor != 0) {
route->mChanged = true;
}
add(session, route);
}
}
audio_devices_t SessionRouteMap::getActiveDeviceForStream(audio_stream_type_t streamType,
const DeviceVector& availableDevices)
{
audio_devices_t device = AUDIO_DEVICE_NONE;
for (size_t index = 0; index < size(); index++) {
sp<SessionRoute> route = valueAt(index);
if (streamType == route->mStreamType && route->isActiveOrChanged()
&& route->mDeviceDescriptor != 0) {
device = route->mDeviceDescriptor->type();
if (!availableDevices.getDevicesFromType(device).isEmpty()) {
break;
}
}
}
return device;
}
} // namespace android