xref: /linux/drivers/net/wireless/intel/iwlwifi/fw/acpi.c (revision 001821b0e79716c4e17c71d8e053a23599a7a508)
1 // SPDX-License-Identifier: GPL-2.0 OR BSD-3-Clause
2 /*
3  * Copyright (C) 2017 Intel Deutschland GmbH
4  * Copyright (C) 2019-2024 Intel Corporation
5  */
6 #include <linux/uuid.h>
7 #include "iwl-drv.h"
8 #include "iwl-debug.h"
9 #include "acpi.h"
10 #include "fw/runtime.h"
11 
12 const guid_t iwl_guid = GUID_INIT(0xF21202BF, 0x8F78, 0x4DC6,
13 				  0xA5, 0xB3, 0x1F, 0x73,
14 				  0x8E, 0x28, 0x5A, 0xDE);
15 
16 static const size_t acpi_dsm_size[DSM_FUNC_NUM_FUNCS] = {
17 	[DSM_FUNC_QUERY] =			sizeof(u32),
18 	[DSM_FUNC_DISABLE_SRD] =		sizeof(u8),
19 	[DSM_FUNC_ENABLE_INDONESIA_5G2] =	sizeof(u8),
20 	[DSM_FUNC_ENABLE_6E] =			sizeof(u32),
21 	[DSM_FUNC_REGULATORY_CONFIG] =		sizeof(u32),
22 	/* Not supported in driver */
23 	[5] =					(size_t)0,
24 	[DSM_FUNC_11AX_ENABLEMENT] =		sizeof(u32),
25 	[DSM_FUNC_ENABLE_UNII4_CHAN] =		sizeof(u32),
26 	[DSM_FUNC_ACTIVATE_CHANNEL] =		sizeof(u32),
27 	[DSM_FUNC_FORCE_DISABLE_CHANNELS] =	sizeof(u32),
28 	[DSM_FUNC_ENERGY_DETECTION_THRESHOLD] =	sizeof(u32),
29 	[DSM_FUNC_RFI_CONFIG] =			sizeof(u32),
30 };
31 
32 static int iwl_acpi_get_handle(struct device *dev, acpi_string method,
33 			       acpi_handle *ret_handle)
34 {
35 	acpi_handle root_handle;
36 	acpi_status status;
37 
38 	root_handle = ACPI_HANDLE(dev);
39 	if (!root_handle) {
40 		IWL_DEBUG_DEV_RADIO(dev,
41 				    "ACPI: Could not retrieve root port handle\n");
42 		return -ENOENT;
43 	}
44 
45 	status = acpi_get_handle(root_handle, method, ret_handle);
46 	if (ACPI_FAILURE(status)) {
47 		IWL_DEBUG_DEV_RADIO(dev,
48 				    "ACPI: %s method not found\n", method);
49 		return -ENOENT;
50 	}
51 	return 0;
52 }
53 
54 static void *iwl_acpi_get_object(struct device *dev, acpi_string method)
55 {
56 	struct acpi_buffer buf = {ACPI_ALLOCATE_BUFFER, NULL};
57 	acpi_handle handle;
58 	acpi_status status;
59 	int ret;
60 
61 	ret = iwl_acpi_get_handle(dev, method, &handle);
62 	if (ret)
63 		return ERR_PTR(-ENOENT);
64 
65 	/* Call the method with no arguments */
66 	status = acpi_evaluate_object(handle, NULL, NULL, &buf);
67 	if (ACPI_FAILURE(status)) {
68 		IWL_DEBUG_DEV_RADIO(dev,
69 				    "ACPI: %s method invocation failed (status: 0x%x)\n",
70 				    method, status);
71 		return ERR_PTR(-ENOENT);
72 	}
73 	return buf.pointer;
74 }
75 
76 /*
77  * Generic function for evaluating a method defined in the device specific
78  * method (DSM) interface. The returned acpi object must be freed by calling
79  * function.
80  */
81 static void *iwl_acpi_get_dsm_object(struct device *dev, int rev, int func,
82 				     union acpi_object *args,
83 				     const guid_t *guid)
84 {
85 	union acpi_object *obj;
86 
87 	obj = acpi_evaluate_dsm(ACPI_HANDLE(dev), guid, rev, func,
88 				args);
89 	if (!obj) {
90 		IWL_DEBUG_DEV_RADIO(dev,
91 				    "ACPI: DSM method invocation failed (rev: %d, func:%d)\n",
92 				    rev, func);
93 		return ERR_PTR(-ENOENT);
94 	}
95 	return obj;
96 }
97 
98 /*
99  * Generic function to evaluate a DSM with no arguments
100  * and an integer return value,
101  * (as an integer object or inside a buffer object),
102  * verify and assign the value in the "value" parameter.
103  * return 0 in success and the appropriate errno otherwise.
104  */
105 static int iwl_acpi_get_dsm_integer(struct device *dev, int rev, int func,
106 				    const guid_t *guid, u64 *value,
107 				    size_t expected_size)
108 {
109 	union acpi_object *obj;
110 	int ret = 0;
111 
112 	obj = iwl_acpi_get_dsm_object(dev, rev, func, NULL, guid);
113 	if (IS_ERR(obj)) {
114 		IWL_DEBUG_DEV_RADIO(dev,
115 				    "Failed to get  DSM object. func= %d\n",
116 				    func);
117 		return -ENOENT;
118 	}
119 
120 	if (obj->type == ACPI_TYPE_INTEGER) {
121 		*value = obj->integer.value;
122 	} else if (obj->type == ACPI_TYPE_BUFFER) {
123 		__le64 le_value = 0;
124 
125 		if (WARN_ON_ONCE(expected_size > sizeof(le_value)))
126 			return -EINVAL;
127 
128 		/* if the buffer size doesn't match the expected size */
129 		if (obj->buffer.length != expected_size)
130 			IWL_DEBUG_DEV_RADIO(dev,
131 					    "ACPI: DSM invalid buffer size, padding or truncating (%d)\n",
132 					    obj->buffer.length);
133 
134 		 /* assuming LE from Intel BIOS spec */
135 		memcpy(&le_value, obj->buffer.pointer,
136 		       min_t(size_t, expected_size, (size_t)obj->buffer.length));
137 		*value = le64_to_cpu(le_value);
138 	} else {
139 		IWL_DEBUG_DEV_RADIO(dev,
140 				    "ACPI: DSM method did not return a valid object, type=%d\n",
141 				    obj->type);
142 		ret = -EINVAL;
143 		goto out;
144 	}
145 
146 	IWL_DEBUG_DEV_RADIO(dev,
147 			    "ACPI: DSM method evaluated: func=%d, ret=%d\n",
148 			    func, ret);
149 out:
150 	ACPI_FREE(obj);
151 	return ret;
152 }
153 
154 /*
155  * This function receives a DSM function number, calculates its expected size
156  * according to Intel BIOS spec, and fills in the value in a 32-bit field.
157  * In case the expected size is smaller than 32-bit, padding will be added.
158  */
159 int iwl_acpi_get_dsm(struct iwl_fw_runtime *fwrt,
160 		     enum iwl_dsm_funcs func, u32 *value)
161 {
162 	size_t expected_size;
163 	u64 tmp;
164 	int ret;
165 
166 	BUILD_BUG_ON(ARRAY_SIZE(acpi_dsm_size) != DSM_FUNC_NUM_FUNCS);
167 
168 	if (WARN_ON(func >= ARRAY_SIZE(acpi_dsm_size)))
169 		return -EINVAL;
170 
171 	expected_size = acpi_dsm_size[func];
172 
173 	/* Currently all ACPI DSMs are either 8-bit or 32-bit */
174 	if (expected_size != sizeof(u8) && expected_size != sizeof(u32))
175 		return -EOPNOTSUPP;
176 
177 	ret = iwl_acpi_get_dsm_integer(fwrt->dev, ACPI_DSM_REV, func,
178 				       &iwl_guid, &tmp, expected_size);
179 	if (ret)
180 		return ret;
181 
182 	if ((expected_size == sizeof(u8) && tmp != (u8)tmp) ||
183 	    (expected_size == sizeof(u32) && tmp != (u32)tmp))
184 		IWL_DEBUG_RADIO(fwrt,
185 				"DSM value overflows the expected size, truncating\n");
186 	*value = (u32)tmp;
187 
188 	return 0;
189 }
190 
191 static union acpi_object *
192 iwl_acpi_get_wifi_pkg_range(struct device *dev,
193 			    union acpi_object *data,
194 			    int min_data_size,
195 			    int max_data_size,
196 			    int *tbl_rev)
197 {
198 	int i;
199 	union acpi_object *wifi_pkg;
200 
201 	/*
202 	 * We need at least one entry in the wifi package that
203 	 * describes the domain, and one more entry, otherwise there's
204 	 * no point in reading it.
205 	 */
206 	if (WARN_ON_ONCE(min_data_size < 2 || min_data_size > max_data_size))
207 		return ERR_PTR(-EINVAL);
208 
209 	/*
210 	 * We need at least two packages, one for the revision and one
211 	 * for the data itself.  Also check that the revision is valid
212 	 * (i.e. it is an integer (each caller has to check by itself
213 	 * if the returned revision is supported)).
214 	 */
215 	if (data->type != ACPI_TYPE_PACKAGE ||
216 	    data->package.count < 2 ||
217 	    data->package.elements[0].type != ACPI_TYPE_INTEGER) {
218 		IWL_DEBUG_DEV_RADIO(dev, "Invalid packages structure\n");
219 		return ERR_PTR(-EINVAL);
220 	}
221 
222 	*tbl_rev = data->package.elements[0].integer.value;
223 
224 	/* loop through all the packages to find the one for WiFi */
225 	for (i = 1; i < data->package.count; i++) {
226 		union acpi_object *domain;
227 
228 		wifi_pkg = &data->package.elements[i];
229 
230 		/* skip entries that are not a package with the right size */
231 		if (wifi_pkg->type != ACPI_TYPE_PACKAGE ||
232 		    wifi_pkg->package.count < min_data_size ||
233 		    wifi_pkg->package.count > max_data_size)
234 			continue;
235 
236 		domain = &wifi_pkg->package.elements[0];
237 		if (domain->type == ACPI_TYPE_INTEGER &&
238 		    domain->integer.value == ACPI_WIFI_DOMAIN)
239 			goto found;
240 	}
241 
242 	return ERR_PTR(-ENOENT);
243 
244 found:
245 	return wifi_pkg;
246 }
247 
248 static union acpi_object *
249 iwl_acpi_get_wifi_pkg(struct device *dev,
250 		      union acpi_object *data,
251 		      int data_size, int *tbl_rev)
252 {
253 	return iwl_acpi_get_wifi_pkg_range(dev, data, data_size, data_size,
254 					   tbl_rev);
255 }
256 
257 int iwl_acpi_get_tas_table(struct iwl_fw_runtime *fwrt,
258 			   struct iwl_tas_data *tas_data)
259 {
260 	union acpi_object *wifi_pkg, *data;
261 	int ret, tbl_rev, i, block_list_size, enabled;
262 
263 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WTAS_METHOD);
264 	if (IS_ERR(data))
265 		return PTR_ERR(data);
266 
267 	/* try to read wtas table revision 1 or revision 0*/
268 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
269 					 ACPI_WTAS_WIFI_DATA_SIZE,
270 					 &tbl_rev);
271 	if (IS_ERR(wifi_pkg)) {
272 		ret = PTR_ERR(wifi_pkg);
273 		goto out_free;
274 	}
275 
276 	if (tbl_rev == 1 && wifi_pkg->package.elements[1].type ==
277 		ACPI_TYPE_INTEGER) {
278 		u32 tas_selection =
279 			(u32)wifi_pkg->package.elements[1].integer.value;
280 
281 		enabled = iwl_parse_tas_selection(fwrt, tas_data,
282 						  tas_selection);
283 
284 	} else if (tbl_rev == 0 &&
285 		wifi_pkg->package.elements[1].type == ACPI_TYPE_INTEGER) {
286 		enabled = !!wifi_pkg->package.elements[1].integer.value;
287 	} else {
288 		ret = -EINVAL;
289 		goto out_free;
290 	}
291 
292 	if (!enabled) {
293 		IWL_DEBUG_RADIO(fwrt, "TAS not enabled\n");
294 		ret = 0;
295 		goto out_free;
296 	}
297 
298 	IWL_DEBUG_RADIO(fwrt, "Reading TAS table revision %d\n", tbl_rev);
299 	if (wifi_pkg->package.elements[2].type != ACPI_TYPE_INTEGER ||
300 	    wifi_pkg->package.elements[2].integer.value >
301 	    IWL_WTAS_BLACK_LIST_MAX) {
302 		IWL_DEBUG_RADIO(fwrt, "TAS invalid array size %llu\n",
303 				wifi_pkg->package.elements[2].integer.value);
304 		ret = -EINVAL;
305 		goto out_free;
306 	}
307 	block_list_size = wifi_pkg->package.elements[2].integer.value;
308 	tas_data->block_list_size = cpu_to_le32(block_list_size);
309 
310 	IWL_DEBUG_RADIO(fwrt, "TAS array size %u\n", block_list_size);
311 
312 	for (i = 0; i < block_list_size; i++) {
313 		u32 country;
314 
315 		if (wifi_pkg->package.elements[3 + i].type !=
316 		    ACPI_TYPE_INTEGER) {
317 			IWL_DEBUG_RADIO(fwrt,
318 					"TAS invalid array elem %d\n", 3 + i);
319 			ret = -EINVAL;
320 			goto out_free;
321 		}
322 
323 		country = wifi_pkg->package.elements[3 + i].integer.value;
324 		tas_data->block_list_array[i] = cpu_to_le32(country);
325 		IWL_DEBUG_RADIO(fwrt, "TAS block list country %d\n", country);
326 	}
327 
328 	ret = 1;
329 out_free:
330 	kfree(data);
331 	return ret;
332 }
333 
334 int iwl_acpi_get_mcc(struct iwl_fw_runtime *fwrt, char *mcc)
335 {
336 	union acpi_object *wifi_pkg, *data;
337 	u32 mcc_val;
338 	int ret, tbl_rev;
339 
340 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDD_METHOD);
341 	if (IS_ERR(data))
342 		return PTR_ERR(data);
343 
344 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
345 					 ACPI_WRDD_WIFI_DATA_SIZE,
346 					 &tbl_rev);
347 	if (IS_ERR(wifi_pkg)) {
348 		ret = PTR_ERR(wifi_pkg);
349 		goto out_free;
350 	}
351 
352 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER ||
353 	    tbl_rev != 0) {
354 		ret = -EINVAL;
355 		goto out_free;
356 	}
357 
358 	mcc_val = wifi_pkg->package.elements[1].integer.value;
359 
360 	mcc[0] = (mcc_val >> 8) & 0xff;
361 	mcc[1] = mcc_val & 0xff;
362 	mcc[2] = '\0';
363 
364 	ret = 0;
365 out_free:
366 	kfree(data);
367 	return ret;
368 }
369 
370 int iwl_acpi_get_pwr_limit(struct iwl_fw_runtime *fwrt, u64 *dflt_pwr_limit)
371 {
372 	union acpi_object *data, *wifi_pkg;
373 	int tbl_rev, ret = -EINVAL;
374 
375 	*dflt_pwr_limit = 0;
376 	data = iwl_acpi_get_object(fwrt->dev, ACPI_SPLC_METHOD);
377 	if (IS_ERR(data))
378 		goto out;
379 
380 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
381 					 ACPI_SPLC_WIFI_DATA_SIZE, &tbl_rev);
382 	if (IS_ERR(wifi_pkg) || tbl_rev != 0 ||
383 	    wifi_pkg->package.elements[1].integer.value != ACPI_TYPE_INTEGER)
384 		goto out_free;
385 
386 	*dflt_pwr_limit = wifi_pkg->package.elements[1].integer.value;
387 	ret = 0;
388 out_free:
389 	kfree(data);
390 out:
391 	return ret;
392 }
393 
394 int iwl_acpi_get_eckv(struct iwl_fw_runtime *fwrt, u32 *extl_clk)
395 {
396 	union acpi_object *wifi_pkg, *data;
397 	int ret, tbl_rev;
398 
399 	data = iwl_acpi_get_object(fwrt->dev, ACPI_ECKV_METHOD);
400 	if (IS_ERR(data))
401 		return PTR_ERR(data);
402 
403 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
404 					 ACPI_ECKV_WIFI_DATA_SIZE,
405 					 &tbl_rev);
406 	if (IS_ERR(wifi_pkg)) {
407 		ret = PTR_ERR(wifi_pkg);
408 		goto out_free;
409 	}
410 
411 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER ||
412 	    tbl_rev != 0) {
413 		ret = -EINVAL;
414 		goto out_free;
415 	}
416 
417 	*extl_clk = wifi_pkg->package.elements[1].integer.value;
418 
419 	ret = 0;
420 
421 out_free:
422 	kfree(data);
423 	return ret;
424 }
425 
426 static int iwl_acpi_sar_set_profile(union acpi_object *table,
427 				    struct iwl_sar_profile *profile,
428 				    bool enabled, u8 num_chains,
429 				    u8 num_sub_bands)
430 {
431 	int i, j, idx = 0;
432 
433 	/*
434 	 * The table from ACPI is flat, but we store it in a
435 	 * structured array.
436 	 */
437 	for (i = 0; i < BIOS_SAR_MAX_CHAINS_PER_PROFILE; i++) {
438 		for (j = 0; j < BIOS_SAR_MAX_SUB_BANDS_NUM; j++) {
439 			/* if we don't have the values, use the default */
440 			if (i >= num_chains || j >= num_sub_bands) {
441 				profile->chains[i].subbands[j] = 0;
442 			} else {
443 				if (table[idx].type != ACPI_TYPE_INTEGER ||
444 				    table[idx].integer.value > U8_MAX)
445 					return -EINVAL;
446 
447 				profile->chains[i].subbands[j] =
448 					table[idx].integer.value;
449 
450 				idx++;
451 			}
452 		}
453 	}
454 
455 	/* Only if all values were valid can the profile be enabled */
456 	profile->enabled = enabled;
457 
458 	return 0;
459 }
460 
461 int iwl_acpi_get_wrds_table(struct iwl_fw_runtime *fwrt)
462 {
463 	union acpi_object *wifi_pkg, *table, *data;
464 	int ret, tbl_rev;
465 	u32 flags;
466 	u8 num_chains, num_sub_bands;
467 
468 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WRDS_METHOD);
469 	if (IS_ERR(data))
470 		return PTR_ERR(data);
471 
472 	/* start by trying to read revision 2 */
473 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
474 					 ACPI_WRDS_WIFI_DATA_SIZE_REV2,
475 					 &tbl_rev);
476 	if (!IS_ERR(wifi_pkg)) {
477 		if (tbl_rev != 2) {
478 			ret = -EINVAL;
479 			goto out_free;
480 		}
481 
482 		num_chains = ACPI_SAR_NUM_CHAINS_REV2;
483 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV2;
484 
485 		goto read_table;
486 	}
487 
488 	/* then try revision 1 */
489 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
490 					 ACPI_WRDS_WIFI_DATA_SIZE_REV1,
491 					 &tbl_rev);
492 	if (!IS_ERR(wifi_pkg)) {
493 		if (tbl_rev != 1) {
494 			ret = -EINVAL;
495 			goto out_free;
496 		}
497 
498 		num_chains = ACPI_SAR_NUM_CHAINS_REV1;
499 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV1;
500 
501 		goto read_table;
502 	}
503 
504 	/* then finally revision 0 */
505 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
506 					 ACPI_WRDS_WIFI_DATA_SIZE_REV0,
507 					 &tbl_rev);
508 	if (!IS_ERR(wifi_pkg)) {
509 		if (tbl_rev != 0) {
510 			ret = -EINVAL;
511 			goto out_free;
512 		}
513 
514 		num_chains = ACPI_SAR_NUM_CHAINS_REV0;
515 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV0;
516 
517 		goto read_table;
518 	}
519 
520 	ret = PTR_ERR(wifi_pkg);
521 	goto out_free;
522 
523 read_table:
524 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER) {
525 		ret = -EINVAL;
526 		goto out_free;
527 	}
528 
529 	IWL_DEBUG_RADIO(fwrt, "Reading WRDS tbl_rev=%d\n", tbl_rev);
530 
531 	flags = wifi_pkg->package.elements[1].integer.value;
532 	fwrt->reduced_power_flags = flags >> IWL_REDUCE_POWER_FLAGS_POS;
533 
534 	/* position of the actual table */
535 	table = &wifi_pkg->package.elements[2];
536 
537 	/* The profile from WRDS is officially profile 1, but goes
538 	 * into sar_profiles[0] (because we don't have a profile 0).
539 	 */
540 	ret = iwl_acpi_sar_set_profile(table, &fwrt->sar_profiles[0],
541 				       flags & IWL_SAR_ENABLE_MSK,
542 				       num_chains, num_sub_bands);
543 out_free:
544 	kfree(data);
545 	return ret;
546 }
547 
548 int iwl_acpi_get_ewrd_table(struct iwl_fw_runtime *fwrt)
549 {
550 	union acpi_object *wifi_pkg, *data;
551 	bool enabled;
552 	int i, n_profiles, tbl_rev, pos;
553 	int ret = 0;
554 	u8 num_chains, num_sub_bands;
555 
556 	data = iwl_acpi_get_object(fwrt->dev, ACPI_EWRD_METHOD);
557 	if (IS_ERR(data))
558 		return PTR_ERR(data);
559 
560 	/* start by trying to read revision 2 */
561 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
562 					 ACPI_EWRD_WIFI_DATA_SIZE_REV2,
563 					 &tbl_rev);
564 	if (!IS_ERR(wifi_pkg)) {
565 		if (tbl_rev != 2) {
566 			ret = -EINVAL;
567 			goto out_free;
568 		}
569 
570 		num_chains = ACPI_SAR_NUM_CHAINS_REV2;
571 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV2;
572 
573 		goto read_table;
574 	}
575 
576 	/* then try revision 1 */
577 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
578 					 ACPI_EWRD_WIFI_DATA_SIZE_REV1,
579 					 &tbl_rev);
580 	if (!IS_ERR(wifi_pkg)) {
581 		if (tbl_rev != 1) {
582 			ret = -EINVAL;
583 			goto out_free;
584 		}
585 
586 		num_chains = ACPI_SAR_NUM_CHAINS_REV1;
587 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV1;
588 
589 		goto read_table;
590 	}
591 
592 	/* then finally revision 0 */
593 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
594 					 ACPI_EWRD_WIFI_DATA_SIZE_REV0,
595 					 &tbl_rev);
596 	if (!IS_ERR(wifi_pkg)) {
597 		if (tbl_rev != 0) {
598 			ret = -EINVAL;
599 			goto out_free;
600 		}
601 
602 		num_chains = ACPI_SAR_NUM_CHAINS_REV0;
603 		num_sub_bands = ACPI_SAR_NUM_SUB_BANDS_REV0;
604 
605 		goto read_table;
606 	}
607 
608 	ret = PTR_ERR(wifi_pkg);
609 	goto out_free;
610 
611 read_table:
612 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER ||
613 	    wifi_pkg->package.elements[2].type != ACPI_TYPE_INTEGER) {
614 		ret = -EINVAL;
615 		goto out_free;
616 	}
617 
618 	enabled = !!(wifi_pkg->package.elements[1].integer.value);
619 	n_profiles = wifi_pkg->package.elements[2].integer.value;
620 
621 	/*
622 	 * Check the validity of n_profiles.  The EWRD profiles start
623 	 * from index 1, so the maximum value allowed here is
624 	 * ACPI_SAR_PROFILES_NUM - 1.
625 	 */
626 	if (n_profiles >= BIOS_SAR_MAX_PROFILE_NUM) {
627 		ret = -EINVAL;
628 		goto out_free;
629 	}
630 
631 	/* the tables start at element 3 */
632 	pos = 3;
633 
634 	for (i = 0; i < n_profiles; i++) {
635 		union acpi_object *table = &wifi_pkg->package.elements[pos];
636 		/* The EWRD profiles officially go from 2 to 4, but we
637 		 * save them in sar_profiles[1-3] (because we don't
638 		 * have profile 0).  So in the array we start from 1.
639 		 */
640 		ret = iwl_acpi_sar_set_profile(table,
641 					       &fwrt->sar_profiles[i + 1],
642 					       enabled, num_chains,
643 					       num_sub_bands);
644 		if (ret < 0)
645 			break;
646 
647 		/* go to the next table */
648 		pos += num_chains * num_sub_bands;
649 	}
650 
651 out_free:
652 	kfree(data);
653 	return ret;
654 }
655 
656 int iwl_acpi_get_wgds_table(struct iwl_fw_runtime *fwrt)
657 {
658 	union acpi_object *wifi_pkg, *data;
659 	int i, j, k, ret, tbl_rev;
660 	u8 num_bands, num_profiles;
661 	static const struct {
662 		u8 revisions;
663 		u8 bands;
664 		u8 profiles;
665 		u8 min_profiles;
666 	} rev_data[] = {
667 		{
668 			.revisions = BIT(3),
669 			.bands = ACPI_GEO_NUM_BANDS_REV2,
670 			.profiles = ACPI_NUM_GEO_PROFILES_REV3,
671 			.min_profiles = BIOS_GEO_MIN_PROFILE_NUM,
672 		},
673 		{
674 			.revisions = BIT(2),
675 			.bands = ACPI_GEO_NUM_BANDS_REV2,
676 			.profiles = ACPI_NUM_GEO_PROFILES,
677 		},
678 		{
679 			.revisions = BIT(0) | BIT(1),
680 			.bands = ACPI_GEO_NUM_BANDS_REV0,
681 			.profiles = ACPI_NUM_GEO_PROFILES,
682 		},
683 	};
684 	int idx;
685 	/* start from one to skip the domain */
686 	int entry_idx = 1;
687 
688 	BUILD_BUG_ON(ACPI_NUM_GEO_PROFILES_REV3 != IWL_NUM_GEO_PROFILES_V3);
689 	BUILD_BUG_ON(ACPI_NUM_GEO_PROFILES != IWL_NUM_GEO_PROFILES);
690 
691 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WGDS_METHOD);
692 	if (IS_ERR(data))
693 		return PTR_ERR(data);
694 
695 	/* read the highest revision we understand first */
696 	for (idx = 0; idx < ARRAY_SIZE(rev_data); idx++) {
697 		/* min_profiles != 0 requires num_profiles header */
698 		u32 hdr_size = 1 + !!rev_data[idx].min_profiles;
699 		u32 profile_size = ACPI_GEO_PER_CHAIN_SIZE *
700 				   rev_data[idx].bands;
701 		u32 max_size = hdr_size + profile_size * rev_data[idx].profiles;
702 		u32 min_size;
703 
704 		if (!rev_data[idx].min_profiles)
705 			min_size = max_size;
706 		else
707 			min_size = hdr_size +
708 				   profile_size * rev_data[idx].min_profiles;
709 
710 		wifi_pkg = iwl_acpi_get_wifi_pkg_range(fwrt->dev, data,
711 						       min_size, max_size,
712 						       &tbl_rev);
713 		if (!IS_ERR(wifi_pkg)) {
714 			if (!(BIT(tbl_rev) & rev_data[idx].revisions))
715 				continue;
716 
717 			num_bands = rev_data[idx].bands;
718 			num_profiles = rev_data[idx].profiles;
719 
720 			if (rev_data[idx].min_profiles) {
721 				/* read header that says # of profiles */
722 				union acpi_object *entry;
723 
724 				entry = &wifi_pkg->package.elements[entry_idx];
725 				entry_idx++;
726 				if (entry->type != ACPI_TYPE_INTEGER ||
727 				    entry->integer.value > num_profiles) {
728 					ret = -EINVAL;
729 					goto out_free;
730 				}
731 				num_profiles = entry->integer.value;
732 
733 				/*
734 				 * this also validates >= min_profiles since we
735 				 * otherwise wouldn't have gotten the data when
736 				 * looking up in ACPI
737 				 */
738 				if (wifi_pkg->package.count !=
739 				    hdr_size + profile_size * num_profiles) {
740 					ret = -EINVAL;
741 					goto out_free;
742 				}
743 			}
744 			goto read_table;
745 		}
746 	}
747 
748 	if (idx < ARRAY_SIZE(rev_data))
749 		ret = PTR_ERR(wifi_pkg);
750 	else
751 		ret = -ENOENT;
752 	goto out_free;
753 
754 read_table:
755 	fwrt->geo_rev = tbl_rev;
756 	for (i = 0; i < num_profiles; i++) {
757 		for (j = 0; j < BIOS_GEO_MAX_NUM_BANDS; j++) {
758 			union acpi_object *entry;
759 
760 			/*
761 			 * num_bands is either 2 or 3, if it's only 2 then
762 			 * fill the third band (6 GHz) with the values from
763 			 * 5 GHz (second band)
764 			 */
765 			if (j >= num_bands) {
766 				fwrt->geo_profiles[i].bands[j].max =
767 					fwrt->geo_profiles[i].bands[1].max;
768 			} else {
769 				entry = &wifi_pkg->package.elements[entry_idx];
770 				entry_idx++;
771 				if (entry->type != ACPI_TYPE_INTEGER ||
772 				    entry->integer.value > U8_MAX) {
773 					ret = -EINVAL;
774 					goto out_free;
775 				}
776 
777 				fwrt->geo_profiles[i].bands[j].max =
778 					entry->integer.value;
779 			}
780 
781 			for (k = 0; k < BIOS_GEO_NUM_CHAINS; k++) {
782 				/* same here as above */
783 				if (j >= num_bands) {
784 					fwrt->geo_profiles[i].bands[j].chains[k] =
785 						fwrt->geo_profiles[i].bands[1].chains[k];
786 				} else {
787 					entry = &wifi_pkg->package.elements[entry_idx];
788 					entry_idx++;
789 					if (entry->type != ACPI_TYPE_INTEGER ||
790 					    entry->integer.value > U8_MAX) {
791 						ret = -EINVAL;
792 						goto out_free;
793 					}
794 
795 					fwrt->geo_profiles[i].bands[j].chains[k] =
796 						entry->integer.value;
797 				}
798 			}
799 		}
800 	}
801 
802 	fwrt->geo_num_profiles = num_profiles;
803 	fwrt->geo_enabled = true;
804 	ret = 0;
805 out_free:
806 	kfree(data);
807 	return ret;
808 }
809 
810 int iwl_acpi_get_ppag_table(struct iwl_fw_runtime *fwrt)
811 {
812 	union acpi_object *wifi_pkg, *data, *flags;
813 	int i, j, ret, tbl_rev, num_sub_bands = 0;
814 	int idx = 2;
815 
816 	data = iwl_acpi_get_object(fwrt->dev, ACPI_PPAG_METHOD);
817 	if (IS_ERR(data))
818 		return PTR_ERR(data);
819 
820 	/* try to read ppag table rev 3, 2 or 1 (all have the same data size) */
821 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
822 				ACPI_PPAG_WIFI_DATA_SIZE_V2, &tbl_rev);
823 
824 	if (!IS_ERR(wifi_pkg)) {
825 		if (tbl_rev >= 1 && tbl_rev <= 3) {
826 			num_sub_bands = IWL_NUM_SUB_BANDS_V2;
827 			IWL_DEBUG_RADIO(fwrt,
828 					"Reading PPAG table (tbl_rev=%d)\n",
829 					tbl_rev);
830 			goto read_table;
831 		} else {
832 			ret = -EINVAL;
833 			goto out_free;
834 		}
835 	}
836 
837 	/* try to read ppag table revision 0 */
838 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
839 			ACPI_PPAG_WIFI_DATA_SIZE_V1, &tbl_rev);
840 
841 	if (!IS_ERR(wifi_pkg)) {
842 		if (tbl_rev != 0) {
843 			ret = -EINVAL;
844 			goto out_free;
845 		}
846 		num_sub_bands = IWL_NUM_SUB_BANDS_V1;
847 		IWL_DEBUG_RADIO(fwrt, "Reading PPAG table v1 (tbl_rev=0)\n");
848 		goto read_table;
849 	}
850 
851 	ret = PTR_ERR(wifi_pkg);
852 	goto out_free;
853 
854 read_table:
855 	fwrt->ppag_ver = tbl_rev;
856 	flags = &wifi_pkg->package.elements[1];
857 
858 	if (flags->type != ACPI_TYPE_INTEGER) {
859 		ret = -EINVAL;
860 		goto out_free;
861 	}
862 
863 	fwrt->ppag_flags = iwl_bios_get_ppag_flags(flags->integer.value,
864 						   fwrt->ppag_ver);
865 
866 	/*
867 	 * read, verify gain values and save them into the PPAG table.
868 	 * first sub-band (j=0) corresponds to Low-Band (2.4GHz), and the
869 	 * following sub-bands to High-Band (5GHz).
870 	 */
871 	for (i = 0; i < IWL_NUM_CHAIN_LIMITS; i++) {
872 		for (j = 0; j < num_sub_bands; j++) {
873 			union acpi_object *ent;
874 
875 			ent = &wifi_pkg->package.elements[idx++];
876 			if (ent->type != ACPI_TYPE_INTEGER) {
877 				ret = -EINVAL;
878 				goto out_free;
879 			}
880 
881 			fwrt->ppag_chains[i].subbands[j] = ent->integer.value;
882 		}
883 	}
884 
885 	ret = 0;
886 
887 out_free:
888 	kfree(data);
889 	return ret;
890 }
891 
892 void iwl_acpi_get_phy_filters(struct iwl_fw_runtime *fwrt,
893 			      struct iwl_phy_specific_cfg *filters)
894 {
895 	struct iwl_phy_specific_cfg tmp = {};
896 	union acpi_object *wifi_pkg, *data;
897 	int tbl_rev, i;
898 
899 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WPFC_METHOD);
900 	if (IS_ERR(data))
901 		return;
902 
903 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
904 					 ACPI_WPFC_WIFI_DATA_SIZE,
905 					 &tbl_rev);
906 	if (IS_ERR(wifi_pkg))
907 		goto out_free;
908 
909 	if (tbl_rev != 0)
910 		goto out_free;
911 
912 	BUILD_BUG_ON(ARRAY_SIZE(filters->filter_cfg_chains) !=
913 		     ACPI_WPFC_WIFI_DATA_SIZE - 1);
914 
915 	for (i = 0; i < ARRAY_SIZE(filters->filter_cfg_chains); i++) {
916 		if (wifi_pkg->package.elements[i + 1].type != ACPI_TYPE_INTEGER)
917 			goto out_free;
918 		tmp.filter_cfg_chains[i] =
919 			cpu_to_le32(wifi_pkg->package.elements[i + 1].integer.value);
920 	}
921 
922 	IWL_DEBUG_RADIO(fwrt, "Loaded WPFC filter config from ACPI\n");
923 	*filters = tmp;
924 out_free:
925 	kfree(data);
926 }
927 IWL_EXPORT_SYMBOL(iwl_acpi_get_phy_filters);
928 
929 void iwl_acpi_get_guid_lock_status(struct iwl_fw_runtime *fwrt)
930 {
931 	union acpi_object *wifi_pkg, *data;
932 	int tbl_rev;
933 
934 	data = iwl_acpi_get_object(fwrt->dev, ACPI_GLAI_METHOD);
935 	if (IS_ERR(data))
936 		return;
937 
938 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
939 					 ACPI_GLAI_WIFI_DATA_SIZE,
940 					 &tbl_rev);
941 	if (IS_ERR(wifi_pkg))
942 		goto out_free;
943 
944 	if (tbl_rev != 0) {
945 		IWL_DEBUG_RADIO(fwrt, "Invalid GLAI revision: %d\n", tbl_rev);
946 		goto out_free;
947 	}
948 
949 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER ||
950 	    wifi_pkg->package.elements[1].integer.value > ACPI_GLAI_MAX_STATUS)
951 		goto out_free;
952 
953 	fwrt->uefi_tables_lock_status =
954 		wifi_pkg->package.elements[1].integer.value;
955 
956 	IWL_DEBUG_RADIO(fwrt,
957 			"Loaded UEFI WIFI GUID lock status: %d from ACPI\n",
958 			fwrt->uefi_tables_lock_status);
959 out_free:
960 	kfree(data);
961 }
962 IWL_EXPORT_SYMBOL(iwl_acpi_get_guid_lock_status);
963 
964 int iwl_acpi_get_wbem(struct iwl_fw_runtime *fwrt, u32 *value)
965 {
966 	union acpi_object *wifi_pkg, *data;
967 	int ret = -ENOENT;
968 	int tbl_rev;
969 
970 	data = iwl_acpi_get_object(fwrt->dev, ACPI_WBEM_METHOD);
971 	if (IS_ERR(data))
972 		return ret;
973 
974 	wifi_pkg = iwl_acpi_get_wifi_pkg(fwrt->dev, data,
975 					 ACPI_WBEM_WIFI_DATA_SIZE,
976 					 &tbl_rev);
977 	if (IS_ERR(wifi_pkg))
978 		goto out_free;
979 
980 	if (tbl_rev != IWL_ACPI_WBEM_REVISION) {
981 		IWL_DEBUG_RADIO(fwrt, "Unsupported ACPI WBEM revision:%d\n",
982 				tbl_rev);
983 		goto out_free;
984 	}
985 
986 	if (wifi_pkg->package.elements[1].type != ACPI_TYPE_INTEGER)
987 		goto out_free;
988 
989 	*value = wifi_pkg->package.elements[1].integer.value &
990 		 IWL_ACPI_WBEM_REV0_MASK;
991 	IWL_DEBUG_RADIO(fwrt, "Loaded WBEM config from ACPI\n");
992 	ret = 0;
993 out_free:
994 	kfree(data);
995 	return ret;
996 }
997