416 lines
12 KiB
C
416 lines
12 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
//
|
|
// Common code for Cirrus Logic Smart Amplifiers
|
|
//
|
|
// Copyright (C) 2024 Cirrus Logic, Inc. and
|
|
// Cirrus Logic International Semiconductor Ltd.
|
|
|
|
#include <asm/byteorder.h>
|
|
#include <kunit/static_stub.h>
|
|
#include <linux/dev_printk.h>
|
|
#include <linux/efi.h>
|
|
#include <linux/firmware/cirrus/cs_dsp.h>
|
|
#include <linux/module.h>
|
|
#include <linux/overflow.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/types.h>
|
|
#include <sound/cs-amp-lib.h>
|
|
|
|
#define CIRRUS_LOGIC_CALIBRATION_EFI_NAME L"CirrusSmartAmpCalibrationData"
|
|
#define CIRRUS_LOGIC_CALIBRATION_EFI_GUID \
|
|
EFI_GUID(0x02f9af02, 0x7734, 0x4233, 0xb4, 0x3d, 0x93, 0xfe, 0x5a, 0xa3, 0x5d, 0xb3)
|
|
|
|
#define LENOVO_SPEAKER_ID_EFI_NAME L"SdwSpeaker"
|
|
#define LENOVO_SPEAKER_ID_EFI_GUID \
|
|
EFI_GUID(0x48df970e, 0xe27f, 0x460a, 0xb5, 0x86, 0x77, 0x19, 0x80, 0x1d, 0x92, 0x82)
|
|
|
|
#define HP_SPEAKER_ID_EFI_NAME L"HPSpeakerID"
|
|
#define HP_SPEAKER_ID_EFI_GUID \
|
|
EFI_GUID(0xc49593a4, 0xd099, 0x419b, 0xa2, 0xc3, 0x67, 0xe9, 0x80, 0xe6, 0x1d, 0x1e)
|
|
|
|
#define HP_CALIBRATION_EFI_NAME L"SmartAmpCalibrationData"
|
|
#define HP_CALIBRATION_EFI_GUID \
|
|
EFI_GUID(0x53559579, 0x8753, 0x4f5c, 0x91, 0x30, 0xe8, 0x2a, 0xcf, 0xb8, 0xd8, 0x93)
|
|
|
|
static const struct cs_amp_lib_cal_efivar {
|
|
efi_char16_t *name;
|
|
efi_guid_t *guid;
|
|
} cs_amp_lib_cal_efivars[] = {
|
|
{
|
|
.name = HP_CALIBRATION_EFI_NAME,
|
|
.guid = &HP_CALIBRATION_EFI_GUID,
|
|
},
|
|
{
|
|
.name = CIRRUS_LOGIC_CALIBRATION_EFI_NAME,
|
|
.guid = &CIRRUS_LOGIC_CALIBRATION_EFI_GUID,
|
|
},
|
|
};
|
|
|
|
static int cs_amp_write_cal_coeff(struct cs_dsp *dsp,
|
|
const struct cirrus_amp_cal_controls *controls,
|
|
const char *ctl_name, u32 val)
|
|
{
|
|
struct cs_dsp_coeff_ctl *cs_ctl;
|
|
__be32 beval = cpu_to_be32(val);
|
|
int ret;
|
|
|
|
KUNIT_STATIC_STUB_REDIRECT(cs_amp_write_cal_coeff, dsp, controls, ctl_name, val);
|
|
|
|
if (IS_REACHABLE(CONFIG_FW_CS_DSP)) {
|
|
mutex_lock(&dsp->pwr_lock);
|
|
cs_ctl = cs_dsp_get_ctl(dsp, ctl_name, controls->mem_region, controls->alg_id);
|
|
ret = cs_dsp_coeff_write_ctrl(cs_ctl, 0, &beval, sizeof(beval));
|
|
mutex_unlock(&dsp->pwr_lock);
|
|
|
|
if (ret < 0) {
|
|
dev_err(dsp->dev, "Failed to write to '%s': %d\n", ctl_name, ret);
|
|
return ret;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
return -ENODEV;
|
|
}
|
|
|
|
static int _cs_amp_write_cal_coeffs(struct cs_dsp *dsp,
|
|
const struct cirrus_amp_cal_controls *controls,
|
|
const struct cirrus_amp_cal_data *data)
|
|
{
|
|
int ret;
|
|
|
|
dev_dbg(dsp->dev, "Calibration: Ambient=%#x, Status=%#x, CalR=%d\n",
|
|
data->calAmbient, data->calStatus, data->calR);
|
|
|
|
if (list_empty(&dsp->ctl_list)) {
|
|
dev_info(dsp->dev, "Calibration disabled due to missing firmware controls\n");
|
|
return -ENOENT;
|
|
}
|
|
|
|
ret = cs_amp_write_cal_coeff(dsp, controls, controls->ambient, data->calAmbient);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = cs_amp_write_cal_coeff(dsp, controls, controls->calr, data->calR);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = cs_amp_write_cal_coeff(dsp, controls, controls->status, data->calStatus);
|
|
if (ret)
|
|
return ret;
|
|
|
|
ret = cs_amp_write_cal_coeff(dsp, controls, controls->checksum, data->calR + 1);
|
|
if (ret)
|
|
return ret;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* cs_amp_write_cal_coeffs - Write calibration data to firmware controls.
|
|
* @dsp: Pointer to struct cs_dsp.
|
|
* @controls: Pointer to definition of firmware controls to be written.
|
|
* @data: Pointer to calibration data.
|
|
*
|
|
* Returns: 0 on success, else negative error value.
|
|
*/
|
|
int cs_amp_write_cal_coeffs(struct cs_dsp *dsp,
|
|
const struct cirrus_amp_cal_controls *controls,
|
|
const struct cirrus_amp_cal_data *data)
|
|
{
|
|
if (IS_REACHABLE(CONFIG_FW_CS_DSP) || IS_ENABLED(CONFIG_SND_SOC_CS_AMP_LIB_TEST))
|
|
return _cs_amp_write_cal_coeffs(dsp, controls, data);
|
|
else
|
|
return -ENODEV;
|
|
}
|
|
EXPORT_SYMBOL_NS_GPL(cs_amp_write_cal_coeffs, "SND_SOC_CS_AMP_LIB");
|
|
|
|
static efi_status_t cs_amp_get_efi_variable(efi_char16_t *name,
|
|
efi_guid_t *guid,
|
|
unsigned long *size,
|
|
void *buf)
|
|
{
|
|
u32 attr;
|
|
|
|
KUNIT_STATIC_STUB_REDIRECT(cs_amp_get_efi_variable, name, guid, size, buf);
|
|
|
|
if (efi_rt_services_supported(EFI_RT_SUPPORTED_GET_VARIABLE))
|
|
return efi.get_variable(name, guid, &attr, size, buf);
|
|
|
|
return EFI_NOT_FOUND;
|
|
}
|
|
|
|
static int cs_amp_convert_efi_status(efi_status_t status)
|
|
{
|
|
switch (status) {
|
|
case EFI_SUCCESS:
|
|
return 0;
|
|
case EFI_NOT_FOUND:
|
|
return -ENOENT;
|
|
case EFI_BUFFER_TOO_SMALL:
|
|
return -EFBIG;
|
|
case EFI_UNSUPPORTED:
|
|
case EFI_ACCESS_DENIED:
|
|
case EFI_SECURITY_VIOLATION:
|
|
return -EACCES;
|
|
default:
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
static struct cirrus_amp_efi_data *cs_amp_get_cal_efi_buffer(struct device *dev)
|
|
{
|
|
struct cirrus_amp_efi_data *efi_data;
|
|
unsigned long data_size = 0;
|
|
u8 *data;
|
|
efi_status_t status;
|
|
int i, ret;
|
|
|
|
/* Find EFI variable and get size */
|
|
for (i = 0; i < ARRAY_SIZE(cs_amp_lib_cal_efivars); i++) {
|
|
status = cs_amp_get_efi_variable(cs_amp_lib_cal_efivars[i].name,
|
|
cs_amp_lib_cal_efivars[i].guid,
|
|
&data_size, NULL);
|
|
if (status == EFI_BUFFER_TOO_SMALL)
|
|
break;
|
|
}
|
|
|
|
if (status != EFI_BUFFER_TOO_SMALL)
|
|
return ERR_PTR(-ENOENT);
|
|
|
|
if (data_size < sizeof(*efi_data)) {
|
|
dev_err(dev, "EFI cal variable truncated\n");
|
|
return ERR_PTR(-EOVERFLOW);
|
|
}
|
|
|
|
/* Get variable contents into buffer */
|
|
data = kmalloc(data_size, GFP_KERNEL);
|
|
if (!data)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
status = cs_amp_get_efi_variable(cs_amp_lib_cal_efivars[i].name,
|
|
cs_amp_lib_cal_efivars[i].guid,
|
|
&data_size, data);
|
|
if (status != EFI_SUCCESS) {
|
|
ret = -EINVAL;
|
|
goto err;
|
|
}
|
|
|
|
efi_data = (struct cirrus_amp_efi_data *)data;
|
|
dev_dbg(dev, "Calibration: Size=%d, Amp Count=%d\n", efi_data->size, efi_data->count);
|
|
|
|
if ((efi_data->count > 128) ||
|
|
struct_size(efi_data, data, efi_data->count) > data_size) {
|
|
dev_err(dev, "EFI cal variable truncated\n");
|
|
ret = -EOVERFLOW;
|
|
goto err;
|
|
}
|
|
|
|
return efi_data;
|
|
|
|
err:
|
|
kfree(data);
|
|
dev_err(dev, "Failed to read calibration data from EFI: %d\n", ret);
|
|
|
|
return ERR_PTR(ret);
|
|
}
|
|
|
|
static u64 cs_amp_cal_target_u64(const struct cirrus_amp_cal_data *data)
|
|
{
|
|
return ((u64)data->calTarget[1] << 32) | data->calTarget[0];
|
|
}
|
|
|
|
static int _cs_amp_get_efi_calibration_data(struct device *dev, u64 target_uid, int amp_index,
|
|
struct cirrus_amp_cal_data *out_data)
|
|
{
|
|
struct cirrus_amp_efi_data *efi_data;
|
|
struct cirrus_amp_cal_data *cal = NULL;
|
|
int i, ret;
|
|
|
|
efi_data = cs_amp_get_cal_efi_buffer(dev);
|
|
if (IS_ERR(efi_data))
|
|
return PTR_ERR(efi_data);
|
|
|
|
if (target_uid) {
|
|
for (i = 0; i < efi_data->count; ++i) {
|
|
u64 cal_target = cs_amp_cal_target_u64(&efi_data->data[i]);
|
|
|
|
/* Skip empty entries */
|
|
if (!efi_data->data[i].calTime[0] && !efi_data->data[i].calTime[1])
|
|
continue;
|
|
|
|
/* Skip entries with unpopulated silicon ID */
|
|
if (cal_target == 0)
|
|
continue;
|
|
|
|
if (cal_target == target_uid) {
|
|
cal = &efi_data->data[i];
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!cal && (amp_index >= 0) && (amp_index < efi_data->count) &&
|
|
(efi_data->data[amp_index].calTime[0] || efi_data->data[amp_index].calTime[1])) {
|
|
u64 cal_target = cs_amp_cal_target_u64(&efi_data->data[amp_index]);
|
|
|
|
/*
|
|
* Treat unpopulated cal_target as a wildcard.
|
|
* If target_uid != 0 we can only get here if cal_target == 0
|
|
* or it didn't match any cal_target value.
|
|
* If target_uid == 0 it is a wildcard.
|
|
*/
|
|
if ((cal_target == 0) || (target_uid == 0))
|
|
cal = &efi_data->data[amp_index];
|
|
else
|
|
dev_warn(dev, "Calibration entry %d does not match silicon ID", amp_index);
|
|
}
|
|
|
|
if (cal) {
|
|
memcpy(out_data, cal, sizeof(*out_data));
|
|
ret = 0;
|
|
} else {
|
|
dev_warn(dev, "No calibration for silicon ID %#llx\n", target_uid);
|
|
ret = -ENOENT;
|
|
}
|
|
|
|
kfree(efi_data);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* cs_amp_get_efi_calibration_data - get an entry from calibration data in EFI.
|
|
* @dev: struct device of the caller.
|
|
* @target_uid: UID to match, or zero to ignore UID matching.
|
|
* @amp_index: Entry index to use, or -1 to prevent lookup by index.
|
|
* @out_data: struct cirrus_amp_cal_data where the entry will be copied.
|
|
*
|
|
* This function can perform 3 types of lookup:
|
|
*
|
|
* (target_uid > 0, amp_index >= 0)
|
|
* UID search with fallback to using the array index.
|
|
* Search the calibration data for a non-zero calTarget that matches
|
|
* target_uid, and if found return that entry. Else, if the entry at
|
|
* [amp_index] has calTarget == 0, return that entry. Else fail.
|
|
*
|
|
* (target_uid > 0, amp_index < 0)
|
|
* UID search only.
|
|
* Search the calibration data for a non-zero calTarget that matches
|
|
* target_uid, and if found return that entry. Else fail.
|
|
*
|
|
* (target_uid == 0, amp_index >= 0)
|
|
* Array index fetch only.
|
|
* Return the entry at [amp_index].
|
|
*
|
|
* An array lookup will be skipped if amp_index exceeds the number of
|
|
* entries in the calibration array, and in this case the return will
|
|
* be -ENOENT. An out-of-range amp_index does not prevent matching by
|
|
* target_uid - it has the same effect as passing amp_index < 0.
|
|
*
|
|
* If the EFI data is too short to be a valid entry, or the entry count
|
|
* in the EFI data overflows the actual length of the data, this function
|
|
* returns -EOVERFLOW.
|
|
*
|
|
* Return: 0 if the entry was found, -ENOENT if no entry was found,
|
|
* -EOVERFLOW if the EFI file is corrupt, else other error value.
|
|
*/
|
|
int cs_amp_get_efi_calibration_data(struct device *dev, u64 target_uid, int amp_index,
|
|
struct cirrus_amp_cal_data *out_data)
|
|
{
|
|
if (IS_ENABLED(CONFIG_EFI) || IS_ENABLED(CONFIG_SND_SOC_CS_AMP_LIB_TEST))
|
|
return _cs_amp_get_efi_calibration_data(dev, target_uid, amp_index, out_data);
|
|
else
|
|
return -ENOENT;
|
|
}
|
|
EXPORT_SYMBOL_NS_GPL(cs_amp_get_efi_calibration_data, "SND_SOC_CS_AMP_LIB");
|
|
|
|
struct cs_amp_spkid_efi {
|
|
efi_char16_t *name;
|
|
efi_guid_t *guid;
|
|
u8 values[2];
|
|
};
|
|
|
|
static int cs_amp_get_efi_byte_spkid(struct device *dev, const struct cs_amp_spkid_efi *info)
|
|
{
|
|
efi_status_t status;
|
|
unsigned long size;
|
|
u8 spkid;
|
|
int i, ret;
|
|
|
|
size = sizeof(spkid);
|
|
status = cs_amp_get_efi_variable(info->name, info->guid, &size, &spkid);
|
|
ret = cs_amp_convert_efi_status(status);
|
|
if (ret < 0)
|
|
return ret;
|
|
|
|
if (size == 0)
|
|
return -ENOENT;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(info->values); i++) {
|
|
if (info->values[i] == spkid)
|
|
return i;
|
|
}
|
|
|
|
dev_err(dev, "EFI speaker ID bad value %#x\n", spkid);
|
|
|
|
return -EINVAL;
|
|
}
|
|
|
|
static const struct cs_amp_spkid_efi cs_amp_spkid_byte_types[] = {
|
|
{
|
|
.name = LENOVO_SPEAKER_ID_EFI_NAME,
|
|
.guid = &LENOVO_SPEAKER_ID_EFI_GUID,
|
|
.values = { 0xd0, 0xd1 },
|
|
},
|
|
{
|
|
.name = HP_SPEAKER_ID_EFI_NAME,
|
|
.guid = &HP_SPEAKER_ID_EFI_GUID,
|
|
.values = { 0x30, 0x31 },
|
|
},
|
|
};
|
|
|
|
/**
|
|
* cs_amp_get_vendor_spkid - get a speaker ID from vendor-specific storage
|
|
* @dev: pointer to struct device
|
|
*
|
|
* Known vendor-specific methods of speaker ID are checked and if one is
|
|
* found its speaker ID value is returned.
|
|
*
|
|
* Return: >=0 is a valid speaker ID. -ENOENT if a vendor-specific method
|
|
* was not found. -EACCES if the vendor-specific storage could not
|
|
* be read. Other error values indicate that the data from the
|
|
* vendor-specific storage was found but could not be understood.
|
|
*/
|
|
int cs_amp_get_vendor_spkid(struct device *dev)
|
|
{
|
|
int i, ret;
|
|
|
|
if (!efi_rt_services_supported(EFI_RT_SUPPORTED_GET_VARIABLE) &&
|
|
!IS_ENABLED(CONFIG_SND_SOC_CS_AMP_LIB_TEST))
|
|
return -ENOENT;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(cs_amp_spkid_byte_types); i++) {
|
|
ret = cs_amp_get_efi_byte_spkid(dev, &cs_amp_spkid_byte_types[i]);
|
|
if (ret != -ENOENT)
|
|
return ret;
|
|
}
|
|
|
|
return -ENOENT;
|
|
}
|
|
EXPORT_SYMBOL_NS_GPL(cs_amp_get_vendor_spkid, "SND_SOC_CS_AMP_LIB");
|
|
|
|
static const struct cs_amp_test_hooks cs_amp_test_hook_ptrs = {
|
|
.get_efi_variable = cs_amp_get_efi_variable,
|
|
.write_cal_coeff = cs_amp_write_cal_coeff,
|
|
};
|
|
|
|
const struct cs_amp_test_hooks * const cs_amp_test_hooks =
|
|
PTR_IF(IS_ENABLED(CONFIG_SND_SOC_CS_AMP_LIB_TEST), &cs_amp_test_hook_ptrs);
|
|
EXPORT_SYMBOL_NS_GPL(cs_amp_test_hooks, "SND_SOC_CS_AMP_LIB");
|
|
|
|
MODULE_DESCRIPTION("Cirrus Logic amplifier library");
|
|
MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
|
|
MODULE_LICENSE("GPL");
|
|
MODULE_IMPORT_NS("FW_CS_DSP");
|