diff options
| author | robert.moore@intel.com <robert.moore@intel.com> | 2008-12-31 00:07:26 -0500 |
|---|---|---|
| committer | Len Brown <len.brown@intel.com> | 2008-12-31 01:17:56 -0500 |
| commit | 4b67a0e467a57e24da6b761dbf95fa5d0225ff19 (patch) | |
| tree | efbfb11d74b7b522dab8b4c7f8d332629d66166c /drivers/acpi/tables | |
| parent | 06f5541960d02d5e0ddd8fd5c9a1554d85d94fa9 (diff) | |
ACPICA: FADT: set acpi_gbl_use_default_register_widths to TRUE by default
This returns the FADT support to the original behavior, which is
to use default register widths. However, now check each register
definition and report a warning if it differs from the default.
This is a first step to moving away from the default widths,
rather than outright believing the widths in all FADTs for all
machines, considered rather dangerous until more data is obtained.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Lin Ming <ming.m.lin@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
Diffstat (limited to 'drivers/acpi/tables')
| -rw-r--r-- | drivers/acpi/tables/tbfadt.c | 173 |
1 files changed, 105 insertions, 68 deletions
diff --git a/drivers/acpi/tables/tbfadt.c b/drivers/acpi/tables/tbfadt.c index b4ce2074c91f..7dec0af3f26a 100644 --- a/drivers/acpi/tables/tbfadt.c +++ b/drivers/acpi/tables/tbfadt.c | |||
| @@ -61,9 +61,10 @@ static void acpi_tb_validate_fadt(void); | |||
| 61 | 61 | ||
| 62 | typedef struct acpi_fadt_info { | 62 | typedef struct acpi_fadt_info { |
| 63 | char *name; | 63 | char *name; |
| 64 | u8 target; | 64 | u8 address64; |
| 65 | u8 source; | 65 | u8 address32; |
| 66 | u8 length; | 66 | u8 length; |
| 67 | u8 default_length; | ||
| 67 | u8 type; | 68 | u8 type; |
| 68 | 69 | ||
| 69 | } acpi_fadt_info; | 70 | } acpi_fadt_info; |
| @@ -72,37 +73,61 @@ typedef struct acpi_fadt_info { | |||
| 72 | #define ACPI_FADT_SEPARATE_LENGTH 2 | 73 | #define ACPI_FADT_SEPARATE_LENGTH 2 |
| 73 | 74 | ||
| 74 | static struct acpi_fadt_info fadt_info_table[] = { | 75 | static struct acpi_fadt_info fadt_info_table[] = { |
| 75 | {"Pm1aEventBlock", ACPI_FADT_OFFSET(xpm1a_event_block), | 76 | {"Pm1aEventBlock", |
| 77 | ACPI_FADT_OFFSET(xpm1a_event_block), | ||
| 76 | ACPI_FADT_OFFSET(pm1a_event_block), | 78 | ACPI_FADT_OFFSET(pm1a_event_block), |
| 77 | ACPI_FADT_OFFSET(pm1_event_length), ACPI_FADT_REQUIRED}, | 79 | ACPI_FADT_OFFSET(pm1_event_length), |
| 80 | ACPI_PM1_REGISTER_WIDTH * 2, /* Enable + Status register */ | ||
| 81 | ACPI_FADT_REQUIRED}, | ||
| 78 | 82 | ||
| 79 | {"Pm1bEventBlock", ACPI_FADT_OFFSET(xpm1b_event_block), | 83 | {"Pm1bEventBlock", |
| 84 | ACPI_FADT_OFFSET(xpm1b_event_block), | ||
| 80 | ACPI_FADT_OFFSET(pm1b_event_block), | 85 | ACPI_FADT_OFFSET(pm1b_event_block), |
| 81 | ACPI_FADT_OFFSET(pm1_event_length), 0}, | 86 | ACPI_FADT_OFFSET(pm1_event_length), |
| 87 | ACPI_PM1_REGISTER_WIDTH * 2, /* Enable + Status register */ | ||
| 88 | 0}, | ||
| 82 | 89 | ||
| 83 | {"Pm1aControlBlock", ACPI_FADT_OFFSET(xpm1a_control_block), | 90 | {"Pm1aControlBlock", |
| 91 | ACPI_FADT_OFFSET(xpm1a_control_block), | ||
| 84 | ACPI_FADT_OFFSET(pm1a_control_block), | 92 | ACPI_FADT_OFFSET(pm1a_control_block), |
| 85 | ACPI_FADT_OFFSET(pm1_control_length), ACPI_FADT_REQUIRED}, | 93 | ACPI_FADT_OFFSET(pm1_control_length), |
| 94 | ACPI_PM1_REGISTER_WIDTH, | ||
| 95 | ACPI_FADT_REQUIRED}, | ||
| 86 | 96 | ||
| 87 | {"Pm1bControlBlock", ACPI_FADT_OFFSET(xpm1b_control_block), | 97 | {"Pm1bControlBlock", |
| 98 | ACPI_FADT_OFFSET(xpm1b_control_block), | ||
| 88 | ACPI_FADT_OFFSET(pm1b_control_block), | 99 | ACPI_FADT_OFFSET(pm1b_control_block), |
| 89 | ACPI_FADT_OFFSET(pm1_control_length), 0}, | 100 | ACPI_FADT_OFFSET(pm1_control_length), |
| 101 | ACPI_PM1_REGISTER_WIDTH, | ||
| 102 | 0}, | ||
| 90 | 103 | ||
| 91 | {"Pm2ControlBlock", ACPI_FADT_OFFSET(xpm2_control_block), | 104 | {"Pm2ControlBlock", |
| 105 | ACPI_FADT_OFFSET(xpm2_control_block), | ||
| 92 | ACPI_FADT_OFFSET(pm2_control_block), | 106 | ACPI_FADT_OFFSET(pm2_control_block), |
| 93 | ACPI_FADT_OFFSET(pm2_control_length), ACPI_FADT_SEPARATE_LENGTH}, | 107 | ACPI_FADT_OFFSET(pm2_control_length), |
| 108 | ACPI_PM2_REGISTER_WIDTH, | ||
| 109 | ACPI_FADT_SEPARATE_LENGTH}, | ||
| 94 | 110 | ||
| 95 | {"PmTimerBlock", ACPI_FADT_OFFSET(xpm_timer_block), | 111 | {"PmTimerBlock", |
| 112 | ACPI_FADT_OFFSET(xpm_timer_block), | ||
| 96 | ACPI_FADT_OFFSET(pm_timer_block), | 113 | ACPI_FADT_OFFSET(pm_timer_block), |
| 97 | ACPI_FADT_OFFSET(pm_timer_length), ACPI_FADT_REQUIRED}, | 114 | ACPI_FADT_OFFSET(pm_timer_length), |
| 115 | ACPI_PM_TIMER_WIDTH, | ||
| 116 | ACPI_FADT_REQUIRED}, | ||
| 98 | 117 | ||
| 99 | {"Gpe0Block", ACPI_FADT_OFFSET(xgpe0_block), | 118 | {"Gpe0Block", |
| 119 | ACPI_FADT_OFFSET(xgpe0_block), | ||
| 100 | ACPI_FADT_OFFSET(gpe0_block), | 120 | ACPI_FADT_OFFSET(gpe0_block), |
| 101 | ACPI_FADT_OFFSET(gpe0_block_length), ACPI_FADT_SEPARATE_LENGTH}, | 121 | ACPI_FADT_OFFSET(gpe0_block_length), |
| 122 | 0, | ||
| 123 | ACPI_FADT_SEPARATE_LENGTH}, | ||
| 102 | 124 | ||
| 103 | {"Gpe1Block", ACPI_FADT_OFFSET(xgpe1_block), | 125 | {"Gpe1Block", |
| 126 | ACPI_FADT_OFFSET(xgpe1_block), | ||
| 104 | ACPI_FADT_OFFSET(gpe1_block), | 127 | ACPI_FADT_OFFSET(gpe1_block), |
| 105 | ACPI_FADT_OFFSET(gpe1_block_length), ACPI_FADT_SEPARATE_LENGTH} | 128 | ACPI_FADT_OFFSET(gpe1_block_length), |
| 129 | 0, | ||
| 130 | ACPI_FADT_SEPARATE_LENGTH} | ||
| 106 | }; | 131 | }; |
| 107 | 132 | ||
| 108 | #define ACPI_FADT_INFO_ENTRIES (sizeof (fadt_info_table) / sizeof (struct acpi_fadt_info)) | 133 | #define ACPI_FADT_INFO_ENTRIES (sizeof (fadt_info_table) / sizeof (struct acpi_fadt_info)) |
| @@ -279,8 +304,9 @@ void acpi_tb_create_local_fadt(struct acpi_table_header *table, u32 length) | |||
| 279 | 304 | ||
| 280 | static void acpi_tb_convert_fadt(void) | 305 | static void acpi_tb_convert_fadt(void) |
| 281 | { | 306 | { |
| 282 | u8 pm1_register_length; | 307 | u8 pm1_register_bit_width; |
| 283 | struct acpi_generic_address *target; | 308 | u8 pm1_register_byte_width; |
| 309 | struct acpi_generic_address *target64; | ||
| 284 | u32 i; | 310 | u32 i; |
| 285 | 311 | ||
| 286 | /* Update the local FADT table header length */ | 312 | /* Update the local FADT table header length */ |
| @@ -326,21 +352,22 @@ static void acpi_tb_convert_fadt(void) | |||
| 326 | } | 352 | } |
| 327 | 353 | ||
| 328 | /* | 354 | /* |
| 329 | * Expand the ACPI 1.0 32-bit V1.0 addresses to the ACPI 2.0 64-bit "X" | 355 | * Expand the ACPI 1.0 32-bit addresses to the ACPI 2.0 64-bit "X" |
| 330 | * generic address structures as necessary. | 356 | * generic address structures as necessary. Later code will always use |
| 357 | * the 64-bit address structures. | ||
| 331 | */ | 358 | */ |
| 332 | for (i = 0; i < ACPI_FADT_INFO_ENTRIES; i++) { | 359 | for (i = 0; i < ACPI_FADT_INFO_ENTRIES; i++) { |
| 333 | target = | 360 | target64 = |
| 334 | ACPI_ADD_PTR(struct acpi_generic_address, &acpi_gbl_FADT, | 361 | ACPI_ADD_PTR(struct acpi_generic_address, &acpi_gbl_FADT, |
| 335 | fadt_info_table[i].target); | 362 | fadt_info_table[i].address64); |
| 336 | 363 | ||
| 337 | /* Expand only if the X target is null */ | 364 | /* Expand only if the 64-bit X target is null */ |
| 338 | 365 | ||
| 339 | if (!target->address) { | 366 | if (!target64->address) { |
| 340 | 367 | ||
| 341 | /* The space_id is always I/O for the legacy address fields */ | 368 | /* The space_id is always I/O for the 32-bit legacy address fields */ |
| 342 | 369 | ||
| 343 | acpi_tb_init_generic_address(target, | 370 | acpi_tb_init_generic_address(target64, |
| 344 | ACPI_ADR_SPACE_SYSTEM_IO, | 371 | ACPI_ADR_SPACE_SYSTEM_IO, |
| 345 | *ACPI_ADD_PTR(u8, | 372 | *ACPI_ADD_PTR(u8, |
| 346 | &acpi_gbl_FADT, | 373 | &acpi_gbl_FADT, |
| @@ -350,7 +377,7 @@ static void acpi_tb_convert_fadt(void) | |||
| 350 | &acpi_gbl_FADT, | 377 | &acpi_gbl_FADT, |
| 351 | fadt_info_table | 378 | fadt_info_table |
| 352 | [i]. | 379 | [i]. |
| 353 | source)); | 380 | address32)); |
| 354 | } | 381 | } |
| 355 | } | 382 | } |
| 356 | 383 | ||
| @@ -359,19 +386,53 @@ static void acpi_tb_convert_fadt(void) | |||
| 359 | acpi_tb_validate_fadt(); | 386 | acpi_tb_validate_fadt(); |
| 360 | 387 | ||
| 361 | /* | 388 | /* |
| 362 | * Get the length of the individual PM1 registers. Each register is | 389 | * Optionally check all register lengths against the default values and |
| 363 | * defined to be the event block length / 2. | 390 | * update them if they are incorrect. |
| 364 | */ | 391 | */ |
| 365 | pm1_register_length = (u8)ACPI_DIV_2(acpi_gbl_FADT.pm1_event_length); | 392 | if (acpi_gbl_use_default_register_widths) { |
| 393 | for (i = 0; i < ACPI_FADT_INFO_ENTRIES; i++) { | ||
| 394 | target64 = | ||
| 395 | ACPI_ADD_PTR(struct acpi_generic_address, | ||
| 396 | &acpi_gbl_FADT, | ||
| 397 | fadt_info_table[i].address64); | ||
| 398 | |||
| 399 | /* | ||
| 400 | * If a valid register (Address != 0) and the (default_length > 0) | ||
| 401 | * (Not a GPE register), then check the width against the default. | ||
| 402 | */ | ||
| 403 | if ((target64->address) && | ||
| 404 | (fadt_info_table[i].default_length > 0) && | ||
| 405 | (fadt_info_table[i].default_length != | ||
| 406 | target64->bit_width)) { | ||
| 407 | ACPI_WARNING((AE_INFO, | ||
| 408 | "Invalid length for %s: %d, using default %d", | ||
| 409 | fadt_info_table[i].name, | ||
| 410 | target64->bit_width, | ||
| 411 | fadt_info_table[i]. | ||
| 412 | default_length)); | ||
| 413 | |||
| 414 | /* Incorrect size, set width to the default */ | ||
| 415 | |||
| 416 | target64->bit_width = | ||
| 417 | fadt_info_table[i].default_length; | ||
| 418 | } | ||
| 419 | } | ||
| 420 | } | ||
| 421 | |||
| 422 | /* | ||
| 423 | * Get the length of the individual PM1 registers (enable and status). | ||
| 424 | * Each register is defined to be (event block length / 2). | ||
| 425 | */ | ||
| 426 | pm1_register_bit_width = | ||
| 427 | (u8)ACPI_DIV_2(acpi_gbl_FADT.xpm1a_event_block.bit_width); | ||
| 428 | pm1_register_byte_width = (u8)ACPI_DIV_8(pm1_register_bit_width); | ||
| 366 | 429 | ||
| 367 | /* | 430 | /* |
| 368 | * Adjust the lengths of the PM1 Event Blocks so that they can be used to | 431 | * Adjust the lengths of the PM1 Event Blocks so that they can be used to |
| 369 | * access the PM1 status register(s). | 432 | * access the PM1 status register(s). Use (width / 2) |
| 370 | */ | 433 | */ |
| 371 | acpi_gbl_FADT.xpm1a_event_block.bit_width = | 434 | acpi_gbl_FADT.xpm1a_event_block.bit_width = pm1_register_bit_width; |
| 372 | (u8)ACPI_MUL_8(pm1_register_length); | 435 | acpi_gbl_FADT.xpm1b_event_block.bit_width = pm1_register_bit_width; |
| 373 | acpi_gbl_FADT.xpm1b_event_block.bit_width = | ||
| 374 | (u8)ACPI_MUL_8(pm1_register_length); | ||
| 375 | 436 | ||
| 376 | /* | 437 | /* |
| 377 | * Calculate separate GAS structs for the PM1 Enable registers. | 438 | * Calculate separate GAS structs for the PM1 Enable registers. |
| @@ -398,9 +459,9 @@ static void acpi_tb_convert_fadt(void) | |||
| 398 | 459 | ||
| 399 | acpi_tb_init_generic_address(&acpi_gbl_xpm1a_enable, | 460 | acpi_tb_init_generic_address(&acpi_gbl_xpm1a_enable, |
| 400 | acpi_gbl_FADT.xpm1a_event_block.space_id, | 461 | acpi_gbl_FADT.xpm1a_event_block.space_id, |
| 401 | pm1_register_length, | 462 | pm1_register_byte_width, |
| 402 | (acpi_gbl_FADT.xpm1a_event_block.address + | 463 | (acpi_gbl_FADT.xpm1a_event_block.address + |
| 403 | pm1_register_length)); | 464 | pm1_register_byte_width)); |
| 404 | /* Don't forget to copy space_id of the GAS */ | 465 | /* Don't forget to copy space_id of the GAS */ |
| 405 | acpi_gbl_xpm1a_enable.space_id = | 466 | acpi_gbl_xpm1a_enable.space_id = |
| 406 | acpi_gbl_FADT.xpm1a_event_block.space_id; | 467 | acpi_gbl_FADT.xpm1a_event_block.space_id; |
| @@ -417,38 +478,14 @@ static void acpi_tb_convert_fadt(void) | |||
| 417 | acpi_gbl_FADT.pm1_event_length); | 478 | acpi_gbl_FADT.pm1_event_length); |
| 418 | acpi_tb_init_generic_address(&acpi_gbl_xpm1b_enable, | 479 | acpi_tb_init_generic_address(&acpi_gbl_xpm1b_enable, |
| 419 | acpi_gbl_FADT.xpm1b_event_block.space_id, | 480 | acpi_gbl_FADT.xpm1b_event_block.space_id, |
| 420 | pm1_register_length, | 481 | pm1_register_byte_width, |
| 421 | (acpi_gbl_FADT.xpm1b_event_block. | 482 | (acpi_gbl_FADT.xpm1b_event_block. |
| 422 | address + pm1_register_length)); | 483 | address + pm1_register_byte_width)); |
| 423 | /* Don't forget to copy space_id of the GAS */ | 484 | /* Don't forget to copy space_id of the GAS */ |
| 424 | acpi_gbl_xpm1b_enable.space_id = | 485 | acpi_gbl_xpm1b_enable.space_id = |
| 425 | acpi_gbl_FADT.xpm1b_event_block.space_id; | 486 | acpi_gbl_FADT.xpm1b_event_block.space_id; |
| 426 | 487 | ||
| 427 | } | 488 | } |
| 428 | |||
| 429 | if (acpi_gbl_use_default_register_widths) { | ||
| 430 | /* | ||
| 431 | * Optionally, use the default sizes for the ACPI registers. | ||
| 432 | * Some FADTs do not have the correct length(s). | ||
| 433 | * | ||
| 434 | * Note: Xpm1a_event_block and Xpm1b_event_block are used to access the PM1 | ||
| 435 | * status registers. The PM1 enable registers are created above. | ||
| 436 | */ | ||
| 437 | acpi_gbl_xpm1a_enable.bit_width = ACPI_PM1_REGISTER_WIDTH; | ||
| 438 | acpi_gbl_xpm1b_enable.bit_width = ACPI_PM1_REGISTER_WIDTH; | ||
| 439 | |||
| 440 | acpi_gbl_FADT.xpm1a_event_block.bit_width = | ||
| 441 | ACPI_PM1_REGISTER_WIDTH; | ||
| 442 | acpi_gbl_FADT.xpm1b_event_block.bit_width = | ||
| 443 | ACPI_PM1_REGISTER_WIDTH; | ||
| 444 | acpi_gbl_FADT.xpm1a_control_block.bit_width = | ||
| 445 | ACPI_PM1_REGISTER_WIDTH; | ||
| 446 | acpi_gbl_FADT.xpm1b_control_block.bit_width = | ||
| 447 | ACPI_PM1_REGISTER_WIDTH; | ||
| 448 | acpi_gbl_FADT.xpm2_control_block.bit_width = | ||
| 449 | ACPI_PM2_REGISTER_WIDTH; | ||
| 450 | acpi_gbl_FADT.xpm_timer_block.bit_width = ACPI_PM_TIMER_WIDTH; | ||
| 451 | } | ||
| 452 | } | 489 | } |
| 453 | 490 | ||
| 454 | /****************************************************************************** | 491 | /****************************************************************************** |
| @@ -511,10 +548,10 @@ static void acpi_tb_validate_fadt(void) | |||
| 511 | */ | 548 | */ |
| 512 | address64 = ACPI_ADD_PTR(struct acpi_generic_address, | 549 | address64 = ACPI_ADD_PTR(struct acpi_generic_address, |
| 513 | &acpi_gbl_FADT, | 550 | &acpi_gbl_FADT, |
| 514 | fadt_info_table[i].target); | 551 | fadt_info_table[i].address64); |
| 515 | address32 = | 552 | address32 = |
| 516 | ACPI_ADD_PTR(u32, &acpi_gbl_FADT, | 553 | ACPI_ADD_PTR(u32, &acpi_gbl_FADT, |
| 517 | fadt_info_table[i].source); | 554 | fadt_info_table[i].address32); |
| 518 | length = | 555 | length = |
| 519 | *ACPI_ADD_PTR(u8, &acpi_gbl_FADT, | 556 | *ACPI_ADD_PTR(u8, &acpi_gbl_FADT, |
| 520 | fadt_info_table[i].length); | 557 | fadt_info_table[i].length); |
| @@ -522,7 +559,7 @@ static void acpi_tb_validate_fadt(void) | |||
| 522 | 559 | ||
| 523 | /* | 560 | /* |
| 524 | * For each extended field, check for length mismatch between the | 561 | * For each extended field, check for length mismatch between the |
| 525 | * legacy length field and the corresonding 64-bit X length field. | 562 | * legacy length field and the corresponding 64-bit X length field. |
| 526 | */ | 563 | */ |
| 527 | if (address64 && (address64->bit_width != ACPI_MUL_8(length))) { | 564 | if (address64 && (address64->bit_width != ACPI_MUL_8(length))) { |
| 528 | ACPI_WARNING((AE_INFO, | 565 | ACPI_WARNING((AE_INFO, |
