2013-01-11 16:08:51 +04:00
/******************************************************************************
*
* Module Name : nsprepkg - Validation of package objects for predefined names
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
/*
2014-02-08 05:42:25 +04:00
* Copyright ( C ) 2000 - 2014 , Intel Corp .
2013-01-11 16:08:51 +04:00
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions , and the following disclaimer ,
* without modification .
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
* substantially similar to the " NO WARRANTY " disclaimer below
* ( " Disclaimer " ) and any redistribution must be conditioned upon
* including a substantially similar Disclaimer requirement for further
* binary redistribution .
* 3. Neither the names of the above - listed copyright holders nor the names
* of any contributors may be used to endorse or promote products derived
* from this software without specific prior written permission .
*
* Alternatively , this software may be distributed under the terms of the
* GNU General Public License ( " GPL " ) version 2 as published by the Free
* Software Foundation .
*
* NO WARRANTY
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT ,
* STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING
* IN ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGES .
*/
# include <acpi/acpi.h>
# include "accommon.h"
# include "acnamesp.h"
# include "acpredef.h"
# define _COMPONENT ACPI_NAMESPACE
ACPI_MODULE_NAME ( " nsprepkg " )
/* Local prototypes */
static acpi_status
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_list ( struct acpi_evaluate_info * info ,
2013-01-11 16:08:51 +04:00
const union acpi_predefined_info * package ,
union acpi_operand_object * * elements , u32 count ) ;
static acpi_status
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( struct acpi_evaluate_info * info ,
2013-01-11 16:08:51 +04:00
union acpi_operand_object * * elements ,
u8 type1 ,
u32 count1 ,
u8 type2 , u32 count2 , u32 start_index ) ;
/*******************************************************************************
*
* FUNCTION : acpi_ns_check_package
*
2013-05-30 06:00:01 +04:00
* PARAMETERS : info - Method execution information block
2013-01-11 16:08:51 +04:00
* return_object_ptr - Pointer to the object returned from the
* evaluation of a method or object
*
* RETURN : Status
*
* DESCRIPTION : Check a returned package object for the correct count and
* correct type of all sub - objects .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
2013-05-30 06:00:01 +04:00
acpi_ns_check_package ( struct acpi_evaluate_info * info ,
2013-01-11 16:08:51 +04:00
union acpi_operand_object * * return_object_ptr )
{
union acpi_operand_object * return_object = * return_object_ptr ;
const union acpi_predefined_info * package ;
union acpi_operand_object * * elements ;
acpi_status status = AE_OK ;
u32 expected_count ;
u32 count ;
u32 i ;
ACPI_FUNCTION_NAME ( ns_check_package ) ;
/* The package info for this name is in the next table entry */
2013-05-30 06:00:01 +04:00
package = info - > predefined + 1 ;
2013-01-11 16:08:51 +04:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_NAMES ,
" %s Validating return Package of Type %X, Count %X \n " ,
2013-05-30 06:00:01 +04:00
info - > full_pathname , package - > ret_info . type ,
2013-01-11 16:08:51 +04:00
return_object - > package . count ) ) ;
/*
* For variable - length Packages , we can safely remove all embedded
* and trailing NULL package elements
*/
2013-05-30 06:00:01 +04:00
acpi_ns_remove_null_elements ( info , package - > ret_info . type ,
2013-01-11 16:08:51 +04:00
return_object ) ;
/* Extract package count and elements array */
elements = return_object - > package . elements ;
count = return_object - > package . count ;
2013-03-08 13:23:51 +04:00
/*
* Most packages must have at least one element . The only exception
* is the variable - length package ( ACPI_PTYPE1_VAR ) .
*/
2013-01-11 16:08:51 +04:00
if ( ! count ) {
2013-03-08 13:23:51 +04:00
if ( package - > ret_info . type = = ACPI_PTYPE1_VAR ) {
return ( AE_OK ) ;
}
2013-05-30 06:00:01 +04:00
ACPI_WARN_PREDEFINED ( ( AE_INFO , info - > full_pathname ,
info - > node_flags ,
2013-01-11 16:08:51 +04:00
" Return Package has no elements (empty) " ) ) ;
return ( AE_AML_OPERAND_VALUE ) ;
}
/*
* Decode the type of the expected package contents
*
* PTYPE1 packages contain no subpackages
2014-02-26 06:31:18 +04:00
* PTYPE2 packages contain subpackages
2013-01-11 16:08:51 +04:00
*/
switch ( package - > ret_info . type ) {
case ACPI_PTYPE1_FIXED :
/*
2014-02-26 06:31:18 +04:00
* The package count is fixed and there are no subpackages
2013-01-11 16:08:51 +04:00
*
* If package is too small , exit .
* If package is larger than expected , issue warning but continue
*/
expected_count =
package - > ret_info . count1 + package - > ret_info . count2 ;
if ( count < expected_count ) {
goto package_too_small ;
} else if ( count > expected_count ) {
ACPI_DEBUG_PRINT ( ( ACPI_DB_REPAIR ,
" %s: Return Package is larger than needed - "
" found %u, expected %u \n " ,
2013-05-30 06:00:01 +04:00
info - > full_pathname , count ,
2013-01-11 16:08:51 +04:00
expected_count ) ) ;
}
/* Validate all elements of the returned package */
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_package_elements ( info , elements ,
2013-01-11 16:08:51 +04:00
package - > ret_info .
object_type1 ,
package - > ret_info .
count1 ,
package - > ret_info .
object_type2 ,
package - > ret_info .
count2 , 0 ) ;
break ;
case ACPI_PTYPE1_VAR :
/*
2014-02-26 06:31:18 +04:00
* The package count is variable , there are no subpackages , and all
2013-01-11 16:08:51 +04:00
* elements must be of the same type
*/
for ( i = 0 ; i < count ; i + + ) {
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , elements ,
2013-01-11 16:08:51 +04:00
package - > ret_info .
object_type1 , i ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
elements + + ;
}
break ;
case ACPI_PTYPE1_OPTION :
/*
2014-02-26 06:31:18 +04:00
* The package count is variable , there are no subpackages . There are
2013-01-11 16:08:51 +04:00
* a fixed number of required elements , and a variable number of
* optional elements .
*
* Check if package is at least as large as the minimum required
*/
expected_count = package - > ret_info3 . count ;
if ( count < expected_count ) {
goto package_too_small ;
}
/* Variable number of sub-objects */
for ( i = 0 ; i < count ; i + + ) {
if ( i < package - > ret_info3 . count ) {
/* These are the required package elements (0, 1, or 2) */
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_object_type ( info , elements ,
2013-01-11 16:08:51 +04:00
package - >
ret_info3 .
object_type [ i ] ,
i ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
} else {
/* These are the optional package elements */
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_object_type ( info , elements ,
2013-01-11 16:08:51 +04:00
package - >
ret_info3 .
tail_object_type ,
i ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
}
elements + + ;
}
break ;
case ACPI_PTYPE2_REV_FIXED :
/* First element is the (Integer) revision */
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , elements ,
2013-01-11 16:08:51 +04:00
ACPI_RTYPE_INTEGER , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
elements + + ;
count - - ;
2014-02-26 06:31:18 +04:00
/* Examine the subpackages */
2013-01-11 16:08:51 +04:00
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_list ( info , package , elements , count ) ;
2013-01-11 16:08:51 +04:00
break ;
case ACPI_PTYPE2_PKG_COUNT :
2014-02-26 06:31:18 +04:00
/* First element is the (Integer) count of subpackages to follow */
2013-01-11 16:08:51 +04:00
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , elements ,
2013-01-11 16:08:51 +04:00
ACPI_RTYPE_INTEGER , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
/*
* Count cannot be larger than the parent package length , but allow it
* to be smaller . The > = accounts for the Integer above .
*/
expected_count = ( u32 ) ( * elements ) - > integer . value ;
if ( expected_count > = count ) {
goto package_too_small ;
}
count = expected_count ;
elements + + ;
2014-02-26 06:31:18 +04:00
/* Examine the subpackages */
2013-01-11 16:08:51 +04:00
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_list ( info , package , elements , count ) ;
2013-01-11 16:08:51 +04:00
break ;
case ACPI_PTYPE2 :
case ACPI_PTYPE2_FIXED :
case ACPI_PTYPE2_MIN :
case ACPI_PTYPE2_COUNT :
case ACPI_PTYPE2_FIX_VAR :
/*
* These types all return a single Package that consists of a
2014-02-26 06:31:18 +04:00
* variable number of subpackages .
2013-01-11 16:08:51 +04:00
*
2014-02-26 06:31:18 +04:00
* First , ensure that the first element is a subpackage . If not ,
2013-01-11 16:08:51 +04:00
* the BIOS may have incorrectly returned the object as a single
* package instead of a Package of Packages ( a common error if
* there is only one entry ) . We may be able to repair this by
* wrapping the returned Package with a new outer Package .
*/
if ( * elements
& & ( ( * elements ) - > common . type ! = ACPI_TYPE_PACKAGE ) ) {
/* Create the new outer package and populate it */
status =
2013-05-30 06:00:01 +04:00
acpi_ns_wrap_with_package ( info , return_object ,
2013-01-11 16:08:51 +04:00
return_object_ptr ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
/* Update locals to point to the new package (of 1 element) */
return_object = * return_object_ptr ;
elements = return_object - > package . elements ;
count = 1 ;
}
2014-02-26 06:31:18 +04:00
/* Examine the subpackages */
2013-01-11 16:08:51 +04:00
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_list ( info , package , elements , count ) ;
2013-01-11 16:08:51 +04:00
break ;
2014-08-29 05:32:42 +04:00
case ACPI_PTYPE2_UUID_PAIR :
/* The package must contain pairs of (UUID + type) */
if ( count & 1 ) {
expected_count = count + 1 ;
goto package_too_small ;
}
while ( count > 0 ) {
status = acpi_ns_check_object_type ( info , elements ,
package - > ret_info .
object_type1 , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
/* Validate length of the UUID buffer */
if ( ( * elements ) - > buffer . length ! = 16 ) {
ACPI_WARN_PREDEFINED ( ( AE_INFO ,
info - > full_pathname ,
info - > node_flags ,
" Invalid length for UUID Buffer " ) ) ;
return ( AE_AML_OPERAND_VALUE ) ;
}
status = acpi_ns_check_object_type ( info , elements + 1 ,
package - > ret_info .
object_type2 , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
elements + = 2 ;
count - = 2 ;
}
break ;
2013-01-11 16:08:51 +04:00
default :
/* Should not get here if predefined info table is correct */
2013-05-30 06:00:01 +04:00
ACPI_WARN_PREDEFINED ( ( AE_INFO , info - > full_pathname ,
info - > node_flags ,
2013-01-11 16:08:51 +04:00
" Invalid internal return type in table entry: %X " ,
package - > ret_info . type ) ) ;
return ( AE_AML_INTERNAL ) ;
}
return ( status ) ;
2013-10-29 05:30:02 +04:00
package_too_small :
2013-01-11 16:08:51 +04:00
/* Error exit for the case with an incorrect package count */
2013-05-30 06:00:01 +04:00
ACPI_WARN_PREDEFINED ( ( AE_INFO , info - > full_pathname , info - > node_flags ,
2013-01-11 16:08:51 +04:00
" Return Package is too small - found %u elements, expected %u " ,
count , expected_count ) ) ;
return ( AE_AML_OPERAND_VALUE ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ns_check_package_list
*
2013-05-30 06:00:01 +04:00
* PARAMETERS : info - Method execution information block
2013-01-11 16:08:51 +04:00
* package - Pointer to package - specific info for method
* elements - Element list of parent package . All elements
* of this list should be of type Package .
* count - Count of subpackages
*
* RETURN : Status
*
* DESCRIPTION : Examine a list of subpackages
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static acpi_status
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_list ( struct acpi_evaluate_info * info ,
2013-01-11 16:08:51 +04:00
const union acpi_predefined_info * package ,
union acpi_operand_object * * elements , u32 count )
{
union acpi_operand_object * sub_package ;
union acpi_operand_object * * sub_elements ;
acpi_status status ;
u32 expected_count ;
u32 i ;
u32 j ;
/*
2014-02-26 06:31:18 +04:00
* Validate each subpackage in the parent Package
2013-01-11 16:08:51 +04:00
*
2014-02-26 06:31:18 +04:00
* NOTE : assumes list of subpackages contains no NULL elements .
2013-01-11 16:08:51 +04:00
* Any NULL elements should have been removed by earlier call
* to acpi_ns_remove_null_elements .
*/
for ( i = 0 ; i < count ; i + + ) {
sub_package = * elements ;
sub_elements = sub_package - > package . elements ;
2013-05-30 06:00:01 +04:00
info - > parent_package = sub_package ;
2013-01-11 16:08:51 +04:00
/* Each sub-object must be of type Package */
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , & sub_package ,
2013-01-11 16:08:51 +04:00
ACPI_RTYPE_PACKAGE , i ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
2014-02-26 06:31:18 +04:00
/* Examine the different types of expected subpackages */
2013-01-11 16:08:51 +04:00
2013-05-30 06:00:01 +04:00
info - > parent_package = sub_package ;
2013-01-11 16:08:51 +04:00
switch ( package - > ret_info . type ) {
case ACPI_PTYPE2 :
case ACPI_PTYPE2_PKG_COUNT :
case ACPI_PTYPE2_REV_FIXED :
/* Each subpackage has a fixed number of elements */
expected_count =
package - > ret_info . count1 + package - > ret_info . count2 ;
if ( sub_package - > package . count < expected_count ) {
goto package_too_small ;
}
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( info , sub_elements ,
2013-01-11 16:08:51 +04:00
package - > ret_info .
object_type1 ,
package - > ret_info .
count1 ,
package - > ret_info .
object_type2 ,
package - > ret_info .
count2 , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
break ;
case ACPI_PTYPE2_FIX_VAR :
/*
* Each subpackage has a fixed number of elements and an
* optional element
*/
expected_count =
package - > ret_info . count1 + package - > ret_info . count2 ;
if ( sub_package - > package . count < expected_count ) {
goto package_too_small ;
}
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( info , sub_elements ,
2013-01-11 16:08:51 +04:00
package - > ret_info .
object_type1 ,
package - > ret_info .
count1 ,
package - > ret_info .
object_type2 ,
sub_package - > package .
count -
package - > ret_info .
count1 , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
break ;
case ACPI_PTYPE2_FIXED :
2014-02-26 06:31:18 +04:00
/* Each subpackage has a fixed length */
2013-01-11 16:08:51 +04:00
expected_count = package - > ret_info2 . count ;
if ( sub_package - > package . count < expected_count ) {
goto package_too_small ;
}
2014-02-26 06:31:18 +04:00
/* Check the type of each subpackage element */
2013-01-11 16:08:51 +04:00
for ( j = 0 ; j < expected_count ; j + + ) {
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_object_type ( info ,
2013-01-11 16:08:51 +04:00
& sub_elements [ j ] ,
package - >
ret_info2 .
object_type [ j ] ,
j ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
}
break ;
case ACPI_PTYPE2_MIN :
2014-02-26 06:31:18 +04:00
/* Each subpackage has a variable but minimum length */
2013-01-11 16:08:51 +04:00
expected_count = package - > ret_info . count1 ;
if ( sub_package - > package . count < expected_count ) {
goto package_too_small ;
}
2014-02-26 06:31:18 +04:00
/* Check the type of each subpackage element */
2013-01-11 16:08:51 +04:00
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( info , sub_elements ,
2013-01-11 16:08:51 +04:00
package - > ret_info .
object_type1 ,
sub_package - > package .
count , 0 , 0 , 0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
break ;
case ACPI_PTYPE2_COUNT :
/*
* First element is the ( Integer ) count of elements , including
* the count field ( the ACPI name is num_elements )
*/
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , sub_elements ,
2013-01-11 16:08:51 +04:00
ACPI_RTYPE_INTEGER ,
0 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
/*
* Make sure package is large enough for the Count and is
* is as large as the minimum size
*/
expected_count = ( u32 ) ( * sub_elements ) - > integer . value ;
if ( sub_package - > package . count < expected_count ) {
goto package_too_small ;
}
if ( sub_package - > package . count <
package - > ret_info . count1 ) {
expected_count = package - > ret_info . count1 ;
goto package_too_small ;
}
if ( expected_count = = 0 ) {
/*
* Either the num_entries element was originally zero or it was
* a NULL element and repaired to an Integer of value zero .
* In either case , repair it by setting num_entries to be the
* actual size of the subpackage .
*/
expected_count = sub_package - > package . count ;
( * sub_elements ) - > integer . value = expected_count ;
}
2014-02-26 06:31:18 +04:00
/* Check the type of each subpackage element */
2013-01-11 16:08:51 +04:00
status =
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( info ,
2013-01-11 16:08:51 +04:00
( sub_elements + 1 ) ,
package - > ret_info .
object_type1 ,
( expected_count - 1 ) ,
0 , 0 , 1 ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
break ;
default : /* Should not get here, type was validated by caller */
return ( AE_AML_INTERNAL ) ;
}
elements + + ;
}
return ( AE_OK ) ;
2013-10-29 05:30:02 +04:00
package_too_small :
2013-01-11 16:08:51 +04:00
2014-02-26 06:31:18 +04:00
/* The subpackage count was smaller than required */
2013-01-11 16:08:51 +04:00
2013-05-30 06:00:01 +04:00
ACPI_WARN_PREDEFINED ( ( AE_INFO , info - > full_pathname , info - > node_flags ,
2014-02-26 06:31:18 +04:00
" Return SubPackage[%u] is too small - found %u elements, expected %u " ,
2013-01-11 16:08:51 +04:00
i , sub_package - > package . count , expected_count ) ) ;
return ( AE_AML_OPERAND_VALUE ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ns_check_package_elements
*
2013-05-30 06:00:01 +04:00
* PARAMETERS : info - Method execution information block
2013-01-11 16:08:51 +04:00
* elements - Pointer to the package elements array
* type1 - Object type for first group
* count1 - Count for first group
* type2 - Object type for second group
* count2 - Count for second group
* start_index - Start of the first group of elements
*
* RETURN : Status
*
* DESCRIPTION : Check that all elements of a package are of the correct object
* type . Supports up to two groups of different object types .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static acpi_status
2013-05-30 06:00:01 +04:00
acpi_ns_check_package_elements ( struct acpi_evaluate_info * info ,
2013-01-11 16:08:51 +04:00
union acpi_operand_object * * elements ,
u8 type1 ,
u32 count1 ,
u8 type2 , u32 count2 , u32 start_index )
{
union acpi_operand_object * * this_element = elements ;
acpi_status status ;
u32 i ;
/*
* Up to two groups of package elements are supported by the data
* structure . All elements in each group must be of the same type .
* The second group can have a count of zero .
*/
for ( i = 0 ; i < count1 ; i + + ) {
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , this_element ,
2013-01-11 16:08:51 +04:00
type1 , i + start_index ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
this_element + + ;
}
for ( i = 0 ; i < count2 ; i + + ) {
2013-05-30 06:00:01 +04:00
status = acpi_ns_check_object_type ( info , this_element ,
2013-01-11 16:08:51 +04:00
type2 ,
( i + count1 + start_index ) ) ;
if ( ACPI_FAILURE ( status ) ) {
return ( status ) ;
}
this_element + + ;
}
return ( AE_OK ) ;
}