/* * 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. */ #include <binder/IServiceManager.h> #include <media/PlayerBase.h> #define max(a, b) ((a) > (b) ? (a) : (b)) #define min(a, b) ((a) < (b) ? (a) : (b)) namespace android { using media::VolumeShaper; //-------------------------------------------------------------------------------------------------- PlayerBase::PlayerBase() : BnPlayer(), mPanMultiplierL(1.0f), mPanMultiplierR(1.0f), mVolumeMultiplierL(1.0f), mVolumeMultiplierR(1.0f), mPIId(PLAYER_PIID_INVALID), mLastReportedEvent(PLAYER_STATE_UNKNOWN) { ALOGD("PlayerBase::PlayerBase()"); // use checkService() to avoid blocking if audio service is not up yet sp<IBinder> binder = defaultServiceManager()->checkService(String16("audio")); if (binder == 0) { ALOGE("PlayerBase(): binding to audio service failed, service up?"); } else { mAudioManager = interface_cast<IAudioManager>(binder); } } PlayerBase::~PlayerBase() { ALOGD("PlayerBase::~PlayerBase()"); baseDestroy(); } void PlayerBase::init(player_type_t playerType, audio_usage_t usage) { if (mAudioManager == 0) { ALOGE("AudioPlayer realize: no audio service, player will not be registered"); } else { mPIId = mAudioManager->trackPlayer(playerType, usage, AUDIO_CONTENT_TYPE_UNKNOWN, this); } } void PlayerBase::baseDestroy() { serviceReleasePlayer(); if (mAudioManager != 0) { mAudioManager.clear(); } } //------------------------------------------------------------------------------ void PlayerBase::servicePlayerEvent(player_state_t event) { if (mAudioManager != 0) { // only report state change Mutex::Autolock _l(mPlayerStateLock); if (event != mLastReportedEvent && mPIId != PLAYER_PIID_INVALID) { mLastReportedEvent = event; mAudioManager->playerEvent(mPIId, event); } } } void PlayerBase::serviceReleasePlayer() { if (mAudioManager != 0 && mPIId != PLAYER_PIID_INVALID) { mAudioManager->releasePlayer(mPIId); } } //FIXME temporary method while some player state is outside of this class void PlayerBase::reportEvent(player_state_t event) { servicePlayerEvent(event); } status_t PlayerBase::startWithStatus() { status_t status = playerStart(); if (status == NO_ERROR) { servicePlayerEvent(PLAYER_STATE_STARTED); } else { ALOGW("PlayerBase::start() error %d", status); } return status; } status_t PlayerBase::pauseWithStatus() { status_t status = playerPause(); if (status == NO_ERROR) { servicePlayerEvent(PLAYER_STATE_PAUSED); } else { ALOGW("PlayerBase::pause() error %d", status); } return status; } status_t PlayerBase::stopWithStatus() { status_t status = playerStop(); if (status == NO_ERROR) { servicePlayerEvent(PLAYER_STATE_STOPPED); } else { ALOGW("PlayerBase::stop() error %d", status); } return status; } //------------------------------------------------------------------------------ // Implementation of IPlayer binder::Status PlayerBase::start() { ALOGD("PlayerBase::start() from IPlayer"); (void)startWithStatus(); return binder::Status::ok(); } binder::Status PlayerBase::pause() { ALOGD("PlayerBase::pause() from IPlayer"); (void)pauseWithStatus(); return binder::Status::ok(); } binder::Status PlayerBase::stop() { ALOGD("PlayerBase::stop() from IPlayer"); (void)stopWithStatus(); return binder::Status::ok(); } binder::Status PlayerBase::setVolume(float vol) { ALOGD("PlayerBase::setVolume() from IPlayer"); { Mutex::Autolock _l(mSettingsLock); mVolumeMultiplierL = vol; mVolumeMultiplierR = vol; } status_t status = playerSetVolume(); if (status != NO_ERROR) { ALOGW("PlayerBase::setVolume() error %d", status); } return binder::Status::fromStatusT(status); } binder::Status PlayerBase::setPan(float pan) { ALOGD("PlayerBase::setPan() from IPlayer"); { Mutex::Autolock _l(mSettingsLock); pan = min(max(-1.0f, pan), 1.0f); if (pan >= 0.0f) { mPanMultiplierL = 1.0f - pan; mPanMultiplierR = 1.0f; } else { mPanMultiplierL = 1.0f; mPanMultiplierR = 1.0f + pan; } } status_t status = playerSetVolume(); if (status != NO_ERROR) { ALOGW("PlayerBase::setPan() error %d", status); } return binder::Status::fromStatusT(status); } binder::Status PlayerBase::setStartDelayMs(int32_t delayMs __unused) { ALOGW("setStartDelay() is not supported"); return binder::Status::ok(); } binder::Status PlayerBase::applyVolumeShaper( const VolumeShaper::Configuration& configuration __unused, const VolumeShaper::Operation& operation __unused) { ALOGW("applyVolumeShaper() is not supported"); return binder::Status::ok(); } } // namespace android