/* * 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