evhandler.c revision 306536
1/******************************************************************************
2 *
3 * Module Name: evhandler - Support for Address Space handlers
4 *
5 *****************************************************************************/
6
7/*
8 * Copyright (C) 2000 - 2016, Intel Corp.
9 * All rights reserved.
10 *
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
13 * are met:
14 * 1. Redistributions of source code must retain the above copyright
15 *    notice, this list of conditions, and the following disclaimer,
16 *    without modification.
17 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
18 *    substantially similar to the "NO WARRANTY" disclaimer below
19 *    ("Disclaimer") and any redistribution must be conditioned upon
20 *    including a substantially similar Disclaimer requirement for further
21 *    binary redistribution.
22 * 3. Neither the names of the above-listed copyright holders nor the names
23 *    of any contributors may be used to endorse or promote products derived
24 *    from this software without specific prior written permission.
25 *
26 * Alternatively, this software may be distributed under the terms of the
27 * GNU General Public License ("GPL") version 2 as published by the Free
28 * Software Foundation.
29 *
30 * NO WARRANTY
31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
35 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
36 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
37 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
40 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
41 * POSSIBILITY OF SUCH DAMAGES.
42 */
43
44#include <contrib/dev/acpica/include/acpi.h>
45#include <contrib/dev/acpica/include/accommon.h>
46#include <contrib/dev/acpica/include/acevents.h>
47#include <contrib/dev/acpica/include/acnamesp.h>
48#include <contrib/dev/acpica/include/acinterp.h>
49
50#define _COMPONENT          ACPI_EVENTS
51        ACPI_MODULE_NAME    ("evhandler")
52
53
54/* Local prototypes */
55
56static ACPI_STATUS
57AcpiEvInstallHandler (
58    ACPI_HANDLE             ObjHandle,
59    UINT32                  Level,
60    void                    *Context,
61    void                    **ReturnValue);
62
63
64/* These are the address spaces that will get default handlers */
65
66UINT8        AcpiGbl_DefaultAddressSpaces[ACPI_NUM_DEFAULT_SPACES] =
67{
68    ACPI_ADR_SPACE_SYSTEM_MEMORY,
69    ACPI_ADR_SPACE_SYSTEM_IO,
70    ACPI_ADR_SPACE_PCI_CONFIG,
71    ACPI_ADR_SPACE_DATA_TABLE
72};
73
74
75/*******************************************************************************
76 *
77 * FUNCTION:    AcpiEvInstallRegionHandlers
78 *
79 * PARAMETERS:  None
80 *
81 * RETURN:      Status
82 *
83 * DESCRIPTION: Installs the core subsystem default address space handlers.
84 *
85 ******************************************************************************/
86
87ACPI_STATUS
88AcpiEvInstallRegionHandlers (
89    void)
90{
91    ACPI_STATUS             Status;
92    UINT32                  i;
93
94
95    ACPI_FUNCTION_TRACE (EvInstallRegionHandlers);
96
97
98    Status = AcpiUtAcquireMutex (ACPI_MTX_NAMESPACE);
99    if (ACPI_FAILURE (Status))
100    {
101        return_ACPI_STATUS (Status);
102    }
103
104    /*
105     * All address spaces (PCI Config, EC, SMBus) are scope dependent and
106     * registration must occur for a specific device.
107     *
108     * In the case of the system memory and IO address spaces there is
109     * currently no device associated with the address space. For these we
110     * use the root.
111     *
112     * We install the default PCI config space handler at the root so that
113     * this space is immediately available even though the we have not
114     * enumerated all the PCI Root Buses yet. This is to conform to the ACPI
115     * specification which states that the PCI config space must be always
116     * available -- even though we are nowhere near ready to find the PCI root
117     * buses at this point.
118     *
119     * NOTE: We ignore AE_ALREADY_EXISTS because this means that a handler
120     * has already been installed (via AcpiInstallAddressSpaceHandler).
121     * Similar for AE_SAME_HANDLER.
122     */
123    for (i = 0; i < ACPI_NUM_DEFAULT_SPACES; i++)
124    {
125        Status = AcpiEvInstallSpaceHandler (AcpiGbl_RootNode,
126            AcpiGbl_DefaultAddressSpaces[i],
127            ACPI_DEFAULT_HANDLER, NULL, NULL);
128        switch (Status)
129        {
130        case AE_OK:
131        case AE_SAME_HANDLER:
132        case AE_ALREADY_EXISTS:
133
134            /* These exceptions are all OK */
135
136            Status = AE_OK;
137            break;
138
139        default:
140
141            goto UnlockAndExit;
142        }
143    }
144
145UnlockAndExit:
146    (void) AcpiUtReleaseMutex (ACPI_MTX_NAMESPACE);
147    return_ACPI_STATUS (Status);
148}
149
150
151/*******************************************************************************
152 *
153 * FUNCTION:    AcpiEvHasDefaultHandler
154 *
155 * PARAMETERS:  Node                - Namespace node for the device
156 *              SpaceId             - The address space ID
157 *
158 * RETURN:      TRUE if default handler is installed, FALSE otherwise
159 *
160 * DESCRIPTION: Check if the default handler is installed for the requested
161 *              space ID.
162 *
163 ******************************************************************************/
164
165BOOLEAN
166AcpiEvHasDefaultHandler (
167    ACPI_NAMESPACE_NODE     *Node,
168    ACPI_ADR_SPACE_TYPE     SpaceId)
169{
170    ACPI_OPERAND_OBJECT     *ObjDesc;
171    ACPI_OPERAND_OBJECT     *HandlerObj;
172
173
174    /* Must have an existing internal object */
175
176    ObjDesc = AcpiNsGetAttachedObject (Node);
177    if (ObjDesc)
178    {
179        HandlerObj = ObjDesc->CommonNotify.Handler;
180
181        /* Walk the linked list of handlers for this object */
182
183        while (HandlerObj)
184        {
185            if (HandlerObj->AddressSpace.SpaceId == SpaceId)
186            {
187                if (HandlerObj->AddressSpace.HandlerFlags &
188                    ACPI_ADDR_HANDLER_DEFAULT_INSTALLED)
189                {
190                    return (TRUE);
191                }
192            }
193
194            HandlerObj = HandlerObj->AddressSpace.Next;
195        }
196    }
197
198    return (FALSE);
199}
200
201
202/*******************************************************************************
203 *
204 * FUNCTION:    AcpiEvInstallHandler
205 *
206 * PARAMETERS:  WalkNamespace callback
207 *
208 * DESCRIPTION: This routine installs an address handler into objects that are
209 *              of type Region or Device.
210 *
211 *              If the Object is a Device, and the device has a handler of
212 *              the same type then the search is terminated in that branch.
213 *
214 *              This is because the existing handler is closer in proximity
215 *              to any more regions than the one we are trying to install.
216 *
217 ******************************************************************************/
218
219static ACPI_STATUS
220AcpiEvInstallHandler (
221    ACPI_HANDLE             ObjHandle,
222    UINT32                  Level,
223    void                    *Context,
224    void                    **ReturnValue)
225{
226    ACPI_OPERAND_OBJECT     *HandlerObj;
227    ACPI_OPERAND_OBJECT     *NextHandlerObj;
228    ACPI_OPERAND_OBJECT     *ObjDesc;
229    ACPI_NAMESPACE_NODE     *Node;
230    ACPI_STATUS             Status;
231
232
233    ACPI_FUNCTION_NAME (EvInstallHandler);
234
235
236    HandlerObj = (ACPI_OPERAND_OBJECT  *) Context;
237
238    /* Parameter validation */
239
240    if (!HandlerObj)
241    {
242        return (AE_OK);
243    }
244
245    /* Convert and validate the device handle */
246
247    Node = AcpiNsValidateHandle (ObjHandle);
248    if (!Node)
249    {
250        return (AE_BAD_PARAMETER);
251    }
252
253    /*
254     * We only care about regions and objects that are allowed to have
255     * address space handlers
256     */
257    if ((Node->Type != ACPI_TYPE_DEVICE) &&
258        (Node->Type != ACPI_TYPE_REGION) &&
259        (Node != AcpiGbl_RootNode))
260    {
261        return (AE_OK);
262    }
263
264    /* Check for an existing internal object */
265
266    ObjDesc = AcpiNsGetAttachedObject (Node);
267    if (!ObjDesc)
268    {
269        /* No object, just exit */
270
271        return (AE_OK);
272    }
273
274    /* Devices are handled different than regions */
275
276    if (ObjDesc->Common.Type == ACPI_TYPE_DEVICE)
277    {
278        /* Check if this Device already has a handler for this address space */
279
280        NextHandlerObj = AcpiEvFindRegionHandler (
281            HandlerObj->AddressSpace.SpaceId, ObjDesc->CommonNotify.Handler);
282        if (NextHandlerObj)
283        {
284            /* Found a handler, is it for the same address space? */
285
286            ACPI_DEBUG_PRINT ((ACPI_DB_OPREGION,
287                "Found handler for region [%s] in device %p(%p) handler %p\n",
288                AcpiUtGetRegionName (HandlerObj->AddressSpace.SpaceId),
289                ObjDesc, NextHandlerObj, HandlerObj));
290
291            /*
292             * Since the object we found it on was a device, then it means
293             * that someone has already installed a handler for the branch
294             * of the namespace from this device on. Just bail out telling
295             * the walk routine to not traverse this branch. This preserves
296             * the scoping rule for handlers.
297             */
298            return (AE_CTRL_DEPTH);
299        }
300
301        /*
302         * As long as the device didn't have a handler for this space we
303         * don't care about it. We just ignore it and proceed.
304         */
305        return (AE_OK);
306    }
307
308    /* Object is a Region */
309
310    if (ObjDesc->Region.SpaceId != HandlerObj->AddressSpace.SpaceId)
311    {
312        /* This region is for a different address space, just ignore it */
313
314        return (AE_OK);
315    }
316
317    /*
318     * Now we have a region and it is for the handler's address space type.
319     *
320     * First disconnect region for any previous handler (if any)
321     */
322    AcpiEvDetachRegion (ObjDesc, FALSE);
323
324    /* Connect the region to the new handler */
325
326    Status = AcpiEvAttachRegion (HandlerObj, ObjDesc, FALSE);
327    return (Status);
328}
329
330
331/*******************************************************************************
332 *
333 * FUNCTION:    AcpiEvFindRegionHandler
334 *
335 * PARAMETERS:  SpaceId         - The address space ID
336 *              HandlerObj      - Head of the handler object list
337 *
338 * RETURN:      Matching handler object. NULL if space ID not matched
339 *
340 * DESCRIPTION: Search a handler object list for a match on the address
341 *              space ID.
342 *
343 ******************************************************************************/
344
345ACPI_OPERAND_OBJECT *
346AcpiEvFindRegionHandler (
347    ACPI_ADR_SPACE_TYPE     SpaceId,
348    ACPI_OPERAND_OBJECT     *HandlerObj)
349{
350
351    /* Walk the handler list for this device */
352
353    while (HandlerObj)
354    {
355        /* Same SpaceId indicates a handler is installed */
356
357        if (HandlerObj->AddressSpace.SpaceId == SpaceId)
358        {
359            return (HandlerObj);
360        }
361
362        /* Next handler object */
363
364        HandlerObj = HandlerObj->AddressSpace.Next;
365    }
366
367    return (NULL);
368}
369
370
371/*******************************************************************************
372 *
373 * FUNCTION:    AcpiEvInstallSpaceHandler
374 *
375 * PARAMETERS:  Node            - Namespace node for the device
376 *              SpaceId         - The address space ID
377 *              Handler         - Address of the handler
378 *              Setup           - Address of the setup function
379 *              Context         - Value passed to the handler on each access
380 *
381 * RETURN:      Status
382 *
383 * DESCRIPTION: Install a handler for all OpRegions of a given SpaceId.
384 *              Assumes namespace is locked
385 *
386 ******************************************************************************/
387
388ACPI_STATUS
389AcpiEvInstallSpaceHandler (
390    ACPI_NAMESPACE_NODE     *Node,
391    ACPI_ADR_SPACE_TYPE     SpaceId,
392    ACPI_ADR_SPACE_HANDLER  Handler,
393    ACPI_ADR_SPACE_SETUP    Setup,
394    void                    *Context)
395{
396    ACPI_OPERAND_OBJECT     *ObjDesc;
397    ACPI_OPERAND_OBJECT     *HandlerObj;
398    ACPI_STATUS             Status = AE_OK;
399    ACPI_OBJECT_TYPE        Type;
400    UINT8                   Flags = 0;
401
402
403    ACPI_FUNCTION_TRACE (EvInstallSpaceHandler);
404
405
406    /*
407     * This registration is valid for only the types below and the root.
408     * The root node is where the default handlers get installed.
409     */
410    if ((Node->Type != ACPI_TYPE_DEVICE)     &&
411        (Node->Type != ACPI_TYPE_PROCESSOR)  &&
412        (Node->Type != ACPI_TYPE_THERMAL)    &&
413        (Node != AcpiGbl_RootNode))
414    {
415        Status = AE_BAD_PARAMETER;
416        goto UnlockAndExit;
417    }
418
419    if (Handler == ACPI_DEFAULT_HANDLER)
420    {
421        Flags = ACPI_ADDR_HANDLER_DEFAULT_INSTALLED;
422
423        switch (SpaceId)
424        {
425        case ACPI_ADR_SPACE_SYSTEM_MEMORY:
426
427            Handler = AcpiExSystemMemorySpaceHandler;
428            Setup   = AcpiEvSystemMemoryRegionSetup;
429            break;
430
431        case ACPI_ADR_SPACE_SYSTEM_IO:
432
433            Handler = AcpiExSystemIoSpaceHandler;
434            Setup   = AcpiEvIoSpaceRegionSetup;
435            break;
436
437        case ACPI_ADR_SPACE_PCI_CONFIG:
438
439            Handler = AcpiExPciConfigSpaceHandler;
440            Setup   = AcpiEvPciConfigRegionSetup;
441            break;
442
443        case ACPI_ADR_SPACE_CMOS:
444
445            Handler = AcpiExCmosSpaceHandler;
446            Setup   = AcpiEvCmosRegionSetup;
447            break;
448
449        case ACPI_ADR_SPACE_PCI_BAR_TARGET:
450
451            Handler = AcpiExPciBarSpaceHandler;
452            Setup   = AcpiEvPciBarRegionSetup;
453            break;
454
455        case ACPI_ADR_SPACE_DATA_TABLE:
456
457            Handler = AcpiExDataTableSpaceHandler;
458            Setup   = NULL;
459            break;
460
461        default:
462
463            Status = AE_BAD_PARAMETER;
464            goto UnlockAndExit;
465        }
466    }
467
468    /* If the caller hasn't specified a setup routine, use the default */
469
470    if (!Setup)
471    {
472        Setup = AcpiEvDefaultRegionSetup;
473    }
474
475    /* Check for an existing internal object */
476
477    ObjDesc = AcpiNsGetAttachedObject (Node);
478    if (ObjDesc)
479    {
480        /*
481         * The attached device object already exists. Now make sure
482         * the handler is not already installed.
483         */
484        HandlerObj = AcpiEvFindRegionHandler (SpaceId,
485            ObjDesc->CommonNotify.Handler);
486
487        if (HandlerObj)
488        {
489            if (HandlerObj->AddressSpace.Handler == Handler)
490            {
491                /*
492                 * It is (relatively) OK to attempt to install the SAME
493                 * handler twice. This can easily happen with the
494                 * PCI_Config space.
495                 */
496                Status = AE_SAME_HANDLER;
497                goto UnlockAndExit;
498            }
499            else
500            {
501                /* A handler is already installed */
502
503                Status = AE_ALREADY_EXISTS;
504            }
505
506            goto UnlockAndExit;
507        }
508    }
509    else
510    {
511        ACPI_DEBUG_PRINT ((ACPI_DB_OPREGION,
512            "Creating object on Device %p while installing handler\n",
513            Node));
514
515        /* ObjDesc does not exist, create one */
516
517        if (Node->Type == ACPI_TYPE_ANY)
518        {
519            Type = ACPI_TYPE_DEVICE;
520        }
521        else
522        {
523            Type = Node->Type;
524        }
525
526        ObjDesc = AcpiUtCreateInternalObject (Type);
527        if (!ObjDesc)
528        {
529            Status = AE_NO_MEMORY;
530            goto UnlockAndExit;
531        }
532
533        /* Init new descriptor */
534
535        ObjDesc->Common.Type = (UINT8) Type;
536
537        /* Attach the new object to the Node */
538
539        Status = AcpiNsAttachObject (Node, ObjDesc, Type);
540
541        /* Remove local reference to the object */
542
543        AcpiUtRemoveReference (ObjDesc);
544
545        if (ACPI_FAILURE (Status))
546        {
547            goto UnlockAndExit;
548        }
549    }
550
551    ACPI_DEBUG_PRINT ((ACPI_DB_OPREGION,
552        "Installing address handler for region %s(%X) "
553        "on Device %4.4s %p(%p)\n",
554        AcpiUtGetRegionName (SpaceId), SpaceId,
555        AcpiUtGetNodeName (Node), Node, ObjDesc));
556
557    /*
558     * Install the handler
559     *
560     * At this point there is no existing handler. Just allocate the object
561     * for the handler and link it into the list.
562     */
563    HandlerObj = AcpiUtCreateInternalObject (ACPI_TYPE_LOCAL_ADDRESS_HANDLER);
564    if (!HandlerObj)
565    {
566        Status = AE_NO_MEMORY;
567        goto UnlockAndExit;
568    }
569
570    /* Init handler obj */
571
572    HandlerObj->AddressSpace.SpaceId = (UINT8) SpaceId;
573    HandlerObj->AddressSpace.HandlerFlags = Flags;
574    HandlerObj->AddressSpace.RegionList = NULL;
575    HandlerObj->AddressSpace.Node = Node;
576    HandlerObj->AddressSpace.Handler = Handler;
577    HandlerObj->AddressSpace.Context = Context;
578    HandlerObj->AddressSpace.Setup = Setup;
579
580    /* Install at head of Device.AddressSpace list */
581
582    HandlerObj->AddressSpace.Next = ObjDesc->CommonNotify.Handler;
583
584    /*
585     * The Device object is the first reference on the HandlerObj.
586     * Each region that uses the handler adds a reference.
587     */
588    ObjDesc->CommonNotify.Handler = HandlerObj;
589
590    /*
591     * Walk the namespace finding all of the regions this handler will
592     * manage.
593     *
594     * Start at the device and search the branch toward the leaf nodes
595     * until either the leaf is encountered or a device is detected that
596     * has an address handler of the same type.
597     *
598     * In either case, back up and search down the remainder of the branch
599     */
600    Status = AcpiNsWalkNamespace (ACPI_TYPE_ANY, Node,
601        ACPI_UINT32_MAX, ACPI_NS_WALK_UNLOCK,
602        AcpiEvInstallHandler, NULL, HandlerObj, NULL);
603
604UnlockAndExit:
605    return_ACPI_STATUS (Status);
606}
607