blob: e0587c85bafdf73a299c40d531eefa27deab4761 [file] [log] [blame]
Bob Moore96db2552005-11-02 00:00:00 -05001/*******************************************************************************
2 *
Joe Perches3c5f9be42008-02-03 17:06:17 +02003 * Module Name: utresrc - Resource management utilities
Bob Moore96db2552005-11-02 00:00:00 -05004 *
5 ******************************************************************************/
6
7/*
Bob Moore7735ca02017-02-08 11:00:08 +08008 * Copyright (C) 2000 - 2017, Intel Corp.
Bob Moore96db2552005-11-02 00:00:00 -05009 * All rights reserved.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions, and the following disclaimer,
16 * without modification.
17 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18 * substantially similar to the "NO WARRANTY" disclaimer below
19 * ("Disclaimer") and any redistribution must be conditioned upon
20 * including a substantially similar Disclaimer requirement for further
21 * binary redistribution.
22 * 3. Neither the names of the above-listed copyright holders nor the names
23 * of any contributors may be used to endorse or promote products derived
24 * from this software without specific prior written permission.
25 *
26 * Alternatively, this software may be distributed under the terms of the
27 * GNU General Public License ("GPL") version 2 as published by the Free
28 * Software Foundation.
29 *
30 * NO WARRANTY
31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41 * POSSIBILITY OF SUCH DAMAGES.
42 */
43
44#include <acpi/acpi.h>
Len Browne2f7a772009-01-09 00:30:03 -050045#include "accommon.h"
Lin Minge0fe0a82011-11-16 14:38:13 +080046#include "acresrc.h"
Bob Moore96db2552005-11-02 00:00:00 -050047
48#define _COMPONENT ACPI_UTILITIES
Bob Mooreb229cf92006-04-21 17:15:00 -040049ACPI_MODULE_NAME("utresrc")
Lv Zheng33348612014-02-08 09:42:46 +080050
51#if defined(ACPI_DEBUG_OUTPUT) || defined (ACPI_DISASSEMBLER) || defined (ACPI_DEBUGGER)
Bob Mooredefba1d2005-12-16 17:05:00 -050052/*
53 * Strings used to decode resource descriptors.
Justin P. Mattock70f23fd2011-05-10 10:16:21 +020054 * Used by both the disassembler and the debugger resource dump routines
Bob Mooredefba1d2005-12-16 17:05:00 -050055 */
Bob Mooreb229cf92006-04-21 17:15:00 -040056const char *acpi_gbl_bm_decode[] = {
57 "NotBusMaster",
58 "BusMaster"
Bob Mooredefba1d2005-12-16 17:05:00 -050059};
60
Bob Mooreb229cf92006-04-21 17:15:00 -040061const char *acpi_gbl_config_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -050062 "0 - Good Configuration",
63 "1 - Acceptable Configuration",
64 "2 - Suboptimal Configuration",
65 "3 - ***Invalid Configuration***",
66};
67
Bob Mooreb229cf92006-04-21 17:15:00 -040068const char *acpi_gbl_consume_decode[] = {
69 "ResourceProducer",
70 "ResourceConsumer"
Bob Mooredefba1d2005-12-16 17:05:00 -050071};
72
Bob Mooreb229cf92006-04-21 17:15:00 -040073const char *acpi_gbl_dec_decode[] = {
74 "PosDecode",
75 "SubDecode"
Bob Mooredefba1d2005-12-16 17:05:00 -050076};
77
Bob Mooreb229cf92006-04-21 17:15:00 -040078const char *acpi_gbl_he_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -050079 "Level",
80 "Edge"
81};
82
Bob Mooreb229cf92006-04-21 17:15:00 -040083const char *acpi_gbl_io_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -050084 "Decode10",
85 "Decode16"
86};
87
Bob Mooreb229cf92006-04-21 17:15:00 -040088const char *acpi_gbl_ll_decode[] = {
89 "ActiveHigh",
Bob Moore4cc09092014-09-02 08:27:27 +080090 "ActiveLow",
91 "ActiveBoth",
92 "Reserved"
Bob Mooredefba1d2005-12-16 17:05:00 -050093};
94
Bob Mooreb229cf92006-04-21 17:15:00 -040095const char *acpi_gbl_max_decode[] = {
96 "MaxNotFixed",
97 "MaxFixed"
Bob Mooredefba1d2005-12-16 17:05:00 -050098};
99
Bob Mooreb229cf92006-04-21 17:15:00 -0400100const char *acpi_gbl_mem_decode[] = {
101 "NonCacheable",
Bob Mooredefba1d2005-12-16 17:05:00 -0500102 "Cacheable",
Bob Mooreb229cf92006-04-21 17:15:00 -0400103 "WriteCombining",
Bob Mooredefba1d2005-12-16 17:05:00 -0500104 "Prefetchable"
105};
106
Bob Mooreb229cf92006-04-21 17:15:00 -0400107const char *acpi_gbl_min_decode[] = {
108 "MinNotFixed",
109 "MinFixed"
Bob Mooredefba1d2005-12-16 17:05:00 -0500110};
111
Bob Mooreb229cf92006-04-21 17:15:00 -0400112const char *acpi_gbl_mtp_decode[] = {
113 "AddressRangeMemory",
114 "AddressRangeReserved",
115 "AddressRangeACPI",
116 "AddressRangeNVS"
Bob Mooredefba1d2005-12-16 17:05:00 -0500117};
118
Bob Mooreb229cf92006-04-21 17:15:00 -0400119const char *acpi_gbl_rng_decode[] = {
120 "InvalidRanges",
121 "NonISAOnlyRanges",
122 "ISAOnlyRanges",
123 "EntireRange"
Bob Mooredefba1d2005-12-16 17:05:00 -0500124};
125
Bob Mooreb229cf92006-04-21 17:15:00 -0400126const char *acpi_gbl_rw_decode[] = {
127 "ReadOnly",
128 "ReadWrite"
Bob Mooredefba1d2005-12-16 17:05:00 -0500129};
130
Bob Mooreb229cf92006-04-21 17:15:00 -0400131const char *acpi_gbl_shr_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -0500132 "Exclusive",
Bob Moore1f06c922012-12-19 05:38:12 +0000133 "Shared",
134 "ExclusiveAndWake", /* ACPI 5.0 */
135 "SharedAndWake" /* ACPI 5.0 */
Bob Mooredefba1d2005-12-16 17:05:00 -0500136};
137
Bob Mooreb229cf92006-04-21 17:15:00 -0400138const char *acpi_gbl_siz_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -0500139 "Transfer8",
140 "Transfer8_16",
141 "Transfer16",
Bob Mooreb229cf92006-04-21 17:15:00 -0400142 "InvalidSize"
Bob Mooredefba1d2005-12-16 17:05:00 -0500143};
144
Bob Mooreb229cf92006-04-21 17:15:00 -0400145const char *acpi_gbl_trs_decode[] = {
146 "DenseTranslation",
147 "SparseTranslation"
Bob Mooredefba1d2005-12-16 17:05:00 -0500148};
149
Bob Mooreb229cf92006-04-21 17:15:00 -0400150const char *acpi_gbl_ttp_decode[] = {
151 "TypeStatic",
152 "TypeTranslation"
Bob Mooredefba1d2005-12-16 17:05:00 -0500153};
154
Bob Mooreb229cf92006-04-21 17:15:00 -0400155const char *acpi_gbl_typ_decode[] = {
Bob Mooredefba1d2005-12-16 17:05:00 -0500156 "Compatibility",
Bob Mooreb229cf92006-04-21 17:15:00 -0400157 "TypeA",
158 "TypeB",
159 "TypeF"
Bob Mooredefba1d2005-12-16 17:05:00 -0500160};
161
Lin Minge0fe0a82011-11-16 14:38:13 +0800162const char *acpi_gbl_ppc_decode[] = {
163 "PullDefault",
164 "PullUp",
165 "PullDown",
166 "PullNone"
167};
168
169const char *acpi_gbl_ior_decode[] = {
170 "IoRestrictionNone",
171 "IoRestrictionInputOnly",
172 "IoRestrictionOutputOnly",
173 "IoRestrictionNoneAndPreserve"
174};
175
176const char *acpi_gbl_dts_decode[] = {
177 "Width8bit",
178 "Width16bit",
179 "Width32bit",
180 "Width64bit",
181 "Width128bit",
182 "Width256bit",
183};
184
185/* GPIO connection type */
186
187const char *acpi_gbl_ct_decode[] = {
188 "Interrupt",
189 "I/O"
190};
191
192/* Serial bus type */
193
194const char *acpi_gbl_sbt_decode[] = {
195 "/* UNKNOWN serial bus type */",
196 "I2C",
197 "SPI",
198 "UART"
199};
200
201/* I2C serial bus access mode */
202
203const char *acpi_gbl_am_decode[] = {
204 "AddressingMode7Bit",
205 "AddressingMode10Bit"
206};
207
208/* I2C serial bus slave mode */
209
210const char *acpi_gbl_sm_decode[] = {
211 "ControllerInitiated",
212 "DeviceInitiated"
213};
214
215/* SPI serial bus wire mode */
216
217const char *acpi_gbl_wm_decode[] = {
218 "FourWireMode",
219 "ThreeWireMode"
220};
221
222/* SPI serial clock phase */
223
224const char *acpi_gbl_cph_decode[] = {
225 "ClockPhaseFirst",
226 "ClockPhaseSecond"
227};
228
229/* SPI serial bus clock polarity */
230
231const char *acpi_gbl_cpo_decode[] = {
232 "ClockPolarityLow",
233 "ClockPolarityHigh"
234};
235
236/* SPI serial bus device polarity */
237
238const char *acpi_gbl_dp_decode[] = {
239 "PolarityLow",
240 "PolarityHigh"
241};
242
243/* UART serial bus endian */
244
245const char *acpi_gbl_ed_decode[] = {
246 "LittleEndian",
247 "BigEndian"
248};
249
250/* UART serial bus bits per byte */
251
252const char *acpi_gbl_bpb_decode[] = {
253 "DataBitsFive",
254 "DataBitsSix",
255 "DataBitsSeven",
256 "DataBitsEight",
257 "DataBitsNine",
258 "/* UNKNOWN Bits per byte */",
259 "/* UNKNOWN Bits per byte */",
260 "/* UNKNOWN Bits per byte */"
261};
262
263/* UART serial bus stop bits */
264
265const char *acpi_gbl_sb_decode[] = {
Bob Moore11811732014-11-27 14:26:28 +0800266 "StopBitsZero",
Lin Minge0fe0a82011-11-16 14:38:13 +0800267 "StopBitsOne",
268 "StopBitsOnePlusHalf",
269 "StopBitsTwo"
270};
271
272/* UART serial bus flow control */
273
274const char *acpi_gbl_fc_decode[] = {
275 "FlowControlNone",
276 "FlowControlHardware",
277 "FlowControlXON",
278 "/* UNKNOWN flow control keyword */"
279};
280
281/* UART serial bus parity type */
282
283const char *acpi_gbl_pt_decode[] = {
284 "ParityTypeNone",
285 "ParityTypeEven",
286 "ParityTypeOdd",
287 "ParityTypeMark",
288 "ParityTypeSpace",
289 "/* UNKNOWN parity keyword */",
290 "/* UNKNOWN parity keyword */",
291 "/* UNKNOWN parity keyword */"
292};
293
Bob Mooredefba1d2005-12-16 17:05:00 -0500294#endif
295
Bob Moore96db2552005-11-02 00:00:00 -0500296/*
297 * Base sizes of the raw AML resource descriptors, indexed by resource type.
298 * Zero indicates a reserved (and therefore invalid) resource type.
299 */
300const u8 acpi_gbl_resource_aml_sizes[] = {
301 /* Small descriptors */
302
303 0,
304 0,
305 0,
306 0,
307 ACPI_AML_SIZE_SMALL(struct aml_resource_irq),
308 ACPI_AML_SIZE_SMALL(struct aml_resource_dma),
309 ACPI_AML_SIZE_SMALL(struct aml_resource_start_dependent),
310 ACPI_AML_SIZE_SMALL(struct aml_resource_end_dependent),
311 ACPI_AML_SIZE_SMALL(struct aml_resource_io),
312 ACPI_AML_SIZE_SMALL(struct aml_resource_fixed_io),
Lin Minge0fe0a82011-11-16 14:38:13 +0800313 ACPI_AML_SIZE_SMALL(struct aml_resource_fixed_dma),
Bob Moore96db2552005-11-02 00:00:00 -0500314 0,
315 0,
316 0,
317 ACPI_AML_SIZE_SMALL(struct aml_resource_vendor_small),
318 ACPI_AML_SIZE_SMALL(struct aml_resource_end_tag),
319
320 /* Large descriptors */
321
322 0,
323 ACPI_AML_SIZE_LARGE(struct aml_resource_memory24),
324 ACPI_AML_SIZE_LARGE(struct aml_resource_generic_register),
325 0,
326 ACPI_AML_SIZE_LARGE(struct aml_resource_vendor_large),
327 ACPI_AML_SIZE_LARGE(struct aml_resource_memory32),
328 ACPI_AML_SIZE_LARGE(struct aml_resource_fixed_memory32),
329 ACPI_AML_SIZE_LARGE(struct aml_resource_address32),
330 ACPI_AML_SIZE_LARGE(struct aml_resource_address16),
331 ACPI_AML_SIZE_LARGE(struct aml_resource_extended_irq),
332 ACPI_AML_SIZE_LARGE(struct aml_resource_address64),
Lin Minge0fe0a82011-11-16 14:38:13 +0800333 ACPI_AML_SIZE_LARGE(struct aml_resource_extended_address64),
334 ACPI_AML_SIZE_LARGE(struct aml_resource_gpio),
335 0,
336 ACPI_AML_SIZE_LARGE(struct aml_resource_common_serialbus),
337};
338
339const u8 acpi_gbl_resource_aml_serial_bus_sizes[] = {
340 0,
341 ACPI_AML_SIZE_LARGE(struct aml_resource_i2c_serialbus),
342 ACPI_AML_SIZE_LARGE(struct aml_resource_spi_serialbus),
343 ACPI_AML_SIZE_LARGE(struct aml_resource_uart_serialbus),
Bob Moore96db2552005-11-02 00:00:00 -0500344};
345
346/*
347 * Resource types, used to validate the resource length field.
348 * The length of fixed-length types must match exactly, variable
349 * lengths must meet the minimum required length, etc.
350 * Zero indicates a reserved (and therefore invalid) resource type.
351 */
352static const u8 acpi_gbl_resource_types[] = {
353 /* Small descriptors */
354
355 0,
356 0,
357 0,
358 0,
Lin Minge0fe0a82011-11-16 14:38:13 +0800359 ACPI_SMALL_VARIABLE_LENGTH, /* 04 IRQ */
360 ACPI_FIXED_LENGTH, /* 05 DMA */
361 ACPI_SMALL_VARIABLE_LENGTH, /* 06 start_dependent_functions */
362 ACPI_FIXED_LENGTH, /* 07 end_dependent_functions */
363 ACPI_FIXED_LENGTH, /* 08 IO */
Bob Mooreba494be2012-07-12 09:40:10 +0800364 ACPI_FIXED_LENGTH, /* 09 fixed_IO */
365 ACPI_FIXED_LENGTH, /* 0A fixed_DMA */
Bob Moore96db2552005-11-02 00:00:00 -0500366 0,
367 0,
368 0,
Bob Mooreba494be2012-07-12 09:40:10 +0800369 ACPI_VARIABLE_LENGTH, /* 0E vendor_short */
370 ACPI_FIXED_LENGTH, /* 0F end_tag */
Bob Moore96db2552005-11-02 00:00:00 -0500371
372 /* Large descriptors */
373
374 0,
Lin Minge0fe0a82011-11-16 14:38:13 +0800375 ACPI_FIXED_LENGTH, /* 01 Memory24 */
376 ACPI_FIXED_LENGTH, /* 02 generic_register */
Bob Moore96db2552005-11-02 00:00:00 -0500377 0,
Lin Minge0fe0a82011-11-16 14:38:13 +0800378 ACPI_VARIABLE_LENGTH, /* 04 vendor_long */
379 ACPI_FIXED_LENGTH, /* 05 Memory32 */
380 ACPI_FIXED_LENGTH, /* 06 memory32_fixed */
381 ACPI_VARIABLE_LENGTH, /* 07 Dword* address */
382 ACPI_VARIABLE_LENGTH, /* 08 Word* address */
Bob Mooreba494be2012-07-12 09:40:10 +0800383 ACPI_VARIABLE_LENGTH, /* 09 extended_IRQ */
384 ACPI_VARIABLE_LENGTH, /* 0A Qword* address */
385 ACPI_FIXED_LENGTH, /* 0B Extended* address */
386 ACPI_VARIABLE_LENGTH, /* 0C Gpio* */
Lin Minge0fe0a82011-11-16 14:38:13 +0800387 0,
Bob Mooreba494be2012-07-12 09:40:10 +0800388 ACPI_VARIABLE_LENGTH /* 0E *serial_bus */
Bob Moore96db2552005-11-02 00:00:00 -0500389};
390
391/*******************************************************************************
392 *
Bob Moore616861242006-03-17 16:44:00 -0500393 * FUNCTION: acpi_ut_walk_aml_resources
394 *
Bob Moore886308e2012-12-19 05:38:07 +0000395 * PARAMETERS: walk_state - Current walk info
396 * PARAMETERS: aml - Pointer to the raw AML resource template
397 * aml_length - Length of the entire template
398 * user_function - Called once for each descriptor found. If
399 * NULL, a pointer to the end_tag is returned
400 * context - Passed to user_function
Bob Moore616861242006-03-17 16:44:00 -0500401 *
402 * RETURN: Status
403 *
404 * DESCRIPTION: Walk a raw AML resource list(buffer). User function called
405 * once for each resource found.
406 *
407 ******************************************************************************/
408
409acpi_status
Bob Moore886308e2012-12-19 05:38:07 +0000410acpi_ut_walk_aml_resources(struct acpi_walk_state *walk_state,
411 u8 *aml,
Bob Moore616861242006-03-17 16:44:00 -0500412 acpi_size aml_length,
Bob Moore793c2382006-03-31 00:00:00 -0500413 acpi_walk_aml_callback user_function, void **context)
Bob Moore616861242006-03-17 16:44:00 -0500414{
415 acpi_status status;
416 u8 *end_aml;
417 u8 resource_index;
418 u32 length;
419 u32 offset = 0;
Lin Minge0fe0a82011-11-16 14:38:13 +0800420 u8 end_tag[2] = { 0x79, 0x00 };
Bob Moore616861242006-03-17 16:44:00 -0500421
Bob Mooreb229cf92006-04-21 17:15:00 -0400422 ACPI_FUNCTION_TRACE(ut_walk_aml_resources);
Bob Moore616861242006-03-17 16:44:00 -0500423
Rafael J. Wysocki1315f012017-04-13 18:14:55 +0200424 /* The absolute minimum resource template is one end_tag descriptor */
425
Bob Moore616861242006-03-17 16:44:00 -0500426 if (aml_length < sizeof(struct aml_resource_end_tag)) {
427 return_ACPI_STATUS(AE_AML_NO_RESOURCE_END_TAG);
428 }
429
430 /* Point to the end of the resource template buffer */
431
432 end_aml = aml + aml_length;
433
434 /* Walk the byte list, abort on any invalid descriptor type or length */
435
436 while (aml < end_aml) {
437
438 /* Validate the Resource Type and Resource Length */
439
Bob Moore886308e2012-12-19 05:38:07 +0000440 status =
441 acpi_ut_validate_resource(walk_state, aml, &resource_index);
Bob Moore616861242006-03-17 16:44:00 -0500442 if (ACPI_FAILURE(status)) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800443 /*
Bob Moore1fad8732015-12-29 13:54:36 +0800444 * Exit on failure. Cannot continue because the descriptor
445 * length may be bogus also.
Lin Minge0fe0a82011-11-16 14:38:13 +0800446 */
Bob Moore616861242006-03-17 16:44:00 -0500447 return_ACPI_STATUS(status);
448 }
449
450 /* Get the length of this descriptor */
451
452 length = acpi_ut_get_descriptor_length(aml);
453
454 /* Invoke the user function */
455
456 if (user_function) {
Rafael J. Wysocki1315f012017-04-13 18:14:55 +0200457 status =
458 user_function(aml, length, offset, resource_index,
459 context);
Bob Moore616861242006-03-17 16:44:00 -0500460 if (ACPI_FAILURE(status)) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800461 return_ACPI_STATUS(status);
Bob Moore616861242006-03-17 16:44:00 -0500462 }
463 }
464
465 /* An end_tag descriptor terminates this resource template */
466
467 if (acpi_ut_get_resource_type(aml) ==
468 ACPI_RESOURCE_NAME_END_TAG) {
469 /*
470 * There must be at least one more byte in the buffer for
471 * the 2nd byte of the end_tag
472 */
473 if ((aml + 1) >= end_aml) {
474 return_ACPI_STATUS(AE_AML_NO_RESOURCE_END_TAG);
475 }
476
Bob Mooreda28e192017-04-26 16:17:43 +0800477 /*
478 * The end_tag opcode must be followed by a zero byte.
479 * Although this byte is technically defined to be a checksum,
480 * in practice, all ASL compilers set this byte to zero.
481 */
482 if (*(aml + 1) != 0) {
483 return_ACPI_STATUS(AE_AML_NO_RESOURCE_END_TAG);
484 }
485
Bob Moore616861242006-03-17 16:44:00 -0500486 /* Return the pointer to the end_tag if requested */
487
488 if (!user_function) {
Bob Moore793c2382006-03-31 00:00:00 -0500489 *context = aml;
Bob Moore616861242006-03-17 16:44:00 -0500490 }
491
492 /* Normal exit */
493
494 return_ACPI_STATUS(AE_OK);
495 }
496
497 aml += length;
498 offset += length;
499 }
500
501 /* Did not find an end_tag descriptor */
502
Lin Minge0fe0a82011-11-16 14:38:13 +0800503 if (user_function) {
504
505 /* Insert an end_tag anyway. acpi_rs_get_list_length always leaves room */
506
Bob Moore886308e2012-12-19 05:38:07 +0000507 (void)acpi_ut_validate_resource(walk_state, end_tag,
508 &resource_index);
Lin Minge0fe0a82011-11-16 14:38:13 +0800509 status =
510 user_function(end_tag, 2, offset, resource_index, context);
511 if (ACPI_FAILURE(status)) {
512 return_ACPI_STATUS(status);
513 }
514 }
515
516 return_ACPI_STATUS(AE_AML_NO_RESOURCE_END_TAG);
Bob Moore616861242006-03-17 16:44:00 -0500517}
518
519/*******************************************************************************
520 *
Bob Moore96db2552005-11-02 00:00:00 -0500521 * FUNCTION: acpi_ut_validate_resource
522 *
Bob Moore886308e2012-12-19 05:38:07 +0000523 * PARAMETERS: walk_state - Current walk info
524 * aml - Pointer to the raw AML resource descriptor
525 * return_index - Where the resource index is returned. NULL
526 * if the index is not required.
Bob Moore96db2552005-11-02 00:00:00 -0500527 *
528 * RETURN: Status, and optionally the Index into the global resource tables
529 *
530 * DESCRIPTION: Validate an AML resource descriptor by checking the Resource
531 * Type and Resource Length. Returns an index into the global
532 * resource information/dispatch tables for later use.
533 *
534 ******************************************************************************/
535
Bob Moore886308e2012-12-19 05:38:07 +0000536acpi_status
537acpi_ut_validate_resource(struct acpi_walk_state *walk_state,
538 void *aml, u8 *return_index)
Bob Moore96db2552005-11-02 00:00:00 -0500539{
Lin Minge0fe0a82011-11-16 14:38:13 +0800540 union aml_resource *aml_resource;
Bob Moore96db2552005-11-02 00:00:00 -0500541 u8 resource_type;
542 u8 resource_index;
543 acpi_rs_length resource_length;
544 acpi_rs_length minimum_resource_length;
545
546 ACPI_FUNCTION_ENTRY();
547
548 /*
549 * 1) Validate the resource_type field (Byte 0)
550 */
Bob Moorec51a4de2005-11-17 13:07:00 -0500551 resource_type = ACPI_GET8(aml);
Bob Moore96db2552005-11-02 00:00:00 -0500552
553 /*
554 * Byte 0 contains the descriptor name (Resource Type)
555 * Examine the large/small bit in the resource header
556 */
557 if (resource_type & ACPI_RESOURCE_NAME_LARGE) {
Bob Moore52fc0b02006-10-02 00:00:00 -0400558
Bob Moore96db2552005-11-02 00:00:00 -0500559 /* Verify the large resource type (name) against the max */
560
561 if (resource_type > ACPI_RESOURCE_NAME_LARGE_MAX) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800562 goto invalid_resource;
Bob Moore96db2552005-11-02 00:00:00 -0500563 }
564
565 /*
566 * Large Resource Type -- bits 6:0 contain the name
567 * Translate range 0x80-0x8B to index range 0x10-0x1B
568 */
569 resource_index = (u8) (resource_type - 0x70);
570 } else {
571 /*
572 * Small Resource Type -- bits 6:3 contain the name
573 * Shift range to index range 0x00-0x0F
574 */
575 resource_index = (u8)
576 ((resource_type & ACPI_RESOURCE_NAME_SMALL_MASK) >> 3);
577 }
578
Lin Minge0fe0a82011-11-16 14:38:13 +0800579 /*
Bob Moore1fad8732015-12-29 13:54:36 +0800580 * Check validity of the resource type, via acpi_gbl_resource_types.
581 * Zero indicates an invalid resource.
Lin Minge0fe0a82011-11-16 14:38:13 +0800582 */
Bob Moore96db2552005-11-02 00:00:00 -0500583 if (!acpi_gbl_resource_types[resource_index]) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800584 goto invalid_resource;
Bob Moore96db2552005-11-02 00:00:00 -0500585 }
586
587 /*
Lin Minge0fe0a82011-11-16 14:38:13 +0800588 * Validate the resource_length field. This ensures that the length
589 * is at least reasonable, and guarantees that it is non-zero.
Bob Moore96db2552005-11-02 00:00:00 -0500590 */
591 resource_length = acpi_ut_get_resource_length(aml);
592 minimum_resource_length = acpi_gbl_resource_aml_sizes[resource_index];
593
594 /* Validate based upon the type of resource - fixed length or variable */
595
596 switch (acpi_gbl_resource_types[resource_index]) {
597 case ACPI_FIXED_LENGTH:
598
599 /* Fixed length resource, length must match exactly */
600
601 if (resource_length != minimum_resource_length) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800602 goto bad_resource_length;
Bob Moore96db2552005-11-02 00:00:00 -0500603 }
604 break;
605
606 case ACPI_VARIABLE_LENGTH:
607
608 /* Variable length resource, length must be at least the minimum */
609
610 if (resource_length < minimum_resource_length) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800611 goto bad_resource_length;
Bob Moore96db2552005-11-02 00:00:00 -0500612 }
613 break;
614
615 case ACPI_SMALL_VARIABLE_LENGTH:
616
617 /* Small variable length resource, length can be (Min) or (Min-1) */
618
619 if ((resource_length > minimum_resource_length) ||
620 (resource_length < (minimum_resource_length - 1))) {
Lin Minge0fe0a82011-11-16 14:38:13 +0800621 goto bad_resource_length;
Bob Moore96db2552005-11-02 00:00:00 -0500622 }
623 break;
624
625 default:
626
627 /* Shouldn't happen (because of validation earlier), but be sure */
628
Lin Minge0fe0a82011-11-16 14:38:13 +0800629 goto invalid_resource;
630 }
631
632 aml_resource = ACPI_CAST_PTR(union aml_resource, aml);
633 if (resource_type == ACPI_RESOURCE_NAME_SERIAL_BUS) {
634
635 /* Validate the bus_type field */
636
637 if ((aml_resource->common_serial_bus.type == 0) ||
638 (aml_resource->common_serial_bus.type >
639 AML_RESOURCE_MAX_SERIALBUSTYPE)) {
Bob Moore886308e2012-12-19 05:38:07 +0000640 if (walk_state) {
641 ACPI_ERROR((AE_INFO,
642 "Invalid/unsupported SerialBus resource descriptor: BusType 0x%2.2X",
643 aml_resource->common_serial_bus.
644 type));
645 }
Lin Minge0fe0a82011-11-16 14:38:13 +0800646 return (AE_AML_INVALID_RESOURCE_TYPE);
647 }
Bob Moore96db2552005-11-02 00:00:00 -0500648 }
649
650 /* Optionally return the resource table index */
651
652 if (return_index) {
653 *return_index = resource_index;
654 }
655
656 return (AE_OK);
Lin Minge0fe0a82011-11-16 14:38:13 +0800657
Lv Zheng10622bf2013-10-29 09:30:02 +0800658invalid_resource:
Lin Minge0fe0a82011-11-16 14:38:13 +0800659
Bob Moore886308e2012-12-19 05:38:07 +0000660 if (walk_state) {
661 ACPI_ERROR((AE_INFO,
662 "Invalid/unsupported resource descriptor: Type 0x%2.2X",
663 resource_type));
664 }
Lin Minge0fe0a82011-11-16 14:38:13 +0800665 return (AE_AML_INVALID_RESOURCE_TYPE);
666
Lv Zheng10622bf2013-10-29 09:30:02 +0800667bad_resource_length:
Lin Minge0fe0a82011-11-16 14:38:13 +0800668
Bob Moore886308e2012-12-19 05:38:07 +0000669 if (walk_state) {
670 ACPI_ERROR((AE_INFO,
671 "Invalid resource descriptor length: Type "
672 "0x%2.2X, Length 0x%4.4X, MinLength 0x%4.4X",
673 resource_type, resource_length,
674 minimum_resource_length));
675 }
Lin Minge0fe0a82011-11-16 14:38:13 +0800676 return (AE_AML_BAD_RESOURCE_LENGTH);
Bob Moore96db2552005-11-02 00:00:00 -0500677}
678
679/*******************************************************************************
680 *
681 * FUNCTION: acpi_ut_get_resource_type
682 *
Bob Mooreba494be2012-07-12 09:40:10 +0800683 * PARAMETERS: aml - Pointer to the raw AML resource descriptor
Bob Moore96db2552005-11-02 00:00:00 -0500684 *
685 * RETURN: The Resource Type with no extraneous bits (except the
686 * Large/Small descriptor bit -- this is left alone)
687 *
688 * DESCRIPTION: Extract the Resource Type/Name from the first byte of
689 * a resource descriptor.
690 *
691 ******************************************************************************/
692
693u8 acpi_ut_get_resource_type(void *aml)
694{
695 ACPI_FUNCTION_ENTRY();
696
697 /*
698 * Byte 0 contains the descriptor name (Resource Type)
699 * Examine the large/small bit in the resource header
700 */
Bob Moorec51a4de2005-11-17 13:07:00 -0500701 if (ACPI_GET8(aml) & ACPI_RESOURCE_NAME_LARGE) {
Bob Moore52fc0b02006-10-02 00:00:00 -0400702
Bob Moore96db2552005-11-02 00:00:00 -0500703 /* Large Resource Type -- bits 6:0 contain the name */
704
Bob Moorec51a4de2005-11-17 13:07:00 -0500705 return (ACPI_GET8(aml));
Bob Moore96db2552005-11-02 00:00:00 -0500706 } else {
707 /* Small Resource Type -- bits 6:3 contain the name */
708
Bob Moorec51a4de2005-11-17 13:07:00 -0500709 return ((u8) (ACPI_GET8(aml) & ACPI_RESOURCE_NAME_SMALL_MASK));
Bob Moore96db2552005-11-02 00:00:00 -0500710 }
711}
712
713/*******************************************************************************
714 *
715 * FUNCTION: acpi_ut_get_resource_length
716 *
Bob Mooreba494be2012-07-12 09:40:10 +0800717 * PARAMETERS: aml - Pointer to the raw AML resource descriptor
Bob Moore96db2552005-11-02 00:00:00 -0500718 *
719 * RETURN: Byte Length
720 *
721 * DESCRIPTION: Get the "Resource Length" of a raw AML descriptor. By
722 * definition, this does not include the size of the descriptor
723 * header or the length field itself.
724 *
725 ******************************************************************************/
726
727u16 acpi_ut_get_resource_length(void *aml)
728{
729 acpi_rs_length resource_length;
730
731 ACPI_FUNCTION_ENTRY();
732
733 /*
734 * Byte 0 contains the descriptor name (Resource Type)
735 * Examine the large/small bit in the resource header
736 */
Bob Moorec51a4de2005-11-17 13:07:00 -0500737 if (ACPI_GET8(aml) & ACPI_RESOURCE_NAME_LARGE) {
Bob Moore52fc0b02006-10-02 00:00:00 -0400738
Bob Moore96db2552005-11-02 00:00:00 -0500739 /* Large Resource type -- bytes 1-2 contain the 16-bit length */
740
Bob Moorec51a4de2005-11-17 13:07:00 -0500741 ACPI_MOVE_16_TO_16(&resource_length, ACPI_ADD_PTR(u8, aml, 1));
Bob Moore96db2552005-11-02 00:00:00 -0500742
743 } else {
744 /* Small Resource type -- bits 2:0 of byte 0 contain the length */
745
Bob Moorec51a4de2005-11-17 13:07:00 -0500746 resource_length = (u16) (ACPI_GET8(aml) &
Bob Moore96db2552005-11-02 00:00:00 -0500747 ACPI_RESOURCE_NAME_SMALL_LENGTH_MASK);
748 }
749
750 return (resource_length);
751}
752
753/*******************************************************************************
754 *
755 * FUNCTION: acpi_ut_get_resource_header_length
756 *
Bob Mooreba494be2012-07-12 09:40:10 +0800757 * PARAMETERS: aml - Pointer to the raw AML resource descriptor
Bob Moore96db2552005-11-02 00:00:00 -0500758 *
759 * RETURN: Length of the AML header (depends on large/small descriptor)
760 *
761 * DESCRIPTION: Get the length of the header for this resource.
762 *
763 ******************************************************************************/
764
765u8 acpi_ut_get_resource_header_length(void *aml)
766{
767 ACPI_FUNCTION_ENTRY();
768
769 /* Examine the large/small bit in the resource header */
770
Bob Moorec51a4de2005-11-17 13:07:00 -0500771 if (ACPI_GET8(aml) & ACPI_RESOURCE_NAME_LARGE) {
Bob Moore96db2552005-11-02 00:00:00 -0500772 return (sizeof(struct aml_resource_large_header));
773 } else {
774 return (sizeof(struct aml_resource_small_header));
775 }
776}
777
778/*******************************************************************************
779 *
780 * FUNCTION: acpi_ut_get_descriptor_length
781 *
Bob Mooreba494be2012-07-12 09:40:10 +0800782 * PARAMETERS: aml - Pointer to the raw AML resource descriptor
Bob Moore96db2552005-11-02 00:00:00 -0500783 *
784 * RETURN: Byte length
785 *
786 * DESCRIPTION: Get the total byte length of a raw AML descriptor, including the
787 * length of the descriptor header and the length field itself.
788 * Used to walk descriptor lists.
789 *
790 ******************************************************************************/
791
792u32 acpi_ut_get_descriptor_length(void *aml)
793{
794 ACPI_FUNCTION_ENTRY();
795
796 /*
797 * Get the Resource Length (does not include header length) and add
798 * the header length (depends on if this is a small or large resource)
799 */
800 return (acpi_ut_get_resource_length(aml) +
801 acpi_ut_get_resource_header_length(aml));
802}
803
804/*******************************************************************************
805 *
806 * FUNCTION: acpi_ut_get_resource_end_tag
807 *
808 * PARAMETERS: obj_desc - The resource template buffer object
Bob Moorec51a4de2005-11-17 13:07:00 -0500809 * end_tag - Where the pointer to the end_tag is returned
Bob Moore96db2552005-11-02 00:00:00 -0500810 *
Bob Moorec51a4de2005-11-17 13:07:00 -0500811 * RETURN: Status, pointer to the end tag
Bob Moore96db2552005-11-02 00:00:00 -0500812 *
813 * DESCRIPTION: Find the end_tag resource descriptor in an AML resource template
Bob Mooreb8e4d892006-01-27 16:43:00 -0500814 * Note: allows a buffer length of zero.
Bob Moore96db2552005-11-02 00:00:00 -0500815 *
816 ******************************************************************************/
817
818acpi_status
Lv Zheng3e8214e2012-12-19 05:37:15 +0000819acpi_ut_get_resource_end_tag(union acpi_operand_object *obj_desc, u8 **end_tag)
Bob Moore96db2552005-11-02 00:00:00 -0500820{
821 acpi_status status;
Bob Moore96db2552005-11-02 00:00:00 -0500822
Bob Mooreb229cf92006-04-21 17:15:00 -0400823 ACPI_FUNCTION_TRACE(ut_get_resource_end_tag);
Bob Moore96db2552005-11-02 00:00:00 -0500824
Bob Mooreb8e4d892006-01-27 16:43:00 -0500825 /* Allow a buffer length of zero */
826
827 if (!obj_desc->buffer.length) {
Bob Moore616861242006-03-17 16:44:00 -0500828 *end_tag = obj_desc->buffer.pointer;
Bob Mooreb8e4d892006-01-27 16:43:00 -0500829 return_ACPI_STATUS(AE_OK);
830 }
831
Bob Moore616861242006-03-17 16:44:00 -0500832 /* Validate the template and get a pointer to the end_tag */
Bob Moore96db2552005-11-02 00:00:00 -0500833
Bob Moore886308e2012-12-19 05:38:07 +0000834 status = acpi_ut_walk_aml_resources(NULL, obj_desc->buffer.pointer,
Bob Moore616861242006-03-17 16:44:00 -0500835 obj_desc->buffer.length, NULL,
Bob Moore793c2382006-03-31 00:00:00 -0500836 (void **)end_tag);
Bob Moore52fc0b02006-10-02 00:00:00 -0400837
Bob Moore616861242006-03-17 16:44:00 -0500838 return_ACPI_STATUS(status);
Bob Moore96db2552005-11-02 00:00:00 -0500839}