blob: 5fb44b54e81a2864c0709e7c86a857552f2e176b [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::common::getChannelCount;
using aidl::android::hardware::audio::effect::IEffect;
using aidl::android::media::audio::common::AudioChannelLayout;
using aidl::android::media::audio::common::AudioConfig;
namespace aidl::android::hardware::audio::effect {
namespace {
inline bool isChannelMaskValid(const 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;
}
inline bool isStereoChannelMask(const AudioChannelLayout& channelMask) {
if (channelMask.getTag() != AudioChannelLayout::layoutMask) return false;
return channelMask.get<AudioChannelLayout::layoutMask>() == AudioChannelLayout::LAYOUT_STEREO;
}
} // namespace
bool DownmixContext::validateCommonConfig(const Parameter::Common& common) {
const AudioConfig& input = common.input;
const AudioConfig& output = common.output;
if (input.base.sampleRate != output.base.sampleRate) {
LOG(ERROR) << __func__ << ": SRC not supported, input: " << input.toString()
<< " output: " << output.toString();
return false;
}
if (!isStereoChannelMask(output.base.channelMask)) {
LOG(ERROR) << __func__ << ": output should be stereo, not "
<< output.base.channelMask.toString();
return false;
}
if (!isChannelMaskValid(input.base.channelMask)) {
LOG(ERROR) << __func__ << ": invalid input channel, " << input.base.channelMask.toString();
return false;
}
return true;
}
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::downmixProcess(float* in, float* out, int samples) {
IEffect::Status status = {EX_ILLEGAL_ARGUMENT, 0, 0};
if (in == nullptr || out == nullptr ||
getCommon().input.frameCount != getCommon().output.frameCount || 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) {
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 += mInputChannelCount;
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;
}
}
int producedSamples = (samples / mInputChannelCount) << 1;
LOG(DEBUG) << __func__ << " done processing " << samples << " samples, generated "
<< producedSamples << " frameSize: " << getInputFrameSize() << " - "
<< getOutputFrameSize();
return {STATUS_OK, samples, producedSamples};
}
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;
}
}
} // namespace aidl::android::hardware::audio::effect