You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
213 lines
8.3 KiB
213 lines
8.3 KiB
/*
|
|
* 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.
|
|
*/
|
|
|
|
#define LOG_TAG "IAudioPolicyServiceClient"
|
|
#include <utils/Log.h>
|
|
|
|
#include <stdint.h>
|
|
#include <sys/types.h>
|
|
|
|
#include <binder/Parcel.h>
|
|
|
|
#include <media/IAudioPolicyServiceClient.h>
|
|
#include <media/AudioSystem.h>
|
|
|
|
namespace android {
|
|
|
|
enum {
|
|
PORT_LIST_UPDATE = IBinder::FIRST_CALL_TRANSACTION,
|
|
PATCH_LIST_UPDATE,
|
|
MIX_STATE_UPDATE,
|
|
RECORDING_CONFIGURATION_UPDATE,
|
|
VOLUME_GROUP_CHANGED,
|
|
};
|
|
|
|
// ----------------------------------------------------------------------
|
|
inline void readAudioConfigBaseFromParcel(const Parcel& data, audio_config_base_t *config) {
|
|
config->sample_rate = data.readUint32();
|
|
config->channel_mask = (audio_channel_mask_t) data.readInt32();
|
|
config->format = (audio_format_t) data.readInt32();
|
|
}
|
|
|
|
inline void writeAudioConfigBaseToParcel(Parcel& data, const audio_config_base_t *config)
|
|
{
|
|
data.writeUint32(config->sample_rate);
|
|
data.writeInt32((int32_t) config->channel_mask);
|
|
data.writeInt32((int32_t) config->format);
|
|
}
|
|
|
|
inline void readRecordClientInfoFromParcel(const Parcel& data, record_client_info_t *clientInfo) {
|
|
clientInfo->riid = (audio_unique_id_t) data.readInt32();
|
|
clientInfo->uid = (uid_t) data.readUint32();
|
|
clientInfo->session = (audio_session_t) data.readInt32();
|
|
clientInfo->source = (audio_source_t) data.readInt32();
|
|
data.read(&clientInfo->port_id, sizeof(audio_port_handle_t));
|
|
clientInfo->silenced = data.readBool();
|
|
}
|
|
|
|
inline void writeRecordClientInfoToParcel(Parcel& data, const record_client_info_t *clientInfo) {
|
|
data.writeInt32((int32_t) clientInfo->riid);
|
|
data.writeUint32((uint32_t) clientInfo->uid);
|
|
data.writeInt32((int32_t) clientInfo->session);
|
|
data.writeInt32((int32_t) clientInfo->source);
|
|
data.write(&clientInfo->port_id, sizeof(audio_port_handle_t));
|
|
data.writeBool(clientInfo->silenced);
|
|
}
|
|
|
|
inline void readEffectVectorFromParcel(const Parcel& data,
|
|
std::vector<effect_descriptor_t> *effects) {
|
|
int32_t numEffects = data.readInt32();
|
|
for (int32_t i = 0; i < numEffects; i++) {
|
|
effect_descriptor_t effect;
|
|
if (data.read(&effect, sizeof(effect_descriptor_t)) != NO_ERROR) {
|
|
break;
|
|
}
|
|
(*effects).push_back(effect);
|
|
}
|
|
}
|
|
|
|
inline void writeEffectVectorToParcel(Parcel& data, std::vector<effect_descriptor_t> effects) {
|
|
data.writeUint32((uint32_t) effects.size());
|
|
for (const auto& effect : effects) {
|
|
if (data.write(&effect, sizeof(effect_descriptor_t)) != NO_ERROR) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------
|
|
class BpAudioPolicyServiceClient : public BpInterface<IAudioPolicyServiceClient>
|
|
{
|
|
public:
|
|
explicit BpAudioPolicyServiceClient(const sp<IBinder>& impl)
|
|
: BpInterface<IAudioPolicyServiceClient>(impl)
|
|
{
|
|
}
|
|
|
|
void onAudioPortListUpdate()
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(IAudioPolicyServiceClient::getInterfaceDescriptor());
|
|
remote()->transact(PORT_LIST_UPDATE, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
void onAudioPatchListUpdate()
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(IAudioPolicyServiceClient::getInterfaceDescriptor());
|
|
remote()->transact(PATCH_LIST_UPDATE, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
void onAudioVolumeGroupChanged(volume_group_t group, int flags)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(IAudioPolicyServiceClient::getInterfaceDescriptor());
|
|
data.writeUint32(group);
|
|
data.writeInt32(flags);
|
|
remote()->transact(VOLUME_GROUP_CHANGED, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
void onDynamicPolicyMixStateUpdate(String8 regId, int32_t state)
|
|
{
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(IAudioPolicyServiceClient::getInterfaceDescriptor());
|
|
data.writeString8(regId);
|
|
data.writeInt32(state);
|
|
remote()->transact(MIX_STATE_UPDATE, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
|
|
void onRecordingConfigurationUpdate(int event,
|
|
const record_client_info_t *clientInfo,
|
|
const audio_config_base_t *clientConfig,
|
|
std::vector<effect_descriptor_t> clientEffects,
|
|
const audio_config_base_t *deviceConfig,
|
|
std::vector<effect_descriptor_t> effects,
|
|
audio_patch_handle_t patchHandle,
|
|
audio_source_t source) {
|
|
Parcel data, reply;
|
|
data.writeInterfaceToken(IAudioPolicyServiceClient::getInterfaceDescriptor());
|
|
data.writeInt32(event);
|
|
writeRecordClientInfoToParcel(data, clientInfo);
|
|
writeAudioConfigBaseToParcel(data, clientConfig);
|
|
writeEffectVectorToParcel(data, clientEffects);
|
|
writeAudioConfigBaseToParcel(data, deviceConfig);
|
|
writeEffectVectorToParcel(data, effects);
|
|
data.writeInt32(patchHandle);
|
|
data.writeInt32((int32_t) source);
|
|
remote()->transact(RECORDING_CONFIGURATION_UPDATE, data, &reply, IBinder::FLAG_ONEWAY);
|
|
}
|
|
};
|
|
|
|
IMPLEMENT_META_INTERFACE(AudioPolicyServiceClient, "android.media.IAudioPolicyServiceClient");
|
|
|
|
// ----------------------------------------------------------------------
|
|
|
|
status_t BnAudioPolicyServiceClient::onTransact(
|
|
uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags)
|
|
{
|
|
switch (code) {
|
|
case PORT_LIST_UPDATE: {
|
|
CHECK_INTERFACE(IAudioPolicyServiceClient, data, reply);
|
|
onAudioPortListUpdate();
|
|
return NO_ERROR;
|
|
} break;
|
|
case PATCH_LIST_UPDATE: {
|
|
CHECK_INTERFACE(IAudioPolicyServiceClient, data, reply);
|
|
onAudioPatchListUpdate();
|
|
return NO_ERROR;
|
|
} break;
|
|
case VOLUME_GROUP_CHANGED: {
|
|
CHECK_INTERFACE(IAudioPolicyServiceClient, data, reply);
|
|
volume_group_t group = static_cast<volume_group_t>(data.readUint32());
|
|
int flags = data.readInt32();
|
|
onAudioVolumeGroupChanged(group, flags);
|
|
return NO_ERROR;
|
|
} break;
|
|
case MIX_STATE_UPDATE: {
|
|
CHECK_INTERFACE(IAudioPolicyServiceClient, data, reply);
|
|
String8 regId = data.readString8();
|
|
int32_t state = data.readInt32();
|
|
onDynamicPolicyMixStateUpdate(regId, state);
|
|
return NO_ERROR;
|
|
} break;
|
|
case RECORDING_CONFIGURATION_UPDATE: {
|
|
CHECK_INTERFACE(IAudioPolicyServiceClient, data, reply);
|
|
int event = (int) data.readInt32();
|
|
record_client_info_t clientInfo;
|
|
audio_config_base_t clientConfig;
|
|
audio_config_base_t deviceConfig;
|
|
readRecordClientInfoFromParcel(data, &clientInfo);
|
|
readAudioConfigBaseFromParcel(data, &clientConfig);
|
|
std::vector<effect_descriptor_t> clientEffects;
|
|
readEffectVectorFromParcel(data, &clientEffects);
|
|
readAudioConfigBaseFromParcel(data, &deviceConfig);
|
|
std::vector<effect_descriptor_t> effects;
|
|
readEffectVectorFromParcel(data, &effects);
|
|
audio_patch_handle_t patchHandle = (audio_patch_handle_t) data.readInt32();
|
|
audio_source_t source = (audio_source_t) data.readInt32();
|
|
onRecordingConfigurationUpdate(event, &clientInfo, &clientConfig, clientEffects,
|
|
&deviceConfig, effects, patchHandle, source);
|
|
return NO_ERROR;
|
|
} break;
|
|
default:
|
|
return BBinder::onTransact(code, data, reply, flags);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
} // namespace android
|