Commit 20935f3e authored by Miri Korenblit's avatar Miri Korenblit Committed by Johannes Berg

wifi: iwlwifi: read ECKV table from UEFI

Try to read the ECKV table from UEFI first,
and if the WIFI UEFI tables are unlocked or the
table doesn't exist - try to read it from ACPI.

Change iwl_acpi_get_eckv() to receive fwrt as argument so
it will be the same as all iwl_acpi_get_x() functions,
so it could  be generated by the macro.

While at it - move the reading of ECKV to INIT stage. There is no
reason to read it each time we load the FW.
Signed-off-by: default avatarMiri Korenblit <miriam.rachel.korenblit@intel.com>
Reviewed-by: default avatarGregory Greenman <gregory.greenman@intel.com>
Link: https://msgid.link/20240201155157.d4937cc00727.I36e5fc7f7850229b9b377c80b5203aa47137c97c@changeidSigned-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent 669761e8
...@@ -386,16 +386,17 @@ int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit) ...@@ -386,16 +386,17 @@ int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit)
return ret; return ret;
} }
int iwl_acpi_get_eckv(struct device *dev, u32 *extl_clk) int iwl_acpi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk)
{ {
union acpi_object *wifi_pkg, *data; union acpi_object *wifi_pkg, *data;
int ret, tbl_rev; int ret, tbl_rev;
data = iwl_acpi_get_object(dev, ACPI_ECKV_METHOD); data = iwl_acpi_get_object(fwrt->dev, ACPI_ECKV_METHOD);
if (IS_ERR(data)) if (IS_ERR(data))
return PTR_ERR(data); return PTR_ERR(data);
wifi_pkg = iwl_acpi_get_wifi_pkg(dev, data, ACPI_ECKV_WIFI_DATA_SIZE, wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
ACPI_ECKV_WIFI_DATA_SIZE,
&tbl_rev); &tbl_rev);
if (IS_ERR(wifi_pkg)) { if (IS_ERR(wifi_pkg)) {
ret = PTR_ERR(wifi_pkg); ret = PTR_ERR(wifi_pkg);
...@@ -416,7 +417,6 @@ int iwl_acpi_get_eckv(struct device *dev, u32 *extl_clk) ...@@ -416,7 +417,6 @@ int iwl_acpi_get_eckv(struct device *dev, u32 *extl_clk)
kfree(data); kfree(data);
return ret; return ret;
} }
IWL_EXPORT_SYMBOL(iwl_acpi_get_eckv);
static int iwl_acpi_sar_set_profile(union acpi_object *table, static int iwl_acpi_sar_set_profile(union acpi_object *table,
struct iwl_sar_profile *profile, struct iwl_sar_profile *profile,
......
...@@ -161,13 +161,13 @@ int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit); ...@@ -161,13 +161,13 @@ int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit);
/* /*
* iwl_acpi_get_eckv - read external clock validation from ACPI, if available * iwl_acpi_get_eckv - read external clock validation from ACPI, if available
* *
* @dev: the struct device * @fwrt: the fw runtime struct
* @extl_clk: output var (2 bytes) that will get the clk indication. * @extl_clk: output var (2 bytes) that will get the clk indication.
* *
* This function tries to read the external clock indication * This function tries to read the external clock indication
* from ACPI if available. * from ACPI if available.
*/ */
int iwl_acpi_get_eckv(struct device *dev, u32 *extl_clk); int iwl_acpi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk);
int iwl_acpi_get_wrds_table(struct iwl_fw_runtime *fwrt); int iwl_acpi_get_wrds_table(struct iwl_fw_runtime *fwrt);
...@@ -219,7 +219,7 @@ static inline int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, ...@@ -219,7 +219,7 @@ static inline int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt,
return 0; return 0;
} }
static inline int iwl_acpi_get_eckv(struct device *dev, u32 *extl_clk) static inline int iwl_acpi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk)
{ {
return -ENOENT; return -ENOENT;
} }
......
...@@ -37,6 +37,7 @@ IWL_BIOS_TABLE_LOADER(ppag_table); ...@@ -37,6 +37,7 @@ IWL_BIOS_TABLE_LOADER(ppag_table);
IWL_BIOS_TABLE_LOADER_DATA(tas_table, struct iwl_tas_data); IWL_BIOS_TABLE_LOADER_DATA(tas_table, struct iwl_tas_data);
IWL_BIOS_TABLE_LOADER_DATA(pwr_limit, u64); IWL_BIOS_TABLE_LOADER_DATA(pwr_limit, u64);
IWL_BIOS_TABLE_LOADER_DATA(mcc, char); IWL_BIOS_TABLE_LOADER_DATA(mcc, char);
IWL_BIOS_TABLE_LOADER_DATA(eckv, u32);
static const struct dmi_system_id dmi_ppag_approved_list[] = { static const struct dmi_system_id dmi_ppag_approved_list[] = {
......
...@@ -141,4 +141,5 @@ int iwl_bios_get_pwr_limit(struct iwl_fw_runtime *fwrt, ...@@ -141,4 +141,5 @@ int iwl_bios_get_pwr_limit(struct iwl_fw_runtime *fwrt,
u64 *dflt_pwr_limit); u64 *dflt_pwr_limit);
int iwl_bios_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc); int iwl_bios_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc);
int iwl_bios_get_eckv(struct iwl_fw_runtime *fwrt, u32 *ext_clk);
#endif /* __fw_regulatory_h__ */ #endif /* __fw_regulatory_h__ */
...@@ -652,3 +652,25 @@ int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc) ...@@ -652,3 +652,25 @@ int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc)
kfree(data); kfree(data);
return ret; return ret;
} }
int iwl_uefi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk)
{
struct uefi_cnv_var_eckv *data;
int ret = 0;
data = iwl_uefi_get_verified_variable(fwrt->trans, IWL_UEFI_ECKV_NAME,
"ECKV", sizeof(*data), NULL);
if (IS_ERR(data))
return -EINVAL;
if (data->revision != IWL_UEFI_ECKV_REVISION) {
ret = -EINVAL;
IWL_DEBUG_RADIO(fwrt, "Unsupported UEFI WRDD revision:%d\n",
data->revision);
goto out;
}
*extl_clk = data->ext_clock_valid;
out:
kfree(data);
return ret;
}
...@@ -19,6 +19,7 @@ ...@@ -19,6 +19,7 @@
#define IWL_UEFI_WTAS_NAME L"UefiCnvWlanWTAS" #define IWL_UEFI_WTAS_NAME L"UefiCnvWlanWTAS"
#define IWL_UEFI_SPLC_NAME L"UefiCnvWlanSPLC" #define IWL_UEFI_SPLC_NAME L"UefiCnvWlanSPLC"
#define IWL_UEFI_WRDD_NAME L"UefiCnvWlanWRDD" #define IWL_UEFI_WRDD_NAME L"UefiCnvWlanWRDD"
#define IWL_UEFI_ECKV_NAME L"UefiCnvWlanECKV"
#define IWL_SGOM_MAP_SIZE 339 #define IWL_SGOM_MAP_SIZE 339
...@@ -32,6 +33,7 @@ ...@@ -32,6 +33,7 @@
#define IWL_UEFI_WTAS_REVISION 1 #define IWL_UEFI_WTAS_REVISION 1
#define IWL_UEFI_SPLC_REVISION 0 #define IWL_UEFI_SPLC_REVISION 0
#define IWL_UEFI_WRDD_REVISION 0 #define IWL_UEFI_WRDD_REVISION 0
#define IWL_UEFI_ECKV_REVISION 0
struct pnvm_sku_package { struct pnvm_sku_package {
u8 rev; u8 rev;
...@@ -155,6 +157,15 @@ struct uefi_cnv_var_wrdd { ...@@ -155,6 +157,15 @@ struct uefi_cnv_var_wrdd {
u32 mcc; u32 mcc;
} __packed; } __packed;
/* struct uefi_cnv_var_eckv - ECKV table as defined in UEFI
* @revision: the revision of the table
* @ext_clock_valid: indicates if external 32KHz clock is valid
*/
struct uefi_cnv_var_eckv {
u8 revision;
u32 ext_clock_valid;
} __packed;
/* /*
* This is known to be broken on v4.19 and to work on v5.4. Until we * This is known to be broken on v4.19 and to work on v5.4. Until we
* figure out why this is the case and how to make it work, simply * figure out why this is the case and how to make it work, simply
...@@ -178,6 +189,7 @@ int iwl_uefi_get_tas_table(struct iwl_fw_runtime *fwrt, ...@@ -178,6 +189,7 @@ int iwl_uefi_get_tas_table(struct iwl_fw_runtime *fwrt,
int iwl_uefi_get_pwr_limit(struct iwl_fw_runtime *fwrt, int iwl_uefi_get_pwr_limit(struct iwl_fw_runtime *fwrt,
u64 *dflt_pwr_limit); u64 *dflt_pwr_limit);
int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc); int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc);
int iwl_uefi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk);
#else /* CONFIG_EFI */ #else /* CONFIG_EFI */
static inline void *iwl_uefi_get_pnvm(struct iwl_trans *trans, size_t *len) static inline void *iwl_uefi_get_pnvm(struct iwl_trans *trans, size_t *len)
{ {
...@@ -246,6 +258,11 @@ static inline int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc) ...@@ -246,6 +258,11 @@ static inline int iwl_uefi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc)
{ {
return -ENOENT; return -ENOENT;
} }
static inline int iwl_uefi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk)
{
return -ENOENT;
}
#endif /* CONFIG_EFI */ #endif /* CONFIG_EFI */
#if defined(CONFIG_EFI) && defined(CONFIG_ACPI) #if defined(CONFIG_EFI) && defined(CONFIG_ACPI)
......
...@@ -1409,6 +1409,9 @@ void iwl_mvm_get_bios_tables(struct iwl_mvm *mvm) ...@@ -1409,6 +1409,9 @@ void iwl_mvm_get_bios_tables(struct iwl_mvm *mvm)
} }
iwl_acpi_get_phy_filters(&mvm->fwrt, &mvm->phy_filters); iwl_acpi_get_phy_filters(&mvm->fwrt, &mvm->phy_filters);
if (iwl_bios_get_eckv(&mvm->fwrt, &mvm->ext_clock_valid))
IWL_DEBUG_RADIO(mvm, "ECKV table doesn't exist in BIOS\n");
} }
static void iwl_mvm_disconnect_iterator(void *data, u8 *mac, static void iwl_mvm_disconnect_iterator(void *data, u8 *mac,
...@@ -1705,9 +1708,6 @@ int iwl_mvm_up(struct iwl_mvm *mvm) ...@@ -1705,9 +1708,6 @@ int iwl_mvm_up(struct iwl_mvm *mvm)
if (!mvm->ptp_data.ptp_clock) if (!mvm->ptp_data.ptp_clock)
iwl_mvm_ptp_init(mvm); iwl_mvm_ptp_init(mvm);
if (iwl_acpi_get_eckv(mvm->dev, &mvm->ext_clock_valid))
IWL_DEBUG_INFO(mvm, "ECKV table doesn't exist in BIOS\n");
ret = iwl_mvm_ppag_init(mvm); ret = iwl_mvm_ppag_init(mvm);
if (ret) if (ret)
goto error; goto error;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment