| // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause |
| /* |
| * Copyright (C) 2017 Intel Deutschland GmbH |
| * Copyright (C) 2019-2024 Intel Corporation |
| */ |
| #include <linux/uuid.h> |
| #include "iwl-drv.h" |
| #include "iwl-debug.h" |
| #include "acpi.h" |
| #include "fw/runtime.h" |
| |
| const guid_t iwl_guid = GUID_INIT(0xF21202BF, 0x8F78, 0x4DC6, |
| 0xA5, 0xB3, 0x1F, 0x73, |
| 0x8E, 0x28, 0x5A, 0xDE); |
| |
| static const size_t acpi_dsm_size[DSM_FUNC_NUM_FUNCS] = { |
| [DSM_FUNC_QUERY] = sizeof(u32), |
| [DSM_FUNC_DISABLE_SRD] = sizeof(u8), |
| [DSM_FUNC_ENABLE_INDONESIA_5G2] = sizeof(u8), |
| [DSM_FUNC_ENABLE_6E] = sizeof(u32), |
| [DSM_FUNC_REGULATORY_CONFIG] = sizeof(u32), |
| /* Not supported in driver */ |
| [5] = (size_t)0, |
| [DSM_FUNC_11AX_ENABLEMENT] = sizeof(u32), |
| [DSM_FUNC_ENABLE_UNII4_CHAN] = sizeof(u32), |
| [DSM_FUNC_ACTIVATE_CHANNEL] = sizeof(u32), |
| [DSM_FUNC_FORCE_DISABLE_CHANNELS] = sizeof(u32), |
| [DSM_FUNC_ENERGY_DETECTION_THRESHOLD] = sizeof(u32), |
| [DSM_FUNC_RFI_CONFIG] = sizeof(u32), |
| [DSM_FUNC_ENABLE_11BE] = sizeof(u32), |
| }; |
| |
| static int iwl_acpi_get_handle(struct device *dev, acpi_string method, |
| acpi_handle *ret_handle) |
| { |
| acpi_handle root_handle; |
| acpi_status status; |
| |
| root_handle = ACPI_HANDLE(dev); |
| if (!root_handle) { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: Could not retrieve root port handle\n"); |
| return -ENOENT; |
| } |
| |
| status = acpi_get_handle(root_handle, method, ret_handle); |
| if (ACPI_FAILURE(status)) { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: %s method not found\n", method); |
| return -ENOENT; |
| } |
| return 0; |
| } |
| |
| static void *iwl_acpi_get_object(struct device *dev, acpi_string method) |
| { |
| struct acpi_buffer buf = {ACPI_ALLOCATE_BUFFER, NULL}; |
| acpi_handle handle; |
| acpi_status status; |
| int ret; |
| |
| ret = iwl_acpi_get_handle(dev, method, &handle); |
| if (ret) |
| return ERR_PTR(-ENOENT); |
| |
| /* Call the method with no arguments */ |
| status = acpi_evaluate_object(handle, NULL, NULL, &buf); |
| if (ACPI_FAILURE(status)) { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: %s method invocation failed (status: 0x%x)\n", |
| method, status); |
| return ERR_PTR(-ENOENT); |
| } |
| return buf.pointer; |
| } |
| |
| /* |
| * Generic function for evaluating a method defined in the device specific |
| * method (DSM) interface. The returned acpi object must be freed by calling |
| * function. |
| */ |
| static void *iwl_acpi_get_dsm_object(struct device *dev, int rev, int func, |
| union acpi_object *args, |
| const guid_t *guid) |
| { |
| union acpi_object *obj; |
| |
| obj = acpi_evaluate_dsm(ACPI_HANDLE(dev), guid, rev, func, |
| args); |
| if (!obj) { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: DSM method invocation failed (rev: %d, func:%d)\n", |
| rev, func); |
| return ERR_PTR(-ENOENT); |
| } |
| return obj; |
| } |
| |
| /* |
| * Generic function to evaluate a DSM with no arguments |
| * and an integer return value, |
| * (as an integer object or inside a buffer object), |
| * verify and assign the value in the "value" parameter. |
| * return 0 in success and the appropriate errno otherwise. |
| */ |
| static int iwl_acpi_get_dsm_integer(struct device *dev, int rev, int func, |
| const guid_t *guid, u64 *value, |
| size_t expected_size) |
| { |
| union acpi_object *obj; |
| int ret = 0; |
| |
| obj = iwl_acpi_get_dsm_object(dev, rev, func, NULL, guid); |
| if (IS_ERR(obj)) { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "Failed to get DSM object. func= %d\n", |
| func); |
| return -ENOENT; |
| } |
| |
| if (obj->type == ACPI_TYPE_INTEGER) { |
| *value = obj->integer.value; |
| } else if (obj->type == ACPI_TYPE_BUFFER) { |
| __le64 le_value = 0; |
| |
| if (WARN_ON_ONCE(expected_size > sizeof(le_value))) |
| return -EINVAL; |
| |
| /* if the buffer size doesn't match the expected size */ |
| if (obj->buffer.length != expected_size) |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: DSM invalid buffer size, padding or truncating (%d)\n", |
| obj->buffer.length); |
| |
| /* assuming LE from Intel BIOS spec */ |
| memcpy(&le_value, obj->buffer.pointer, |
| min_t(size_t, expected_size, (size_t)obj->buffer.length)); |
| *value = le64_to_cpu(le_value); |
| } else { |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: DSM method did not return a valid object, type=%d\n", |
| obj->type); |
| ret = -EINVAL; |
| goto out; |
| } |
| |
| IWL_DEBUG_DEV_RADIO(dev, |
| "ACPI: DSM method evaluated: func=%d, ret=%d\n", |
| func, ret); |
| out: |
| ACPI_FREE(obj); |
| return ret; |
| } |
| |
| /* |
| * This function receives a DSM function number, calculates its expected size |
| * according to Intel BIOS spec, and fills in the value in a 32-bit field. |
| * In case the expected size is smaller than 32-bit, padding will be added. |
| */ |
| int iwl_acpi_get_dsm(struct iwl_fw_runtime *fwrt, |
| enum iwl_dsm_funcs func, u32 *value) |
| { |
| size_t expected_size; |
| u64 tmp; |
| int ret; |
| |
| BUILD_BUG_ON(ARRAY_SIZE(acpi_dsm_size) != DSM_FUNC_NUM_FUNCS); |
| |
| if (WARN_ON(func >= ARRAY_SIZE(acpi_dsm_size))) |
| return -EINVAL; |
| |
| expected_size = acpi_dsm_size[func]; |
| |
| /* Currently all ACPI DSMs are either 8-bit or 32-bit */ |
| if (expected_size != sizeof(u8) && expected_size != sizeof(u32)) |
| return -EOPNOTSUPP; |
| |
| ret = iwl_acpi_get_dsm_integer(fwrt->dev, ACPI_DSM_REV, func, |
| &iwl_guid, &tmp, expected_size); |
| if (ret) |
| return ret; |
| |
| if ((expected_size == sizeof(u8) && tmp != (u8)tmp) || |
| (expected_size == sizeof(u32) && tmp != (u32)tmp)) |
| IWL_DEBUG_RADIO(fwrt, |
| "DSM value overflows the expected size, truncating\n"); |
| *value = (u32)tmp; |
| |
| return 0; |
| } |
| |
| static union acpi_object * |
| iwl_acpi_get_wifi_pkg_range(struct device *dev, |
| union acpi_object *data, |
| int min_data_size, |
| int max_data_size, |
| int *tbl_rev) |
| { |
| int i; |
| union acpi_object *wifi_pkg; |
| |
| /* |
| * We need at least one entry in the wifi package that |
| * describes the domain, and one more entry, otherwise there's |
| * no point in reading it. |
| */ |
| if (WARN_ON_ONCE(min_data_size < 2 || min_data_size > max_data_size)) |
| return ERR_PTR(-EINVAL); |
| |
| /* |
| * We need at least two packages, one for the revision and one |
| * for the data itself. Also check that the revision is valid |
| * (i.e. it is an integer (each caller has to check by itself |
| * if the returned revision is supported)). |
| */ |
| if (data->type != ACPI_TYPE_PACKAGE || |
| data->package.count < 2 || |
| data->package.elements[0].type != ACPI_TYPE_INTEGER) { |
| IWL_DEBUG_DEV_RADIO(dev, "Invalid packages structure\n"); |
| return ERR_PTR(-EINVAL); |
| } |
| |
| *tbl_rev = data->package.elements[0].integer.value; |
| |
| /* loop through all the packages to find the one for WiFi */ |
| for (i = 1; i < data->package.count; i++) { |
| union acpi_object *domain; |
| |
| wifi_pkg = &data->package.elements[i]; |
| |
| /* skip entries that are not a package with the right size */ |
| if (wifi_pkg->type != ACPI_TYPE_PACKAGE || |
| wifi_pkg->package.count < min_data_size || |
| wifi_pkg->package.count > max_data_size) |
| continue; |
| |
| domain = &wifi_pkg->package.elements[0]; |
| if (domain->type == ACPI_TYPE_INTEGER && |
| domain->integer.value == ACPI_WIFI_DOMAIN) |
| goto found; |
| } |
| |
| return ERR_PTR(-ENOENT); |
| |
| found: |
| return wifi_pkg; |
| } |
| |
| static union acpi_object * |
| iwl_acpi_get_wifi_pkg(struct device *dev, |
| union acpi_object *data, |
| int data_size, int *tbl_rev) |
| { |
| return iwl_acpi_get_wifi_pkg_range(dev, data, data_size, data_size, |
| tbl_rev); |
| } |
| |
| int iwl_acpi_get_tas_table(struct iwl_fw_runtime *fwrt, |
| struct iwl_tas_data *tas_data) |
| { |
| union acpi_object *wifi_pkg, *data; |
| int ret, tbl_rev, i, block_list_size, enabled; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WTAS_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| /* try to read wtas table revision 1 or revision 0*/ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WTAS_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) { |
| ret = PTR_ERR(wifi_pkg); |
| goto out_free; |
| } |
| |
| if (tbl_rev == 1 && wifi_pkg->package.elements[1].type == |
| ACPI_TYPE_INTEGER) { |
| u32 tas_selection = |
| (u32)wifi_pkg->package.elements[1].integer.value; |
| |
| enabled = iwl_parse_tas_selection(fwrt, tas_data, |
| tas_selection); |
| |
| } else if (tbl_rev == 0 && |
| wifi_pkg->package.elements[1].type == ACPI_TYPE_INTEGER) { |
| enabled = !!wifi_pkg->package.elements[1].integer.value; |
| } else { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| if (!enabled) { |
| IWL_DEBUG_RADIO(fwrt, "TAS not enabled\n"); |
| ret = 0; |
| goto out_free; |
| } |
| |
| IWL_DEBUG_RADIO(fwrt, "Reading TAS table revision %d\n", tbl_rev); |
| if (wifi_pkg->package.elements[2].type != ACPI_TYPE_INTEGER || |
| wifi_pkg->package.elements[2].integer.value > |
| IWL_WTAS_BLACK_LIST_MAX) { |
| IWL_DEBUG_RADIO(fwrt, "TAS invalid array size %llu\n", |
| wifi_pkg->package.elements[2].integer.value); |
| ret = -EINVAL; |
| goto out_free; |
| } |
| block_list_size = wifi_pkg->package.elements[2].integer.value; |
| tas_data->block_list_size = cpu_to_le32(block_list_size); |
| |
| IWL_DEBUG_RADIO(fwrt, "TAS array size %u\n", block_list_size); |
| |
| for (i = 0; i < block_list_size; i++) { |
| u32 country; |
| |
| if (wifi_pkg->package.elements[3 + i].type != |
| ACPI_TYPE_INTEGER) { |
| IWL_DEBUG_RADIO(fwrt, |
| "TAS invalid array elem %d\n", 3 + i); |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| country = wifi_pkg->package.elements[3 + i].integer.value; |
| tas_data->block_list_array[i] = cpu_to_le32(country); |
| IWL_DEBUG_RADIO(fwrt, "TAS block list country %d\n", country); |
| } |
| |
| ret = 1; |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| int iwl_acpi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc) |
| { |
| union acpi_object *wifi_pkg, *data; |
| u32 mcc_val; |
| int ret, tbl_rev; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDD_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WRDD_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) { |
| ret = PTR_ERR(wifi_pkg); |
| goto out_free; |
| } |
| |
| if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER || |
| tbl_rev != 0) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| mcc_val = wifi_pkg->package.elements[1].integer.value; |
| |
| mcc[0] = (mcc_val >> 8) & 0xff; |
| mcc[1] = mcc_val & 0xff; |
| mcc[2] = '\0'; |
| |
| ret = 0; |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit) |
| { |
| union acpi_object *data, *wifi_pkg; |
| int tbl_rev, ret = -EINVAL; |
| |
| *dflt_pwr_limit = 0; |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_SPLC_METHOD); |
| if (IS_ERR(data)) |
| goto out; |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_SPLC_WIFI_DATA_SIZE, &tbl_rev); |
| if (IS_ERR(wifi_pkg) || tbl_rev != 0 || |
| wifi_pkg->package.elements[1].integer.value != ACPI_TYPE_INTEGER) |
| goto out_free; |
| |
| *dflt_pwr_limit = wifi_pkg->package.elements[1].integer.value; |
| ret = 0; |
| out_free: |
| kfree(data); |
| out: |
| return ret; |
| } |
| |
| int iwl_acpi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk) |
| { |
| union acpi_object *wifi_pkg, *data; |
| int ret, tbl_rev; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_ECKV_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_ECKV_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) { |
| ret = PTR_ERR(wifi_pkg); |
| goto out_free; |
| } |
| |
| if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER || |
| tbl_rev != 0) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| *extl_clk = wifi_pkg->package.elements[1].integer.value; |
| |
| ret = 0; |
| |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| static int iwl_acpi_sar_set_profile(union acpi_object *table, |
| struct iwl_sar_profile *profile, |
| bool enabled, u8 num_chains, |
| u8 num_sub_bands) |
| { |
| int i, j, idx = 0; |
| |
| /* |
| * The table from ACPI is flat, but we store it in a |
| * structured array. |
| */ |
| for (i = 0; i < BIOS_SAR_MAX_CHAINS_PER_PROFILE; i++) { |
| for (j = 0; j < BIOS_SAR_MAX_SUB_BANDS_NUM; j++) { |
| /* if we don't have the values, use the default */ |
| if (i >= num_chains || j >= num_sub_bands) { |
| profile->chains[i].subbands[j] = 0; |
| } else { |
| if (table[idx].type != ACPI_TYPE_INTEGER || |
| table[idx].integer.value > U8_MAX) |
| return -EINVAL; |
| |
| profile->chains[i].subbands[j] = |
| table[idx].integer.value; |
| |
| idx++; |
| } |
| } |
| } |
| |
| /* Only if all values were valid can the profile be enabled */ |
| profile->enabled = enabled; |
| |
| return 0; |
| } |
| |
| int iwl_acpi_get_wrds_table(struct iwl_fw_runtime *fwrt) |
| { |
| union acpi_object *wifi_pkg, *table, *data; |
| int ret, tbl_rev; |
| u32 flags; |
| u8 num_chains, num_sub_bands; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDS_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| /* start by trying to read revision 2 */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WRDS_WIFI_DATA_SIZE_REV2, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 2) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| num_chains = ACPI_SAR_NUM_CHAINS_REV2; |
| num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV2; |
| |
| goto read_table; |
| } |
| |
| /* then try revision 1 */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WRDS_WIFI_DATA_SIZE_REV1, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 1) { |
| ret = -EINVAL; |
| 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 = -EINVAL; |
| 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) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| IWL_DEBUG_RADIO(fwrt, "Reading WRDS tbl_rev=%d\n", tbl_rev); |
| |
| flags = wifi_pkg->package.elements[1].integer.value; |
| fwrt->reduced_power_flags = flags >> IWL_REDUCE_POWER_FLAGS_POS; |
| |
| /* position of the actual table */ |
| table = &wifi_pkg->package.elements[2]; |
| |
| /* The profile from WRDS is officially profile 1, but goes |
| * into sar_profiles[0] (because we don't have a profile 0). |
| */ |
| ret = iwl_acpi_sar_set_profile(table, &fwrt->sar_profiles[0], |
| flags & IWL_SAR_ENABLE_MSK, |
| num_chains, num_sub_bands); |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| int iwl_acpi_get_ewrd_table(struct iwl_fw_runtime *fwrt) |
| { |
| union acpi_object *wifi_pkg, *data; |
| bool enabled; |
| int i, n_profiles, tbl_rev, pos; |
| int ret = 0; |
| u8 num_chains, num_sub_bands; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_EWRD_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| /* start by trying to read revision 2 */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_EWRD_WIFI_DATA_SIZE_REV2, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 2) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| num_chains = ACPI_SAR_NUM_CHAINS_REV2; |
| num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV2; |
| |
| goto read_table; |
| } |
| |
| /* then try revision 1 */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_EWRD_WIFI_DATA_SIZE_REV1, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 1) { |
| ret = -EINVAL; |
| 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_EWRD_WIFI_DATA_SIZE_REV0, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 0) { |
| ret = -EINVAL; |
| 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 || |
| wifi_pkg->package.elements[2].type != ACPI_TYPE_INTEGER) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| enabled = !!(wifi_pkg->package.elements[1].integer.value); |
| n_profiles = wifi_pkg->package.elements[2].integer.value; |
| |
| /* |
| * Check the validity of n_profiles. The EWRD profiles start |
| * from index 1, so the maximum value allowed here is |
| * ACPI_SAR_PROFILES_NUM - 1. |
| */ |
| if (n_profiles >= BIOS_SAR_MAX_PROFILE_NUM) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| /* the tables start at element 3 */ |
| pos = 3; |
| |
| for (i = 0; i < n_profiles; i++) { |
| union acpi_object *table = &wifi_pkg->package.elements[pos]; |
| /* The EWRD profiles officially go from 2 to 4, but we |
| * save them in sar_profiles[1-3] (because we don't |
| * have profile 0). So in the array we start from 1. |
| */ |
| ret = iwl_acpi_sar_set_profile(table, |
| &fwrt->sar_profiles[i + 1], |
| enabled, num_chains, |
| num_sub_bands); |
| if (ret < 0) |
| break; |
| |
| /* go to the next table */ |
| pos += num_chains * num_sub_bands; |
| } |
| |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| int iwl_acpi_get_wgds_table(struct iwl_fw_runtime *fwrt) |
| { |
| union acpi_object *wifi_pkg, *data; |
| int i, j, k, ret, tbl_rev; |
| u8 num_bands, num_profiles; |
| static const struct { |
| u8 revisions; |
| u8 bands; |
| u8 profiles; |
| u8 min_profiles; |
| } rev_data[] = { |
| { |
| .revisions = BIT(3), |
| .bands = ACPI_GEO_NUM_BANDS_REV2, |
| .profiles = ACPI_NUM_GEO_PROFILES_REV3, |
| .min_profiles = BIOS_GEO_MIN_PROFILE_NUM, |
| }, |
| { |
| .revisions = BIT(2), |
| .bands = ACPI_GEO_NUM_BANDS_REV2, |
| .profiles = ACPI_NUM_GEO_PROFILES, |
| }, |
| { |
| .revisions = BIT(0) | BIT(1), |
| .bands = ACPI_GEO_NUM_BANDS_REV0, |
| .profiles = ACPI_NUM_GEO_PROFILES, |
| }, |
| }; |
| int idx; |
| /* start from one to skip the domain */ |
| int entry_idx = 1; |
| |
| BUILD_BUG_ON(ACPI_NUM_GEO_PROFILES_REV3 != IWL_NUM_GEO_PROFILES_V3); |
| BUILD_BUG_ON(ACPI_NUM_GEO_PROFILES != IWL_NUM_GEO_PROFILES); |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WGDS_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| /* read the highest revision we understand first */ |
| for (idx = 0; idx < ARRAY_SIZE(rev_data); idx++) { |
| /* min_profiles != 0 requires num_profiles header */ |
| u32 hdr_size = 1 + !!rev_data[idx].min_profiles; |
| u32 profile_size = ACPI_GEO_PER_CHAIN_SIZE * |
| rev_data[idx].bands; |
| u32 max_size = hdr_size + profile_size * rev_data[idx].profiles; |
| u32 min_size; |
| |
| if (!rev_data[idx].min_profiles) |
| min_size = max_size; |
| else |
| min_size = hdr_size + |
| profile_size * rev_data[idx].min_profiles; |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg_range(fwrt->dev, data, |
| min_size, max_size, |
| &tbl_rev); |
| if (!IS_ERR(wifi_pkg)) { |
| if (!(BIT(tbl_rev) & rev_data[idx].revisions)) |
| continue; |
| |
| num_bands = rev_data[idx].bands; |
| num_profiles = rev_data[idx].profiles; |
| |
| if (rev_data[idx].min_profiles) { |
| /* read header that says # of profiles */ |
| union acpi_object *entry; |
| |
| entry = &wifi_pkg->package.elements[entry_idx]; |
| entry_idx++; |
| if (entry->type != ACPI_TYPE_INTEGER || |
| entry->integer.value > num_profiles || |
| entry->integer.value < |
| rev_data[idx].min_profiles) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| /* |
| * Check to see if we received package count |
| * same as max # of profiles |
| */ |
| if (wifi_pkg->package.count != |
| hdr_size + profile_size * num_profiles) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| /* Number of valid profiles */ |
| num_profiles = entry->integer.value; |
| } |
| goto read_table; |
| } |
| } |
| |
| if (idx < ARRAY_SIZE(rev_data)) |
| ret = PTR_ERR(wifi_pkg); |
| else |
| ret = -ENOENT; |
| goto out_free; |
| |
| read_table: |
| fwrt->geo_rev = tbl_rev; |
| for (i = 0; i < num_profiles; i++) { |
| for (j = 0; j < BIOS_GEO_MAX_NUM_BANDS; j++) { |
| union acpi_object *entry; |
| |
| /* |
| * num_bands is either 2 or 3, if it's only 2 then |
| * fill the third band (6 GHz) with the values from |
| * 5 GHz (second band) |
| */ |
| if (j >= num_bands) { |
| fwrt->geo_profiles[i].bands[j].max = |
| fwrt->geo_profiles[i].bands[1].max; |
| } else { |
| entry = &wifi_pkg->package.elements[entry_idx]; |
| entry_idx++; |
| if (entry->type != ACPI_TYPE_INTEGER || |
| entry->integer.value > U8_MAX) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| fwrt->geo_profiles[i].bands[j].max = |
| entry->integer.value; |
| } |
| |
| for (k = 0; k < BIOS_GEO_NUM_CHAINS; k++) { |
| /* same here as above */ |
| if (j >= num_bands) { |
| fwrt->geo_profiles[i].bands[j].chains[k] = |
| fwrt->geo_profiles[i].bands[1].chains[k]; |
| } else { |
| entry = &wifi_pkg->package.elements[entry_idx]; |
| entry_idx++; |
| if (entry->type != ACPI_TYPE_INTEGER || |
| entry->integer.value > U8_MAX) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| fwrt->geo_profiles[i].bands[j].chains[k] = |
| entry->integer.value; |
| } |
| } |
| } |
| } |
| |
| fwrt->geo_num_profiles = num_profiles; |
| fwrt->geo_enabled = true; |
| ret = 0; |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| int iwl_acpi_get_ppag_table(struct iwl_fw_runtime *fwrt) |
| { |
| union acpi_object *wifi_pkg, *data, *flags; |
| int i, j, ret, tbl_rev, num_sub_bands = 0; |
| int idx = 2; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_PPAG_METHOD); |
| if (IS_ERR(data)) |
| return PTR_ERR(data); |
| |
| /* try to read ppag table rev 3, 2 or 1 (all have the same data size) */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_PPAG_WIFI_DATA_SIZE_V2, &tbl_rev); |
| |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev >= 1 && tbl_rev <= 3) { |
| num_sub_bands = IWL_NUM_SUB_BANDS_V2; |
| IWL_DEBUG_RADIO(fwrt, |
| "Reading PPAG table (tbl_rev=%d)\n", |
| tbl_rev); |
| goto read_table; |
| } else { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| } |
| |
| /* try to read ppag table revision 0 */ |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_PPAG_WIFI_DATA_SIZE_V1, &tbl_rev); |
| |
| if (!IS_ERR(wifi_pkg)) { |
| if (tbl_rev != 0) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| num_sub_bands = IWL_NUM_SUB_BANDS_V1; |
| IWL_DEBUG_RADIO(fwrt, "Reading PPAG table v1 (tbl_rev=0)\n"); |
| goto read_table; |
| } |
| |
| ret = PTR_ERR(wifi_pkg); |
| goto out_free; |
| |
| read_table: |
| fwrt->ppag_ver = tbl_rev; |
| flags = &wifi_pkg->package.elements[1]; |
| |
| if (flags->type != ACPI_TYPE_INTEGER) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| fwrt->ppag_flags = iwl_bios_get_ppag_flags(flags->integer.value, |
| fwrt->ppag_ver); |
| |
| /* |
| * read, verify gain values and save them into the PPAG table. |
| * first sub-band (j=0) corresponds to Low-Band (2.4GHz), and the |
| * following sub-bands to High-Band (5GHz). |
| */ |
| for (i = 0; i < IWL_NUM_CHAIN_LIMITS; i++) { |
| for (j = 0; j < num_sub_bands; j++) { |
| union acpi_object *ent; |
| |
| ent = &wifi_pkg->package.elements[idx++]; |
| if (ent->type != ACPI_TYPE_INTEGER) { |
| ret = -EINVAL; |
| goto out_free; |
| } |
| |
| fwrt->ppag_chains[i].subbands[j] = ent->integer.value; |
| } |
| } |
| |
| ret = 0; |
| |
| out_free: |
| kfree(data); |
| return ret; |
| } |
| |
| void iwl_acpi_get_phy_filters(struct iwl_fw_runtime *fwrt, |
| struct iwl_phy_specific_cfg *filters) |
| { |
| struct iwl_phy_specific_cfg tmp = {}; |
| union acpi_object *wifi_pkg, *data; |
| int tbl_rev, i; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WPFC_METHOD); |
| if (IS_ERR(data)) |
| return; |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WPFC_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) |
| goto out_free; |
| |
| if (tbl_rev != 0) |
| goto out_free; |
| |
| BUILD_BUG_ON(ARRAY_SIZE(filters->filter_cfg_chains) != |
| ACPI_WPFC_WIFI_DATA_SIZE - 1); |
| |
| for (i = 0; i < ARRAY_SIZE(filters->filter_cfg_chains); i++) { |
| if (wifi_pkg->package.elements[i + 1].type != ACPI_TYPE_INTEGER) |
| goto out_free; |
| tmp.filter_cfg_chains[i] = |
| cpu_to_le32(wifi_pkg->package.elements[i + 1].integer.value); |
| } |
| |
| IWL_DEBUG_RADIO(fwrt, "Loaded WPFC filter config from ACPI\n"); |
| *filters = tmp; |
| out_free: |
| kfree(data); |
| } |
| IWL_EXPORT_SYMBOL(iwl_acpi_get_phy_filters); |
| |
| void iwl_acpi_get_guid_lock_status(struct iwl_fw_runtime *fwrt) |
| { |
| union acpi_object *wifi_pkg, *data; |
| int tbl_rev; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_GLAI_METHOD); |
| if (IS_ERR(data)) |
| return; |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_GLAI_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) |
| goto out_free; |
| |
| if (tbl_rev != 0) { |
| IWL_DEBUG_RADIO(fwrt, "Invalid GLAI revision: %d\n", tbl_rev); |
| goto out_free; |
| } |
| |
| if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER || |
| wifi_pkg->package.elements[1].integer.value > ACPI_GLAI_MAX_STATUS) |
| goto out_free; |
| |
| fwrt->uefi_tables_lock_status = |
| wifi_pkg->package.elements[1].integer.value; |
| |
| IWL_DEBUG_RADIO(fwrt, |
| "Loaded UEFI WIFI GUID lock status: %d from ACPI\n", |
| fwrt->uefi_tables_lock_status); |
| out_free: |
| kfree(data); |
| } |
| IWL_EXPORT_SYMBOL(iwl_acpi_get_guid_lock_status); |
| |
| int iwl_acpi_get_wbem(struct iwl_fw_runtime *fwrt, u32 *value) |
| { |
| union acpi_object *wifi_pkg, *data; |
| int ret = -ENOENT; |
| int tbl_rev; |
| |
| data = iwl_acpi_get_object(fwrt->dev, ACPI_WBEM_METHOD); |
| if (IS_ERR(data)) |
| return ret; |
| |
| wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data, |
| ACPI_WBEM_WIFI_DATA_SIZE, |
| &tbl_rev); |
| if (IS_ERR(wifi_pkg)) |
| goto out_free; |
| |
| if (tbl_rev != IWL_ACPI_WBEM_REVISION) { |
| IWL_DEBUG_RADIO(fwrt, "Unsupported ACPI WBEM revision:%d\n", |
| tbl_rev); |
| goto out_free; |
| } |
| |
| if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER) |
| goto out_free; |
| |
| *value = wifi_pkg->package.elements[1].integer.value & |
| IWL_ACPI_WBEM_REV0_MASK; |
| IWL_DEBUG_RADIO(fwrt, "Loaded WBEM config from ACPI\n"); |
| ret = 0; |
| out_free: |
| kfree(data); |
| return ret; |
| } |