blob: ac893d8f25af6d2e88ac5200348f7f1a4ebfe61d [file] [log] [blame]
/*
* Copyright (C) 2022 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 "AHAL_DownmixContext"
#include <android-base/logging.h>
#include "DownmixContext.h"
using aidl::android::hardware::audio::effect::IEffect;
using aidl::android::hardware::audio::common::getChannelCount;
using aidl::android::media::audio::common::AudioChannelLayout;
namespace aidl::android::hardware::audio::effect {
DownmixContext::DownmixContext(int statusDepth, const Parameter::Common& common)
: EffectContext(statusDepth, common) {
LOG(DEBUG) << __func__;
mState = DOWNMIX_STATE_UNINITIALIZED;
init_params(common);
}
DownmixContext::~DownmixContext() {
LOG(DEBUG) << __func__;
mState = DOWNMIX_STATE_UNINITIALIZED;
}
RetCode DownmixContext::enable() {
LOG(DEBUG) << __func__;
if (mState != DOWNMIX_STATE_INITIALIZED) {
return RetCode::ERROR_EFFECT_LIB_ERROR;
}
mState = DOWNMIX_STATE_ACTIVE;
return RetCode::SUCCESS;
}
RetCode DownmixContext::disable() {
LOG(DEBUG) << __func__;
if (mState != DOWNMIX_STATE_ACTIVE) {
return RetCode::ERROR_EFFECT_LIB_ERROR;
}
mState = DOWNMIX_STATE_INITIALIZED;
return RetCode::SUCCESS;
}
void DownmixContext::reset() {
LOG(DEBUG) << __func__;
disable();
resetBuffer();
}
IEffect::Status DownmixContext::lvmProcess(float* in, float* out, int samples) {
LOG(DEBUG) << __func__ << " in " << in << " out " << out << " sample " << samples;
IEffect::Status status = {EX_ILLEGAL_ARGUMENT, 0, 0};
if (in == nullptr || out == nullptr || getInputFrameSize() != getOutputFrameSize() ||
getInputFrameSize() == 0) {
return status;
}
status = {EX_ILLEGAL_STATE, 0, 0};
if (mState == DOWNMIX_STATE_UNINITIALIZED) {
LOG(ERROR) << __func__ << "Trying to use an uninitialized downmixer";
return status;
} else if (mState == DOWNMIX_STATE_INITIALIZED) {
LOG(ERROR) << __func__ << "Trying to use a non-configured downmixer";
return status;
}
LOG(DEBUG) << __func__ << " start processing";
bool accumulate = false;
int frames = samples * sizeof(float) / getInputFrameSize();
if (mType == Downmix::Type::STRIP) {
int inputChannelCount = getChannelCount(mChMask);
while (frames) {
if (accumulate) {
out[0] = std::clamp(out[0] + in[0], -1.f, 1.f);
out[1] = std::clamp(out[1] + in[1], -1.f, 1.f);
} else {
out[0] = in[0];
out[1] = in[1];
}
in += inputChannelCount;
out += 2;
frames--;
}
} else {
int chMask = mChMask.get<AudioChannelLayout::layoutMask>();
if (!mChannelMix.process(in, out, frames, accumulate, (audio_channel_mask_t)chMask)) {
LOG(ERROR) << "Multichannel configuration " << mChMask.toString()
<< " is not supported";
return status;
}
}
LOG(DEBUG) << __func__ << " done processing";
return {STATUS_OK, samples, samples};
}
void DownmixContext::init_params(const Parameter::Common& common) {
// when configuring the effect, do not allow a blank or unsupported channel mask
AudioChannelLayout channelMask = common.input.base.channelMask;
if (isChannelMaskValid(channelMask)) {
LOG(ERROR) << "Downmix_Configure error: input channel mask " << channelMask.toString()
<< " not supported";
} else {
mType = Downmix::Type::FOLD;
mChMask = channelMask;
mState = DOWNMIX_STATE_INITIALIZED;
}
}
bool DownmixContext::isChannelMaskValid(AudioChannelLayout channelMask) {
if (channelMask.getTag() == AudioChannelLayout::layoutMask) return false;
int chMask = channelMask.get<AudioChannelLayout::layoutMask>();
// check against unsupported channels (up to FCC_26)
constexpr uint32_t MAXIMUM_CHANNEL_MASK = AudioChannelLayout::LAYOUT_22POINT2 |
AudioChannelLayout::CHANNEL_FRONT_WIDE_LEFT |
AudioChannelLayout::CHANNEL_FRONT_WIDE_RIGHT;
if (chMask & ~MAXIMUM_CHANNEL_MASK) {
LOG(ERROR) << "Unsupported channels in " << (chMask & ~MAXIMUM_CHANNEL_MASK);
return false;
}
return true;
}
} // namespace aidl::android::hardware::audio::effect