Commit a647b5c3 authored by Bob Moore's avatar Bob Moore Committed by Len Brown

ACPICA: Add Buffer->String conversion for predefined methods

For predefined methods (such as _BIF), add automatic conversion for
objects that are required to be a String, but a Buffer was found
instead. This can happen when reading string battery data from
an operation region, because it used to be difficult to convert
the data from buffer to string from within the ASL. Linux BZ 11822.

http://bugzilla.kernel.org/show_bug.cgi?id=11822Signed-off-by: default avatarBob Moore <robert.moore@intel.com>
Signed-off-by: default avatarLin Ming <ming.m.lin@intel.com>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent d037c5fd
...@@ -282,8 +282,7 @@ acpi_status acpi_ns_evaluate(struct acpi_evaluate_info * info) ...@@ -282,8 +282,7 @@ acpi_status acpi_ns_evaluate(struct acpi_evaluate_info * info)
* the method on invalid return object. * the method on invalid return object.
*/ */
(void)acpi_ns_check_predefined_names(node, (void)acpi_ns_check_predefined_names(node,
info-> &info->return_object);
return_object);
} }
/* Mark the node as having been evaluated */ /* Mark the node as having been evaluated */
......
...@@ -72,7 +72,7 @@ ACPI_MODULE_NAME("nspredef") ...@@ -72,7 +72,7 @@ ACPI_MODULE_NAME("nspredef")
/* Local prototypes */ /* Local prototypes */
static acpi_status static acpi_status
acpi_ns_check_package(char *pathname, acpi_ns_check_package(char *pathname,
union acpi_operand_object *return_object, union acpi_operand_object **return_object_ptr,
const union acpi_predefined_info *predefined); const union acpi_predefined_info *predefined);
static acpi_status static acpi_status
...@@ -82,13 +82,18 @@ acpi_ns_check_package_elements(char *pathname, ...@@ -82,13 +82,18 @@ acpi_ns_check_package_elements(char *pathname,
static acpi_status static acpi_status
acpi_ns_check_object_type(char *pathname, acpi_ns_check_object_type(char *pathname,
union acpi_operand_object *return_object, union acpi_operand_object **return_object_ptr,
u32 expected_btypes, u32 package_index); u32 expected_btypes, u32 package_index);
static acpi_status static acpi_status
acpi_ns_check_reference(char *pathname, acpi_ns_check_reference(char *pathname,
union acpi_operand_object *return_object); union acpi_operand_object *return_object);
static acpi_status
acpi_ns_repair_object(u32 expected_btypes,
u32 package_index,
union acpi_operand_object **return_object_ptr);
/* /*
* Names for the types that can be returned by the predefined objects. * Names for the types that can be returned by the predefined objects.
* Used for warning messages. Must be in the same order as the ACPI_RTYPEs * Used for warning messages. Must be in the same order as the ACPI_RTYPEs
...@@ -108,8 +113,8 @@ static const char *acpi_rtype_names[] = { ...@@ -108,8 +113,8 @@ static const char *acpi_rtype_names[] = {
* FUNCTION: acpi_ns_check_predefined_names * FUNCTION: acpi_ns_check_predefined_names
* *
* PARAMETERS: Node - Namespace node for the method/object * PARAMETERS: Node - Namespace node for the method/object
* return_object - Object returned from the evaluation of this * return_object_ptr - Pointer to the object returned from the
* method/object * evaluation of a method or object
* *
* RETURN: Status * RETURN: Status
* *
...@@ -119,8 +124,9 @@ static const char *acpi_rtype_names[] = { ...@@ -119,8 +124,9 @@ static const char *acpi_rtype_names[] = {
acpi_status acpi_status
acpi_ns_check_predefined_names(struct acpi_namespace_node *node, acpi_ns_check_predefined_names(struct acpi_namespace_node *node,
union acpi_operand_object *return_object) union acpi_operand_object **return_object_ptr)
{ {
union acpi_operand_object *return_object = *return_object_ptr;
acpi_status status = AE_OK; acpi_status status = AE_OK;
const union acpi_predefined_info *predefined; const union acpi_predefined_info *predefined;
char *pathname; char *pathname;
...@@ -182,7 +188,7 @@ acpi_ns_check_predefined_names(struct acpi_namespace_node *node, ...@@ -182,7 +188,7 @@ acpi_ns_check_predefined_names(struct acpi_namespace_node *node,
* Check that the type of the return object is what is expected for * Check that the type of the return object is what is expected for
* this predefined name * this predefined name
*/ */
status = acpi_ns_check_object_type(pathname, return_object, status = acpi_ns_check_object_type(pathname, return_object_ptr,
predefined->info.expected_btypes, predefined->info.expected_btypes,
ACPI_NOT_PACKAGE); ACPI_NOT_PACKAGE);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
...@@ -193,7 +199,8 @@ acpi_ns_check_predefined_names(struct acpi_namespace_node *node, ...@@ -193,7 +199,8 @@ acpi_ns_check_predefined_names(struct acpi_namespace_node *node,
if (ACPI_GET_OBJECT_TYPE(return_object) == ACPI_TYPE_PACKAGE) { if (ACPI_GET_OBJECT_TYPE(return_object) == ACPI_TYPE_PACKAGE) {
status = status =
acpi_ns_check_package(pathname, return_object, predefined); acpi_ns_check_package(pathname, return_object_ptr,
predefined);
} }
exit: exit:
...@@ -307,8 +314,8 @@ const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct ...@@ -307,8 +314,8 @@ const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct
* FUNCTION: acpi_ns_check_package * FUNCTION: acpi_ns_check_package
* *
* PARAMETERS: Pathname - Full pathname to the node (for error msgs) * PARAMETERS: Pathname - Full pathname to the node (for error msgs)
* return_object - Object returned from the evaluation of a * return_object_ptr - Pointer to the object returned from the
* method or object * evaluation of a method or object
* Predefined - Pointer to entry in predefined name table * Predefined - Pointer to entry in predefined name table
* *
* RETURN: Status * RETURN: Status
...@@ -320,9 +327,10 @@ const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct ...@@ -320,9 +327,10 @@ const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct
static acpi_status static acpi_status
acpi_ns_check_package(char *pathname, acpi_ns_check_package(char *pathname,
union acpi_operand_object *return_object, union acpi_operand_object **return_object_ptr,
const union acpi_predefined_info *predefined) const union acpi_predefined_info *predefined)
{ {
union acpi_operand_object *return_object = *return_object_ptr;
const union acpi_predefined_info *package; const union acpi_predefined_info *package;
union acpi_operand_object *sub_package; union acpi_operand_object *sub_package;
union acpi_operand_object **elements; union acpi_operand_object **elements;
...@@ -408,7 +416,7 @@ acpi_ns_check_package(char *pathname, ...@@ -408,7 +416,7 @@ acpi_ns_check_package(char *pathname,
* elements must be of the same type * elements must be of the same type
*/ */
for (i = 0; i < count; i++) { for (i = 0; i < count; i++) {
status = acpi_ns_check_object_type(pathname, *elements, status = acpi_ns_check_object_type(pathname, elements,
package->ret_info. package->ret_info.
object_type1, i); object_type1, i);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
...@@ -441,7 +449,7 @@ acpi_ns_check_package(char *pathname, ...@@ -441,7 +449,7 @@ acpi_ns_check_package(char *pathname,
status = status =
acpi_ns_check_object_type(pathname, acpi_ns_check_object_type(pathname,
*elements, elements,
package-> package->
ret_info3. ret_info3.
object_type[i], object_type[i],
...@@ -454,7 +462,7 @@ acpi_ns_check_package(char *pathname, ...@@ -454,7 +462,7 @@ acpi_ns_check_package(char *pathname,
status = status =
acpi_ns_check_object_type(pathname, acpi_ns_check_object_type(pathname,
*elements, elements,
package-> package->
ret_info3. ret_info3.
tail_object_type, tail_object_type,
...@@ -471,7 +479,7 @@ acpi_ns_check_package(char *pathname, ...@@ -471,7 +479,7 @@ acpi_ns_check_package(char *pathname,
/* First element is the (Integer) count of sub-packages to follow */ /* First element is the (Integer) count of sub-packages to follow */
status = acpi_ns_check_object_type(pathname, *elements, status = acpi_ns_check_object_type(pathname, elements,
ACPI_RTYPE_INTEGER, 0); ACPI_RTYPE_INTEGER, 0);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
return (status); return (status);
...@@ -509,7 +517,7 @@ acpi_ns_check_package(char *pathname, ...@@ -509,7 +517,7 @@ acpi_ns_check_package(char *pathname,
/* Each sub-object must be of type Package */ /* Each sub-object must be of type Package */
status = status =
acpi_ns_check_object_type(pathname, sub_package, acpi_ns_check_object_type(pathname, &sub_package,
ACPI_RTYPE_PACKAGE, i); ACPI_RTYPE_PACKAGE, i);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
return (status); return (status);
...@@ -567,12 +575,8 @@ acpi_ns_check_package(char *pathname, ...@@ -567,12 +575,8 @@ acpi_ns_check_package(char *pathname,
for (j = 0; j < expected_count; j++) { for (j = 0; j < expected_count; j++) {
status = status =
acpi_ns_check_object_type(pathname, acpi_ns_check_object_type(pathname,
sub_elements &sub_elements[j],
[j], package->ret_info2.object_type[j], j);
package->
ret_info2.
object_type
[j], j);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
return (status); return (status);
} }
...@@ -611,7 +615,7 @@ acpi_ns_check_package(char *pathname, ...@@ -611,7 +615,7 @@ acpi_ns_check_package(char *pathname,
status = status =
acpi_ns_check_object_type(pathname, acpi_ns_check_object_type(pathname,
*sub_elements, sub_elements,
ACPI_RTYPE_INTEGER, ACPI_RTYPE_INTEGER,
0); 0);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
...@@ -708,7 +712,7 @@ acpi_ns_check_package_elements(char *pathname, ...@@ -708,7 +712,7 @@ acpi_ns_check_package_elements(char *pathname,
* The second group can have a count of zero. * The second group can have a count of zero.
*/ */
for (i = 0; i < count1; i++) { for (i = 0; i < count1; i++) {
status = acpi_ns_check_object_type(pathname, *this_element, status = acpi_ns_check_object_type(pathname, this_element,
type1, i); type1, i);
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
return (status); return (status);
...@@ -717,7 +721,7 @@ acpi_ns_check_package_elements(char *pathname, ...@@ -717,7 +721,7 @@ acpi_ns_check_package_elements(char *pathname,
} }
for (i = 0; i < count2; i++) { for (i = 0; i < count2; i++) {
status = acpi_ns_check_object_type(pathname, *this_element, status = acpi_ns_check_object_type(pathname, this_element,
type2, (i + count1)); type2, (i + count1));
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
return (status); return (status);
...@@ -733,8 +737,8 @@ acpi_ns_check_package_elements(char *pathname, ...@@ -733,8 +737,8 @@ acpi_ns_check_package_elements(char *pathname,
* FUNCTION: acpi_ns_check_object_type * FUNCTION: acpi_ns_check_object_type
* *
* PARAMETERS: Pathname - Full pathname to the node (for error msgs) * PARAMETERS: Pathname - Full pathname to the node (for error msgs)
* return_object - Object return from the execution of this * return_object_ptr - Pointer to the object returned from the
* method/object * evaluation of a method or object
* expected_btypes - Bitmap of expected return type(s) * expected_btypes - Bitmap of expected return type(s)
* package_index - Index of object within parent package (if * package_index - Index of object within parent package (if
* applicable - ACPI_NOT_PACKAGE otherwise) * applicable - ACPI_NOT_PACKAGE otherwise)
...@@ -748,9 +752,10 @@ acpi_ns_check_package_elements(char *pathname, ...@@ -748,9 +752,10 @@ acpi_ns_check_package_elements(char *pathname,
static acpi_status static acpi_status
acpi_ns_check_object_type(char *pathname, acpi_ns_check_object_type(char *pathname,
union acpi_operand_object *return_object, union acpi_operand_object **return_object_ptr,
u32 expected_btypes, u32 package_index) u32 expected_btypes, u32 package_index)
{ {
union acpi_operand_object *return_object = *return_object_ptr;
acpi_status status = AE_OK; acpi_status status = AE_OK;
u32 return_btype; u32 return_btype;
char type_buffer[48]; /* Room for 5 types */ char type_buffer[48]; /* Room for 5 types */
...@@ -814,6 +819,14 @@ acpi_ns_check_object_type(char *pathname, ...@@ -814,6 +819,14 @@ acpi_ns_check_object_type(char *pathname,
/* Is the object one of the expected types? */ /* Is the object one of the expected types? */
if (!(return_btype & expected_btypes)) { if (!(return_btype & expected_btypes)) {
/* Type mismatch -- attempt repair of the returned object */
status = acpi_ns_repair_object(expected_btypes, package_index,
return_object_ptr);
if (ACPI_SUCCESS(status)) {
return (status);
}
goto type_error_exit; goto type_error_exit;
} }
...@@ -898,3 +911,86 @@ acpi_ns_check_reference(char *pathname, ...@@ -898,3 +911,86 @@ acpi_ns_check_reference(char *pathname,
return (AE_AML_OPERAND_TYPE); return (AE_AML_OPERAND_TYPE);
} }
/*******************************************************************************
*
* FUNCTION: acpi_ns_repair_object
*
* PARAMETERS: Pathname - Full pathname to the node (for error msgs)
* package_index - Used to determine if target is in a package
* return_object_ptr - Pointer to the object returned from the
* evaluation of a method or object
*
* RETURN: Status. AE_OK if repair was successful.
*
* DESCRIPTION: Attempt to repair/convert a return object of a type that was
* not expected.
*
******************************************************************************/
static acpi_status
acpi_ns_repair_object(u32 expected_btypes,
u32 package_index,
union acpi_operand_object **return_object_ptr)
{
union acpi_operand_object *return_object = *return_object_ptr;
union acpi_operand_object *new_object;
acpi_size length;
switch (ACPI_GET_OBJECT_TYPE(return_object)) {
case ACPI_TYPE_BUFFER:
if (!(expected_btypes & ACPI_RTYPE_STRING)) {
return (AE_AML_OPERAND_TYPE);
}
/*
* Have a Buffer, expected a String, convert. Use a to_string
* conversion, no transform performed on the buffer data. The best
* example of this is the _BIF method, where the string data from
* the battery is often (incorrectly) returned as buffer object(s).
*/
length = 0;
while ((length < return_object->buffer.length) &&
(return_object->buffer.pointer[length])) {
length++;
}
/* Allocate a new string object */
new_object = acpi_ut_create_string_object(length);
if (!new_object) {
return (AE_NO_MEMORY);
}
/*
* Copy the raw buffer data with no transform. String is already NULL
* terminated at Length+1.
*/
ACPI_MEMCPY(new_object->string.pointer,
return_object->buffer.pointer, length);
/* Install the new return object */
acpi_ut_remove_reference(return_object);
*return_object_ptr = new_object;
/*
* If the object is a package element, we need to:
* 1. Decrement the reference count of the orignal object, it was
* incremented when building the package
* 2. Increment the reference count of the new object, it will be
* decremented when releasing the package
*/
if (package_index != ACPI_NOT_PACKAGE) {
acpi_ut_remove_reference(return_object);
acpi_ut_add_reference(new_object);
}
return (AE_OK);
default:
break;
}
return (AE_AML_OPERAND_TYPE);
}
...@@ -182,7 +182,7 @@ acpi_status acpi_ns_evaluate(struct acpi_evaluate_info *info); ...@@ -182,7 +182,7 @@ acpi_status acpi_ns_evaluate(struct acpi_evaluate_info *info);
*/ */
acpi_status acpi_status
acpi_ns_check_predefined_names(struct acpi_namespace_node *node, acpi_ns_check_predefined_names(struct acpi_namespace_node *node,
union acpi_operand_object *return_object); union acpi_operand_object **return_object);
const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct const union acpi_predefined_info *acpi_ns_check_for_predefined_name(struct
acpi_namespace_node acpi_namespace_node
......
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