Skip to content

Commit 1ab69f2

Browse files
Lv Zhengrafaeljw
authored andcommitted
ACPI: EC: Fix an EC event IRQ storming issue
The EC event IRQ (SCI_EVT) can only be handled by submitting QR_EC. As the EC driver handles SCI_EVT in a workqueue, after SCI_EVT is flagged and before QR_EC is submitted, there is a period risking IRQ storming. EC IRQ must be masked for this period but linux EC driver never does so. No end user notices the IRQ storming and no developer fixes this known issue because: 1. The EC IRQ is always edge triggered GPE, and 2. The kernel can execute no-op EC IRQ handler very fast. For edge-triggered EC GPE platforms, it is only reported of post-resume EC event lost issues, there won't be an IRQ storming. For level triggered EC GPE platforms, fortunately the kernel is always fast enough to execute such a no-op EC IRQ handler so that the IRQ handler won't be accumulated to starve the task contexts, causing a real IRQ storming. But the IRQ storming actually can still happen when: 1. The EC IRQ performs like level triggered GPE, and 2. The kernel EC debugging log is turned on but the console is slow enough. There are more and more platforms using EC GPE as wake GPE where the EC GPE is likely designed as level triggered. Then when EC debugging log is enabled, the EC IRQ handler is no longer a no-op but dumps IRQ status to the consoles. If the consoles are slow enough, the EC IRQs can arrive much faster than executing the handler. Finally the accumulated EC event IRQ handlers starve the task contexts, causing the IRQ storming to occur, and the kernel hangs can be observed during boot/resume. This patch fixes this issue by masking EC IRQ for this period: 1. Begins when there is an SCI_EVT IRQ pending, and 2. Ends when there is a QR_EC completed (SCI_EVT acknowledged). Tested-by: Wang Wendy <[email protected]> Tested-by: Feng Chenzhou <[email protected]> Signed-off-by: Lv Zheng <[email protected]> Signed-off-by: Rafael J. Wysocki <[email protected]>
1 parent c0bc126 commit 1ab69f2

File tree

1 file changed

+6
-5
lines changed

1 file changed

+6
-5
lines changed

drivers/acpi/ec.c

Lines changed: 6 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -459,8 +459,10 @@ static bool acpi_ec_submit_flushable_request(struct acpi_ec *ec)
459459

460460
static void acpi_ec_submit_query(struct acpi_ec *ec)
461461
{
462-
if (acpi_ec_event_enabled(ec) &&
463-
!test_and_set_bit(EC_FLAGS_QUERY_PENDING, &ec->flags)) {
462+
acpi_ec_set_storm(ec, EC_FLAGS_COMMAND_STORM);
463+
if (!acpi_ec_event_enabled(ec))
464+
return;
465+
if (!test_and_set_bit(EC_FLAGS_QUERY_PENDING, &ec->flags)) {
464466
ec_dbg_evt("Command(%s) submitted/blocked",
465467
acpi_ec_cmd_string(ACPI_EC_COMMAND_QUERY));
466468
ec->nr_pending_queries++;
@@ -470,11 +472,10 @@ static void acpi_ec_submit_query(struct acpi_ec *ec)
470472

471473
static void acpi_ec_complete_query(struct acpi_ec *ec)
472474
{
473-
if (test_bit(EC_FLAGS_QUERY_PENDING, &ec->flags)) {
474-
clear_bit(EC_FLAGS_QUERY_PENDING, &ec->flags);
475+
if (test_and_clear_bit(EC_FLAGS_QUERY_PENDING, &ec->flags))
475476
ec_dbg_evt("Command(%s) unblocked",
476477
acpi_ec_cmd_string(ACPI_EC_COMMAND_QUERY));
477-
}
478+
acpi_ec_clear_storm(ec, EC_FLAGS_COMMAND_STORM);
478479
}
479480

480481
static inline void __acpi_ec_enable_event(struct acpi_ec *ec)

0 commit comments

Comments
 (0)