/* * 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. */ #pragma once #include #include #include #include #include "AudioIODescriptorInterface.h" #include "ClientDescriptor.h" #include "DeviceDescriptor.h" #include "EffectDescriptor.h" #include "IOProfile.h" #include "PolicyAudioPort.h" namespace android { class AudioPolicyMix; class AudioPolicyClientInterface; // descriptor for audio inputs. Used to maintain current configuration of each opened audio input // and keep track of the usage of this input. class AudioInputDescriptor: public AudioPortConfig, public PolicyAudioPortConfig, public AudioIODescriptorInterface, public ClientMapHandler { public: AudioInputDescriptor(const sp& profile, AudioPolicyClientInterface *clientInterface); virtual ~AudioInputDescriptor() = default; audio_module_handle_t getModuleHandle() const; audio_devices_t getDeviceType() const { return (mDevice != nullptr) ? mDevice->type() : AUDIO_DEVICE_NONE; } sp getDevice() const { return mDevice; } void setDevice(const sp &device) { mDevice = device; } DeviceVector supportedDevices() const { return mProfile != nullptr ? mProfile->getSupportedDevices() : DeviceVector(); } void dump(String8 *dst) const override; audio_io_handle_t mIoHandle = AUDIO_IO_HANDLE_NONE; // input handle wp mPolicyMix; // non NULL when used by a dynamic policy const sp mProfile; // I/O profile this output derives from // PolicyAudioPortConfig virtual sp getPolicyAudioPort() const { return mProfile; } // AudioPortConfig virtual status_t applyAudioPortConfig(const struct audio_port_config *config, struct audio_port_config *backupConfig = NULL); virtual void toAudioPortConfig(struct audio_port_config *dstConfig, const struct audio_port_config *srcConfig = NULL) const; virtual sp getAudioPort() const { return mProfile; } void toAudioPort(struct audio_port *port) const; void setPreemptedSessions(const SortedVector& sessions); SortedVector getPreemptedSessions() const; bool hasPreemptedSession(audio_session_t session) const; void clearPreemptedSessions(); bool isActive() const { return mGlobalActiveCount > 0; } bool isSourceActive(audio_source_t source) const; audio_source_t source() const; bool isSoundTrigger() const; sp getHighestPriorityClient() const; audio_attributes_t getHighestPriorityAttributes() const; void setClientActive(const sp& client, bool active); int32_t activeCount() { return mGlobalActiveCount; } void trackEffectEnabled(const sp &effect, bool enabled); EffectDescriptorCollection getEnabledEffects() const; EffectDescriptorCollection getActiveEffects() const; // enabled and not suspended // implementation of AudioIODescriptorInterface audio_config_base_t getConfig() const override; audio_patch_handle_t getPatchHandle() const override; void setPatchHandle(audio_patch_handle_t handle) override; bool isMmap() override { if (getPolicyAudioPort() != nullptr) { return getPolicyAudioPort()->isMmap(); } return false; } status_t open(const audio_config_t *config, const sp &device, audio_source_t source, audio_input_flags_t flags, audio_io_handle_t *input); // Called when a stream is about to be started. // Note: called after setClientActive(client, true) status_t start(); // Called after a stream is stopped // Note: called after setClientActive(client, false) void stop(); void close(); RecordClientVector getClientsForSession(audio_session_t session); RecordClientVector clientsList(bool activeOnly = false, audio_source_t source = AUDIO_SOURCE_DEFAULT, bool preferredDeviceOnly = false) const; void setAppState(audio_port_handle_t portId, app_state_t state); // implementation of ClientMapHandler void addClient(const sp &client) override; // Go over all active clients and suspend or restore effects according highest priority // active use case void checkSuspendEffects(); private: void updateClientRecordingConfiguration(int event, const sp& client); audio_patch_handle_t mPatchHandle = AUDIO_PATCH_HANDLE_NONE; sp mDevice = nullptr; /**< current device this input is routed to */ // Because a preemptible capture session can preempt another one, we end up in an endless loop // situation were each session is allowed to restart after being preempted, // thus preempting the other one which restarts and so on. // To avoid this situation, we store which audio session was preempted when // a particular input started and prevent preemption of this active input by this session. // We also inherit sessions from the preempted input to avoid a 3 way preemption loop etc... SortedVector mPreemptedSessions; AudioPolicyClientInterface * const mClientInterface; int32_t mGlobalActiveCount = 0; // non-client-specific activity ref count EffectDescriptorCollection mEnabledEffects; }; class AudioInputCollection : public DefaultKeyedVector< audio_io_handle_t, sp > { public: bool isSourceActive(audio_source_t source) const; sp getInputFromId(audio_port_handle_t id) const; // count active capture sessions using one of the specified devices. // ignore devices if empty vector is passed uint32_t activeInputsCountOnDevices(const DeviceVector &devices) const; /** * return io handle of active input or 0 if no input is active * Only considers inputs from physical devices (e.g. main mic, headset mic) when * ignoreVirtualInputs is true. */ Vector > getActiveInputs(); sp getInputForClient(audio_port_handle_t portId); void trackEffectEnabled(const sp &effect, bool enabled); /** * @brief clearSessionRoutesForDevice: when a device is disconnected, and if this device has * been chosen as the preferred device by any client, the policy manager shall * prevent from using this device any more by clearing all the session routes involving this * device. * In other words, the preferred device port id of these clients will be resetted to NONE. * @param disconnectedDevice device to be disconnected */ void clearSessionRoutesForDevice(const sp &disconnectedDevice); void dump(String8 *dst) const; }; } // namespace android