2018-03-15 07:13:07 +08:00
|
|
|
/* SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0 */
|
2005-04-17 06:20:36 +08:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
2006-03-18 05:44:00 +08:00
|
|
|
* Name: acobject.h - Definition of union acpi_operand_object (Internal object only)
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
2021-01-16 02:48:25 +08:00
|
|
|
* Copyright (C) 2000 - 2021, Intel Corp.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
2018-03-15 07:13:07 +08:00
|
|
|
*****************************************************************************/
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
#ifndef _ACOBJECT_H
|
|
|
|
#define _ACOBJECT_H
|
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
/* acpisrc:struct_defs -- for acpisrc conversion */
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
2006-03-18 05:44:00 +08:00
|
|
|
* The union acpi_operand_object is used to pass AML operands from the dispatcher
|
2005-04-17 06:20:36 +08:00
|
|
|
* to the interpreter, and to keep track of the various handlers such as
|
2006-03-18 05:44:00 +08:00
|
|
|
* address space handlers and notify handlers. The object is a constant
|
2005-04-17 06:20:36 +08:00
|
|
|
* size in order to allow it to be cached and reused.
|
2007-02-03 00:48:23 +08:00
|
|
|
*
|
|
|
|
* Note: The object is optimized to be aligned and will not work if it is
|
|
|
|
* byte-packed.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2007-02-03 00:48:23 +08:00
|
|
|
#if ACPI_MACHINE_WIDTH == 64
|
|
|
|
#pragma pack(8)
|
|
|
|
#else
|
|
|
|
#pragma pack(4)
|
|
|
|
#endif
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
*
|
|
|
|
* Common Descriptors
|
|
|
|
*
|
|
|
|
******************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Common area for all objects.
|
|
|
|
*
|
2006-03-18 05:44:00 +08:00
|
|
|
* descriptor_type is used to differentiate between internal descriptors, and
|
|
|
|
* must be in the same place across all descriptors
|
ACPI: ACPICA 20060331
Implemented header file support for the following
additional ACPI tables: ASF!, BOOT, CPEP, DBGP, MCFG, SPCR,
SPMI, TCPA, and WDRT. With this support, all current and
known ACPI tables are now defined in the ACPICA headers and
are available for use by device drivers and other software.
Implemented support to allow tables that contain ACPI
names with invalid characters to be loaded. Previously,
this would cause the table load to fail, but since
there are several known cases of such tables on
existing machines, this change was made to enable
ACPI support for them. Also, this matches the
behavior of the Microsoft ACPI implementation.
https://bugzilla.novell.com/show_bug.cgi?id=147621
Fixed a couple regressions introduced during the memory
optimization in the 20060317 release. The namespace
node definition required additional reorganization and
an internal datatype that had been changed to 8-bit was
restored to 32-bit. (Valery Podrezov)
Fixed a problem where a null pointer passed to
acpi_ut_delete_generic_state() could be passed through
to acpi_os_release_object which is unexpected. Such
null pointers are now trapped and ignored, matching
the behavior of the previous implementation before the
deployment of acpi_os_release_object(). (Valery Podrezov,
Fiodor Suietov)
Fixed a memory mapping leak during the deletion of
a SystemMemory operation region where a cached memory
mapping was not deleted. This became a noticeable problem
for operation regions that are defined within frequently
used control methods. (Dana Meyers)
Reorganized the ACPI table header files into two main
files: one for the ACPI tables consumed by the ACPICA core,
and another for the miscellaneous ACPI tables that are
consumed by the drivers and other software. The various
FADT definitions were merged into one common section and
three different tables (ACPI 1.0, 1.0+, and 2.0)
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-03-31 13:00:00 +08:00
|
|
|
*
|
|
|
|
* Note: The descriptor_type and Type fields must appear in the identical
|
|
|
|
* position in both the struct acpi_namespace_node and union acpi_operand_object
|
|
|
|
* structures.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2006-03-18 05:44:00 +08:00
|
|
|
#define ACPI_OBJECT_COMMON_HEADER \
|
|
|
|
union acpi_operand_object *next_object; /* Objects linked to parent NS node */\
|
|
|
|
u8 descriptor_type; /* To differentiate various internal objs */\
|
|
|
|
u8 type; /* acpi_object_type */\
|
|
|
|
u16 reference_count; /* For object deletion management */\
|
|
|
|
u8 flags;
|
|
|
|
/*
|
|
|
|
* Note: There are 3 bytes available here before the
|
|
|
|
* next natural alignment boundary (for both 32/64 cases)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Values for Flag byte above */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2010-05-27 09:27:34 +08:00
|
|
|
#define AOPOBJ_AML_CONSTANT 0x01 /* Integer is an AML constant */
|
|
|
|
#define AOPOBJ_STATIC_POINTER 0x02 /* Data is part of an ACPI table, don't delete */
|
tree-wide: fix comment/printk typos
"gadget", "through", "command", "maintain", "maintain", "controller", "address",
"between", "initiali[zs]e", "instead", "function", "select", "already",
"equal", "access", "management", "hierarchy", "registration", "interest",
"relative", "memory", "offset", "already",
Signed-off-by: Uwe Kleine-König <u.kleine-koenig@pengutronix.de>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
2010-11-02 03:38:34 +08:00
|
|
|
#define AOPOBJ_DATA_VALID 0x04 /* Object is initialized and data is valid */
|
2015-12-29 14:03:08 +08:00
|
|
|
#define AOPOBJ_OBJECT_INITIALIZED 0x08 /* Region is initialized */
|
|
|
|
#define AOPOBJ_REG_CONNECTED 0x10 /* _REG was run */
|
|
|
|
#define AOPOBJ_SETUP_COMPLETE 0x20 /* Region setup is complete */
|
|
|
|
#define AOPOBJ_INVALID 0x40 /* Host OS won't allow a Region address */
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Basic data types
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_common {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER};
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_integer {
|
2007-02-03 00:48:23 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 fill[3]; /* Prevent warning on some compilers */
|
2010-01-21 10:06:32 +08:00
|
|
|
u64 value;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
2012-07-16 09:52:27 +08:00
|
|
|
* Note: The String and Buffer object must be identical through the
|
|
|
|
* pointer and length elements. There is code that depends on this.
|
2006-03-18 05:44:00 +08:00
|
|
|
*
|
|
|
|
* Fields common to both Strings and Buffers
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2006-03-18 05:44:00 +08:00
|
|
|
#define ACPI_COMMON_BUFFER_INFO(_type) \
|
|
|
|
_type *pointer; \
|
|
|
|
u32 length;
|
|
|
|
|
2017-07-10 15:23:15 +08:00
|
|
|
/* Null terminated, ASCII characters only */
|
|
|
|
|
|
|
|
struct acpi_object_string {
|
2006-03-18 05:44:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_BUFFER_INFO(char) /* String in AML stream or allocated string */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_buffer {
|
2006-03-18 05:44:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_BUFFER_INFO(u8) /* Buffer in AML stream or allocated buffer */
|
2005-08-05 12:44:28 +08:00
|
|
|
u32 aml_length;
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 *aml_start;
|
|
|
|
struct acpi_namespace_node *node; /* Link back to parent node */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_package {
|
2006-03-18 05:44:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER struct acpi_namespace_node *node; /* Link back to parent node */
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object **elements; /* Array of pointers to acpi_objects */
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 *aml_start;
|
|
|
|
u32 aml_length;
|
|
|
|
u32 count; /* # of elements in package */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Complex data types
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_event {
|
2006-06-24 05:04:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER acpi_semaphore os_semaphore; /* Actual OS synchronization object */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_mutex {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 sync_level; /* 0-15, specified in Mutex() call */
|
|
|
|
u16 acquisition_depth; /* Allow multiple Acquires, same thread */
|
2006-06-24 05:04:00 +08:00
|
|
|
acpi_mutex os_mutex; /* Actual OS synchronization object */
|
2008-04-10 23:06:37 +08:00
|
|
|
acpi_thread_id thread_id; /* Current owner of the mutex */
|
|
|
|
struct acpi_thread_state *owner_thread; /* Current owner of the mutex */
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object *prev; /* Link for list of acquired mutexes */
|
|
|
|
union acpi_operand_object *next; /* Link for list of acquired mutexes */
|
|
|
|
struct acpi_namespace_node *node; /* Containing namespace node */
|
|
|
|
u8 original_sync_level; /* Owner's original sync level (0-15) */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_region {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 space_id;
|
|
|
|
struct acpi_namespace_node *node; /* Containing namespace node */
|
2006-03-18 05:44:00 +08:00
|
|
|
union acpi_operand_object *handler; /* Handler for region access */
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object *next;
|
|
|
|
acpi_physical_address address;
|
2006-03-18 05:44:00 +08:00
|
|
|
u32 length;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct acpi_object_method {
|
2011-01-12 09:19:43 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 info_flags;
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 param_count;
|
2006-06-24 05:04:00 +08:00
|
|
|
u8 sync_level;
|
|
|
|
union acpi_operand_object *mutex;
|
2015-07-23 12:52:31 +08:00
|
|
|
union acpi_operand_object *node;
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 *aml_start;
|
2009-12-11 15:28:27 +08:00
|
|
|
union {
|
2012-10-31 10:25:36 +08:00
|
|
|
acpi_internal_method implementation;
|
2009-12-11 15:28:27 +08:00
|
|
|
union acpi_operand_object *handler;
|
2011-01-12 09:19:43 +08:00
|
|
|
} dispatch;
|
2009-12-11 15:28:27 +08:00
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
u32 aml_length;
|
|
|
|
acpi_owner_id owner_id;
|
2019-08-17 05:43:21 +08:00
|
|
|
u8 thread_count;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2011-01-12 09:19:43 +08:00
|
|
|
/* Flags for info_flags field above */
|
|
|
|
|
|
|
|
#define ACPI_METHOD_MODULE_LEVEL 0x01 /* Method is actually module-level code */
|
|
|
|
#define ACPI_METHOD_INTERNAL_ONLY 0x02 /* Method is implemented internally (_OSI) */
|
|
|
|
#define ACPI_METHOD_SERIALIZED 0x04 /* Method is serialized */
|
|
|
|
#define ACPI_METHOD_SERIALIZED_PENDING 0x08 /* Method is to be marked serialized */
|
2014-03-24 14:49:13 +08:00
|
|
|
#define ACPI_METHOD_IGNORE_SYNC_LEVEL 0x10 /* Method was auto-serialized at table load time */
|
|
|
|
#define ACPI_METHOD_MODIFIED_NAMESPACE 0x20 /* Method modified the namespace */
|
2011-01-12 09:19:43 +08:00
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
2012-10-31 10:26:55 +08:00
|
|
|
* Objects that can be notified. All share a common notify_info area.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
/*
|
|
|
|
* Common fields for objects that support ASL notifications
|
|
|
|
*/
|
|
|
|
#define ACPI_COMMON_NOTIFY_INFO \
|
2012-05-03 11:08:19 +08:00
|
|
|
union acpi_operand_object *notify_list[2]; /* Handlers for system/device notifies */\
|
2006-03-18 05:44:00 +08:00
|
|
|
union acpi_operand_object *handler; /* Handler for Address space */
|
|
|
|
|
2017-07-10 15:23:15 +08:00
|
|
|
/* COMMON NOTIFY for POWER, PROCESSOR, DEVICE, and THERMAL */
|
|
|
|
|
|
|
|
struct acpi_object_notify_common {
|
2005-08-05 12:44:28 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_NOTIFY_INFO};
|
2005-04-17 06:20:36 +08:00
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_device {
|
2005-04-17 06:20:36 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER
|
2005-08-05 12:44:28 +08:00
|
|
|
ACPI_COMMON_NOTIFY_INFO struct acpi_gpe_block_info *gpe_block;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_power_resource {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_NOTIFY_INFO u32 system_level;
|
|
|
|
u32 resource_order;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_processor {
|
2007-02-03 00:48:23 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER
|
|
|
|
/* The next two fields take advantage of the 3-byte space before NOTIFY_INFO */
|
|
|
|
u8 proc_id;
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 length;
|
2007-05-10 11:34:35 +08:00
|
|
|
ACPI_COMMON_NOTIFY_INFO acpi_io_address address;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_thermal_zone {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_NOTIFY_INFO};
|
2005-04-17 06:20:36 +08:00
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
2012-10-31 10:26:55 +08:00
|
|
|
* Fields. All share a common header/info field.
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
/*
|
|
|
|
* Common bitfield for the field objects
|
|
|
|
* "Field Datum" -- a datum from the actual field object
|
|
|
|
* "Buffer Datum" -- a datum from a user buffer, read from or to be written to the field
|
|
|
|
*/
|
|
|
|
#define ACPI_COMMON_FIELD_INFO \
|
|
|
|
u8 field_flags; /* Access, update, and lock bits */\
|
|
|
|
u8 attribute; /* From access_as keyword */\
|
|
|
|
u8 access_byte_width; /* Read/Write size in bytes */\
|
|
|
|
struct acpi_namespace_node *node; /* Link back to parent node */\
|
|
|
|
u32 bit_length; /* Length of field in bits */\
|
|
|
|
u32 base_byte_offset; /* Byte offset within containing object */\
|
|
|
|
u32 value; /* Value to store into the Bank or Index register */\
|
|
|
|
u8 start_field_bit_offset;/* Bit offset within first field datum (0-63) */\
|
2011-11-16 13:39:07 +08:00
|
|
|
u8 access_length; /* For serial regions/fields */
|
2010-08-06 09:09:33 +08:00
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
|
2017-07-10 15:23:15 +08:00
|
|
|
/* COMMON FIELD (for BUFFER, REGION, BANK, and INDEX fields) */
|
|
|
|
|
|
|
|
struct acpi_object_field_common {
|
2006-03-18 05:44:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_FIELD_INFO union acpi_operand_object *region_obj; /* Parent Operation Region object (REGION/BANK fields only) */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_region_field {
|
2011-11-16 13:39:07 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_FIELD_INFO u16 resource_length;
|
|
|
|
union acpi_operand_object *region_obj; /* Containing op_region object */
|
|
|
|
u8 *resource_buffer; /* resource_template for serial regions/fields */
|
2014-09-23 10:35:47 +08:00
|
|
|
u16 pin_number_index; /* Index relative to previous Connection/Template */
|
2019-02-23 08:06:25 +08:00
|
|
|
u8 *internal_pcc_buffer; /* Internal buffer for fields associated with PCC */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_bank_field {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_FIELD_INFO union acpi_operand_object *region_obj; /* Containing op_region object */
|
|
|
|
union acpi_operand_object *bank_obj; /* bank_select Register object */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_index_field {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_FIELD_INFO
|
|
|
|
/*
|
ACPI: ACPICA 20060421
Removed a device initialization optimization introduced in
20051216 where the _STA method was not run unless an _INI
was also present for the same device. This optimization
could cause problems because it could allow _INI methods
to be run within a not-present device subtree (If a
not-present device had no _INI, _STA would not be run,
the not-present status would not be discovered, and the
children of the device would be incorrectly traversed.)
Implemented a new _STA optimization where namespace
subtrees that do not contain _INI are identified and
ignored during device initialization. Selectively running
_STA can significantly improve boot time on large machines
(with assistance from Len Brown.)
Implemented support for the device initialization case
where the returned _STA flags indicate a device not-present
but functioning. In this case, _INI is not run, but the
device children are examined for presence, as per the
ACPI specification.
Implemented an additional change to the IndexField support
in order to conform to MS behavior. The value written to
the Index Register is not simply a byte offset, it is a
byte offset in units of the access width of the parent
Index Field. (Fiodor Suietov)
Defined and deployed a new OSL interface,
acpi_os_validate_address(). This interface is called during
the creation of all AML operation regions, and allows
the host OS to exert control over what addresses it will
allow the AML code to access. Operation Regions whose
addresses are disallowed will cause a runtime exception
when they are actually accessed (will not affect or abort
table loading.)
Defined and deployed a new OSL interface,
acpi_os_validate_interface(). This interface allows the host OS
to match the various "optional" interface/behavior strings
for the _OSI predefined control method as appropriate
(with assistance from Bjorn Helgaas.)
Restructured and corrected various problems in the
exception handling code paths within DsCallControlMethod
and DsTerminateControlMethod in dsmethod (with assistance
from Takayoshi Kochi.)
Modified the Linux source converter to ignore quoted string
literals while converting identifiers from mixed to lower
case. This will correct problems with the disassembler
and other areas where such strings must not be modified.
The ACPI_FUNCTION_* macros no longer require quotes around
the function name. This allows the Linux source converter
to convert the names, now that the converter ignores
quoted strings.
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Len Brown <len.brown@intel.com>
2006-04-22 05:15:00 +08:00
|
|
|
* No "RegionObj" pointer needed since the Index and Data registers
|
2005-08-05 12:44:28 +08:00
|
|
|
* are each field definitions unto themselves.
|
|
|
|
*/
|
|
|
|
union acpi_operand_object *index_obj; /* Index register */
|
|
|
|
union acpi_operand_object *data_obj; /* Data register */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* The buffer_field is different in that it is part of a Buffer, not an op_region */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_buffer_field {
|
ACPICA: Dispatcher: always generate buffer objects for ASL create_field() operator
ACPICA commit 79a466b64e6af36cc83102f05915e56cb7dd89ab
According to table 19-419 of the ACPI 6.3 specification, buffer_fields
created using the ASL create_field() Operator have been treated as
integers if the buffer_field is small enough to fit inside of an ASL
integer (32-bits or 64-bits depending on the definition block
revision). If they are larger, buffer fields are treated as ASL
Buffer objects. However, this is not true for other AML interpreter
implementations.
It has been discovered that other AML interpreters always treat
buffer fields created by create_field() as a buffer regardless of the
length of the buffer field.
More specifically, the Microsoft AML interpreter always treats buffer
fields created by the create_field() operator as buffer. ACPICA
currently does this only when the field size is larger than the
maximum integer width. This causes problems with AML code shipped in
Microsoft Surface devices.
More details:
The control methods in these devices determine the success of an ASL
control method execution by examining the type resulting from storing
a buffer field created by a create_field() operator. On success, a
Buffer object is expected, on failure an Integer containing an error
code. This buffer object is created with a dynamic size via the
create_field() operator. Due to the difference in behavior, Buffer
values of small size are however converted to Integers and thus
interpreted by the control method as having failed, whereas in
reality it succeeded. Below is an example of a control method called
TEST that illustrates this behavior.
Method (CBUF) // Create a Buffer field
{
/*
* Depending on the value of RAND, ACPICA interpreter will treat
* BF00 as an integer or buffer.
*/
create_field (BUFF, 0, RAND, BF00)
return (BF00)
}
Method (TEST)
{
/*
* Storing the value returned by CBUF to local0 will result in
* implicit type conversion outlined in the ACPI specification.
*
* ACPICA will treat local0 like an ASL integer if RAND is less
* than or equal to 64 or 32 (depending on the definition_block
* revision). If RAND is greater, it will be treated like an ASL
* buffer. Other implementations treat local0 like an ASL buffer
* regardless of the value of RAND.
*/
local0 = CBUF()
/*
* object_type of 0x03 represents an ASL Buffer
*/
if (object_type (Local0) != 0x03)
{
// Error on ACPICA if RAND is small enough
}
else
{
/*
* Success on APICA if RAND is large enough
* Other implementations always take this path because local0
* is always treated as a buffer.
*/
}
}
This change prohibits the previously mentioned integer conversion to
match other AML interpreter implementations (Microsoft) that do not
conform to the ACPI specification.
Link: https://github.com/acpica/acpica/commit/79a466b6
Signed-off-by: Maximilian Luz <luzmaximilian@gmail.com>
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Erik Kaneda <erik.kaneda@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2019-12-18 03:35:22 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER ACPI_COMMON_FIELD_INFO u8 is_create_field; /* Special case for objects created by create_field() */
|
|
|
|
union acpi_operand_object *buffer_obj; /* Containing Buffer object */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Objects for handlers
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_notify_handler {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER struct acpi_namespace_node *node; /* Parent device */
|
2012-05-03 11:08:19 +08:00
|
|
|
u32 handler_type; /* Type: Device/System/Both */
|
|
|
|
acpi_notify_handler handler; /* Handler address */
|
2005-08-05 12:44:28 +08:00
|
|
|
void *context;
|
2012-05-03 11:08:19 +08:00
|
|
|
union acpi_operand_object *next[2]; /* Device and System handler lists */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_addr_handler {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 space_id;
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 handler_flags;
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_adr_space_handler handler;
|
|
|
|
struct acpi_namespace_node *node; /* Parent device */
|
|
|
|
void *context;
|
ACPICA: Fix race in generic_serial_bus (I2C) and GPIO op_region parameter handling
ACPICA commit c9e0116952363b0fa815143dca7e9a2eb4fefa61
The handling of the generic_serial_bus (I2C) and GPIO op_regions in
acpi_ev_address_space_dispatch() passes a number of extra parameters
to the address-space handler through the address-space Context pointer
(instead of using more function parameters).
The Context is shared between threads, so if multiple threads try to
call the handler for the same address-space at the same time, then
a second thread could change the parameters of a first thread while
the handler is running for the first thread.
An example of this race hitting is the Lenovo Yoga Tablet2 1015L,
where there are both attrib_bytes accesses and attrib_byte accesses
to the same address-space. The attrib_bytes access stores the number
of bytes to transfer in Context->access_length. Where as for the
attrib_byte access the number of bytes to transfer is always 1 and
field_obj->Field.access_length is unused (so 0). Both types of
accesses racing from different threads leads to the following problem:
1. Thread a. starts an attrib_bytes access, stores a non 0 value
from field_obj->Field.access_length in Context->access_length
2. Thread b. starts an attrib_byte access, stores 0 in
Context->access_length
3. Thread a. calls i2c_acpi_space_handler() (under Linux). Which
sees that the access-type is ACPI_GSB_ACCESS_ATTRIB_MULTIBYTE
and calls acpi_gsb_i2c_read_bytes(..., Context->access_length)
4. At this point Context->access_length is 0 (set by thread b.)
rather then the field_obj->Field.access_length value from thread a.
This 0 length reads leads to the following errors being logged:
i2c i2c-0: adapter quirk: no zero length (addr 0x0078, size 0, read)
i2c i2c-0: i2c read 0 bytes from client@0x78 starting at reg 0x0 failed, error: -95
Note this is just an example of the problems which this race can cause.
There are likely many more (sporadic) problems caused by this race.
This commit adds a new context_mutex to struct acpi_object_addr_handler
and makes acpi_ev_address_space_dispatch() take that mutex when
using the shared Context to pass extra parameters to an address-space
handler, fixing this race.
Note the new mutex must be taken *after* exiting the interpreter,
therefor the existing acpi_ex_exit_interpreter() call is moved to above
the code which stores the extra parameters in the Context.
Link: https://github.com/acpica/acpica/commit/c9e01169
Signed-off-by: Hans de Goede <hdegoede@redhat.com>
Signed-off-by: Bob Moore <robert.moore@intel.com>
Signed-off-by: Erik Kaneda <erik.kaneda@intel.com>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
2021-02-19 07:17:07 +08:00
|
|
|
acpi_mutex context_mutex;
|
2005-08-05 12:44:28 +08:00
|
|
|
acpi_adr_space_setup setup;
|
2012-12-19 13:36:49 +08:00
|
|
|
union acpi_operand_object *region_list; /* Regions using this handler */
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object *next;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
/* Flags for address handler (handler_flags) */
|
|
|
|
|
|
|
|
#define ACPI_ADDR_HANDLER_DEFAULT_INSTALLED 0x01
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* Special internal objects
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
2008-09-27 11:08:41 +08:00
|
|
|
* The Reference object is used for these opcodes:
|
|
|
|
* Arg[0-6], Local[0-7], index_op, name_op, ref_of_op, load_op, load_table_op, debug_op
|
|
|
|
* The Reference.Class differentiates these types.
|
2005-04-17 06:20:36 +08:00
|
|
|
*/
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_reference {
|
2008-09-27 11:08:41 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER u8 class; /* Reference Class */
|
|
|
|
u8 target_type; /* Used for Index Op */
|
2017-08-03 14:27:22 +08:00
|
|
|
u8 resolved; /* Reference has been resolved to a value */
|
2006-03-18 05:44:00 +08:00
|
|
|
void *object; /* name_op=>HANDLE to obj, index_op=>union acpi_operand_object */
|
2008-09-27 11:08:41 +08:00
|
|
|
struct acpi_namespace_node *node; /* ref_of or Namepath */
|
|
|
|
union acpi_operand_object **where; /* Target of Index */
|
2015-07-01 14:44:44 +08:00
|
|
|
u8 *index_pointer; /* Used for Buffers and Strings */
|
2017-08-03 14:27:22 +08:00
|
|
|
u8 *aml; /* Used for deferred resolution of the ref */
|
2008-09-27 11:08:41 +08:00
|
|
|
u32 value; /* Used for Local/Arg/Index/ddb_handle */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2008-09-27 11:08:41 +08:00
|
|
|
/* Values for Reference.Class above */
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
ACPI_REFCLASS_LOCAL = 0, /* Method local */
|
|
|
|
ACPI_REFCLASS_ARG = 1, /* Method argument */
|
|
|
|
ACPI_REFCLASS_REFOF = 2, /* Result of ref_of() TBD: Split to Ref/Node and Ref/operand_obj? */
|
|
|
|
ACPI_REFCLASS_INDEX = 3, /* Result of Index() */
|
|
|
|
ACPI_REFCLASS_TABLE = 4, /* ddb_handle - Load(), load_table() */
|
|
|
|
ACPI_REFCLASS_NAME = 5, /* Reference to a named object */
|
|
|
|
ACPI_REFCLASS_DEBUG = 6, /* Debug object */
|
|
|
|
|
|
|
|
ACPI_REFCLASS_MAX = 6
|
|
|
|
} ACPI_REFERENCE_CLASSES;
|
|
|
|
|
2005-04-17 06:20:36 +08:00
|
|
|
/*
|
|
|
|
* Extra object is used as additional storage for types that
|
|
|
|
* have AML code in their declarations (term_args) that must be
|
|
|
|
* evaluated at run time.
|
|
|
|
*
|
|
|
|
* Currently: Region and field_unit types
|
|
|
|
*/
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_extra {
|
2006-03-18 05:44:00 +08:00
|
|
|
ACPI_OBJECT_COMMON_HEADER struct acpi_namespace_node *method_REG; /* _REG method for this region (if any) */
|
2011-11-28 09:46:02 +08:00
|
|
|
struct acpi_namespace_node *scope_node;
|
2005-08-05 12:44:28 +08:00
|
|
|
void *region_context; /* Region-specific data */
|
2006-03-18 05:44:00 +08:00
|
|
|
u8 *aml_start;
|
|
|
|
u32 aml_length;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Additional data that can be attached to namespace nodes */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_data {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER acpi_object_handler handler;
|
|
|
|
void *pointer;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Structure used when objects are cached for reuse */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
struct acpi_object_cache_list {
|
|
|
|
ACPI_OBJECT_COMMON_HEADER union acpi_operand_object *next; /* Link for object cache and internal lists */
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
2012-07-12 09:40:10 +08:00
|
|
|
* union acpi_operand_object descriptor - a giant union of all of the above
|
2005-04-17 06:20:36 +08:00
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object {
|
|
|
|
struct acpi_object_common common;
|
|
|
|
struct acpi_object_integer integer;
|
|
|
|
struct acpi_object_string string;
|
|
|
|
struct acpi_object_buffer buffer;
|
|
|
|
struct acpi_object_package package;
|
|
|
|
struct acpi_object_event event;
|
|
|
|
struct acpi_object_method method;
|
|
|
|
struct acpi_object_mutex mutex;
|
|
|
|
struct acpi_object_region region;
|
|
|
|
struct acpi_object_notify_common common_notify;
|
|
|
|
struct acpi_object_device device;
|
|
|
|
struct acpi_object_power_resource power_resource;
|
|
|
|
struct acpi_object_processor processor;
|
|
|
|
struct acpi_object_thermal_zone thermal_zone;
|
|
|
|
struct acpi_object_field_common common_field;
|
|
|
|
struct acpi_object_region_field field;
|
|
|
|
struct acpi_object_buffer_field buffer_field;
|
|
|
|
struct acpi_object_bank_field bank_field;
|
|
|
|
struct acpi_object_index_field index_field;
|
|
|
|
struct acpi_object_notify_handler notify;
|
|
|
|
struct acpi_object_addr_handler address_space;
|
|
|
|
struct acpi_object_reference reference;
|
|
|
|
struct acpi_object_extra extra;
|
|
|
|
struct acpi_object_data data;
|
|
|
|
struct acpi_object_cache_list cache;
|
2008-09-27 10:22:09 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Add namespace node to union in order to simplify code that accepts both
|
|
|
|
* ACPI_OPERAND_OBJECTs and ACPI_NAMESPACE_NODEs. The structures share
|
|
|
|
* a common descriptor_type field in order to differentiate them.
|
|
|
|
*/
|
|
|
|
struct acpi_namespace_node node;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* union acpi_descriptor - objects that share a common descriptor identifier
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
/* Object descriptor types */
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
#define ACPI_DESC_TYPE_CACHED 0x01 /* Used only when object is cached */
|
2005-04-17 06:20:36 +08:00
|
|
|
#define ACPI_DESC_TYPE_STATE 0x02
|
|
|
|
#define ACPI_DESC_TYPE_STATE_UPDATE 0x03
|
|
|
|
#define ACPI_DESC_TYPE_STATE_PACKAGE 0x04
|
|
|
|
#define ACPI_DESC_TYPE_STATE_CONTROL 0x05
|
|
|
|
#define ACPI_DESC_TYPE_STATE_RPSCOPE 0x06
|
|
|
|
#define ACPI_DESC_TYPE_STATE_PSCOPE 0x07
|
|
|
|
#define ACPI_DESC_TYPE_STATE_WSCOPE 0x08
|
|
|
|
#define ACPI_DESC_TYPE_STATE_RESULT 0x09
|
|
|
|
#define ACPI_DESC_TYPE_STATE_NOTIFY 0x0A
|
|
|
|
#define ACPI_DESC_TYPE_STATE_THREAD 0x0B
|
|
|
|
#define ACPI_DESC_TYPE_WALK 0x0C
|
|
|
|
#define ACPI_DESC_TYPE_PARSER 0x0D
|
|
|
|
#define ACPI_DESC_TYPE_OPERAND 0x0E
|
|
|
|
#define ACPI_DESC_TYPE_NAMED 0x0F
|
|
|
|
#define ACPI_DESC_TYPE_MAX 0x0F
|
|
|
|
|
2006-03-18 05:44:00 +08:00
|
|
|
struct acpi_common_descriptor {
|
|
|
|
void *common_pointer;
|
|
|
|
u8 descriptor_type; /* To differentiate various internal objs */
|
|
|
|
};
|
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_descriptor {
|
2006-03-18 05:44:00 +08:00
|
|
|
struct acpi_common_descriptor common;
|
2005-08-05 12:44:28 +08:00
|
|
|
union acpi_operand_object object;
|
|
|
|
struct acpi_namespace_node node;
|
|
|
|
union acpi_parse_object op;
|
2005-04-17 06:20:36 +08:00
|
|
|
};
|
|
|
|
|
2014-03-05 14:12:01 +08:00
|
|
|
#pragma pack()
|
2007-02-03 00:48:23 +08:00
|
|
|
|
2005-08-05 12:44:28 +08:00
|
|
|
#endif /* _ACOBJECT_H */
|