You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
405 lines
14 KiB
405 lines
14 KiB
/*
|
|
* Copyright (C) 2018 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 "core/default/ParametersUtil.h"
|
|
#include "core/default/Util.h"
|
|
|
|
#include <system/audio.h>
|
|
|
|
#include <util/CoreUtils.h>
|
|
|
|
#include <tinyalsa/asoundlib.h>
|
|
#include <cutils/properties.h>
|
|
|
|
namespace android {
|
|
namespace hardware {
|
|
namespace audio {
|
|
namespace CORE_TYPES_CPP_VERSION {
|
|
namespace implementation {
|
|
|
|
/** Converts a status_t in Result according to the rules of AudioParameter::get*
|
|
* Note: Static method and not private method to avoid leaking status_t dependency
|
|
*/
|
|
static Result getHalStatusToResult(status_t status) {
|
|
switch (status) {
|
|
case OK:
|
|
return Result::OK;
|
|
case BAD_VALUE: // Nothing was returned, probably because the HAL does
|
|
// not handle it
|
|
return Result::NOT_SUPPORTED;
|
|
case INVALID_OPERATION: // Conversion from string to the requested type
|
|
// failed
|
|
return Result::INVALID_ARGUMENTS;
|
|
default: // Should not happen
|
|
ALOGW("Unexpected status returned by getParam: %u", status);
|
|
return Result::INVALID_ARGUMENTS;
|
|
}
|
|
}
|
|
|
|
Result ParametersUtil::getParam(const char* name, bool* value) {
|
|
String8 halValue;
|
|
Result retval = getParam(name, &halValue);
|
|
*value = false;
|
|
if (retval == Result::OK) {
|
|
if (halValue.empty()) {
|
|
return Result::NOT_SUPPORTED;
|
|
}
|
|
*value = !(halValue == AudioParameter::valueOff);
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
Result ParametersUtil::getParam(const char* name, int* value) {
|
|
const String8 halName(name);
|
|
AudioParameter keys;
|
|
keys.addKey(halName);
|
|
std::unique_ptr<AudioParameter> params = getParams(keys);
|
|
return getHalStatusToResult(params->getInt(halName, *value));
|
|
}
|
|
|
|
Result ParametersUtil::getParam(const char* name, String8* value, AudioParameter context) {
|
|
const String8 halName(name);
|
|
context.addKey(halName);
|
|
std::unique_ptr<AudioParameter> params = getParams(context);
|
|
return getHalStatusToResult(params->get(halName, *value));
|
|
}
|
|
|
|
void ParametersUtil::getParametersImpl(
|
|
const hidl_vec<ParameterValue>& context, const hidl_vec<hidl_string>& keys,
|
|
std::function<void(Result retval, const hidl_vec<ParameterValue>& parameters)> cb) {
|
|
AudioParameter halKeys;
|
|
for (auto& pair : context) {
|
|
halKeys.add(String8(pair.key.c_str()), String8(pair.value.c_str()));
|
|
}
|
|
for (size_t i = 0; i < keys.size(); ++i) {
|
|
halKeys.addKey(String8(keys[i].c_str()));
|
|
}
|
|
std::unique_ptr<AudioParameter> halValues = getParams(halKeys);
|
|
Result retval =
|
|
(keys.size() == 0 || halValues->size() != 0) ? Result::OK : Result::NOT_SUPPORTED;
|
|
hidl_vec<ParameterValue> result;
|
|
result.resize(halValues->size());
|
|
String8 halKey, halValue;
|
|
for (size_t i = 0; i < halValues->size(); ++i) {
|
|
status_t status = halValues->getAt(i, halKey, halValue);
|
|
if (status != OK) {
|
|
result.resize(0);
|
|
retval = getHalStatusToResult(status);
|
|
break;
|
|
}
|
|
result[i].key = halKey.string();
|
|
result[i].value = halValue.string();
|
|
}
|
|
cb(retval, result);
|
|
}
|
|
|
|
std::unique_ptr<AudioParameter> ParametersUtil::getParams(const AudioParameter& keys) {
|
|
String8 paramsAndValues;
|
|
char* halValues = halGetParameters(keys.keysToString().string());
|
|
if (halValues != NULL) {
|
|
paramsAndValues.setTo(halValues);
|
|
free(halValues);
|
|
} else {
|
|
paramsAndValues.clear();
|
|
}
|
|
return std::unique_ptr<AudioParameter>(new AudioParameter(paramsAndValues));
|
|
}
|
|
|
|
Result ParametersUtil::setParam(const char* name, const char* value) {
|
|
AudioParameter param;
|
|
param.add(String8(name), String8(value));
|
|
return setParams(param);
|
|
}
|
|
|
|
Result ParametersUtil::setParam(const char* name, bool value) {
|
|
AudioParameter param;
|
|
param.add(String8(name), String8(value ? AudioParameter::valueOn : AudioParameter::valueOff));
|
|
return setParams(param);
|
|
}
|
|
|
|
Result ParametersUtil::setParam(const char* name, int value) {
|
|
AudioParameter param;
|
|
param.addInt(String8(name), value);
|
|
return setParams(param);
|
|
}
|
|
|
|
Result ParametersUtil::setParam(const char* name, float value) {
|
|
AudioParameter param;
|
|
param.addFloat(String8(name), value);
|
|
return setParams(param);
|
|
}
|
|
|
|
int getMixerValueByName(const char *name) {
|
|
const auto mixer = mixer_open(0);
|
|
const auto ctl = mixer_get_ctl_by_name(mixer, name);
|
|
int value = 0;
|
|
|
|
if (mixer == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return 0;
|
|
}
|
|
|
|
if (ctl == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return 0;
|
|
}
|
|
|
|
value = mixer_ctl_get_value(ctl, 0);
|
|
|
|
mixer_close(mixer);
|
|
|
|
return value;
|
|
}
|
|
|
|
const char *getMixerValueString(const char *name) {
|
|
const auto mixer = mixer_open(0);
|
|
const auto ctl = mixer_get_ctl_by_name(mixer, name);
|
|
int control_value = mixer_ctl_get_value(ctl, 0);
|
|
const char *value;
|
|
|
|
if (mixer == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return "-1";
|
|
}
|
|
|
|
if (ctl == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return "-1";
|
|
}
|
|
|
|
value = mixer_ctl_get_enum_string(ctl, control_value);
|
|
|
|
mixer_close(mixer);
|
|
|
|
return value;
|
|
}
|
|
|
|
void setMixerValueByName(mixer *mixer, const char *name, int value) {
|
|
const auto ctl = mixer_get_ctl_by_name(mixer, name);
|
|
|
|
if (ctl == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return;
|
|
}
|
|
|
|
if (mixer_ctl_set_value(ctl, 0, value) < 0) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void setMixerValueByNameString(mixer *mixer, const char *name, const char *value) {
|
|
const auto ctl = mixer_get_ctl_by_name(mixer, name);
|
|
|
|
if (ctl == nullptr) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return;
|
|
}
|
|
|
|
if (mixer_ctl_set_enum_by_string(ctl, value)) {
|
|
ALOGE("Failed to find mixer ctl for %s", name);
|
|
return;
|
|
}
|
|
}
|
|
|
|
void setEarpieceVoipMixers(bool enabled) {
|
|
const auto mixer = mixer_open(0);
|
|
|
|
if (mixer == nullptr) {
|
|
ALOGE("Failed to open mixer");
|
|
return;
|
|
}
|
|
|
|
if (enabled) {
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX0", "ADC0");
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX1", "ADC2");
|
|
setMixerValueByName(mixer, "MultiMedia2 Mixer QUIN_MI2S_TX", 0);
|
|
setMixerValueByNameString(mixer, "TAS256X ASI1 SEL RIGHT", "I2C offset");
|
|
setMixerValueByName(mixer, "TAS256X ASI Right Switch", 0);
|
|
setMixerValueByNameString(mixer, "TAS256X ASI1 SEL LEFT", "LeftRightDiv2");
|
|
setMixerValueByName(mixer, "TAS256X RECEIVER ENABLE LEFT", 1);
|
|
setMixerValueByName(mixer, "TAS256X ASI Left Switch", 1);
|
|
setMixerValueByName(mixer, "TAS256X AMP OUTPUT LVL LEFT", 8);
|
|
setMixerValueByName(mixer, "TAS256X PLAYBACK VOLUME LEFT", 55);
|
|
setMixerValueByName(mixer, "TAS256X PLAYBACK VOLUME RIGHT", 55);
|
|
setMixerValueByName(mixer, "SmartPA Mute", 0);
|
|
setMixerValueByNameString(mixer, "EC Reference Channels", "One");
|
|
setMixerValueByNameString(mixer, "AUDIO_REF_EC_UL10 MUX", "QUIN_MI2S_RX");
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer SLIM_7_TX_Voip", 0);
|
|
setMixerValueByName(mixer, "SLIM_7_RX_Voice Mixer Voip", 0);
|
|
setMixerValueByName(mixer, "QUIN_MI2S_RX_Voice Mixer Voip", 1);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer TX_CDC_DMA_TX_3_Voip", 1);
|
|
setMixerValueByName(mixer, "QUIN_MI2S_RX_Voice Mixer Voip", 0);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer TX_CDC_DMA_TX_3_Voip", 0);
|
|
}
|
|
|
|
mixer_close(mixer);
|
|
}
|
|
|
|
void setSpeakerVoipMixers(bool enabled) {
|
|
const auto mixer = mixer_open(0);
|
|
|
|
if (mixer == nullptr) {
|
|
ALOGE("Failed to open mixer");
|
|
return;
|
|
}
|
|
|
|
if (enabled) {
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX0", "ADC2");
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX1", "ADC0");
|
|
setMixerValueByName(mixer, "MultiMedia2 Mixer QUIN_MI2S_TX", 1);
|
|
setMixerValueByNameString(mixer, "TAS256X ASI1 SEL RIGHT", "Right");
|
|
setMixerValueByName(mixer, "TAS256X ASI Right Switch", 1);
|
|
setMixerValueByNameString(mixer, "TAS256X ASI1 SEL LEFT", "Left");
|
|
setMixerValueByName(mixer, "TAS256X RECEIVER ENABLE LEFT", 0);
|
|
setMixerValueByName(mixer, "TAS256X ASI Left Switch", 1);
|
|
setMixerValueByName(mixer, "TAS256X AMP OUTPUT LVL LEFT", 20);
|
|
setMixerValueByName(mixer, "TAS256X PLAYBACK VOLUME LEFT", 55);
|
|
setMixerValueByName(mixer, "TAS256X PLAYBACK VOLUME RIGHT", 55);
|
|
setMixerValueByName(mixer, "SmartPA Mute", 0);
|
|
setMixerValueByNameString(mixer, "EC Reference Channels", "Two");
|
|
setMixerValueByNameString(mixer, "AUDIO_REF_EC_UL10 MUX", "QUIN_MI2S_RX");
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer SLIM_7_TX_Voip", 0);
|
|
setMixerValueByName(mixer, "SLIM_7_RX_Voice Mixer Voip", 0);
|
|
setMixerValueByName(mixer, "QUIN_MI2S_RX_Voice Mixer Voip", 1);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer TX_CDC_DMA_TX_3_Voip", 1);
|
|
setMixerValueByName(mixer, "QUIN_MI2S_RX_Voice Mixer Voip", 0);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer TX_CDC_DMA_TX_3_Voip", 0);
|
|
}
|
|
|
|
mixer_close(mixer);
|
|
}
|
|
|
|
void setBTVoipMixers(bool enabled) {
|
|
const auto mixer = mixer_open(0);
|
|
|
|
if (mixer == nullptr) {
|
|
ALOGE("Failed to open mixer");
|
|
return;
|
|
}
|
|
|
|
if (enabled) {
|
|
setMixerValueByName(mixer, "TX_AIF1_CAP Mixer DEC0", 0);
|
|
setMixerValueByName(mixer, "TX_AIF1_CAP Mixer DEC1", 0);
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX0", "ZERO");
|
|
setMixerValueByNameString(mixer, "TX SMIC MUX1", "ZERO");
|
|
setMixerValueByNameString(mixer, "TX DEC0 MUX", "MSM_DMIC");
|
|
setMixerValueByNameString(mixer, "TX DEC1 MUX", "MSM_DMIC");
|
|
setMixerValueByName(mixer, "TX_DEC0 Volume", 102);
|
|
setMixerValueByName(mixer, "TX_DEC1 Volume", 102);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer SLIM_7_TX_Voip", 1);
|
|
setMixerValueByName(mixer, "Voip_Tx Mixer TX_CDC_DMA_TX_3_Voip", 0);
|
|
setMixerValueByName(mixer, "SLIM_7_RX_Voice Mixer Voip", 1);
|
|
setMixerValueByName(mixer, "QUIN_MI2S_RX_Voice Mixer Voip", 0);
|
|
setMixerValueByName(mixer, "TAS256X IVSENSE ENABLE", 0);
|
|
setMixerValueByName(mixer, "TAS256X PLAYBACK VOLUME LEFT", 55);
|
|
setMixerValueByName(mixer, "TAS256X RECEIVER ENABLE LEFT", 0);
|
|
setMixerValueByName(mixer, "TAS256X AMP OUTPUT LVL LEFT", 20);
|
|
setMixerValueByName(mixer, "SmartPA Mute", 1);
|
|
setMixerValueByName(mixer, "ADC1 Volume", 12);
|
|
setMixerValueByName(mixer, "ADC3 Volume", 12);
|
|
setMixerValueByNameString(mixer, "TX_CDC_DMA_TX_3 Channels", "One");
|
|
setMixerValueByName(mixer, "TAS256X ASI Left Switch", 0);
|
|
setMixerValueByNameString(mixer, "TAS256X ASI1 SEL LEFT", "I2C Offset");
|
|
setMixerValueByName(mixer, "ADC3_MIXER Switch", 0);
|
|
}
|
|
|
|
mixer_close(mixer);
|
|
}
|
|
|
|
void fixupMixers() {
|
|
char simslot[92];
|
|
char voipdevice[92];
|
|
const char *ampstatus;
|
|
int voipstatus, btvoipstatus;
|
|
|
|
property_get("vendor.calls.slotid", simslot, "-1");
|
|
property_get("vendor.audio.voip.device", voipdevice, "none");
|
|
|
|
voipstatus = getMixerValueByName("QUIN_MI2S_RX_Voice Mixer Voip");
|
|
ampstatus = getMixerValueString("TAS256X RECEIVER ENABLE LEFT");
|
|
btvoipstatus = getMixerValueByName("SLIM_7_RX_Voice Mixer Voip");
|
|
|
|
ALOGD("ampstatus is %s", ampstatus);
|
|
ALOGD("voipstatus is %d", voipstatus);
|
|
ALOGD("bluetooth VoIP status is %d", btvoipstatus);
|
|
|
|
// VoIP call
|
|
if (voipstatus == 1 || btvoipstatus == 1) {
|
|
ALOGD("In VoIP call");
|
|
if (strcmp(voipdevice, "bluetooth") == 0 && btvoipstatus == 0) {
|
|
ALOGD("Setting bluetooth mixers");
|
|
setBTVoipMixers(true);
|
|
}
|
|
|
|
if (strcmp(voipdevice, "earpiece") == 0) {
|
|
if (btvoipstatus == 1 || strcmp(ampstatus, "DISABLE") == 0) {
|
|
ALOGD("Setting earpiece mixers");
|
|
setEarpieceVoipMixers(true);
|
|
}
|
|
}
|
|
|
|
if (strcmp(voipdevice, "speaker") == 0) {
|
|
if (btvoipstatus == 1 || strcmp(ampstatus, "ENABLE") == 0) {
|
|
ALOGD("Setting speaker mixers");
|
|
setSpeakerVoipMixers(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Result ParametersUtil::setParametersImpl(const hidl_vec<ParameterValue>& context,
|
|
const hidl_vec<ParameterValue>& parameters) {
|
|
AudioParameter params;
|
|
for (auto& pair : context) {
|
|
params.add(String8(pair.key.c_str()), String8(pair.value.c_str()));
|
|
}
|
|
for (size_t i = 0; i < parameters.size(); ++i) {
|
|
if (parameters[i].key == "bt_wbs") {
|
|
params.add(String8("g_sco_samplerate"),
|
|
String8(parameters[i].value == AudioParameter::valueOn ? "16000" : "8000"));
|
|
}
|
|
if (parameters[i].key == "voip_call") {
|
|
if (parameters[i].value == "1") {
|
|
fixupMixers();
|
|
}
|
|
}
|
|
params.add(String8(parameters[i].key.c_str()), String8(parameters[i].value.c_str()));
|
|
}
|
|
return setParams(params);
|
|
}
|
|
|
|
Result ParametersUtil::setParam(const char* name, const DeviceAddress& address) {
|
|
audio_devices_t halDeviceType;
|
|
char halDeviceAddress[AUDIO_DEVICE_MAX_ADDRESS_LEN];
|
|
if (CoreUtils::deviceAddressToHal(address, &halDeviceType, halDeviceAddress) != NO_ERROR) {
|
|
return Result::INVALID_ARGUMENTS;
|
|
}
|
|
AudioParameter params{String8(halDeviceAddress)};
|
|
params.addInt(String8(name), halDeviceType);
|
|
return setParams(params);
|
|
}
|
|
|
|
Result ParametersUtil::setParams(const AudioParameter& param) {
|
|
int halStatus = halSetParameters(param.toString().string());
|
|
return util::analyzeStatus(halStatus);
|
|
}
|
|
|
|
} // namespace implementation
|
|
} // namespace CORE_TYPES_CPP_VERSION
|
|
} // namespace audio
|
|
} // namespace hardware
|
|
} // namespace android
|
|
|