Commit 164a08ce authored by Lv Zheng's avatar Lv Zheng Committed by Rafael J. Wysocki

ACPICA: Dispatcher: Introduce timeout mechanism for infinite loop detection

ACPICA commit 9605023e7e6d1f05581502766c8cf2905bcc03d9

This patch implements a new infinite loop detection mechanism to replace
the old one, it uses acpi_os_get_timer() to limit loop execution into a
determined time slice.
This is useful in case some hardware/firmware operations really require the
AML interpreter to wait while the old mechanism could expire too fast on
recent machines.

The new mechanism converts old acpi_gbl_max_loop_iterations to store the user
configurable value for the new mechanism in order to allow users to be
still able to configure this value for acpiexec via command line. This
patch also removes wrong initilization code of acpi_gbl_max_loop_iterations
accordingly (it should have been initialized by ACPI_INIT_GLOBAL, and the
default value is also properly tuned for acpiexec). Reported by M. Foronda,
fixed by Lv Zheng.

Link: https://github.com/acpica/acpica/commit/9605023e
Link: https://bugzilla.kernel.org/show_bug.cgi?id=156501Reported-by: default avatarM. Foronda <josemauricioforonda@gmail.com>
Signed-off-by: default avatarLv Zheng <lv.zheng@intel.com>
Signed-off-by: default avatarBob Moore <robert.moore@intel.com>
Signed-off-by: default avatarErik Schmauss <erik.schmauss@intel.com>
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
parent 29ad1f88
...@@ -622,7 +622,7 @@ struct acpi_control_state { ...@@ -622,7 +622,7 @@ struct acpi_control_state {
union acpi_parse_object *predicate_op; union acpi_parse_object *predicate_op;
u8 *aml_predicate_start; /* Start of if/while predicate */ u8 *aml_predicate_start; /* Start of if/while predicate */
u8 *package_end; /* End of if/while block */ u8 *package_end; /* End of if/while block */
u32 loop_count; /* While() loop counter */ u64 loop_timeout; /* While() loop timeout */
}; };
/* /*
......
...@@ -118,6 +118,8 @@ acpi_ds_exec_begin_control_op(struct acpi_walk_state *walk_state, ...@@ -118,6 +118,8 @@ acpi_ds_exec_begin_control_op(struct acpi_walk_state *walk_state,
control_state->control.package_end = control_state->control.package_end =
walk_state->parser_state.pkg_end; walk_state->parser_state.pkg_end;
control_state->control.opcode = op->common.aml_opcode; control_state->control.opcode = op->common.aml_opcode;
control_state->control.loop_timeout = acpi_os_get_timer() +
(u64)(acpi_gbl_max_loop_iterations * ACPI_100NSEC_PER_SEC);
/* Push the control state on this walk's control stack */ /* Push the control state on this walk's control stack */
...@@ -206,14 +208,14 @@ acpi_ds_exec_end_control_op(struct acpi_walk_state *walk_state, ...@@ -206,14 +208,14 @@ acpi_ds_exec_end_control_op(struct acpi_walk_state *walk_state,
/* Predicate was true, the body of the loop was just executed */ /* Predicate was true, the body of the loop was just executed */
/* /*
* This loop counter mechanism allows the interpreter to escape * This infinite loop detection mechanism allows the interpreter
* possibly infinite loops. This can occur in poorly written AML * to escape possibly infinite loops. This can occur in poorly
* when the hardware does not respond within a while loop and the * written AML when the hardware does not respond within a while
* loop does not implement a timeout. * loop and the loop does not implement a timeout.
*/ */
control_state->control.loop_count++; if (ACPI_TIME_AFTER(acpi_os_get_timer(),
if (control_state->control.loop_count > control_state->control.
acpi_gbl_max_loop_iterations) { loop_timeout)) {
status = AE_AML_INFINITE_LOOP; status = AE_AML_INFINITE_LOOP;
break; break;
} }
......
...@@ -206,7 +206,6 @@ acpi_status acpi_ut_init_globals(void) ...@@ -206,7 +206,6 @@ acpi_status acpi_ut_init_globals(void)
acpi_gbl_next_owner_id_offset = 0; acpi_gbl_next_owner_id_offset = 0;
acpi_gbl_debugger_configuration = DEBUGGER_THREADING; acpi_gbl_debugger_configuration = DEBUGGER_THREADING;
acpi_gbl_osi_mutex = NULL; acpi_gbl_osi_mutex = NULL;
acpi_gbl_max_loop_iterations = ACPI_MAX_LOOP_COUNT;
/* Hardware oriented */ /* Hardware oriented */
......
...@@ -145,9 +145,9 @@ ...@@ -145,9 +145,9 @@
#define ACPI_ADDRESS_RANGE_MAX 2 #define ACPI_ADDRESS_RANGE_MAX 2
/* Maximum number of While() loops before abort */ /* Maximum time (default 30s) of While() loops before abort */
#define ACPI_MAX_LOOP_COUNT 0x000FFFFF #define ACPI_MAX_LOOP_TIMEOUT 30
/****************************************************************************** /******************************************************************************
* *
......
...@@ -260,11 +260,11 @@ ACPI_INIT_GLOBAL(u8, acpi_gbl_osi_data, 0); ...@@ -260,11 +260,11 @@ ACPI_INIT_GLOBAL(u8, acpi_gbl_osi_data, 0);
ACPI_INIT_GLOBAL(u8, acpi_gbl_reduced_hardware, FALSE); ACPI_INIT_GLOBAL(u8, acpi_gbl_reduced_hardware, FALSE);
/* /*
* Maximum number of While() loop iterations before forced method abort. * Maximum timeout for While() loop iterations before forced method abort.
* This mechanism is intended to prevent infinite loops during interpreter * This mechanism is intended to prevent infinite loops during interpreter
* execution within a host kernel. * execution within a host kernel.
*/ */
ACPI_INIT_GLOBAL(u32, acpi_gbl_max_loop_iterations, ACPI_MAX_LOOP_COUNT); ACPI_INIT_GLOBAL(u32, acpi_gbl_max_loop_iterations, ACPI_MAX_LOOP_TIMEOUT);
/* /*
* This mechanism is used to trace a specified AML method. The method is * This mechanism is used to trace a specified AML method. The method is
......
...@@ -468,6 +468,8 @@ typedef void *acpi_handle; /* Actually a ptr to a NS Node */ ...@@ -468,6 +468,8 @@ typedef void *acpi_handle; /* Actually a ptr to a NS Node */
#define ACPI_NSEC_PER_MSEC 1000000L #define ACPI_NSEC_PER_MSEC 1000000L
#define ACPI_NSEC_PER_SEC 1000000000L #define ACPI_NSEC_PER_SEC 1000000000L
#define ACPI_TIME_AFTER(a, b) ((s64)((b) - (a)) < 0)
/* Owner IDs are used to track namespace nodes for selective deletion */ /* Owner IDs are used to track namespace nodes for selective deletion */
typedef u8 acpi_owner_id; typedef u8 acpi_owner_id;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment