xref: /minix3/minix/drivers/power/acpi/utilities/utownerid.c (revision 29492bb71c7148a089a5afafa0c99409161218df)
1*29492bb7SDavid van Moolenbroek /*******************************************************************************
2*29492bb7SDavid van Moolenbroek  *
3*29492bb7SDavid van Moolenbroek  * Module Name: utownerid - Support for Table/Method Owner IDs
4*29492bb7SDavid van Moolenbroek  *
5*29492bb7SDavid van Moolenbroek  ******************************************************************************/
6*29492bb7SDavid van Moolenbroek 
7*29492bb7SDavid van Moolenbroek /*
8*29492bb7SDavid van Moolenbroek  * Copyright (C) 2000 - 2014, Intel Corp.
9*29492bb7SDavid van Moolenbroek  * All rights reserved.
10*29492bb7SDavid van Moolenbroek  *
11*29492bb7SDavid van Moolenbroek  * Redistribution and use in source and binary forms, with or without
12*29492bb7SDavid van Moolenbroek  * modification, are permitted provided that the following conditions
13*29492bb7SDavid van Moolenbroek  * are met:
14*29492bb7SDavid van Moolenbroek  * 1. Redistributions of source code must retain the above copyright
15*29492bb7SDavid van Moolenbroek  *    notice, this list of conditions, and the following disclaimer,
16*29492bb7SDavid van Moolenbroek  *    without modification.
17*29492bb7SDavid van Moolenbroek  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18*29492bb7SDavid van Moolenbroek  *    substantially similar to the "NO WARRANTY" disclaimer below
19*29492bb7SDavid van Moolenbroek  *    ("Disclaimer") and any redistribution must be conditioned upon
20*29492bb7SDavid van Moolenbroek  *    including a substantially similar Disclaimer requirement for further
21*29492bb7SDavid van Moolenbroek  *    binary redistribution.
22*29492bb7SDavid van Moolenbroek  * 3. Neither the names of the above-listed copyright holders nor the names
23*29492bb7SDavid van Moolenbroek  *    of any contributors may be used to endorse or promote products derived
24*29492bb7SDavid van Moolenbroek  *    from this software without specific prior written permission.
25*29492bb7SDavid van Moolenbroek  *
26*29492bb7SDavid van Moolenbroek  * Alternatively, this software may be distributed under the terms of the
27*29492bb7SDavid van Moolenbroek  * GNU General Public License ("GPL") version 2 as published by the Free
28*29492bb7SDavid van Moolenbroek  * Software Foundation.
29*29492bb7SDavid van Moolenbroek  *
30*29492bb7SDavid van Moolenbroek  * NO WARRANTY
31*29492bb7SDavid van Moolenbroek  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32*29492bb7SDavid van Moolenbroek  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33*29492bb7SDavid van Moolenbroek  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34*29492bb7SDavid van Moolenbroek  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35*29492bb7SDavid van Moolenbroek  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36*29492bb7SDavid van Moolenbroek  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37*29492bb7SDavid van Moolenbroek  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38*29492bb7SDavid van Moolenbroek  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39*29492bb7SDavid van Moolenbroek  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40*29492bb7SDavid van Moolenbroek  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41*29492bb7SDavid van Moolenbroek  * POSSIBILITY OF SUCH DAMAGES.
42*29492bb7SDavid van Moolenbroek  */
43*29492bb7SDavid van Moolenbroek 
44*29492bb7SDavid van Moolenbroek #include "acpi.h"
45*29492bb7SDavid van Moolenbroek #include "accommon.h"
46*29492bb7SDavid van Moolenbroek #include "acnamesp.h"
47*29492bb7SDavid van Moolenbroek 
48*29492bb7SDavid van Moolenbroek 
49*29492bb7SDavid van Moolenbroek #define _COMPONENT          ACPI_UTILITIES
50*29492bb7SDavid van Moolenbroek         ACPI_MODULE_NAME    ("utownerid")
51*29492bb7SDavid van Moolenbroek 
52*29492bb7SDavid van Moolenbroek 
53*29492bb7SDavid van Moolenbroek /*******************************************************************************
54*29492bb7SDavid van Moolenbroek  *
55*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiUtAllocateOwnerId
56*29492bb7SDavid van Moolenbroek  *
57*29492bb7SDavid van Moolenbroek  * PARAMETERS:  OwnerId         - Where the new owner ID is returned
58*29492bb7SDavid van Moolenbroek  *
59*29492bb7SDavid van Moolenbroek  * RETURN:      Status
60*29492bb7SDavid van Moolenbroek  *
61*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Allocate a table or method owner ID. The owner ID is used to
62*29492bb7SDavid van Moolenbroek  *              track objects created by the table or method, to be deleted
63*29492bb7SDavid van Moolenbroek  *              when the method exits or the table is unloaded.
64*29492bb7SDavid van Moolenbroek  *
65*29492bb7SDavid van Moolenbroek  ******************************************************************************/
66*29492bb7SDavid van Moolenbroek 
67*29492bb7SDavid van Moolenbroek ACPI_STATUS
AcpiUtAllocateOwnerId(ACPI_OWNER_ID * OwnerId)68*29492bb7SDavid van Moolenbroek AcpiUtAllocateOwnerId (
69*29492bb7SDavid van Moolenbroek     ACPI_OWNER_ID           *OwnerId)
70*29492bb7SDavid van Moolenbroek {
71*29492bb7SDavid van Moolenbroek     UINT32                  i;
72*29492bb7SDavid van Moolenbroek     UINT32                  j;
73*29492bb7SDavid van Moolenbroek     UINT32                  k;
74*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
75*29492bb7SDavid van Moolenbroek 
76*29492bb7SDavid van Moolenbroek 
77*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE (UtAllocateOwnerId);
78*29492bb7SDavid van Moolenbroek 
79*29492bb7SDavid van Moolenbroek 
80*29492bb7SDavid van Moolenbroek     /* Guard against multiple allocations of ID to the same location */
81*29492bb7SDavid van Moolenbroek 
82*29492bb7SDavid van Moolenbroek     if (*OwnerId)
83*29492bb7SDavid van Moolenbroek     {
84*29492bb7SDavid van Moolenbroek         ACPI_ERROR ((AE_INFO, "Owner ID [0x%2.2X] already exists", *OwnerId));
85*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (AE_ALREADY_EXISTS);
86*29492bb7SDavid van Moolenbroek     }
87*29492bb7SDavid van Moolenbroek 
88*29492bb7SDavid van Moolenbroek     /* Mutex for the global ID mask */
89*29492bb7SDavid van Moolenbroek 
90*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_CACHES);
91*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
92*29492bb7SDavid van Moolenbroek     {
93*29492bb7SDavid van Moolenbroek         return_ACPI_STATUS (Status);
94*29492bb7SDavid van Moolenbroek     }
95*29492bb7SDavid van Moolenbroek 
96*29492bb7SDavid van Moolenbroek     /*
97*29492bb7SDavid van Moolenbroek      * Find a free owner ID, cycle through all possible IDs on repeated
98*29492bb7SDavid van Moolenbroek      * allocations. (ACPI_NUM_OWNERID_MASKS + 1) because first index may have
99*29492bb7SDavid van Moolenbroek      * to be scanned twice.
100*29492bb7SDavid van Moolenbroek      */
101*29492bb7SDavid van Moolenbroek     for (i = 0, j = AcpiGbl_LastOwnerIdIndex;
102*29492bb7SDavid van Moolenbroek          i < (ACPI_NUM_OWNERID_MASKS + 1);
103*29492bb7SDavid van Moolenbroek          i++, j++)
104*29492bb7SDavid van Moolenbroek     {
105*29492bb7SDavid van Moolenbroek         if (j >= ACPI_NUM_OWNERID_MASKS)
106*29492bb7SDavid van Moolenbroek         {
107*29492bb7SDavid van Moolenbroek             j = 0;  /* Wraparound to start of mask array */
108*29492bb7SDavid van Moolenbroek         }
109*29492bb7SDavid van Moolenbroek 
110*29492bb7SDavid van Moolenbroek         for (k = AcpiGbl_NextOwnerIdOffset; k < 32; k++)
111*29492bb7SDavid van Moolenbroek         {
112*29492bb7SDavid van Moolenbroek             if (AcpiGbl_OwnerIdMask[j] == ACPI_UINT32_MAX)
113*29492bb7SDavid van Moolenbroek             {
114*29492bb7SDavid van Moolenbroek                 /* There are no free IDs in this mask */
115*29492bb7SDavid van Moolenbroek 
116*29492bb7SDavid van Moolenbroek                 break;
117*29492bb7SDavid van Moolenbroek             }
118*29492bb7SDavid van Moolenbroek 
119*29492bb7SDavid van Moolenbroek             if (!(AcpiGbl_OwnerIdMask[j] & (1 << k)))
120*29492bb7SDavid van Moolenbroek             {
121*29492bb7SDavid van Moolenbroek                 /*
122*29492bb7SDavid van Moolenbroek                  * Found a free ID. The actual ID is the bit index plus one,
123*29492bb7SDavid van Moolenbroek                  * making zero an invalid Owner ID. Save this as the last ID
124*29492bb7SDavid van Moolenbroek                  * allocated and update the global ID mask.
125*29492bb7SDavid van Moolenbroek                  */
126*29492bb7SDavid van Moolenbroek                 AcpiGbl_OwnerIdMask[j] |= (1 << k);
127*29492bb7SDavid van Moolenbroek 
128*29492bb7SDavid van Moolenbroek                 AcpiGbl_LastOwnerIdIndex = (UINT8) j;
129*29492bb7SDavid van Moolenbroek                 AcpiGbl_NextOwnerIdOffset = (UINT8) (k + 1);
130*29492bb7SDavid van Moolenbroek 
131*29492bb7SDavid van Moolenbroek                 /*
132*29492bb7SDavid van Moolenbroek                  * Construct encoded ID from the index and bit position
133*29492bb7SDavid van Moolenbroek                  *
134*29492bb7SDavid van Moolenbroek                  * Note: Last [j].k (bit 255) is never used and is marked
135*29492bb7SDavid van Moolenbroek                  * permanently allocated (prevents +1 overflow)
136*29492bb7SDavid van Moolenbroek                  */
137*29492bb7SDavid van Moolenbroek                 *OwnerId = (ACPI_OWNER_ID) ((k + 1) + ACPI_MUL_32 (j));
138*29492bb7SDavid van Moolenbroek 
139*29492bb7SDavid van Moolenbroek                 ACPI_DEBUG_PRINT ((ACPI_DB_VALUES,
140*29492bb7SDavid van Moolenbroek                     "Allocated OwnerId: %2.2X\n", (unsigned int) *OwnerId));
141*29492bb7SDavid van Moolenbroek                 goto Exit;
142*29492bb7SDavid van Moolenbroek             }
143*29492bb7SDavid van Moolenbroek         }
144*29492bb7SDavid van Moolenbroek 
145*29492bb7SDavid van Moolenbroek         AcpiGbl_NextOwnerIdOffset = 0;
146*29492bb7SDavid van Moolenbroek     }
147*29492bb7SDavid van Moolenbroek 
148*29492bb7SDavid van Moolenbroek     /*
149*29492bb7SDavid van Moolenbroek      * All OwnerIds have been allocated. This typically should
150*29492bb7SDavid van Moolenbroek      * not happen since the IDs are reused after deallocation. The IDs are
151*29492bb7SDavid van Moolenbroek      * allocated upon table load (one per table) and method execution, and
152*29492bb7SDavid van Moolenbroek      * they are released when a table is unloaded or a method completes
153*29492bb7SDavid van Moolenbroek      * execution.
154*29492bb7SDavid van Moolenbroek      *
155*29492bb7SDavid van Moolenbroek      * If this error happens, there may be very deep nesting of invoked control
156*29492bb7SDavid van Moolenbroek      * methods, or there may be a bug where the IDs are not released.
157*29492bb7SDavid van Moolenbroek      */
158*29492bb7SDavid van Moolenbroek     Status = AE_OWNER_ID_LIMIT;
159*29492bb7SDavid van Moolenbroek     ACPI_ERROR ((AE_INFO,
160*29492bb7SDavid van Moolenbroek         "Could not allocate new OwnerId (255 max), AE_OWNER_ID_LIMIT"));
161*29492bb7SDavid van Moolenbroek 
162*29492bb7SDavid van Moolenbroek Exit:
163*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_CACHES);
164*29492bb7SDavid van Moolenbroek     return_ACPI_STATUS (Status);
165*29492bb7SDavid van Moolenbroek }
166*29492bb7SDavid van Moolenbroek 
167*29492bb7SDavid van Moolenbroek 
168*29492bb7SDavid van Moolenbroek /*******************************************************************************
169*29492bb7SDavid van Moolenbroek  *
170*29492bb7SDavid van Moolenbroek  * FUNCTION:    AcpiUtReleaseOwnerId
171*29492bb7SDavid van Moolenbroek  *
172*29492bb7SDavid van Moolenbroek  * PARAMETERS:  OwnerIdPtr          - Pointer to a previously allocated OwnerID
173*29492bb7SDavid van Moolenbroek  *
174*29492bb7SDavid van Moolenbroek  * RETURN:      None. No error is returned because we are either exiting a
175*29492bb7SDavid van Moolenbroek  *              control method or unloading a table. Either way, we would
176*29492bb7SDavid van Moolenbroek  *              ignore any error anyway.
177*29492bb7SDavid van Moolenbroek  *
178*29492bb7SDavid van Moolenbroek  * DESCRIPTION: Release a table or method owner ID. Valid IDs are 1 - 255
179*29492bb7SDavid van Moolenbroek  *
180*29492bb7SDavid van Moolenbroek  ******************************************************************************/
181*29492bb7SDavid van Moolenbroek 
182*29492bb7SDavid van Moolenbroek void
AcpiUtReleaseOwnerId(ACPI_OWNER_ID * OwnerIdPtr)183*29492bb7SDavid van Moolenbroek AcpiUtReleaseOwnerId (
184*29492bb7SDavid van Moolenbroek     ACPI_OWNER_ID           *OwnerIdPtr)
185*29492bb7SDavid van Moolenbroek {
186*29492bb7SDavid van Moolenbroek     ACPI_OWNER_ID           OwnerId = *OwnerIdPtr;
187*29492bb7SDavid van Moolenbroek     ACPI_STATUS             Status;
188*29492bb7SDavid van Moolenbroek     UINT32                  Index;
189*29492bb7SDavid van Moolenbroek     UINT32                  Bit;
190*29492bb7SDavid van Moolenbroek 
191*29492bb7SDavid van Moolenbroek 
192*29492bb7SDavid van Moolenbroek     ACPI_FUNCTION_TRACE_U32 (UtReleaseOwnerId, OwnerId);
193*29492bb7SDavid van Moolenbroek 
194*29492bb7SDavid van Moolenbroek 
195*29492bb7SDavid van Moolenbroek     /* Always clear the input OwnerId (zero is an invalid ID) */
196*29492bb7SDavid van Moolenbroek 
197*29492bb7SDavid van Moolenbroek     *OwnerIdPtr = 0;
198*29492bb7SDavid van Moolenbroek 
199*29492bb7SDavid van Moolenbroek     /* Zero is not a valid OwnerID */
200*29492bb7SDavid van Moolenbroek 
201*29492bb7SDavid van Moolenbroek     if (OwnerId == 0)
202*29492bb7SDavid van Moolenbroek     {
203*29492bb7SDavid van Moolenbroek         ACPI_ERROR ((AE_INFO, "Invalid OwnerId: 0x%2.2X", OwnerId));
204*29492bb7SDavid van Moolenbroek         return_VOID;
205*29492bb7SDavid van Moolenbroek     }
206*29492bb7SDavid van Moolenbroek 
207*29492bb7SDavid van Moolenbroek     /* Mutex for the global ID mask */
208*29492bb7SDavid van Moolenbroek 
209*29492bb7SDavid van Moolenbroek     Status = AcpiUtAcquireMutex (ACPI_MTX_CACHES);
210*29492bb7SDavid van Moolenbroek     if (ACPI_FAILURE (Status))
211*29492bb7SDavid van Moolenbroek     {
212*29492bb7SDavid van Moolenbroek         return_VOID;
213*29492bb7SDavid van Moolenbroek     }
214*29492bb7SDavid van Moolenbroek 
215*29492bb7SDavid van Moolenbroek     /* Normalize the ID to zero */
216*29492bb7SDavid van Moolenbroek 
217*29492bb7SDavid van Moolenbroek     OwnerId--;
218*29492bb7SDavid van Moolenbroek 
219*29492bb7SDavid van Moolenbroek     /* Decode ID to index/offset pair */
220*29492bb7SDavid van Moolenbroek 
221*29492bb7SDavid van Moolenbroek     Index = ACPI_DIV_32 (OwnerId);
222*29492bb7SDavid van Moolenbroek     Bit = 1 << ACPI_MOD_32 (OwnerId);
223*29492bb7SDavid van Moolenbroek 
224*29492bb7SDavid van Moolenbroek     /* Free the owner ID only if it is valid */
225*29492bb7SDavid van Moolenbroek 
226*29492bb7SDavid van Moolenbroek     if (AcpiGbl_OwnerIdMask[Index] & Bit)
227*29492bb7SDavid van Moolenbroek     {
228*29492bb7SDavid van Moolenbroek         AcpiGbl_OwnerIdMask[Index] ^= Bit;
229*29492bb7SDavid van Moolenbroek     }
230*29492bb7SDavid van Moolenbroek     else
231*29492bb7SDavid van Moolenbroek     {
232*29492bb7SDavid van Moolenbroek         ACPI_ERROR ((AE_INFO,
233*29492bb7SDavid van Moolenbroek             "Release of non-allocated OwnerId: 0x%2.2X", OwnerId + 1));
234*29492bb7SDavid van Moolenbroek     }
235*29492bb7SDavid van Moolenbroek 
236*29492bb7SDavid van Moolenbroek     (void) AcpiUtReleaseMutex (ACPI_MTX_CACHES);
237*29492bb7SDavid van Moolenbroek     return_VOID;
238*29492bb7SDavid van Moolenbroek }
239