Commit 2a808414 authored by Luca Coelho's avatar Luca Coelho

iwlwifi: acpi: support reading and storing WRDS revision 1 and 2

Change the SAR profile tables storage to revision 2 regardless of the
revision we read from ACPI.  Revision 2 is a superset of revision 1,
which is in turn a superset of revision 0, so they can all be stored
inside revision 2.

Add support for reading and storing also revisions 1 and 2, whose only
difference is the number of chains and number of sub-bands.  So most
of the code revolves around passing different chain and sub-band
sizes.

With this patch we still pass only revision 0 to the firmware, but
that will be changed in a separate patch.
Signed-off-by: default avatarLuca Coelho <luciano.coelho@intel.com>
Link: https://lore.kernel.org/r/iwlwifi.20210805141826.2edad09415c8.I36297aef90a9ec6a3ea1bf1a151a62f272826d59@changeidSigned-off-by: default avatarLuca Coelho <luciano.coelho@intel.com>
parent 8ecf0477
...@@ -442,10 +442,10 @@ static int iwl_sar_fill_table(struct iwl_fw_runtime *fwrt, ...@@ -442,10 +442,10 @@ static int iwl_sar_fill_table(struct iwl_fw_runtime *fwrt,
__le16 *per_chain, u32 n_subbands, __le16 *per_chain, u32 n_subbands,
int prof_a, int prof_b) int prof_a, int prof_b)
{ {
int profs[ACPI_SAR_NUM_CHAINS] = { prof_a, prof_b }; int profs[ACPI_SAR_NUM_CHAINS_REV0] = { prof_a, prof_b };
int i, j; int i, j;
for (i = 0; i < ACPI_SAR_NUM_CHAINS; i++) { for (i = 0; i < ACPI_SAR_NUM_CHAINS_REV0; i++) {
struct iwl_sar_profile *prof; struct iwl_sar_profile *prof;
/* don't allow SAR to be disabled (profile 0 means disable) */ /* don't allow SAR to be disabled (profile 0 means disable) */
...@@ -494,7 +494,7 @@ int iwl_sar_select_profile(struct iwl_fw_runtime *fwrt, ...@@ -494,7 +494,7 @@ int iwl_sar_select_profile(struct iwl_fw_runtime *fwrt,
for (i = 0; i < n_tables; i++) { for (i = 0; i < n_tables; i++) {
ret = iwl_sar_fill_table(fwrt, ret = iwl_sar_fill_table(fwrt,
&per_chain[i * n_subbands * ACPI_SAR_NUM_CHAINS], &per_chain[i * n_subbands * ACPI_SAR_NUM_CHAINS_REV0],
n_subbands, prof_a, prof_b); n_subbands, prof_a, prof_b);
if (ret) if (ret)
break; break;
...@@ -509,28 +509,71 @@ int iwl_sar_get_wrds_table(struct iwl_fw_runtime *fwrt) ...@@ -509,28 +509,71 @@ int iwl_sar_get_wrds_table(struct iwl_fw_runtime *fwrt)
union acpi_object *wifi_pkg, *table, *data; union acpi_object *wifi_pkg, *table, *data;
bool enabled; bool enabled;
int ret, tbl_rev; int ret, tbl_rev;
u8 num_chains, num_sub_bands;
data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDS_METHOD); data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDS_METHOD);
if (IS_ERR(data)) if (IS_ERR(data))
return PTR_ERR(data); return PTR_ERR(data);
/* start by trying to read revision 2 */
wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
ACPI_WRDS_WIFI_DATA_SIZE, &tbl_rev); ACPI_WRDS_WIFI_DATA_SIZE_REV2,
if (IS_ERR(wifi_pkg)) { &tbl_rev);
ret = PTR_ERR(wifi_pkg); if (!IS_ERR(wifi_pkg)) {
goto out_free; if (tbl_rev != 2) {
ret = PTR_ERR(wifi_pkg);
goto out_free;
}
num_chains = ACPI_SAR_NUM_CHAINS_REV2;
num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV2;
goto read_table;
} }
if (tbl_rev != 0) { /* then try revision 1 */
ret = -EINVAL; wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
goto out_free; ACPI_WRDS_WIFI_DATA_SIZE_REV1,
&tbl_rev);
if (!IS_ERR(wifi_pkg)) {
if (tbl_rev != 1) {
ret = PTR_ERR(wifi_pkg);
goto out_free;
}
num_chains = ACPI_SAR_NUM_CHAINS_REV1;
num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV1;
goto read_table;
} }
/* then finally revision 0 */
wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
ACPI_WRDS_WIFI_DATA_SIZE_REV0,
&tbl_rev);
if (!IS_ERR(wifi_pkg)) {
if (tbl_rev != 0) {
ret = PTR_ERR(wifi_pkg);
goto out_free;
}
num_chains = ACPI_SAR_NUM_CHAINS_REV0;
num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV0;
goto read_table;
}
ret = PTR_ERR(wifi_pkg);
goto out_free;
read_table:
if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER) { if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER) {
ret = -EINVAL; ret = -EINVAL;
goto out_free; goto out_free;
} }
IWL_DEBUG_RADIO(fwrt, "Reading WRDS tbl_rev=%d\n", tbl_rev);
enabled = !!(wifi_pkg->package.elements[1].integer.value); enabled = !!(wifi_pkg->package.elements[1].integer.value);
/* position of the actual table */ /* position of the actual table */
...@@ -540,7 +583,7 @@ int iwl_sar_get_wrds_table(struct iwl_fw_runtime *fwrt) ...@@ -540,7 +583,7 @@ int iwl_sar_get_wrds_table(struct iwl_fw_runtime *fwrt)
* into sar_profiles[0] (because we don't have a profile 0). * into sar_profiles[0] (because we don't have a profile 0).
*/ */
ret = iwl_sar_set_profile(table, &fwrt->sar_profiles[0], enabled, ret = iwl_sar_set_profile(table, &fwrt->sar_profiles[0], enabled,
ACPI_SAR_NUM_CHAINS, ACPI_SAR_NUM_SUB_BANDS); num_chains, num_sub_bands);
out_free: out_free:
kfree(data); kfree(data);
return ret; return ret;
...@@ -598,15 +641,14 @@ int iwl_sar_get_ewrd_table(struct iwl_fw_runtime *fwrt) ...@@ -598,15 +641,14 @@ int iwl_sar_get_ewrd_table(struct iwl_fw_runtime *fwrt)
* have profile 0). So in the array we start from 1. * have profile 0). So in the array we start from 1.
*/ */
ret = iwl_sar_set_profile(&wifi_pkg->package.elements[pos], ret = iwl_sar_set_profile(&wifi_pkg->package.elements[pos],
&fwrt->sar_profiles[i + 1], &fwrt->sar_profiles[i + 1], enabled,
enabled, ACPI_SAR_NUM_CHAINS_REV0,
ACPI_SAR_NUM_CHAINS, ACPI_SAR_NUM_SUB_BANDS_REV0);
ACPI_SAR_NUM_SUB_BANDS);
if (ret < 0) if (ret < 0)
break; break;
/* go to the next table */ /* go to the next table */
pos += ACPI_SAR_NUM_CHAINS * ACPI_SAR_NUM_SUB_BANDS; pos += ACPI_SAR_NUM_CHAINS_REV0 * ACPI_SAR_NUM_SUB_BANDS_REV0;
} }
out_free: out_free:
......
...@@ -32,14 +32,22 @@ ...@@ -32,14 +32,22 @@
#define ACPI_NUM_GEO_PROFILES 3 #define ACPI_NUM_GEO_PROFILES 3
#define ACPI_GEO_PER_CHAIN_SIZE 3 #define ACPI_GEO_PER_CHAIN_SIZE 3
#define ACPI_SAR_NUM_CHAINS 2 #define ACPI_SAR_NUM_CHAINS_REV0 2
#define ACPI_SAR_NUM_SUB_BANDS 5 #define ACPI_SAR_NUM_CHAINS_REV1 2
#define ACPI_SAR_NUM_CHAINS_REV2 4
#define ACPI_WRDS_WIFI_DATA_SIZE (ACPI_SAR_NUM_CHAINS * \ #define ACPI_SAR_NUM_SUB_BANDS_REV0 5
ACPI_SAR_NUM_SUB_BANDS + 2) #define ACPI_SAR_NUM_SUB_BANDS_REV1 11
#define ACPI_SAR_NUM_SUB_BANDS_REV2 11
#define ACPI_WRDS_WIFI_DATA_SIZE_REV0 (ACPI_SAR_NUM_CHAINS_REV0 * \
ACPI_SAR_NUM_SUB_BANDS_REV0 + 2)
#define ACPI_WRDS_WIFI_DATA_SIZE_REV1 (ACPI_SAR_NUM_CHAINS_REV1 * \
ACPI_SAR_NUM_SUB_BANDS_REV1 + 2)
#define ACPI_WRDS_WIFI_DATA_SIZE_REV2 (ACPI_SAR_NUM_CHAINS_REV2 * \
ACPI_SAR_NUM_SUB_BANDS_REV2 + 2)
#define ACPI_EWRD_WIFI_DATA_SIZE ((ACPI_SAR_PROFILE_NUM - 1) * \ #define ACPI_EWRD_WIFI_DATA_SIZE ((ACPI_SAR_PROFILE_NUM - 1) * \
ACPI_SAR_NUM_CHAINS * \ ACPI_SAR_NUM_CHAINS_REV0 * \
ACPI_SAR_NUM_SUB_BANDS + 3) ACPI_SAR_NUM_SUB_BANDS_REV0 + 3)
#define ACPI_WGDS_WIFI_DATA_SIZE 19 #define ACPI_WGDS_WIFI_DATA_SIZE 19
#define ACPI_WRDD_WIFI_DATA_SIZE 2 #define ACPI_WRDD_WIFI_DATA_SIZE 2
#define ACPI_SPLC_WIFI_DATA_SIZE 2 #define ACPI_SPLC_WIFI_DATA_SIZE 2
...@@ -64,13 +72,18 @@ ...@@ -64,13 +72,18 @@
#define ACPI_PPAG_MIN_HB -16 #define ACPI_PPAG_MIN_HB -16
#define ACPI_PPAG_MAX_HB 40 #define ACPI_PPAG_MAX_HB 40
/*
* The profile for revision 2 is a superset of revision 1, which is in
* turn a superset of revision 0. So we can store all revisions
* inside revision 2, which is what we represent here.
*/
struct iwl_sar_profile_chain { struct iwl_sar_profile_chain {
u8 subbands[ACPI_SAR_NUM_SUB_BANDS]; u8 subbands[ACPI_SAR_NUM_SUB_BANDS_REV2];
}; };
struct iwl_sar_profile { struct iwl_sar_profile {
bool enabled; bool enabled;
struct iwl_sar_profile_chain chains[ACPI_SAR_NUM_CHAINS]; struct iwl_sar_profile_chain chains[ACPI_SAR_NUM_CHAINS_REV2];
}; };
struct iwl_geo_profile { struct iwl_geo_profile {
......
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