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