2018-03-14 16:13:07 -07:00
// SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2013-01-11 13:08:51 +01:00
/*******************************************************************************
*
* Module Name : utownerid - Support for Table / Method Owner IDs
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include <acpi/acpi.h>
# include "accommon.h"
# include "acnamesp.h"
# define _COMPONENT ACPI_UTILITIES
ACPI_MODULE_NAME ( " utownerid " )
/*******************************************************************************
*
* FUNCTION : acpi_ut_allocate_owner_id
*
* PARAMETERS : owner_id - Where the new owner ID is returned
*
* RETURN : Status
*
* DESCRIPTION : Allocate a table or method owner ID . The owner ID is used to
* track objects created by the table or method , to be deleted
* when the method exits or the table is unloaded .
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-05-05 12:57:53 +08:00
acpi_status acpi_ut_allocate_owner_id ( acpi_owner_id * owner_id )
2013-01-11 13:08:51 +01:00
{
u32 i ;
u32 j ;
u32 k ;
acpi_status status ;
ACPI_FUNCTION_TRACE ( ut_allocate_owner_id ) ;
/* Guard against multiple allocations of ID to the same location */
if ( * owner_id ) {
2015-12-29 13:54:36 +08:00
ACPI_ERROR ( ( AE_INFO ,
" Owner ID [0x%2.2X] already exists " , * owner_id ) ) ;
2013-01-11 13:08:51 +01:00
return_ACPI_STATUS ( AE_ALREADY_EXISTS ) ;
}
/* Mutex for the global ID mask */
status = acpi_ut_acquire_mutex ( ACPI_MTX_CACHES ) ;
if ( ACPI_FAILURE ( status ) ) {
return_ACPI_STATUS ( status ) ;
}
/*
* Find a free owner ID , cycle through all possible IDs on repeated
2015-12-29 13:54:36 +08:00
* allocations . ( ACPI_NUM_OWNERID_MASKS + 1 ) because first index
* may have to be scanned twice .
2013-01-11 13:08:51 +01:00
*/
for ( i = 0 , j = acpi_gbl_last_owner_id_index ;
i < ( ACPI_NUM_OWNERID_MASKS + 1 ) ; i + + , j + + ) {
if ( j > = ACPI_NUM_OWNERID_MASKS ) {
j = 0 ; /* Wraparound to start of mask array */
}
for ( k = acpi_gbl_next_owner_id_offset ; k < 32 ; k + + ) {
if ( acpi_gbl_owner_id_mask [ j ] = = ACPI_UINT32_MAX ) {
/* There are no free IDs in this mask */
break ;
}
2017-06-05 16:40:15 +08:00
/*
* Note : the u32 cast ensures that 1 is stored as a unsigned
* integer . Omitting the cast may result in 1 being stored as an
* int . Some compilers or runtime error detection may flag this as
* an error .
*/
if ( ! ( acpi_gbl_owner_id_mask [ j ] & ( ( u32 ) 1 < < k ) ) ) {
2013-01-11 13:08:51 +01:00
/*
* Found a free ID . The actual ID is the bit index plus one ,
* making zero an invalid Owner ID . Save this as the last ID
* allocated and update the global ID mask .
*/
2017-06-05 16:40:15 +08:00
acpi_gbl_owner_id_mask [ j ] | = ( ( u32 ) 1 < < k ) ;
2013-01-11 13:08:51 +01:00
acpi_gbl_last_owner_id_index = ( u8 ) j ;
acpi_gbl_next_owner_id_offset = ( u8 ) ( k + 1 ) ;
/*
* Construct encoded ID from the index and bit position
*
* Note : Last [ j ] . k ( bit 255 ) is never used and is marked
* permanently allocated ( prevents + 1 overflow )
*/
* owner_id =
2016-05-05 12:57:53 +08:00
( acpi_owner_id ) ( ( k + 1 ) + ACPI_MUL_32 ( j ) ) ;
2013-01-11 13:08:51 +01:00
ACPI_DEBUG_PRINT ( ( ACPI_DB_VALUES ,
" Allocated OwnerId: %2.2X \n " ,
( unsigned int ) * owner_id ) ) ;
goto exit ;
}
}
acpi_gbl_next_owner_id_offset = 0 ;
}
/*
* All owner_ids have been allocated . This typically should
* not happen since the IDs are reused after deallocation . The IDs are
* allocated upon table load ( one per table ) and method execution , and
* they are released when a table is unloaded or a method completes
* execution .
*
2015-12-29 13:54:36 +08:00
* If this error happens , there may be very deep nesting of invoked
* control methods , or there may be a bug where the IDs are not released .
2013-01-11 13:08:51 +01:00
*/
status = AE_OWNER_ID_LIMIT ;
ACPI_ERROR ( ( AE_INFO ,
" Could not allocate new OwnerId (255 max), AE_OWNER_ID_LIMIT " ) ) ;
2013-10-29 09:30:02 +08:00
exit :
2013-01-11 13:08:51 +01:00
( void ) acpi_ut_release_mutex ( ACPI_MTX_CACHES ) ;
return_ACPI_STATUS ( status ) ;
}
/*******************************************************************************
*
* FUNCTION : acpi_ut_release_owner_id
*
* PARAMETERS : owner_id_ptr - Pointer to a previously allocated owner_ID
*
* RETURN : None . No error is returned because we are either exiting a
* control method or unloading a table . Either way , we would
* ignore any error anyway .
*
* DESCRIPTION : Release a table or method owner ID . Valid IDs are 1 - 255
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2016-05-05 12:57:53 +08:00
void acpi_ut_release_owner_id ( acpi_owner_id * owner_id_ptr )
2013-01-11 13:08:51 +01:00
{
acpi_owner_id owner_id = * owner_id_ptr ;
acpi_status status ;
u32 index ;
u32 bit ;
ACPI_FUNCTION_TRACE_U32 ( ut_release_owner_id , owner_id ) ;
/* Always clear the input owner_id (zero is an invalid ID) */
* owner_id_ptr = 0 ;
/* Zero is not a valid owner_ID */
if ( owner_id = = 0 ) {
ACPI_ERROR ( ( AE_INFO , " Invalid OwnerId: 0x%2.2X " , owner_id ) ) ;
return_VOID ;
}
/* Mutex for the global ID mask */
status = acpi_ut_acquire_mutex ( ACPI_MTX_CACHES ) ;
if ( ACPI_FAILURE ( status ) ) {
return_VOID ;
}
/* Normalize the ID to zero */
owner_id - - ;
/* Decode ID to index/offset pair */
index = ACPI_DIV_32 ( owner_id ) ;
2017-06-05 16:40:15 +08:00
bit = ( u32 ) 1 < < ACPI_MOD_32 ( owner_id ) ;
2013-01-11 13:08:51 +01:00
/* Free the owner ID only if it is valid */
if ( acpi_gbl_owner_id_mask [ index ] & bit ) {
acpi_gbl_owner_id_mask [ index ] ^ = bit ;
} else {
ACPI_ERROR ( ( AE_INFO ,
" Release of non-allocated OwnerId: 0x%2.2X " ,
owner_id + 1 ) ) ;
}
( void ) acpi_ut_release_mutex ( ACPI_MTX_CACHES ) ;
return_VOID ;
}