2018-03-14 16:13:07 -07:00
// SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2009-06-29 13:39:29 +08:00
/******************************************************************************
*
2015-07-01 14:44:04 +08:00
* Module Name : utids - support for device Ids - HID , UID , CID , SUB , CLS
2009-06-29 13:39:29 +08:00
*
2023-04-05 15:38:21 +02:00
* Copyright ( C ) 2000 - 2023 , Intel Corp .
2009-06-29 13:39:29 +08:00
*
2018-03-14 16:13:07 -07:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-06-29 13:39:29 +08:00
# include <acpi/acpi.h>
# include "accommon.h"
# include "acinterp.h"
# define _COMPONENT ACPI_UTILITIES
ACPI_MODULE_NAME ( " utids " )
/*******************************************************************************
*
* FUNCTION : acpi_ut_execute_HID
*
* PARAMETERS : device_node - Node for the device
* return_id - Where the string HID is returned
*
* RETURN : Status
*
* DESCRIPTION : Executes the _HID control method that returns the hardware
* ID of the device . The HID is either an 32 - bit encoded EISAID
* Integer or a String . A string is always returned . An EISAID
* is converted to a string .
*
* NOTE : Internal function , no parameter validation
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
acpi_ut_execute_HID ( struct acpi_namespace_node * device_node ,
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id * * return_id )
2009-06-29 13:39:29 +08:00
{
union acpi_operand_object * obj_desc ;
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id * hid ;
2009-06-29 13:39:29 +08:00
u32 length ;
acpi_status status ;
ACPI_FUNCTION_TRACE ( ut_execute_HID ) ;
status = acpi_ut_evaluate_object ( device_node , METHOD_NAME__HID ,
ACPI_BTYPE_INTEGER | ACPI_BTYPE_STRING ,
& obj_desc ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/* Get the size of the String to be returned, includes null terminator */
if ( obj_desc - > common . type = = ACPI_TYPE_INTEGER ) {
length = ACPI_EISAID_STRING_SIZE ;
} else {
length = obj_desc - > string . length + 1 ;
}
/* Allocate a buffer for the HID */
hid =
2012-10-31 02:25:24 +00:00
ACPI_ALLOCATE_ZEROED ( sizeof ( struct acpi_pnp_device_id ) +
2016-05-05 12:57:53 +08:00
( acpi_size ) length ) ;
2009-06-29 13:39:29 +08:00
if ( ! hid ) {
status = AE_NO_MEMORY ;
goto cleanup ;
}
2012-10-31 02:25:24 +00:00
/* Area for the string starts after PNP_DEVICE_ID struct */
2009-06-29 13:39:29 +08:00
2012-10-31 02:25:24 +00:00
hid - > string =
ACPI_ADD_PTR ( char , hid , sizeof ( struct acpi_pnp_device_id ) ) ;
2009-06-29 13:39:29 +08:00
/* Convert EISAID to a string or simply copy existing string */
if ( obj_desc - > common . type = = ACPI_TYPE_INTEGER ) {
acpi_ex_eisa_id_to_string ( hid - > string , obj_desc - > integer . value ) ;
} else {
2015-07-01 14:45:11 +08:00
strcpy ( hid - > string , obj_desc - > string . pointer ) ;
2009-06-29 13:39:29 +08:00
}
hid - > length = length ;
* return_id = hid ;
cleanup :
/* On exit, we must delete the return object */
acpi_ut_remove_reference ( obj_desc ) ;
return_ACPI_STATUS ( status ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ut_execute_UID
*
* PARAMETERS : device_node - Node for the device
* return_id - Where the string UID is returned
*
* RETURN : Status
*
* DESCRIPTION : Executes the _UID control method that returns the unique
* ID of the device . The UID is either a 64 - bit Integer ( NOT an
* EISAID ) or a string . Always returns a string . A 64 - bit integer
* is converted to a decimal string .
*
* NOTE : Internal function , no parameter validation
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
acpi_ut_execute_UID ( struct acpi_namespace_node * device_node ,
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id * * return_id )
2009-06-29 13:39:29 +08:00
{
union acpi_operand_object * obj_desc ;
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id * uid ;
2009-06-29 13:39:29 +08:00
u32 length ;
acpi_status status ;
ACPI_FUNCTION_TRACE ( ut_execute_UID ) ;
status = acpi_ut_evaluate_object ( device_node , METHOD_NAME__UID ,
ACPI_BTYPE_INTEGER | ACPI_BTYPE_STRING ,
& obj_desc ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/* Get the size of the String to be returned, includes null terminator */
if ( obj_desc - > common . type = = ACPI_TYPE_INTEGER ) {
length = ACPI_MAX64_DECIMAL_DIGITS + 1 ;
} else {
length = obj_desc - > string . length + 1 ;
}
/* Allocate a buffer for the UID */
uid =
2012-10-31 02:25:24 +00:00
ACPI_ALLOCATE_ZEROED ( sizeof ( struct acpi_pnp_device_id ) +
2016-05-05 12:57:53 +08:00
( acpi_size ) length ) ;
2009-06-29 13:39:29 +08:00
if ( ! uid ) {
status = AE_NO_MEMORY ;
goto cleanup ;
}
2012-10-31 02:25:24 +00:00
/* Area for the string starts after PNP_DEVICE_ID struct */
2009-06-29 13:39:29 +08:00
2012-10-31 02:25:24 +00:00
uid - > string =
ACPI_ADD_PTR ( char , uid , sizeof ( struct acpi_pnp_device_id ) ) ;
2009-06-29 13:39:29 +08:00
/* Convert an Integer to string, or just copy an existing string */
if ( obj_desc - > common . type = = ACPI_TYPE_INTEGER ) {
acpi_ex_integer_to_string ( uid - > string , obj_desc - > integer . value ) ;
} else {
2015-07-01 14:45:11 +08:00
strcpy ( uid - > string , obj_desc - > string . pointer ) ;
2009-06-29 13:39:29 +08:00
}
uid - > length = length ;
* return_id = uid ;
cleanup :
/* On exit, we must delete the return object */
acpi_ut_remove_reference ( obj_desc ) ;
return_ACPI_STATUS ( status ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ut_execute_CID
*
* PARAMETERS : device_node - Node for the device
* return_cid_list - Where the CID list is returned
*
* RETURN : Status , list of CID strings
*
* DESCRIPTION : Executes the _CID control method that returns one or more
* compatible hardware IDs for the device .
*
* NOTE : Internal function , no parameter validation
*
* A _CID method can return either a single compatible ID or a package of
* compatible IDs . Each compatible ID can be one of the following :
* 1 ) Integer ( 32 bit compressed EISA ID ) or
* 2 ) String ( PCI ID format , e . g . " PCI \ VEN_vvvv&DEV_dddd&SUBSYS_ssssssss " )
*
* The Integer CIDs are converted to string format by this function .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
acpi_ut_execute_CID ( struct acpi_namespace_node * device_node ,
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id_list * * return_cid_list )
2009-06-29 13:39:29 +08:00
{
union acpi_operand_object * * cid_objects ;
union acpi_operand_object * obj_desc ;
2012-10-31 02:25:24 +00:00
struct acpi_pnp_device_id_list * cid_list ;
2009-06-29 13:39:29 +08:00
char * next_id_string ;
u32 string_area_size ;
u32 length ;
u32 cid_list_size ;
acpi_status status ;
u32 count ;
u32 i ;
ACPI_FUNCTION_TRACE ( ut_execute_CID ) ;
/* Evaluate the _CID method for this device */
status = acpi_ut_evaluate_object ( device_node , METHOD_NAME__CID ,
ACPI_BTYPE_INTEGER | ACPI_BTYPE_STRING
| ACPI_BTYPE_PACKAGE , & obj_desc ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/*
* Get the count and size of the returned _CIDs . _CID can return either
* a Package of Integers / Strings or a single Integer or String .
* Note : This section also validates that all CID elements are of the
* correct type ( Integer or String ) .
*/
if ( obj_desc - > common . type = = ACPI_TYPE_PACKAGE ) {
count = obj_desc - > package . count ;
cid_objects = obj_desc - > package . elements ;
} else { /* Single Integer or String CID */
count = 1 ;
cid_objects = & obj_desc ;
}
string_area_size = 0 ;
for ( i = 0 ; i < count ; i + + ) {
/* String lengths include null terminator */
switch ( cid_objects [ i ] - > common . type ) {
case ACPI_TYPE_INTEGER :
2013-06-08 00:58:14 +00:00
2009-06-29 13:39:29 +08:00
string_area_size + = ACPI_EISAID_STRING_SIZE ;
break ;
case ACPI_TYPE_STRING :
2013-06-08 00:58:14 +00:00
2009-06-29 13:39:29 +08:00
string_area_size + = cid_objects [ i ] - > string . length + 1 ;
break ;
default :
2013-06-08 00:58:14 +00:00
2009-06-29 13:39:29 +08:00
status = AE_TYPE ;
goto cleanup ;
}
}
/*
* Now that we know the length of the CIDs , allocate return buffer :
* 1 ) Size of the base structure +
2012-10-31 02:25:24 +00:00
* 2 ) Size of the CID PNP_DEVICE_ID array +
2009-06-29 13:39:29 +08:00
* 3 ) Size of the actual CID strings
*/
2012-10-31 02:25:24 +00:00
cid_list_size = sizeof ( struct acpi_pnp_device_id_list ) +
2020-07-20 10:31:19 -07:00
( count * sizeof ( struct acpi_pnp_device_id ) ) + string_area_size ;
2009-06-29 13:39:29 +08:00
cid_list = ACPI_ALLOCATE_ZEROED ( cid_list_size ) ;
if ( ! cid_list ) {
status = AE_NO_MEMORY ;
goto cleanup ;
}
2012-10-31 02:25:24 +00:00
/* Area for CID strings starts after the CID PNP_DEVICE_ID array */
2009-06-29 13:39:29 +08:00
next_id_string = ACPI_CAST_PTR ( char , cid_list - > ids ) +
2016-05-05 12:57:53 +08:00
( ( acpi_size ) count * sizeof ( struct acpi_pnp_device_id ) ) ;
2009-06-29 13:39:29 +08:00
/* Copy/convert the CIDs to the return buffer */
for ( i = 0 ; i < count ; i + + ) {
if ( cid_objects [ i ] - > common . type = = ACPI_TYPE_INTEGER ) {
/* Convert the Integer (EISAID) CID to a string */
acpi_ex_eisa_id_to_string ( next_id_string ,
cid_objects [ i ] - > integer .
value ) ;
length = ACPI_EISAID_STRING_SIZE ;
} else { /* ACPI_TYPE_STRING */
/* Copy the String CID from the returned object */
2015-07-01 14:45:11 +08:00
strcpy ( next_id_string , cid_objects [ i ] - > string . pointer ) ;
2009-06-29 13:39:29 +08:00
length = cid_objects [ i ] - > string . length + 1 ;
}
cid_list - > ids [ i ] . string = next_id_string ;
cid_list - > ids [ i ] . length = length ;
next_id_string + = length ;
}
/* Finish the CID list */
cid_list - > count = count ;
cid_list - > list_size = cid_list_size ;
* return_cid_list = cid_list ;
cleanup :
/* On exit, we must delete the _CID return object */
acpi_ut_remove_reference ( obj_desc ) ;
return_ACPI_STATUS ( status ) ;
}
2015-07-01 14:44:04 +08:00
/*******************************************************************************
*
* FUNCTION : acpi_ut_execute_CLS
*
* PARAMETERS : device_node - Node for the device
* return_id - Where the _CLS is returned
*
* RETURN : Status
*
* DESCRIPTION : Executes the _CLS control method that returns PCI - defined
* class code of the device . The _CLS value is always a package
* containing PCI class information as a list of integers .
* The returned string has format " BBSSPP " , where :
* BB = Base - class code
* SS = Sub - class code
* PP = Programming Interface code
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
acpi_status
acpi_ut_execute_CLS ( struct acpi_namespace_node * device_node ,
struct acpi_pnp_device_id * * return_id )
{
union acpi_operand_object * obj_desc ;
union acpi_operand_object * * cls_objects ;
u32 count ;
struct acpi_pnp_device_id * cls ;
u32 length ;
acpi_status status ;
u8 class_code [ 3 ] = { 0 , 0 , 0 } ;
ACPI_FUNCTION_TRACE ( ut_execute_CLS ) ;
status = acpi_ut_evaluate_object ( device_node , METHOD_NAME__CLS ,
ACPI_BTYPE_PACKAGE , & obj_desc ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/* Get the size of the String to be returned, includes null terminator */
length = ACPI_PCICLS_STRING_SIZE ;
cls_objects = obj_desc - > package . elements ;
count = obj_desc - > package . count ;
if ( obj_desc - > common . type = = ACPI_TYPE_PACKAGE ) {
if ( count > 0
& & cls_objects [ 0 ] - > common . type = = ACPI_TYPE_INTEGER ) {
class_code [ 0 ] = ( u8 ) cls_objects [ 0 ] - > integer . value ;
}
if ( count > 1
& & cls_objects [ 1 ] - > common . type = = ACPI_TYPE_INTEGER ) {
class_code [ 1 ] = ( u8 ) cls_objects [ 1 ] - > integer . value ;
}
if ( count > 2
& & cls_objects [ 2 ] - > common . type = = ACPI_TYPE_INTEGER ) {
class_code [ 2 ] = ( u8 ) cls_objects [ 2 ] - > integer . value ;
}
}
/* Allocate a buffer for the CLS */
cls =
ACPI_ALLOCATE_ZEROED ( sizeof ( struct acpi_pnp_device_id ) +
2016-05-05 12:57:53 +08:00
( acpi_size ) length ) ;
2015-07-01 14:44:04 +08:00
if ( ! cls ) {
status = AE_NO_MEMORY ;
goto cleanup ;
}
/* Area for the string starts after PNP_DEVICE_ID struct */
cls - > string =
ACPI_ADD_PTR ( char , cls , sizeof ( struct acpi_pnp_device_id ) ) ;
/* Simply copy existing string */
acpi_ex_pci_cls_to_string ( cls - > string , class_code ) ;
cls - > length = length ;
* return_id = cls ;
cleanup :
/* On exit, we must delete the return object */
acpi_ut_remove_reference ( obj_desc ) ;
return_ACPI_STATUS ( status ) ;
}