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