mirror of
https://github.com/Evolution-X-Devices/device_xiaomi_sdm710-common
synced 2026-01-27 12:13:48 +00:00
sdm710-common: Add audio amplifier HAL for TAS2562
Add audio amplifier HAL for TAS2562 codec used by pyxis and vela. Implementation is based on reversely-engineered stock audio HAL. Change-Id: I6d7daa636b632c8a12af09b09cece78042449353
This commit is contained in:
1
amplifier/.clang-format
Symbolic link
1
amplifier/.clang-format
Symbolic link
@@ -0,0 +1 @@
|
||||
../../../../build/soong/scripts/system-clang-format
|
||||
44
amplifier/Android.mk
Normal file
44
amplifier/Android.mk
Normal file
@@ -0,0 +1,44 @@
|
||||
#
|
||||
# Copyright 2024 The LineageOS 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.
|
||||
#
|
||||
|
||||
LOCAL_PATH := $(call my-dir)
|
||||
|
||||
include $(CLEAR_VARS)
|
||||
|
||||
LOCAL_MODULE := audio_amplifier.tas2562
|
||||
LOCAL_MODULE_RELATIVE_PATH := hw
|
||||
LOCAL_SRC_FILES := tas2562.c
|
||||
LOCAL_VENDOR_MODULE := true
|
||||
|
||||
LOCAL_C_INCLUDES += \
|
||||
$(call include-path-for, audio-route) \
|
||||
$(call include-path-for, audio-utils) \
|
||||
$(call project-path-for, qcom-audio)/hal \
|
||||
$(call project-path-for, qcom-audio)/hal/audio_extn \
|
||||
$(call project-path-for, qcom-audio)/hal/msm8974 \
|
||||
external/tinycompress/include
|
||||
|
||||
LOCAL_HEADER_LIBRARIES += \
|
||||
generated_kernel_headers \
|
||||
libhardware_headers
|
||||
|
||||
LOCAL_SHARED_LIBRARIES += \
|
||||
audio.primary.$(TARGET_BOARD_PLATFORM) \
|
||||
libcutils \
|
||||
liblog \
|
||||
libtinyalsa
|
||||
|
||||
include $(BUILD_SHARED_LIBRARY)
|
||||
406
amplifier/tas2562.c
Normal file
406
amplifier/tas2562.c
Normal file
@@ -0,0 +1,406 @@
|
||||
/*
|
||||
* Copyright (C) 2015 The CyanogenMod Open Source Project
|
||||
* Copyright (C) 2024 The LineageOS 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 "amplifier_tas2562"
|
||||
#define LOG_NDEBUG 0
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdlib.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
#include <cutils/log.h>
|
||||
#include <cutils/str_parms.h>
|
||||
|
||||
#include <hardware/audio_amplifier.h>
|
||||
#include <hardware/hardware.h>
|
||||
|
||||
#include "audio_hw.h"
|
||||
#include "platform.h"
|
||||
#include "platform_api.h"
|
||||
|
||||
#define TAS2562_CAL_FILE "/mnt/vendor/persist/audio/tas2562_cal.bin"
|
||||
|
||||
#define TAS2562_SET_TCAL_LEFT "TAS2562_SET_TCAL_LEFT"
|
||||
#define TAS2562_ALGO_PROFILE "TAS2562_ALGO_PROFILE"
|
||||
#define TAS2562_SET_RE_LEFT "TAS2562_SET_RE_LEFT"
|
||||
#define TAS2562_SMARTPA_ENABLE "TAS2562_SMARTPA_ENABLE"
|
||||
#define TAS2562_SET_SPKID_LEFT "TAS2562_SET_SPKID_LEFT"
|
||||
|
||||
typedef enum tas2562_profile {
|
||||
PROFILE_NONE = -1,
|
||||
PROFILE_MUSIC = 0,
|
||||
PROFILE_RING,
|
||||
PROFILE_VOICE,
|
||||
PROFILE_MAX = PROFILE_VOICE,
|
||||
} tas2562_profile_t;
|
||||
|
||||
#define TAS2562_PROFILE(x) [PROFILE_##x] = #x
|
||||
static const char* tas2562_profile_names[] = {
|
||||
TAS2562_PROFILE(MUSIC),
|
||||
TAS2562_PROFILE(RING),
|
||||
TAS2562_PROFILE(VOICE),
|
||||
};
|
||||
|
||||
static const struct pcm_config tas2562_pcm_config = {
|
||||
.channels = 2,
|
||||
.rate = 48000,
|
||||
.period_size = 256,
|
||||
.period_count = 4,
|
||||
.format = PCM_FORMAT_S24_LE,
|
||||
.start_threshold = 0,
|
||||
.stop_threshold = INT_MAX,
|
||||
.avail_min = 40,
|
||||
};
|
||||
|
||||
typedef struct tas2562_amp {
|
||||
amplifier_device_t amp_dev;
|
||||
tas2562_profile_t profile;
|
||||
struct audio_device* adev;
|
||||
struct pcm* pcm;
|
||||
int re;
|
||||
int tcal;
|
||||
} tas2562_amp_t;
|
||||
|
||||
static int tas2562_parse_cal_file(int* re, int* tcal) {
|
||||
char *ptr, *tok1, *tok2;
|
||||
FILE* cal_file;
|
||||
char buf[256];
|
||||
int rc = 0;
|
||||
|
||||
if (!re || !tcal) return -EINVAL;
|
||||
|
||||
memset(buf, 0, sizeof(buf));
|
||||
cal_file = fopen(TAS2562_CAL_FILE, "r");
|
||||
if (!cal_file) {
|
||||
ALOGE("%s: Failed to open calibration file: %d", __func__, errno);
|
||||
return -ENOENT;
|
||||
}
|
||||
|
||||
if (!fread(buf, 1, sizeof(buf) - 1, cal_file)) {
|
||||
if (ferror(cal_file)) {
|
||||
ALOGE("%s: Failed to read calibration file", __func__);
|
||||
rc = -EIO;
|
||||
} else {
|
||||
ALOGE("%s: Calibration file is empty", __func__);
|
||||
rc = -ENODATA;
|
||||
}
|
||||
}
|
||||
fclose(cal_file);
|
||||
|
||||
if (rc) return rc;
|
||||
|
||||
tok1 = strtok_r(buf, ";", &ptr);
|
||||
if (!tok1) goto format_err;
|
||||
|
||||
tok2 = strtok_r(ptr, ";", &ptr);
|
||||
if (!tok2) goto format_err;
|
||||
|
||||
*re = (int)(atof(tok1) * 524288.0f);
|
||||
*tcal = (int)atof(tok2);
|
||||
|
||||
ALOGI("%s: Read calibration: Re=%d, Tcal=%d", __func__, *re, *tcal);
|
||||
|
||||
return 0;
|
||||
|
||||
format_err:
|
||||
ALOGE("%s: Failed to parse calibration file", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
static int tas2562_mixer_set_enum_by_string(struct mixer* mixer, const char* name,
|
||||
const char* value) {
|
||||
struct mixer_ctl* ctl;
|
||||
int ret = 0;
|
||||
|
||||
ctl = mixer_get_ctl_by_name(mixer, name);
|
||||
if (!ctl) {
|
||||
ALOGE("%s: Could not get mixer ctl '%s'", __func__, name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = mixer_ctl_set_enum_by_string(ctl, value);
|
||||
if (ret < 0) {
|
||||
ALOGE("%s: Failed to set mixer ctl '%s' to enum '%s'", __func__, name, value);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
ALOGI("%s: Set mixer ctl '%s' to enum '%s'", __func__, name, value);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static int tas2562_mixer_set_value(struct mixer* mixer, const char* name, int value) {
|
||||
struct mixer_ctl* ctl;
|
||||
int ret = 0;
|
||||
|
||||
ctl = mixer_get_ctl_by_name(mixer, name);
|
||||
if (!ctl) {
|
||||
ALOGE("%s: Could not get mixer ctl '%s'", __func__, name);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
ret = mixer_ctl_set_value(ctl, 0, value);
|
||||
if (ret < 0) {
|
||||
ALOGE("%s: Failed to set mixer ctl '%s' to '%d'", __func__, name, value);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
ALOGI("%s: Set mixer ctl '%s' to '%d'", __func__, name, value);
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
static bool tas2562_is_speaker(uint32_t device) {
|
||||
bool is_speaker;
|
||||
switch (device) {
|
||||
case SND_DEVICE_OUT_SPEAKER:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_ANC_HEADSET:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_BT_A2DP:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_BT_SCO:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_BT_SCO_WB:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_DISPLAY_PORT:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_HDMI:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_HEADPHONES:
|
||||
case SND_DEVICE_OUT_SPEAKER_AND_LINE:
|
||||
case SND_DEVICE_OUT_SPEAKER_REVERSE:
|
||||
case SND_DEVICE_OUT_VOICE_SPEAKER:
|
||||
case SND_DEVICE_OUT_VOICE_SPEAKER_AND_VOICE_ANC_HEADSET:
|
||||
case SND_DEVICE_OUT_VOICE_SPEAKER_AND_VOICE_HEADPHONES:
|
||||
case SND_DEVICE_OUT_VOICE_SPEAKER_2:
|
||||
is_speaker = true;
|
||||
break;
|
||||
default:
|
||||
is_speaker = false;
|
||||
break;
|
||||
}
|
||||
return is_speaker;
|
||||
}
|
||||
|
||||
static int tas2562_set_mode(amplifier_device_t* device, audio_mode_t mode) {
|
||||
tas2562_amp_t* tas2562 = (tas2562_amp_t*)device;
|
||||
|
||||
if (!tas2562) {
|
||||
ALOGE("%s: Invalid params", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
switch (mode) {
|
||||
case AUDIO_MODE_NORMAL:
|
||||
tas2562->profile = PROFILE_MUSIC;
|
||||
break;
|
||||
case AUDIO_MODE_RINGTONE:
|
||||
tas2562->profile = PROFILE_RING;
|
||||
break;
|
||||
case AUDIO_MODE_IN_CALL:
|
||||
case AUDIO_MODE_IN_COMMUNICATION:
|
||||
tas2562->profile = PROFILE_VOICE;
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
ALOGI("%s: Setting profile to %s", __func__, tas2562_profile_names[tas2562->profile]);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tas2562_start_feedback(tas2562_amp_t* tas2562, uint32_t device) {
|
||||
struct pcm_config pcm_config = tas2562_pcm_config;
|
||||
struct audio_device* adev = tas2562->adev;
|
||||
struct mixer* mixer = adev->mixer;
|
||||
struct audio_usecase* usecase;
|
||||
const char* profile;
|
||||
struct pcm* pcm;
|
||||
int pcm_id, rc = 0;
|
||||
|
||||
if (!tas2562_is_speaker(device)) return 0;
|
||||
|
||||
if (tas2562->pcm) {
|
||||
ALOGE("%s: Invalid state", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
usecase = calloc(1, sizeof(*usecase));
|
||||
if (!usecase) {
|
||||
ALOGE("%s: Failed to allocate memory for usecase", __func__);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
usecase->id = USECASE_AUDIO_SPKR_CALIB_TX;
|
||||
usecase->type = PCM_CAPTURE;
|
||||
usecase->in_snd_device = SND_DEVICE_IN_CAPTURE_VI_FEEDBACK;
|
||||
usecase->out_snd_device = SND_DEVICE_NONE;
|
||||
list_init(&usecase->device_list);
|
||||
list_add_head(&adev->usecase_list, &usecase->list);
|
||||
|
||||
enable_snd_device(adev, SND_DEVICE_IN_CAPTURE_VI_FEEDBACK);
|
||||
enable_audio_route(adev, usecase);
|
||||
|
||||
tas2562_mixer_set_value(mixer, TAS2562_SET_SPKID_LEFT, 0);
|
||||
|
||||
profile = tas2562_profile_names[tas2562->profile];
|
||||
ALOGI("%s: Using profile %s", __func__, profile);
|
||||
tas2562_mixer_set_enum_by_string(mixer, TAS2562_ALGO_PROFILE, profile);
|
||||
|
||||
if (tas2562->tcal != -1) tas2562_mixer_set_value(mixer, TAS2562_SET_TCAL_LEFT, tas2562->tcal);
|
||||
|
||||
if (tas2562->re != -1) tas2562_mixer_set_value(mixer, TAS2562_SET_RE_LEFT, tas2562->re);
|
||||
|
||||
tas2562_mixer_set_enum_by_string(mixer, TAS2562_SMARTPA_ENABLE, "ENABLE");
|
||||
|
||||
pcm_id = platform_get_pcm_device_id(usecase->id, PCM_CAPTURE);
|
||||
if (pcm_id < 0) {
|
||||
ALOGE("%s: Invalid PCM device for usecase %d", __func__, usecase->id);
|
||||
rc = -ENODEV;
|
||||
goto err_no_pcm;
|
||||
}
|
||||
|
||||
pcm = pcm_open(adev->snd_card, pcm_id, PCM_IN, &pcm_config);
|
||||
if (!pcm || !pcm_is_ready(pcm)) {
|
||||
ALOGE("%s: Failed to open PCM device: %s", __func__, pcm_get_error(pcm));
|
||||
rc = -EIO;
|
||||
goto err_no_pcm;
|
||||
}
|
||||
|
||||
rc = pcm_start(pcm);
|
||||
if (rc < 0) {
|
||||
ALOGE("%s: Failed to start PCM: %s", __func__, pcm_get_error(pcm));
|
||||
goto err_pcm_start;
|
||||
}
|
||||
|
||||
tas2562->pcm = pcm;
|
||||
|
||||
ALOGI("%s: Feedback enabled successfully", __func__);
|
||||
|
||||
return 0;
|
||||
|
||||
err_pcm_start:
|
||||
if (pcm) pcm_close(pcm);
|
||||
err_no_pcm:
|
||||
tas2562_mixer_set_enum_by_string(mixer, TAS2562_SMARTPA_ENABLE, "DISABLE");
|
||||
disable_audio_route(adev, usecase);
|
||||
disable_snd_device(adev, SND_DEVICE_IN_CAPTURE_VI_FEEDBACK);
|
||||
list_remove(&usecase->list);
|
||||
free(usecase);
|
||||
|
||||
return rc;
|
||||
}
|
||||
|
||||
static int tas2562_stop_feedback(tas2562_amp_t* tas2562, uint32_t device) {
|
||||
struct audio_device* adev = tas2562->adev;
|
||||
struct mixer* mixer = adev->mixer;
|
||||
struct audio_usecase* usecase;
|
||||
|
||||
if (!tas2562_is_speaker(device)) return 0;
|
||||
|
||||
if (!tas2562->pcm) {
|
||||
ALOGI("%s: Invalid state", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
pcm_close(tas2562->pcm);
|
||||
tas2562->pcm = NULL;
|
||||
|
||||
tas2562_mixer_set_enum_by_string(mixer, TAS2562_SMARTPA_ENABLE, "DISABLE");
|
||||
|
||||
disable_snd_device(adev, SND_DEVICE_IN_CAPTURE_VI_FEEDBACK);
|
||||
|
||||
usecase = get_usecase_from_list(adev, USECASE_AUDIO_SPKR_CALIB_TX);
|
||||
if (usecase) {
|
||||
disable_audio_route(adev, usecase);
|
||||
list_remove(&usecase->list);
|
||||
free(usecase);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tas2562_set_feedback(struct amplifier_device* device, void* adev, uint32_t devices,
|
||||
bool enable) {
|
||||
tas2562_amp_t* tas2562 = (tas2562_amp_t*)device;
|
||||
|
||||
if (!adev || !tas2562) {
|
||||
ALOGE("%s: Invalid parameters", __func__);
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
tas2562->adev = adev;
|
||||
|
||||
if (enable)
|
||||
return tas2562_start_feedback(tas2562, devices);
|
||||
else
|
||||
return tas2562_stop_feedback(tas2562, devices);
|
||||
}
|
||||
|
||||
static int tas2562_dev_close(hw_device_t* device) {
|
||||
if (device) free(device);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int tas2562_module_open(const hw_module_t* module, const char* name, hw_device_t** device) {
|
||||
tas2562_amp_t* tas2562;
|
||||
|
||||
if (strcmp(name, AMPLIFIER_HARDWARE_INTERFACE)) {
|
||||
ALOGE("%s:%d: %s does not match amplifier hardware interface name\n", __func__, __LINE__,
|
||||
name);
|
||||
return -ENODEV;
|
||||
}
|
||||
|
||||
tas2562 = calloc(1, sizeof(*tas2562));
|
||||
if (!tas2562) {
|
||||
ALOGE("%s:%d: Unable to allocate memory for amplifier device\n", __func__, __LINE__);
|
||||
return -ENOMEM;
|
||||
}
|
||||
|
||||
tas2562->amp_dev.common.tag = HARDWARE_DEVICE_TAG;
|
||||
tas2562->amp_dev.common.module = (hw_module_t*)module;
|
||||
tas2562->amp_dev.common.version = HARDWARE_DEVICE_API_VERSION(1, 0);
|
||||
tas2562->amp_dev.common.close = tas2562_dev_close;
|
||||
|
||||
tas2562->amp_dev.set_mode = tas2562_set_mode;
|
||||
tas2562->amp_dev.set_feedback = tas2562_set_feedback;
|
||||
|
||||
tas2562->profile = PROFILE_MUSIC;
|
||||
tas2562->re = -1;
|
||||
tas2562->tcal = -1;
|
||||
|
||||
tas2562_parse_cal_file(&tas2562->re, &tas2562->tcal);
|
||||
|
||||
*device = (hw_device_t*)tas2562;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static struct hw_module_methods_t hal_module_methods = {
|
||||
.open = tas2562_module_open,
|
||||
};
|
||||
|
||||
/* clang-format off */
|
||||
amplifier_module_t HAL_MODULE_INFO_SYM = {
|
||||
.common = {
|
||||
.tag = HARDWARE_MODULE_TAG,
|
||||
.module_api_version = AMPLIFIER_DEVICE_API_VERSION_CURRENT,
|
||||
.hal_api_version = HARDWARE_HAL_API_VERSION,
|
||||
.id = AMPLIFIER_HARDWARE_MODULE_ID,
|
||||
.name = "TAS2562 audio amplifier HAL",
|
||||
.author = "Ivan Vecera <ivan@cera.cz>",
|
||||
.methods = &hal_module_methods,
|
||||
},
|
||||
};
|
||||
Reference in New Issue
Block a user