diff options
author | Alexey Starikovskiy <astarikovskiy@suse.de> | 2010-04-09 21:18:35 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@suse.de> | 2010-04-26 10:48:02 -0400 |
commit | 3064d4245eecba89cc2eb2ce3cf89fd5ffe060de (patch) | |
tree | f06d8248de08c12057d01cdead9f02ed279b76ee | |
parent | d4187adfe63696f0abc60d32320271f1ec7e8110 (diff) |
ACPI: EC: Allow multibyte access to EC
commit dadf28a10c3eb29421837a2e413ab869ebd upstream
http://bugzilla.kernel.org/show_bug.cgi?id=14667
[bwh: Backport to 2.6.32; same applies to 2.6.33]
Signed-off-by: Alexey Starikovskiy <astarikovskiy@suse.de>
Signed-off-by: Len Brown <len.brown@intel.com>
Cc: Ben Hutchings <ben@decadent.org.uk>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
-rw-r--r-- | drivers/acpi/acpica/exprep.c | 12 | ||||
-rw-r--r-- | drivers/acpi/ec.c | 35 |
2 files changed, 21 insertions, 26 deletions
diff --git a/drivers/acpi/acpica/exprep.c b/drivers/acpi/acpica/exprep.c index 52fec07064f0..c02d543a2601 100644 --- a/drivers/acpi/acpica/exprep.c +++ b/drivers/acpi/acpica/exprep.c | |||
@@ -468,6 +468,18 @@ acpi_status acpi_ex_prep_field_value(struct acpi_create_field_info *info) | |||
468 | 468 | ||
469 | acpi_ut_add_reference(obj_desc->field.region_obj); | 469 | acpi_ut_add_reference(obj_desc->field.region_obj); |
470 | 470 | ||
471 | /* allow full data read from EC address space */ | ||
472 | if (obj_desc->field.region_obj->region.space_id == | ||
473 | ACPI_ADR_SPACE_EC) { | ||
474 | if (obj_desc->common_field.bit_length > 8) | ||
475 | obj_desc->common_field.access_bit_width = | ||
476 | ACPI_ROUND_UP(obj_desc->common_field. | ||
477 | bit_length, 8); | ||
478 | obj_desc->common_field.access_byte_width = | ||
479 | ACPI_DIV_8(obj_desc->common_field. | ||
480 | access_bit_width); | ||
481 | } | ||
482 | |||
471 | ACPI_DEBUG_PRINT((ACPI_DB_BFIELD, | 483 | ACPI_DEBUG_PRINT((ACPI_DB_BFIELD, |
472 | "RegionField: BitOff %X, Off %X, Gran %X, Region %p\n", | 484 | "RegionField: BitOff %X, Off %X, Gran %X, Region %p\n", |
473 | obj_desc->field.start_field_bit_offset, | 485 | obj_desc->field.start_field_bit_offset, |
diff --git a/drivers/acpi/ec.c b/drivers/acpi/ec.c index d6471bb6852f..fc67d1175a56 100644 --- a/drivers/acpi/ec.c +++ b/drivers/acpi/ec.c | |||
@@ -589,12 +589,12 @@ static u32 acpi_ec_gpe_handler(void *data) | |||
589 | 589 | ||
590 | static acpi_status | 590 | static acpi_status |
591 | acpi_ec_space_handler(u32 function, acpi_physical_address address, | 591 | acpi_ec_space_handler(u32 function, acpi_physical_address address, |
592 | u32 bits, acpi_integer *value, | 592 | u32 bits, acpi_integer *value64, |
593 | void *handler_context, void *region_context) | 593 | void *handler_context, void *region_context) |
594 | { | 594 | { |
595 | struct acpi_ec *ec = handler_context; | 595 | struct acpi_ec *ec = handler_context; |
596 | int result = 0, i; | 596 | int result = 0, i, bytes = bits / 8; |
597 | u8 temp = 0; | 597 | u8 *value = (u8 *)value64; |
598 | 598 | ||
599 | if ((address > 0xFF) || !value || !handler_context) | 599 | if ((address > 0xFF) || !value || !handler_context) |
600 | return AE_BAD_PARAMETER; | 600 | return AE_BAD_PARAMETER; |
@@ -602,32 +602,15 @@ acpi_ec_space_handler(u32 function, acpi_physical_address address, | |||
602 | if (function != ACPI_READ && function != ACPI_WRITE) | 602 | if (function != ACPI_READ && function != ACPI_WRITE) |
603 | return AE_BAD_PARAMETER; | 603 | return AE_BAD_PARAMETER; |
604 | 604 | ||
605 | if (bits != 8 && acpi_strict) | 605 | if (EC_FLAGS_MSI || bits > 8) |
606 | return AE_BAD_PARAMETER; | ||
607 | |||
608 | if (EC_FLAGS_MSI) | ||
609 | acpi_ec_burst_enable(ec); | 606 | acpi_ec_burst_enable(ec); |
610 | 607 | ||
611 | if (function == ACPI_READ) { | 608 | for (i = 0; i < bytes; ++i, ++address, ++value) |
612 | result = acpi_ec_read(ec, address, &temp); | 609 | result = (function == ACPI_READ) ? |
613 | *value = temp; | 610 | acpi_ec_read(ec, address, value) : |
614 | } else { | 611 | acpi_ec_write(ec, address, *value); |
615 | temp = 0xff & (*value); | ||
616 | result = acpi_ec_write(ec, address, temp); | ||
617 | } | ||
618 | |||
619 | for (i = 8; unlikely(bits - i > 0); i += 8) { | ||
620 | ++address; | ||
621 | if (function == ACPI_READ) { | ||
622 | result = acpi_ec_read(ec, address, &temp); | ||
623 | (*value) |= ((acpi_integer)temp) << i; | ||
624 | } else { | ||
625 | temp = 0xff & ((*value) >> i); | ||
626 | result = acpi_ec_write(ec, address, temp); | ||
627 | } | ||
628 | } | ||
629 | 612 | ||
630 | if (EC_FLAGS_MSI) | 613 | if (EC_FLAGS_MSI || bits > 8) |
631 | acpi_ec_burst_disable(ec); | 614 | acpi_ec_burst_disable(ec); |
632 | 615 | ||
633 | switch (result) { | 616 | switch (result) { |