17c478bd9Sstevel@tonic-gate /******************************************************************************
27c478bd9Sstevel@tonic-gate  *
37c478bd9Sstevel@tonic-gate  * Module Name: exfldio - Aml Field I/O
47c478bd9Sstevel@tonic-gate  *
57c478bd9Sstevel@tonic-gate  *****************************************************************************/
67c478bd9Sstevel@tonic-gate 
7*35786f68SRobert Mustacchi /******************************************************************************
8*35786f68SRobert Mustacchi  *
9*35786f68SRobert Mustacchi  * 1. Copyright Notice
10*35786f68SRobert Mustacchi  *
11*35786f68SRobert Mustacchi  * Some or all of this work - Copyright (c) 1999 - 2018, Intel Corp.
127c478bd9Sstevel@tonic-gate  * All rights reserved.
137c478bd9Sstevel@tonic-gate  *
14*35786f68SRobert Mustacchi  * 2. License
15*35786f68SRobert Mustacchi  *
16*35786f68SRobert Mustacchi  * 2.1. This is your license from Intel Corp. under its intellectual property
17*35786f68SRobert Mustacchi  * rights. You may have additional license terms from the party that provided
18*35786f68SRobert Mustacchi  * you this software, covering your right to use that party's intellectual
19*35786f68SRobert Mustacchi  * property rights.
20*35786f68SRobert Mustacchi  *
21*35786f68SRobert Mustacchi  * 2.2. Intel grants, free of charge, to any person ("Licensee") obtaining a
22*35786f68SRobert Mustacchi  * copy of the source code appearing in this file ("Covered Code") an
23*35786f68SRobert Mustacchi  * irrevocable, perpetual, worldwide license under Intel's copyrights in the
24*35786f68SRobert Mustacchi  * base code distributed originally by Intel ("Original Intel Code") to copy,
25*35786f68SRobert Mustacchi  * make derivatives, distribute, use and display any portion of the Covered
26*35786f68SRobert Mustacchi  * Code in any form, with the right to sublicense such rights; and
27*35786f68SRobert Mustacchi  *
28*35786f68SRobert Mustacchi  * 2.3. Intel grants Licensee a non-exclusive and non-transferable patent
29*35786f68SRobert Mustacchi  * license (with the right to sublicense), under only those claims of Intel
30*35786f68SRobert Mustacchi  * patents that are infringed by the Original Intel Code, to make, use, sell,
31*35786f68SRobert Mustacchi  * offer to sell, and import the Covered Code and derivative works thereof
32*35786f68SRobert Mustacchi  * solely to the minimum extent necessary to exercise the above copyright
33*35786f68SRobert Mustacchi  * license, and in no event shall the patent license extend to any additions
34*35786f68SRobert Mustacchi  * to or modifications of the Original Intel Code. No other license or right
35*35786f68SRobert Mustacchi  * is granted directly or by implication, estoppel or otherwise;
36*35786f68SRobert Mustacchi  *
37*35786f68SRobert Mustacchi  * The above copyright and patent license is granted only if the following
38*35786f68SRobert Mustacchi  * conditions are met:
39*35786f68SRobert Mustacchi  *
40*35786f68SRobert Mustacchi  * 3. Conditions
41*35786f68SRobert Mustacchi  *
42*35786f68SRobert Mustacchi  * 3.1. Redistribution of Source with Rights to Further Distribute Source.
43*35786f68SRobert Mustacchi  * Redistribution of source code of any substantial portion of the Covered
44*35786f68SRobert Mustacchi  * Code or modification with rights to further distribute source must include
45*35786f68SRobert Mustacchi  * the above Copyright Notice, the above License, this list of Conditions,
46*35786f68SRobert Mustacchi  * and the following Disclaimer and Export Compliance provision. In addition,
47*35786f68SRobert Mustacchi  * Licensee must cause all Covered Code to which Licensee contributes to
48*35786f68SRobert Mustacchi  * contain a file documenting the changes Licensee made to create that Covered
49*35786f68SRobert Mustacchi  * Code and the date of any change. Licensee must include in that file the
50*35786f68SRobert Mustacchi  * documentation of any changes made by any predecessor Licensee. Licensee
51*35786f68SRobert Mustacchi  * must include a prominent statement that the modification is derived,
52*35786f68SRobert Mustacchi  * directly or indirectly, from Original Intel Code.
53*35786f68SRobert Mustacchi  *
54*35786f68SRobert Mustacchi  * 3.2. Redistribution of Source with no Rights to Further Distribute Source.
55*35786f68SRobert Mustacchi  * Redistribution of source code of any substantial portion of the Covered
56*35786f68SRobert Mustacchi  * Code or modification without rights to further distribute source must
57*35786f68SRobert Mustacchi  * include the following Disclaimer and Export Compliance provision in the
58*35786f68SRobert Mustacchi  * documentation and/or other materials provided with distribution. In
59*35786f68SRobert Mustacchi  * addition, Licensee may not authorize further sublicense of source of any
60*35786f68SRobert Mustacchi  * portion of the Covered Code, and must include terms to the effect that the
61*35786f68SRobert Mustacchi  * license from Licensee to its licensee is limited to the intellectual
62*35786f68SRobert Mustacchi  * property embodied in the software Licensee provides to its licensee, and
63*35786f68SRobert Mustacchi  * not to intellectual property embodied in modifications its licensee may
64*35786f68SRobert Mustacchi  * make.
65*35786f68SRobert Mustacchi  *
66*35786f68SRobert Mustacchi  * 3.3. Redistribution of Executable. Redistribution in executable form of any
67*35786f68SRobert Mustacchi  * substantial portion of the Covered Code or modification must reproduce the
68*35786f68SRobert Mustacchi  * above Copyright Notice, and the following Disclaimer and Export Compliance
69*35786f68SRobert Mustacchi  * provision in the documentation and/or other materials provided with the
70*35786f68SRobert Mustacchi  * distribution.
71*35786f68SRobert Mustacchi  *
72*35786f68SRobert Mustacchi  * 3.4. Intel retains all right, title, and interest in and to the Original
73*35786f68SRobert Mustacchi  * Intel Code.
74*35786f68SRobert Mustacchi  *
75*35786f68SRobert Mustacchi  * 3.5. Neither the name Intel nor any other trademark owned or controlled by
76*35786f68SRobert Mustacchi  * Intel shall be used in advertising or otherwise to promote the sale, use or
77*35786f68SRobert Mustacchi  * other dealings in products derived from or relating to the Covered Code
78*35786f68SRobert Mustacchi  * without prior written authorization from Intel.
79*35786f68SRobert Mustacchi  *
80*35786f68SRobert Mustacchi  * 4. Disclaimer and Export Compliance
81*35786f68SRobert Mustacchi  *
82*35786f68SRobert Mustacchi  * 4.1. INTEL MAKES NO WARRANTY OF ANY KIND REGARDING ANY SOFTWARE PROVIDED
83*35786f68SRobert Mustacchi  * HERE. ANY SOFTWARE ORIGINATING FROM INTEL OR DERIVED FROM INTEL SOFTWARE
84*35786f68SRobert Mustacchi  * IS PROVIDED "AS IS," AND INTEL WILL NOT PROVIDE ANY SUPPORT, ASSISTANCE,
85*35786f68SRobert Mustacchi  * INSTALLATION, TRAINING OR OTHER SERVICES. INTEL WILL NOT PROVIDE ANY
86*35786f68SRobert Mustacchi  * UPDATES, ENHANCEMENTS OR EXTENSIONS. INTEL SPECIFICALLY DISCLAIMS ANY
87*35786f68SRobert Mustacchi  * IMPLIED WARRANTIES OF MERCHANTABILITY, NONINFRINGEMENT AND FITNESS FOR A
88*35786f68SRobert Mustacchi  * PARTICULAR PURPOSE.
89*35786f68SRobert Mustacchi  *
90*35786f68SRobert Mustacchi  * 4.2. IN NO EVENT SHALL INTEL HAVE ANY LIABILITY TO LICENSEE, ITS LICENSEES
91*35786f68SRobert Mustacchi  * OR ANY OTHER THIRD PARTY, FOR ANY LOST PROFITS, LOST DATA, LOSS OF USE OR
92*35786f68SRobert Mustacchi  * COSTS OF PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, OR FOR ANY INDIRECT,
93*35786f68SRobert Mustacchi  * SPECIAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THIS AGREEMENT, UNDER ANY
94*35786f68SRobert Mustacchi  * CAUSE OF ACTION OR THEORY OF LIABILITY, AND IRRESPECTIVE OF WHETHER INTEL
95*35786f68SRobert Mustacchi  * HAS ADVANCE NOTICE OF THE POSSIBILITY OF SUCH DAMAGES. THESE LIMITATIONS
96*35786f68SRobert Mustacchi  * SHALL APPLY NOTWITHSTANDING THE FAILURE OF THE ESSENTIAL PURPOSE OF ANY
97*35786f68SRobert Mustacchi  * LIMITED REMEDY.
98*35786f68SRobert Mustacchi  *
99*35786f68SRobert Mustacchi  * 4.3. Licensee shall not export, either directly or indirectly, any of this
100*35786f68SRobert Mustacchi  * software or system incorporating such software without first obtaining any
101*35786f68SRobert Mustacchi  * required license or other approval from the U. S. Department of Commerce or
102*35786f68SRobert Mustacchi  * any other agency or department of the United States Government. In the
103*35786f68SRobert Mustacchi  * event Licensee exports any such software from the United States or
104*35786f68SRobert Mustacchi  * re-exports any such software from a foreign destination, Licensee shall
105*35786f68SRobert Mustacchi  * ensure that the distribution and export/re-export of the software is in
106*35786f68SRobert Mustacchi  * compliance with all laws, regulations, orders, or other restrictions of the
107*35786f68SRobert Mustacchi  * U.S. Export Administration Regulations. Licensee agrees that neither it nor
108*35786f68SRobert Mustacchi  * any of its subsidiaries will export/re-export any technical data, process,
109*35786f68SRobert Mustacchi  * software, or service, directly or indirectly, to any country for which the
110*35786f68SRobert Mustacchi  * United States government or any agency thereof requires an export license,
111*35786f68SRobert Mustacchi  * other governmental approval, or letter of assurance, without first obtaining
112*35786f68SRobert Mustacchi  * such license, approval or letter.
113*35786f68SRobert Mustacchi  *
114*35786f68SRobert Mustacchi  *****************************************************************************
115*35786f68SRobert Mustacchi  *
116*35786f68SRobert Mustacchi  * Alternatively, you may choose to be licensed under the terms of the
117*35786f68SRobert Mustacchi  * following license:
118*35786f68SRobert Mustacchi  *
11926f3cdf0SGordon Ross  * Redistribution and use in source and binary forms, with or without
12026f3cdf0SGordon Ross  * modification, are permitted provided that the following conditions
12126f3cdf0SGordon Ross  * are met:
12226f3cdf0SGordon Ross  * 1. Redistributions of source code must retain the above copyright
12326f3cdf0SGordon Ross  *    notice, this list of conditions, and the following disclaimer,
12426f3cdf0SGordon Ross  *    without modification.
12526f3cdf0SGordon Ross  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
12626f3cdf0SGordon Ross  *    substantially similar to the "NO WARRANTY" disclaimer below
12726f3cdf0SGordon Ross  *    ("Disclaimer") and any redistribution must be conditioned upon
12826f3cdf0SGordon Ross  *    including a substantially similar Disclaimer requirement for further
12926f3cdf0SGordon Ross  *    binary redistribution.
13026f3cdf0SGordon Ross  * 3. Neither the names of the above-listed copyright holders nor the names
13126f3cdf0SGordon Ross  *    of any contributors may be used to endorse or promote products derived
13226f3cdf0SGordon Ross  *    from this software without specific prior written permission.
1337c478bd9Sstevel@tonic-gate  *
13426f3cdf0SGordon Ross  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
13526f3cdf0SGordon Ross  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
136*35786f68SRobert Mustacchi  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
13726f3cdf0SGordon Ross  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
138*35786f68SRobert Mustacchi  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
139*35786f68SRobert Mustacchi  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
140*35786f68SRobert Mustacchi  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
141*35786f68SRobert Mustacchi  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
142*35786f68SRobert Mustacchi  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
143*35786f68SRobert Mustacchi  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
144*35786f68SRobert Mustacchi  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
145*35786f68SRobert Mustacchi  *
146*35786f68SRobert Mustacchi  * Alternatively, you may choose to be licensed under the terms of the
147*35786f68SRobert Mustacchi  * GNU General Public License ("GPL") version 2 as published by the Free
148*35786f68SRobert Mustacchi  * Software Foundation.
149*35786f68SRobert Mustacchi  *
150*35786f68SRobert Mustacchi  *****************************************************************************/
1517c478bd9Sstevel@tonic-gate 
1527c478bd9Sstevel@tonic-gate #include "acpi.h"
153aa2aa9a6SDana Myers #include "accommon.h"
1547c478bd9Sstevel@tonic-gate #include "acinterp.h"
1557c478bd9Sstevel@tonic-gate #include "amlcode.h"
1567c478bd9Sstevel@tonic-gate #include "acevents.h"
1577c478bd9Sstevel@tonic-gate #include "acdispat.h"
1587c478bd9Sstevel@tonic-gate 
1597c478bd9Sstevel@tonic-gate 
1607c478bd9Sstevel@tonic-gate #define _COMPONENT          ACPI_EXECUTER
1617c478bd9Sstevel@tonic-gate         ACPI_MODULE_NAME    ("exfldio")
1627c478bd9Sstevel@tonic-gate 
1637c478bd9Sstevel@tonic-gate /* Local prototypes */
1647c478bd9Sstevel@tonic-gate 
1657c478bd9Sstevel@tonic-gate static ACPI_STATUS
1667c478bd9Sstevel@tonic-gate AcpiExFieldDatumIo (
1677c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
1687c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset,
16926f3cdf0SGordon Ross     UINT64                  *Value,
1707c478bd9Sstevel@tonic-gate     UINT32                  ReadWrite);
1717c478bd9Sstevel@tonic-gate 
1727c478bd9Sstevel@tonic-gate static BOOLEAN
1737c478bd9Sstevel@tonic-gate AcpiExRegisterOverflow (
1747c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
17526f3cdf0SGordon Ross     UINT64                  Value);
1767c478bd9Sstevel@tonic-gate 
1777c478bd9Sstevel@tonic-gate static ACPI_STATUS
1787c478bd9Sstevel@tonic-gate AcpiExSetupRegion (
1797c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
1807c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset);
1817c478bd9Sstevel@tonic-gate 
1827c478bd9Sstevel@tonic-gate 
1837c478bd9Sstevel@tonic-gate /*******************************************************************************
1847c478bd9Sstevel@tonic-gate  *
1857c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExSetupRegion
1867c478bd9Sstevel@tonic-gate  *
1877c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc                 - Field to be read or written
1887c478bd9Sstevel@tonic-gate  *              FieldDatumByteOffset    - Byte offset of this datum within the
1897c478bd9Sstevel@tonic-gate  *                                        parent field
1907c478bd9Sstevel@tonic-gate  *
1917c478bd9Sstevel@tonic-gate  * RETURN:      Status
1927c478bd9Sstevel@tonic-gate  *
1937c478bd9Sstevel@tonic-gate  * DESCRIPTION: Common processing for AcpiExExtractFromField and
1947b1019a6SJerry Jelinek  *              AcpiExInsertIntoField. Initialize the Region if necessary and
1957c478bd9Sstevel@tonic-gate  *              validate the request.
1967c478bd9Sstevel@tonic-gate  *
1977c478bd9Sstevel@tonic-gate  ******************************************************************************/
1987c478bd9Sstevel@tonic-gate 
1997c478bd9Sstevel@tonic-gate static ACPI_STATUS
AcpiExSetupRegion(ACPI_OPERAND_OBJECT * ObjDesc,UINT32 FieldDatumByteOffset)2007c478bd9Sstevel@tonic-gate AcpiExSetupRegion (
2017c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
2027c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset)
2037c478bd9Sstevel@tonic-gate {
2047c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status = AE_OK;
2057c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *RgnDesc;
2067b1019a6SJerry Jelinek     UINT8                   SpaceId;
2077c478bd9Sstevel@tonic-gate 
2087c478bd9Sstevel@tonic-gate 
20927f7c583Smyers     ACPI_FUNCTION_TRACE_U32 (ExSetupRegion, FieldDatumByteOffset);
2107c478bd9Sstevel@tonic-gate 
2117c478bd9Sstevel@tonic-gate 
2127c478bd9Sstevel@tonic-gate     RgnDesc = ObjDesc->CommonField.RegionObj;
2137c478bd9Sstevel@tonic-gate 
2147c478bd9Sstevel@tonic-gate     /* We must have a valid region */
2157c478bd9Sstevel@tonic-gate 
216aa2aa9a6SDana Myers     if (RgnDesc->Common.Type != ACPI_TYPE_REGION)
2177c478bd9Sstevel@tonic-gate     {
21826f3cdf0SGordon Ross         ACPI_ERROR ((AE_INFO, "Needed Region, found type 0x%X (%s)",
219aa2aa9a6SDana Myers             RgnDesc->Common.Type,
2207c478bd9Sstevel@tonic-gate             AcpiUtGetObjectTypeName (RgnDesc)));
2217c478bd9Sstevel@tonic-gate 
2227c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (AE_AML_OPERAND_TYPE);
2237c478bd9Sstevel@tonic-gate     }
2247c478bd9Sstevel@tonic-gate 
2257b1019a6SJerry Jelinek     SpaceId = RgnDesc->Region.SpaceId;
2267b1019a6SJerry Jelinek 
2277b1019a6SJerry Jelinek     /* Validate the Space ID */
2287b1019a6SJerry Jelinek 
2297b1019a6SJerry Jelinek     if (!AcpiIsValidSpaceId (SpaceId))
2307b1019a6SJerry Jelinek     {
2317b1019a6SJerry Jelinek         ACPI_ERROR ((AE_INFO,
2327b1019a6SJerry Jelinek             "Invalid/unknown Address Space ID: 0x%2.2X", SpaceId));
2337b1019a6SJerry Jelinek         return_ACPI_STATUS (AE_AML_INVALID_SPACE_ID);
2347b1019a6SJerry Jelinek     }
2357b1019a6SJerry Jelinek 
2367c478bd9Sstevel@tonic-gate     /*
2377c478bd9Sstevel@tonic-gate      * If the Region Address and Length have not been previously evaluated,
2387c478bd9Sstevel@tonic-gate      * evaluate them now and save the results.
2397c478bd9Sstevel@tonic-gate      */
2407c478bd9Sstevel@tonic-gate     if (!(RgnDesc->Common.Flags & AOPOBJ_DATA_VALID))
2417c478bd9Sstevel@tonic-gate     {
2427c478bd9Sstevel@tonic-gate         Status = AcpiDsGetRegionArguments (RgnDesc);
2437c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
2447c478bd9Sstevel@tonic-gate         {
2457c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (Status);
2467c478bd9Sstevel@tonic-gate         }
2477c478bd9Sstevel@tonic-gate     }
2487c478bd9Sstevel@tonic-gate 
24927f7c583Smyers     /*
2507b1019a6SJerry Jelinek      * Exit now for SMBus, GSBus or IPMI address space, it has a non-linear
25126f3cdf0SGordon Ross      * address space and the request cannot be directly validated
25227f7c583Smyers      */
2537b1019a6SJerry Jelinek     if (SpaceId == ACPI_ADR_SPACE_SMBUS ||
2547b1019a6SJerry Jelinek         SpaceId == ACPI_ADR_SPACE_GSBUS ||
2557b1019a6SJerry Jelinek         SpaceId == ACPI_ADR_SPACE_IPMI)
2567c478bd9Sstevel@tonic-gate     {
25757190917SDana Myers         /* SMBus or IPMI has a non-linear address space */
2587c478bd9Sstevel@tonic-gate 
2597c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (AE_OK);
2607c478bd9Sstevel@tonic-gate     }
2617c478bd9Sstevel@tonic-gate 
2627c478bd9Sstevel@tonic-gate #ifdef ACPI_UNDER_DEVELOPMENT
2637c478bd9Sstevel@tonic-gate     /*
2647c478bd9Sstevel@tonic-gate      * If the Field access is AnyAcc, we can now compute the optimal
2657c478bd9Sstevel@tonic-gate      * access (because we know know the length of the parent region)
2667c478bd9Sstevel@tonic-gate      */
2677c478bd9Sstevel@tonic-gate     if (!(ObjDesc->Common.Flags & AOPOBJ_DATA_VALID))
2687c478bd9Sstevel@tonic-gate     {
2697c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
2707c478bd9Sstevel@tonic-gate         {
2717c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (Status);
2727c478bd9Sstevel@tonic-gate         }
2737c478bd9Sstevel@tonic-gate     }
2747c478bd9Sstevel@tonic-gate #endif
2757c478bd9Sstevel@tonic-gate 
2767c478bd9Sstevel@tonic-gate     /*
2777b1019a6SJerry Jelinek      * Validate the request. The entire request from the byte offset for a
2787c478bd9Sstevel@tonic-gate      * length of one field datum (access width) must fit within the region.
2797c478bd9Sstevel@tonic-gate      * (Region length is specified in bytes)
2807c478bd9Sstevel@tonic-gate      */
28127f7c583Smyers     if (RgnDesc->Region.Length <
2827b1019a6SJerry Jelinek         (ObjDesc->CommonField.BaseByteOffset + FieldDatumByteOffset +
2837b1019a6SJerry Jelinek         ObjDesc->CommonField.AccessByteWidth))
2847c478bd9Sstevel@tonic-gate     {
2857c478bd9Sstevel@tonic-gate         if (AcpiGbl_EnableInterpreterSlack)
2867c478bd9Sstevel@tonic-gate         {
2877c478bd9Sstevel@tonic-gate             /*
2887c478bd9Sstevel@tonic-gate              * Slack mode only:  We will go ahead and allow access to this
2897c478bd9Sstevel@tonic-gate              * field if it is within the region length rounded up to the next
290db2bae30SDana Myers              * access width boundary. ACPI_SIZE cast for 64-bit compile.
2917c478bd9Sstevel@tonic-gate              */
2927c478bd9Sstevel@tonic-gate             if (ACPI_ROUND_UP (RgnDesc->Region.Length,
29327f7c583Smyers                     ObjDesc->CommonField.AccessByteWidth) >=
294db2bae30SDana Myers                 ((ACPI_SIZE) ObjDesc->CommonField.BaseByteOffset +
295db2bae30SDana Myers                     ObjDesc->CommonField.AccessByteWidth +
296db2bae30SDana Myers                     FieldDatumByteOffset))
2977c478bd9Sstevel@tonic-gate             {
2987c478bd9Sstevel@tonic-gate                 return_ACPI_STATUS (AE_OK);
2997c478bd9Sstevel@tonic-gate             }
3007c478bd9Sstevel@tonic-gate         }
3017c478bd9Sstevel@tonic-gate 
3027c478bd9Sstevel@tonic-gate         if (RgnDesc->Region.Length < ObjDesc->CommonField.AccessByteWidth)
3037c478bd9Sstevel@tonic-gate         {
3047c478bd9Sstevel@tonic-gate             /*
3057c478bd9Sstevel@tonic-gate              * This is the case where the AccessType (AccWord, etc.) is wider
3067b1019a6SJerry Jelinek              * than the region itself. For example, a region of length one
3077c478bd9Sstevel@tonic-gate              * byte, and a field with Dword access specified.
3087c478bd9Sstevel@tonic-gate              */
30930082d0cSmyers             ACPI_ERROR ((AE_INFO,
3107b1019a6SJerry Jelinek                 "Field [%4.4s] access width (%u bytes) "
3117b1019a6SJerry Jelinek                 "too large for region [%4.4s] (length %u)",
3127c478bd9Sstevel@tonic-gate                 AcpiUtGetNodeName (ObjDesc->CommonField.Node),
3137c478bd9Sstevel@tonic-gate                 ObjDesc->CommonField.AccessByteWidth,
3147c478bd9Sstevel@tonic-gate                 AcpiUtGetNodeName (RgnDesc->Region.Node),
3157c478bd9Sstevel@tonic-gate                 RgnDesc->Region.Length));
3167c478bd9Sstevel@tonic-gate         }
3177c478bd9Sstevel@tonic-gate 
3187c478bd9Sstevel@tonic-gate         /*
3197c478bd9Sstevel@tonic-gate          * Offset rounded up to next multiple of field width
3207c478bd9Sstevel@tonic-gate          * exceeds region length, indicate an error
3217c478bd9Sstevel@tonic-gate          */
32230082d0cSmyers         ACPI_ERROR ((AE_INFO,
3237b1019a6SJerry Jelinek             "Field [%4.4s] Base+Offset+Width %u+%u+%u "
3247b1019a6SJerry Jelinek             "is beyond end of region [%4.4s] (length %u)",
3257c478bd9Sstevel@tonic-gate             AcpiUtGetNodeName (ObjDesc->CommonField.Node),
3267c478bd9Sstevel@tonic-gate             ObjDesc->CommonField.BaseByteOffset,
3277c478bd9Sstevel@tonic-gate             FieldDatumByteOffset, ObjDesc->CommonField.AccessByteWidth,
3287c478bd9Sstevel@tonic-gate             AcpiUtGetNodeName (RgnDesc->Region.Node),
3297c478bd9Sstevel@tonic-gate             RgnDesc->Region.Length));
3307c478bd9Sstevel@tonic-gate 
3317c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (AE_AML_REGION_LIMIT);
3327c478bd9Sstevel@tonic-gate     }
3337c478bd9Sstevel@tonic-gate 
3347c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (AE_OK);
3357c478bd9Sstevel@tonic-gate }
3367c478bd9Sstevel@tonic-gate 
3377c478bd9Sstevel@tonic-gate 
3387c478bd9Sstevel@tonic-gate /*******************************************************************************
3397c478bd9Sstevel@tonic-gate  *
3407c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExAccessRegion
3417c478bd9Sstevel@tonic-gate  *
3427c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc                 - Field to be read
3437c478bd9Sstevel@tonic-gate  *              FieldDatumByteOffset    - Byte offset of this datum within the
3447c478bd9Sstevel@tonic-gate  *                                        parent field
3457c478bd9Sstevel@tonic-gate  *              Value                   - Where to store value (must at least
34626f3cdf0SGordon Ross  *                                        64 bits)
3477c478bd9Sstevel@tonic-gate  *              Function                - Read or Write flag plus other region-
3487c478bd9Sstevel@tonic-gate  *                                        dependent flags
3497c478bd9Sstevel@tonic-gate  *
3507c478bd9Sstevel@tonic-gate  * RETURN:      Status
3517c478bd9Sstevel@tonic-gate  *
3527c478bd9Sstevel@tonic-gate  * DESCRIPTION: Read or Write a single field datum to an Operation Region.
3537c478bd9Sstevel@tonic-gate  *
3547c478bd9Sstevel@tonic-gate  ******************************************************************************/
3557c478bd9Sstevel@tonic-gate 
3567c478bd9Sstevel@tonic-gate ACPI_STATUS
AcpiExAccessRegion(ACPI_OPERAND_OBJECT * ObjDesc,UINT32 FieldDatumByteOffset,UINT64 * Value,UINT32 Function)3577c478bd9Sstevel@tonic-gate AcpiExAccessRegion (
3587c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
3597c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset,
36026f3cdf0SGordon Ross     UINT64                  *Value,
3617c478bd9Sstevel@tonic-gate     UINT32                  Function)
3627c478bd9Sstevel@tonic-gate {
3637c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status;
3647c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *RgnDesc;
365aa2aa9a6SDana Myers     UINT32                  RegionOffset;
3667c478bd9Sstevel@tonic-gate 
3677c478bd9Sstevel@tonic-gate 
36827f7c583Smyers     ACPI_FUNCTION_TRACE (ExAccessRegion);
3697c478bd9Sstevel@tonic-gate 
3707c478bd9Sstevel@tonic-gate 
3717c478bd9Sstevel@tonic-gate     /*
3727c478bd9Sstevel@tonic-gate      * Ensure that the region operands are fully evaluated and verify
3737c478bd9Sstevel@tonic-gate      * the validity of the request
3747c478bd9Sstevel@tonic-gate      */
3757c478bd9Sstevel@tonic-gate     Status = AcpiExSetupRegion (ObjDesc, FieldDatumByteOffset);
3767c478bd9Sstevel@tonic-gate     if (ACPI_FAILURE (Status))
3777c478bd9Sstevel@tonic-gate     {
3787c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (Status);
3797c478bd9Sstevel@tonic-gate     }
3807c478bd9Sstevel@tonic-gate 
3817c478bd9Sstevel@tonic-gate     /*
3827c478bd9Sstevel@tonic-gate      * The physical address of this field datum is:
3837c478bd9Sstevel@tonic-gate      *
3847c478bd9Sstevel@tonic-gate      * 1) The base of the region, plus
3857c478bd9Sstevel@tonic-gate      * 2) The base offset of the field, plus
3867c478bd9Sstevel@tonic-gate      * 3) The current offset into the field
3877c478bd9Sstevel@tonic-gate      */
3887c478bd9Sstevel@tonic-gate     RgnDesc = ObjDesc->CommonField.RegionObj;
389aa2aa9a6SDana Myers     RegionOffset =
390aa2aa9a6SDana Myers         ObjDesc->CommonField.BaseByteOffset +
391aa2aa9a6SDana Myers         FieldDatumByteOffset;
3927c478bd9Sstevel@tonic-gate 
3937c478bd9Sstevel@tonic-gate     if ((Function & ACPI_IO_MASK) == ACPI_READ)
3947c478bd9Sstevel@tonic-gate     {
3957c478bd9Sstevel@tonic-gate         ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD, "[READ]"));
3967c478bd9Sstevel@tonic-gate     }
3977c478bd9Sstevel@tonic-gate     else
3987c478bd9Sstevel@tonic-gate     {
3997c478bd9Sstevel@tonic-gate         ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD, "[WRITE]"));
4007c478bd9Sstevel@tonic-gate     }
4017c478bd9Sstevel@tonic-gate 
4027c478bd9Sstevel@tonic-gate     ACPI_DEBUG_PRINT_RAW ((ACPI_DB_BFIELD,
4037b1019a6SJerry Jelinek         " Region [%s:%X], Width %X, ByteBase %X, Offset %X at %8.8X%8.8X\n",
4047c478bd9Sstevel@tonic-gate         AcpiUtGetRegionName (RgnDesc->Region.SpaceId),
4057c478bd9Sstevel@tonic-gate         RgnDesc->Region.SpaceId,
4067c478bd9Sstevel@tonic-gate         ObjDesc->CommonField.AccessByteWidth,
4077c478bd9Sstevel@tonic-gate         ObjDesc->CommonField.BaseByteOffset,
4087c478bd9Sstevel@tonic-gate         FieldDatumByteOffset,
4097b1019a6SJerry Jelinek         ACPI_FORMAT_UINT64 (RgnDesc->Region.Address + RegionOffset)));
4107c478bd9Sstevel@tonic-gate 
4117c478bd9Sstevel@tonic-gate     /* Invoke the appropriate AddressSpace/OpRegion handler */
4127c478bd9Sstevel@tonic-gate 
4137b1019a6SJerry Jelinek     Status = AcpiEvAddressSpaceDispatch (RgnDesc, ObjDesc,
4147b1019a6SJerry Jelinek         Function, RegionOffset,
4157b1019a6SJerry Jelinek         ACPI_MUL_8 (ObjDesc->CommonField.AccessByteWidth), Value);
4167c478bd9Sstevel@tonic-gate 
4177c478bd9Sstevel@tonic-gate     if (ACPI_FAILURE (Status))
4187c478bd9Sstevel@tonic-gate     {
4197c478bd9Sstevel@tonic-gate         if (Status == AE_NOT_IMPLEMENTED)
4207c478bd9Sstevel@tonic-gate         {
42130082d0cSmyers             ACPI_ERROR ((AE_INFO,
42226f3cdf0SGordon Ross                 "Region %s (ID=%u) not implemented",
4237c478bd9Sstevel@tonic-gate                 AcpiUtGetRegionName (RgnDesc->Region.SpaceId),
4247c478bd9Sstevel@tonic-gate                 RgnDesc->Region.SpaceId));
4257c478bd9Sstevel@tonic-gate         }
4267c478bd9Sstevel@tonic-gate         else if (Status == AE_NOT_EXIST)
4277c478bd9Sstevel@tonic-gate         {
42830082d0cSmyers             ACPI_ERROR ((AE_INFO,
42926f3cdf0SGordon Ross                 "Region %s (ID=%u) has no handler",
4307c478bd9Sstevel@tonic-gate                 AcpiUtGetRegionName (RgnDesc->Region.SpaceId),
4317c478bd9Sstevel@tonic-gate                 RgnDesc->Region.SpaceId));
4327c478bd9Sstevel@tonic-gate         }
4337c478bd9Sstevel@tonic-gate     }
4347c478bd9Sstevel@tonic-gate 
4357c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (Status);
4367c478bd9Sstevel@tonic-gate }
4377c478bd9Sstevel@tonic-gate 
4387c478bd9Sstevel@tonic-gate 
4397c478bd9Sstevel@tonic-gate /*******************************************************************************
4407c478bd9Sstevel@tonic-gate  *
4417c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExRegisterOverflow
4427c478bd9Sstevel@tonic-gate  *
4437c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc                 - Register(Field) to be written
4447c478bd9Sstevel@tonic-gate  *              Value                   - Value to be stored
4457c478bd9Sstevel@tonic-gate  *
4467c478bd9Sstevel@tonic-gate  * RETURN:      TRUE if value overflows the field, FALSE otherwise
4477c478bd9Sstevel@tonic-gate  *
4487c478bd9Sstevel@tonic-gate  * DESCRIPTION: Check if a value is out of range of the field being written.
4497c478bd9Sstevel@tonic-gate  *              Used to check if the values written to Index and Bank registers
4507b1019a6SJerry Jelinek  *              are out of range. Normally, the value is simply truncated
4517c478bd9Sstevel@tonic-gate  *              to fit the field, but this case is most likely a serious
4527c478bd9Sstevel@tonic-gate  *              coding error in the ASL.
4537c478bd9Sstevel@tonic-gate  *
4547c478bd9Sstevel@tonic-gate  ******************************************************************************/
4557c478bd9Sstevel@tonic-gate 
4567c478bd9Sstevel@tonic-gate static BOOLEAN
AcpiExRegisterOverflow(ACPI_OPERAND_OBJECT * ObjDesc,UINT64 Value)4577c478bd9Sstevel@tonic-gate AcpiExRegisterOverflow (
4587c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
45926f3cdf0SGordon Ross     UINT64                  Value)
4607c478bd9Sstevel@tonic-gate {
4617c478bd9Sstevel@tonic-gate 
4627c478bd9Sstevel@tonic-gate     if (ObjDesc->CommonField.BitLength >= ACPI_INTEGER_BIT_SIZE)
4637c478bd9Sstevel@tonic-gate     {
4647c478bd9Sstevel@tonic-gate         /*
4657c478bd9Sstevel@tonic-gate          * The field is large enough to hold the maximum integer, so we can
4667c478bd9Sstevel@tonic-gate          * never overflow it.
4677c478bd9Sstevel@tonic-gate          */
4687c478bd9Sstevel@tonic-gate         return (FALSE);
4697c478bd9Sstevel@tonic-gate     }
4707c478bd9Sstevel@tonic-gate 
47126f3cdf0SGordon Ross     if (Value >= ((UINT64) 1 << ObjDesc->CommonField.BitLength))
4727c478bd9Sstevel@tonic-gate     {
4737c478bd9Sstevel@tonic-gate         /*
4747c478bd9Sstevel@tonic-gate          * The Value is larger than the maximum value that can fit into
4757c478bd9Sstevel@tonic-gate          * the register.
4767c478bd9Sstevel@tonic-gate          */
4777b1019a6SJerry Jelinek         ACPI_ERROR ((AE_INFO,
4787b1019a6SJerry Jelinek             "Index value 0x%8.8X%8.8X overflows field width 0x%X",
4797b1019a6SJerry Jelinek             ACPI_FORMAT_UINT64 (Value),
4807b1019a6SJerry Jelinek             ObjDesc->CommonField.BitLength));
4817b1019a6SJerry Jelinek 
4827c478bd9Sstevel@tonic-gate         return (TRUE);
4837c478bd9Sstevel@tonic-gate     }
4847c478bd9Sstevel@tonic-gate 
4857c478bd9Sstevel@tonic-gate     /* The Value will fit into the field with no truncation */
4867c478bd9Sstevel@tonic-gate 
4877c478bd9Sstevel@tonic-gate     return (FALSE);
4887c478bd9Sstevel@tonic-gate }
4897c478bd9Sstevel@tonic-gate 
4907c478bd9Sstevel@tonic-gate 
4917c478bd9Sstevel@tonic-gate /*******************************************************************************
4927c478bd9Sstevel@tonic-gate  *
4937c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExFieldDatumIo
4947c478bd9Sstevel@tonic-gate  *
4957c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc                 - Field to be read
4967c478bd9Sstevel@tonic-gate  *              FieldDatumByteOffset    - Byte offset of this datum within the
4977c478bd9Sstevel@tonic-gate  *                                        parent field
4987c478bd9Sstevel@tonic-gate  *              Value                   - Where to store value (must be 64 bits)
4997c478bd9Sstevel@tonic-gate  *              ReadWrite               - Read or Write flag
5007c478bd9Sstevel@tonic-gate  *
5017c478bd9Sstevel@tonic-gate  * RETURN:      Status
5027c478bd9Sstevel@tonic-gate  *
5037b1019a6SJerry Jelinek  * DESCRIPTION: Read or Write a single datum of a field. The FieldType is
5047c478bd9Sstevel@tonic-gate  *              demultiplexed here to handle the different types of fields
5057c478bd9Sstevel@tonic-gate  *              (BufferField, RegionField, IndexField, BankField)
5067c478bd9Sstevel@tonic-gate  *
5077c478bd9Sstevel@tonic-gate  ******************************************************************************/
5087c478bd9Sstevel@tonic-gate 
5097c478bd9Sstevel@tonic-gate static ACPI_STATUS
AcpiExFieldDatumIo(ACPI_OPERAND_OBJECT * ObjDesc,UINT32 FieldDatumByteOffset,UINT64 * Value,UINT32 ReadWrite)5107c478bd9Sstevel@tonic-gate AcpiExFieldDatumIo (
5117c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
5127c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset,
51326f3cdf0SGordon Ross     UINT64                  *Value,
5147c478bd9Sstevel@tonic-gate     UINT32                  ReadWrite)
5157c478bd9Sstevel@tonic-gate {
5167c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status;
51726f3cdf0SGordon Ross     UINT64                  LocalValue;
5187c478bd9Sstevel@tonic-gate 
5197c478bd9Sstevel@tonic-gate 
52027f7c583Smyers     ACPI_FUNCTION_TRACE_U32 (ExFieldDatumIo, FieldDatumByteOffset);
5217c478bd9Sstevel@tonic-gate 
5227c478bd9Sstevel@tonic-gate 
5237c478bd9Sstevel@tonic-gate     if (ReadWrite == ACPI_READ)
5247c478bd9Sstevel@tonic-gate     {
5257c478bd9Sstevel@tonic-gate         if (!Value)
5267c478bd9Sstevel@tonic-gate         {
5277c478bd9Sstevel@tonic-gate             LocalValue = 0;
5287c478bd9Sstevel@tonic-gate 
5297c478bd9Sstevel@tonic-gate             /* To support reads without saving return value */
5307c478bd9Sstevel@tonic-gate             Value = &LocalValue;
5317c478bd9Sstevel@tonic-gate         }
5327c478bd9Sstevel@tonic-gate 
5337c478bd9Sstevel@tonic-gate         /* Clear the entire return buffer first, [Very Important!] */
5347c478bd9Sstevel@tonic-gate 
5357c478bd9Sstevel@tonic-gate         *Value = 0;
5367c478bd9Sstevel@tonic-gate     }
5377c478bd9Sstevel@tonic-gate 
5387c478bd9Sstevel@tonic-gate     /*
5397c478bd9Sstevel@tonic-gate      * The four types of fields are:
5407c478bd9Sstevel@tonic-gate      *
5417c478bd9Sstevel@tonic-gate      * BufferField - Read/write from/to a Buffer
5427c478bd9Sstevel@tonic-gate      * RegionField - Read/write from/to a Operation Region.
5437c478bd9Sstevel@tonic-gate      * BankField   - Write to a Bank Register, then read/write from/to an
5447c478bd9Sstevel@tonic-gate      *               OperationRegion
5457c478bd9Sstevel@tonic-gate      * IndexField  - Write to an Index Register, then read/write from/to a
5467c478bd9Sstevel@tonic-gate      *               Data Register
5477c478bd9Sstevel@tonic-gate      */
548aa2aa9a6SDana Myers     switch (ObjDesc->Common.Type)
5497c478bd9Sstevel@tonic-gate     {
5507c478bd9Sstevel@tonic-gate     case ACPI_TYPE_BUFFER_FIELD:
5517c478bd9Sstevel@tonic-gate         /*
5527c478bd9Sstevel@tonic-gate          * If the BufferField arguments have not been previously evaluated,
5537c478bd9Sstevel@tonic-gate          * evaluate them now and save the results.
5547c478bd9Sstevel@tonic-gate          */
5557c478bd9Sstevel@tonic-gate         if (!(ObjDesc->Common.Flags & AOPOBJ_DATA_VALID))
5567c478bd9Sstevel@tonic-gate         {
5577c478bd9Sstevel@tonic-gate             Status = AcpiDsGetBufferFieldArguments (ObjDesc);
5587c478bd9Sstevel@tonic-gate             if (ACPI_FAILURE (Status))
5597c478bd9Sstevel@tonic-gate             {
5607c478bd9Sstevel@tonic-gate                 return_ACPI_STATUS (Status);
5617c478bd9Sstevel@tonic-gate             }
5627c478bd9Sstevel@tonic-gate         }
5637c478bd9Sstevel@tonic-gate 
5647c478bd9Sstevel@tonic-gate         if (ReadWrite == ACPI_READ)
5657c478bd9Sstevel@tonic-gate         {
5667c478bd9Sstevel@tonic-gate             /*
5677c478bd9Sstevel@tonic-gate              * Copy the data from the source buffer.
5687c478bd9Sstevel@tonic-gate              * Length is the field width in bytes.
5697c478bd9Sstevel@tonic-gate              */
5707b1019a6SJerry Jelinek             memcpy (Value,
5717c478bd9Sstevel@tonic-gate                 (ObjDesc->BufferField.BufferObj)->Buffer.Pointer +
5727c478bd9Sstevel@tonic-gate                     ObjDesc->BufferField.BaseByteOffset +
5737c478bd9Sstevel@tonic-gate                     FieldDatumByteOffset,
5747c478bd9Sstevel@tonic-gate                 ObjDesc->CommonField.AccessByteWidth);
5757c478bd9Sstevel@tonic-gate         }
5767c478bd9Sstevel@tonic-gate         else
5777c478bd9Sstevel@tonic-gate         {
5787c478bd9Sstevel@tonic-gate             /*
5797c478bd9Sstevel@tonic-gate              * Copy the data to the target buffer.
5807c478bd9Sstevel@tonic-gate              * Length is the field width in bytes.
5817c478bd9Sstevel@tonic-gate              */
5827b1019a6SJerry Jelinek             memcpy ((ObjDesc->BufferField.BufferObj)->Buffer.Pointer +
58327f7c583Smyers                 ObjDesc->BufferField.BaseByteOffset +
58427f7c583Smyers                 FieldDatumByteOffset,
58527f7c583Smyers                 Value, ObjDesc->CommonField.AccessByteWidth);
5867c478bd9Sstevel@tonic-gate         }
5877c478bd9Sstevel@tonic-gate 
5887c478bd9Sstevel@tonic-gate         Status = AE_OK;
5897c478bd9Sstevel@tonic-gate         break;
5907c478bd9Sstevel@tonic-gate 
5917c478bd9Sstevel@tonic-gate     case ACPI_TYPE_LOCAL_BANK_FIELD:
5927c478bd9Sstevel@tonic-gate         /*
5937c478bd9Sstevel@tonic-gate          * Ensure that the BankValue is not beyond the capacity of
5947c478bd9Sstevel@tonic-gate          * the register
5957c478bd9Sstevel@tonic-gate          */
5967c478bd9Sstevel@tonic-gate         if (AcpiExRegisterOverflow (ObjDesc->BankField.BankObj,
59726f3cdf0SGordon Ross                 (UINT64) ObjDesc->BankField.Value))
5987c478bd9Sstevel@tonic-gate         {
5997c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (AE_AML_REGISTER_LIMIT);
6007c478bd9Sstevel@tonic-gate         }
6017c478bd9Sstevel@tonic-gate 
6027c478bd9Sstevel@tonic-gate         /*
6037c478bd9Sstevel@tonic-gate          * For BankFields, we must write the BankValue to the BankRegister
6047c478bd9Sstevel@tonic-gate          * (itself a RegionField) before we can access the data.
6057c478bd9Sstevel@tonic-gate          */
6067c478bd9Sstevel@tonic-gate         Status = AcpiExInsertIntoField (ObjDesc->BankField.BankObj,
60727f7c583Smyers                     &ObjDesc->BankField.Value,
60827f7c583Smyers                     sizeof (ObjDesc->BankField.Value));
6097c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
6107c478bd9Sstevel@tonic-gate         {
6117c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (Status);
6127c478bd9Sstevel@tonic-gate         }
6137c478bd9Sstevel@tonic-gate 
6147c478bd9Sstevel@tonic-gate         /*
6157c478bd9Sstevel@tonic-gate          * Now that the Bank has been selected, fall through to the
6167c478bd9Sstevel@tonic-gate          * RegionField case and write the datum to the Operation Region
6177c478bd9Sstevel@tonic-gate          */
6187c478bd9Sstevel@tonic-gate 
6197c478bd9Sstevel@tonic-gate         /*lint -fallthrough */
6207c478bd9Sstevel@tonic-gate 
6217c478bd9Sstevel@tonic-gate     case ACPI_TYPE_LOCAL_REGION_FIELD:
6227c478bd9Sstevel@tonic-gate         /*
6237c478bd9Sstevel@tonic-gate          * For simple RegionFields, we just directly access the owning
6247c478bd9Sstevel@tonic-gate          * Operation Region.
6257c478bd9Sstevel@tonic-gate          */
6267b1019a6SJerry Jelinek         Status = AcpiExAccessRegion (
6277b1019a6SJerry Jelinek             ObjDesc, FieldDatumByteOffset, Value, ReadWrite);
6287c478bd9Sstevel@tonic-gate         break;
6297c478bd9Sstevel@tonic-gate 
6307c478bd9Sstevel@tonic-gate     case ACPI_TYPE_LOCAL_INDEX_FIELD:
6317c478bd9Sstevel@tonic-gate         /*
6327c478bd9Sstevel@tonic-gate          * Ensure that the IndexValue is not beyond the capacity of
6337c478bd9Sstevel@tonic-gate          * the register
6347c478bd9Sstevel@tonic-gate          */
6357c478bd9Sstevel@tonic-gate         if (AcpiExRegisterOverflow (ObjDesc->IndexField.IndexObj,
63626f3cdf0SGordon Ross                 (UINT64) ObjDesc->IndexField.Value))
6377c478bd9Sstevel@tonic-gate         {
6387c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (AE_AML_REGISTER_LIMIT);
6397c478bd9Sstevel@tonic-gate         }
6407c478bd9Sstevel@tonic-gate 
6417c478bd9Sstevel@tonic-gate         /* Write the index value to the IndexRegister (itself a RegionField) */
6427c478bd9Sstevel@tonic-gate 
6437c478bd9Sstevel@tonic-gate         FieldDatumByteOffset += ObjDesc->IndexField.Value;
6447c478bd9Sstevel@tonic-gate 
6457c478bd9Sstevel@tonic-gate         ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
64627f7c583Smyers             "Write to Index Register: Value %8.8X\n",
64727f7c583Smyers             FieldDatumByteOffset));
6487c478bd9Sstevel@tonic-gate 
6497c478bd9Sstevel@tonic-gate         Status = AcpiExInsertIntoField (ObjDesc->IndexField.IndexObj,
6507b1019a6SJerry Jelinek             &FieldDatumByteOffset, sizeof (FieldDatumByteOffset));
6517c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
6527c478bd9Sstevel@tonic-gate         {
6537c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (Status);
6547c478bd9Sstevel@tonic-gate         }
6557c478bd9Sstevel@tonic-gate 
6567c478bd9Sstevel@tonic-gate         if (ReadWrite == ACPI_READ)
6577c478bd9Sstevel@tonic-gate         {
6587c478bd9Sstevel@tonic-gate             /* Read the datum from the DataRegister */
6597c478bd9Sstevel@tonic-gate 
660aa2aa9a6SDana Myers             ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
661aa2aa9a6SDana Myers                 "Read from Data Register\n"));
662aa2aa9a6SDana Myers 
6637b1019a6SJerry Jelinek             Status = AcpiExExtractFromField (
6647b1019a6SJerry Jelinek                 ObjDesc->IndexField.DataObj, Value, sizeof (UINT64));
6657c478bd9Sstevel@tonic-gate         }
6667c478bd9Sstevel@tonic-gate         else
6677c478bd9Sstevel@tonic-gate         {
6687c478bd9Sstevel@tonic-gate             /* Write the datum to the DataRegister */
6697c478bd9Sstevel@tonic-gate 
670aa2aa9a6SDana Myers             ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
671aa2aa9a6SDana Myers                 "Write to Data Register: Value %8.8X%8.8X\n",
672aa2aa9a6SDana Myers                 ACPI_FORMAT_UINT64 (*Value)));
673aa2aa9a6SDana Myers 
6747b1019a6SJerry Jelinek             Status = AcpiExInsertIntoField (
6757b1019a6SJerry Jelinek                 ObjDesc->IndexField.DataObj, Value, sizeof (UINT64));
6767c478bd9Sstevel@tonic-gate         }
6777c478bd9Sstevel@tonic-gate         break;
6787c478bd9Sstevel@tonic-gate 
6797c478bd9Sstevel@tonic-gate     default:
6807c478bd9Sstevel@tonic-gate 
68126f3cdf0SGordon Ross         ACPI_ERROR ((AE_INFO, "Wrong object type in field I/O %u",
682aa2aa9a6SDana Myers             ObjDesc->Common.Type));
6837c478bd9Sstevel@tonic-gate         Status = AE_AML_INTERNAL;
6847c478bd9Sstevel@tonic-gate         break;
6857c478bd9Sstevel@tonic-gate     }
6867c478bd9Sstevel@tonic-gate 
6877c478bd9Sstevel@tonic-gate     if (ACPI_SUCCESS (Status))
6887c478bd9Sstevel@tonic-gate     {
6897c478bd9Sstevel@tonic-gate         if (ReadWrite == ACPI_READ)
6907c478bd9Sstevel@tonic-gate         {
6917c478bd9Sstevel@tonic-gate             ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
69226f3cdf0SGordon Ross                 "Value Read %8.8X%8.8X, Width %u\n",
6937c478bd9Sstevel@tonic-gate                 ACPI_FORMAT_UINT64 (*Value),
6947c478bd9Sstevel@tonic-gate                 ObjDesc->CommonField.AccessByteWidth));
6957c478bd9Sstevel@tonic-gate         }
6967c478bd9Sstevel@tonic-gate         else
6977c478bd9Sstevel@tonic-gate         {
6987c478bd9Sstevel@tonic-gate             ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
69926f3cdf0SGordon Ross                 "Value Written %8.8X%8.8X, Width %u\n",
7007c478bd9Sstevel@tonic-gate                 ACPI_FORMAT_UINT64 (*Value),
7017c478bd9Sstevel@tonic-gate                 ObjDesc->CommonField.AccessByteWidth));
7027c478bd9Sstevel@tonic-gate         }
7037c478bd9Sstevel@tonic-gate     }
7047c478bd9Sstevel@tonic-gate 
7057c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (Status);
7067c478bd9Sstevel@tonic-gate }
7077c478bd9Sstevel@tonic-gate 
7087c478bd9Sstevel@tonic-gate 
7097c478bd9Sstevel@tonic-gate /*******************************************************************************
7107c478bd9Sstevel@tonic-gate  *
7117c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExWriteWithUpdateRule
7127c478bd9Sstevel@tonic-gate  *
7137c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc                 - Field to be written
7147c478bd9Sstevel@tonic-gate  *              Mask                    - bitmask within field datum
7157c478bd9Sstevel@tonic-gate  *              FieldValue              - Value to write
7167c478bd9Sstevel@tonic-gate  *              FieldDatumByteOffset    - Offset of datum within field
7177c478bd9Sstevel@tonic-gate  *
7187c478bd9Sstevel@tonic-gate  * RETURN:      Status
7197c478bd9Sstevel@tonic-gate  *
7207c478bd9Sstevel@tonic-gate  * DESCRIPTION: Apply the field update rule to a field write
7217c478bd9Sstevel@tonic-gate  *
7227c478bd9Sstevel@tonic-gate  ******************************************************************************/
7237c478bd9Sstevel@tonic-gate 
7247c478bd9Sstevel@tonic-gate ACPI_STATUS
AcpiExWriteWithUpdateRule(ACPI_OPERAND_OBJECT * ObjDesc,UINT64 Mask,UINT64 FieldValue,UINT32 FieldDatumByteOffset)7257c478bd9Sstevel@tonic-gate AcpiExWriteWithUpdateRule (
7267c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
72726f3cdf0SGordon Ross     UINT64                  Mask,
72826f3cdf0SGordon Ross     UINT64                  FieldValue,
7297c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumByteOffset)
7307c478bd9Sstevel@tonic-gate {
7317c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status = AE_OK;
73226f3cdf0SGordon Ross     UINT64                  MergedValue;
73326f3cdf0SGordon Ross     UINT64                  CurrentValue;
7347c478bd9Sstevel@tonic-gate 
7357c478bd9Sstevel@tonic-gate 
73627f7c583Smyers     ACPI_FUNCTION_TRACE_U32 (ExWriteWithUpdateRule, Mask);
7377c478bd9Sstevel@tonic-gate 
7387c478bd9Sstevel@tonic-gate 
7397c478bd9Sstevel@tonic-gate     /* Start with the new bits  */
7407c478bd9Sstevel@tonic-gate 
7417c478bd9Sstevel@tonic-gate     MergedValue = FieldValue;
7427c478bd9Sstevel@tonic-gate 
7437c478bd9Sstevel@tonic-gate     /* If the mask is all ones, we don't need to worry about the update rule */
7447c478bd9Sstevel@tonic-gate 
74526f3cdf0SGordon Ross     if (Mask != ACPI_UINT64_MAX)
7467c478bd9Sstevel@tonic-gate     {
7477c478bd9Sstevel@tonic-gate         /* Decode the update rule */
7487c478bd9Sstevel@tonic-gate 
7497c478bd9Sstevel@tonic-gate         switch (ObjDesc->CommonField.FieldFlags & AML_FIELD_UPDATE_RULE_MASK)
7507c478bd9Sstevel@tonic-gate         {
7517c478bd9Sstevel@tonic-gate         case AML_FIELD_UPDATE_PRESERVE:
7527c478bd9Sstevel@tonic-gate             /*
7537c478bd9Sstevel@tonic-gate              * Check if update rule needs to be applied (not if mask is all
7547c478bd9Sstevel@tonic-gate              * ones)  The left shift drops the bits we want to ignore.
7557c478bd9Sstevel@tonic-gate              */
7567c478bd9Sstevel@tonic-gate             if ((~Mask << (ACPI_MUL_8 (sizeof (Mask)) -
7577b1019a6SJerry Jelinek                    ACPI_MUL_8 (ObjDesc->CommonField.AccessByteWidth))) != 0)
7587c478bd9Sstevel@tonic-gate             {
7597c478bd9Sstevel@tonic-gate                 /*
7607c478bd9Sstevel@tonic-gate                  * Read the current contents of the byte/word/dword containing
7617c478bd9Sstevel@tonic-gate                  * the field, and merge with the new field value.
7627c478bd9Sstevel@tonic-gate                  */
7637b1019a6SJerry Jelinek                 Status = AcpiExFieldDatumIo (
7647b1019a6SJerry Jelinek                     ObjDesc, FieldDatumByteOffset, &CurrentValue, ACPI_READ);
7657c478bd9Sstevel@tonic-gate                 if (ACPI_FAILURE (Status))
7667c478bd9Sstevel@tonic-gate                 {
7677c478bd9Sstevel@tonic-gate                     return_ACPI_STATUS (Status);
7687c478bd9Sstevel@tonic-gate                 }
7697c478bd9Sstevel@tonic-gate 
7707c478bd9Sstevel@tonic-gate                 MergedValue |= (CurrentValue & ~Mask);
7717c478bd9Sstevel@tonic-gate             }
7727c478bd9Sstevel@tonic-gate             break;
7737c478bd9Sstevel@tonic-gate 
7747c478bd9Sstevel@tonic-gate         case AML_FIELD_UPDATE_WRITE_AS_ONES:
7757c478bd9Sstevel@tonic-gate 
7767c478bd9Sstevel@tonic-gate             /* Set positions outside the field to all ones */
7777c478bd9Sstevel@tonic-gate 
7787c478bd9Sstevel@tonic-gate             MergedValue |= ~Mask;
7797c478bd9Sstevel@tonic-gate             break;
7807c478bd9Sstevel@tonic-gate 
7817c478bd9Sstevel@tonic-gate         case AML_FIELD_UPDATE_WRITE_AS_ZEROS:
7827c478bd9Sstevel@tonic-gate 
7837c478bd9Sstevel@tonic-gate             /* Set positions outside the field to all zeros */
7847c478bd9Sstevel@tonic-gate 
7857c478bd9Sstevel@tonic-gate             MergedValue &= Mask;
7867c478bd9Sstevel@tonic-gate             break;
7877c478bd9Sstevel@tonic-gate 
7887c478bd9Sstevel@tonic-gate         default:
7897c478bd9Sstevel@tonic-gate 
79030082d0cSmyers             ACPI_ERROR ((AE_INFO,
79126f3cdf0SGordon Ross                 "Unknown UpdateRule value: 0x%X",
7927b1019a6SJerry Jelinek                 (ObjDesc->CommonField.FieldFlags &
7937b1019a6SJerry Jelinek                     AML_FIELD_UPDATE_RULE_MASK)));
7947c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (AE_AML_OPERAND_VALUE);
7957c478bd9Sstevel@tonic-gate         }
7967c478bd9Sstevel@tonic-gate     }
7977c478bd9Sstevel@tonic-gate 
7987c478bd9Sstevel@tonic-gate     ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,
7997b1019a6SJerry Jelinek         "Mask %8.8X%8.8X, DatumOffset %X, Width %X, "
8007b1019a6SJerry Jelinek         "Value %8.8X%8.8X, MergedValue %8.8X%8.8X\n",
8017c478bd9Sstevel@tonic-gate         ACPI_FORMAT_UINT64 (Mask),
8027c478bd9Sstevel@tonic-gate         FieldDatumByteOffset,
8037c478bd9Sstevel@tonic-gate         ObjDesc->CommonField.AccessByteWidth,
8047c478bd9Sstevel@tonic-gate         ACPI_FORMAT_UINT64 (FieldValue),
8057c478bd9Sstevel@tonic-gate         ACPI_FORMAT_UINT64 (MergedValue)));
8067c478bd9Sstevel@tonic-gate 
8077c478bd9Sstevel@tonic-gate     /* Write the merged value */
8087c478bd9Sstevel@tonic-gate 
8097b1019a6SJerry Jelinek     Status = AcpiExFieldDatumIo (
8107b1019a6SJerry Jelinek         ObjDesc, FieldDatumByteOffset, &MergedValue, ACPI_WRITE);
8117c478bd9Sstevel@tonic-gate 
8127c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (Status);
8137c478bd9Sstevel@tonic-gate }
8147c478bd9Sstevel@tonic-gate 
8157c478bd9Sstevel@tonic-gate 
8167c478bd9Sstevel@tonic-gate /*******************************************************************************
8177c478bd9Sstevel@tonic-gate  *
8187c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExExtractFromField
8197c478bd9Sstevel@tonic-gate  *
8207c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc             - Field to be read
8217c478bd9Sstevel@tonic-gate  *              Buffer              - Where to store the field data
8227c478bd9Sstevel@tonic-gate  *              BufferLength        - Length of Buffer
8237c478bd9Sstevel@tonic-gate  *
8247c478bd9Sstevel@tonic-gate  * RETURN:      Status
8257c478bd9Sstevel@tonic-gate  *
8267c478bd9Sstevel@tonic-gate  * DESCRIPTION: Retrieve the current value of the given field
8277c478bd9Sstevel@tonic-gate  *
8287c478bd9Sstevel@tonic-gate  ******************************************************************************/
8297c478bd9Sstevel@tonic-gate 
8307c478bd9Sstevel@tonic-gate ACPI_STATUS
AcpiExExtractFromField(ACPI_OPERAND_OBJECT * ObjDesc,void * Buffer,UINT32 BufferLength)8317c478bd9Sstevel@tonic-gate AcpiExExtractFromField (
8327c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
8337c478bd9Sstevel@tonic-gate     void                    *Buffer,
8347c478bd9Sstevel@tonic-gate     UINT32                  BufferLength)
8357c478bd9Sstevel@tonic-gate {
8367c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status;
83726f3cdf0SGordon Ross     UINT64                  RawDatum;
83826f3cdf0SGordon Ross     UINT64                  MergedDatum;
8397c478bd9Sstevel@tonic-gate     UINT32                  FieldOffset = 0;
8407c478bd9Sstevel@tonic-gate     UINT32                  BufferOffset = 0;
8417c478bd9Sstevel@tonic-gate     UINT32                  BufferTailBits;
8427c478bd9Sstevel@tonic-gate     UINT32                  DatumCount;
8437c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumCount;
84426f3cdf0SGordon Ross     UINT32                  AccessBitWidth;
8457c478bd9Sstevel@tonic-gate     UINT32                  i;
8467c478bd9Sstevel@tonic-gate 
8477c478bd9Sstevel@tonic-gate 
84827f7c583Smyers     ACPI_FUNCTION_TRACE (ExExtractFromField);
8497c478bd9Sstevel@tonic-gate 
8507c478bd9Sstevel@tonic-gate 
8517c478bd9Sstevel@tonic-gate     /* Validate target buffer and clear it */
8527c478bd9Sstevel@tonic-gate 
85327f7c583Smyers     if (BufferLength <
85426f3cdf0SGordon Ross         ACPI_ROUND_BITS_UP_TO_BYTES (ObjDesc->CommonField.BitLength))
8557c478bd9Sstevel@tonic-gate     {
85630082d0cSmyers         ACPI_ERROR ((AE_INFO,
85726f3cdf0SGordon Ross             "Field size %u (bits) is too large for buffer (%u)",
8587c478bd9Sstevel@tonic-gate             ObjDesc->CommonField.BitLength, BufferLength));
8597c478bd9Sstevel@tonic-gate 
8607c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (AE_BUFFER_OVERFLOW);
8617c478bd9Sstevel@tonic-gate     }
86226f3cdf0SGordon Ross 
8637b1019a6SJerry Jelinek     memset (Buffer, 0, BufferLength);
86426f3cdf0SGordon Ross     AccessBitWidth = ACPI_MUL_8 (ObjDesc->CommonField.AccessByteWidth);
86526f3cdf0SGordon Ross 
86626f3cdf0SGordon Ross     /* Handle the simple case here */
86726f3cdf0SGordon Ross 
86826f3cdf0SGordon Ross     if ((ObjDesc->CommonField.StartFieldBitOffset == 0) &&
86926f3cdf0SGordon Ross         (ObjDesc->CommonField.BitLength == AccessBitWidth))
87026f3cdf0SGordon Ross     {
8717b1019a6SJerry Jelinek         if (BufferLength >= sizeof (UINT64))
8727b1019a6SJerry Jelinek         {
8737b1019a6SJerry Jelinek             Status = AcpiExFieldDatumIo (ObjDesc, 0, Buffer, ACPI_READ);
8747b1019a6SJerry Jelinek         }
8757b1019a6SJerry Jelinek         else
8767b1019a6SJerry Jelinek         {
8777b1019a6SJerry Jelinek             /* Use RawDatum (UINT64) to handle buffers < 64 bits */
8787b1019a6SJerry Jelinek 
8797b1019a6SJerry Jelinek             Status = AcpiExFieldDatumIo (ObjDesc, 0, &RawDatum, ACPI_READ);
8807b1019a6SJerry Jelinek             memcpy (Buffer, &RawDatum, BufferLength);
8817b1019a6SJerry Jelinek         }
8827b1019a6SJerry Jelinek 
88326f3cdf0SGordon Ross         return_ACPI_STATUS (Status);
88426f3cdf0SGordon Ross     }
88526f3cdf0SGordon Ross 
88626f3cdf0SGordon Ross /* TBD: Move to common setup code */
88726f3cdf0SGordon Ross 
88826f3cdf0SGordon Ross     /* Field algorithm is limited to sizeof(UINT64), truncate if needed */
88926f3cdf0SGordon Ross 
89026f3cdf0SGordon Ross     if (ObjDesc->CommonField.AccessByteWidth > sizeof (UINT64))
89126f3cdf0SGordon Ross     {
89226f3cdf0SGordon Ross         ObjDesc->CommonField.AccessByteWidth = sizeof (UINT64);
89326f3cdf0SGordon Ross         AccessBitWidth = sizeof (UINT64) * 8;
89426f3cdf0SGordon Ross     }
8957c478bd9Sstevel@tonic-gate 
8967c478bd9Sstevel@tonic-gate     /* Compute the number of datums (access width data items) */
8977c478bd9Sstevel@tonic-gate 
8987c478bd9Sstevel@tonic-gate     DatumCount = ACPI_ROUND_UP_TO (
89926f3cdf0SGordon Ross         ObjDesc->CommonField.BitLength, AccessBitWidth);
90026f3cdf0SGordon Ross 
9017c478bd9Sstevel@tonic-gate     FieldDatumCount = ACPI_ROUND_UP_TO (
90226f3cdf0SGordon Ross         ObjDesc->CommonField.BitLength +
90326f3cdf0SGordon Ross         ObjDesc->CommonField.StartFieldBitOffset, AccessBitWidth);
9047c478bd9Sstevel@tonic-gate 
9057c478bd9Sstevel@tonic-gate     /* Priming read from the field */
9067c478bd9Sstevel@tonic-gate 
9077c478bd9Sstevel@tonic-gate     Status = AcpiExFieldDatumIo (ObjDesc, FieldOffset, &RawDatum, ACPI_READ);
9087c478bd9Sstevel@tonic-gate     if (ACPI_FAILURE (Status))
9097c478bd9Sstevel@tonic-gate     {
9107c478bd9Sstevel@tonic-gate         return_ACPI_STATUS (Status);
9117c478bd9Sstevel@tonic-gate     }
9127c478bd9Sstevel@tonic-gate     MergedDatum = RawDatum >> ObjDesc->CommonField.StartFieldBitOffset;
9137c478bd9Sstevel@tonic-gate 
9147c478bd9Sstevel@tonic-gate     /* Read the rest of the field */
9157c478bd9Sstevel@tonic-gate 
9167c478bd9Sstevel@tonic-gate     for (i = 1; i < FieldDatumCount; i++)
9177c478bd9Sstevel@tonic-gate     {
9187c478bd9Sstevel@tonic-gate         /* Get next input datum from the field */
9197c478bd9Sstevel@tonic-gate 
9207c478bd9Sstevel@tonic-gate         FieldOffset += ObjDesc->CommonField.AccessByteWidth;
9217b1019a6SJerry Jelinek         Status = AcpiExFieldDatumIo (
9227b1019a6SJerry Jelinek             ObjDesc, FieldOffset, &RawDatum, ACPI_READ);
9237c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
9247c478bd9Sstevel@tonic-gate         {
9257c478bd9Sstevel@tonic-gate             return_ACPI_STATUS (Status);
9267c478bd9Sstevel@tonic-gate         }
9277c478bd9Sstevel@tonic-gate 
92827f7c583Smyers         /*
92927f7c583Smyers          * Merge with previous datum if necessary.
93027f7c583Smyers          *
93127f7c583Smyers          * Note: Before the shift, check if the shift value will be larger than
93227f7c583Smyers          * the integer size. If so, there is no need to perform the operation.
93327f7c583Smyers          * This avoids the differences in behavior between different compilers
93427f7c583Smyers          * concerning shift values larger than the target data width.
93527f7c583Smyers          */
93626f3cdf0SGordon Ross         if (AccessBitWidth - ObjDesc->CommonField.StartFieldBitOffset <
93726f3cdf0SGordon Ross             ACPI_INTEGER_BIT_SIZE)
93827f7c583Smyers         {
93927f7c583Smyers             MergedDatum |= RawDatum <<
94026f3cdf0SGordon Ross                 (AccessBitWidth - ObjDesc->CommonField.StartFieldBitOffset);
94127f7c583Smyers         }
9427c478bd9Sstevel@tonic-gate 
9437c478bd9Sstevel@tonic-gate         if (i == DatumCount)
9447c478bd9Sstevel@tonic-gate         {
9457c478bd9Sstevel@tonic-gate             break;
9467c478bd9Sstevel@tonic-gate         }
9477c478bd9Sstevel@tonic-gate 
9487c478bd9Sstevel@tonic-gate         /* Write merged datum to target buffer */
9497c478bd9Sstevel@tonic-gate 
9507b1019a6SJerry Jelinek         memcpy (((char *) Buffer) + BufferOffset, &MergedDatum,
9517c478bd9Sstevel@tonic-gate             ACPI_MIN(ObjDesc->CommonField.AccessByteWidth,
95227f7c583Smyers                 BufferLength - BufferOffset));
9537c478bd9Sstevel@tonic-gate 
9547c478bd9Sstevel@tonic-gate         BufferOffset += ObjDesc->CommonField.AccessByteWidth;
9557c478bd9Sstevel@tonic-gate         MergedDatum = RawDatum >> ObjDesc->CommonField.StartFieldBitOffset;
9567c478bd9Sstevel@tonic-gate     }
9577c478bd9Sstevel@tonic-gate 
9587c478bd9Sstevel@tonic-gate     /* Mask off any extra bits in the last datum */
9597c478bd9Sstevel@tonic-gate 
96026f3cdf0SGordon Ross     BufferTailBits = ObjDesc->CommonField.BitLength % AccessBitWidth;
9617c478bd9Sstevel@tonic-gate     if (BufferTailBits)
9627c478bd9Sstevel@tonic-gate     {
9637c478bd9Sstevel@tonic-gate         MergedDatum &= ACPI_MASK_BITS_ABOVE (BufferTailBits);
9647c478bd9Sstevel@tonic-gate     }
9657c478bd9Sstevel@tonic-gate 
9667c478bd9Sstevel@tonic-gate     /* Write the last datum to the buffer */
9677c478bd9Sstevel@tonic-gate 
9687b1019a6SJerry Jelinek     memcpy (((char *) Buffer) + BufferOffset, &MergedDatum,
9697c478bd9Sstevel@tonic-gate         ACPI_MIN(ObjDesc->CommonField.AccessByteWidth,
97027f7c583Smyers             BufferLength - BufferOffset));
9717c478bd9Sstevel@tonic-gate 
9727c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (AE_OK);
9737c478bd9Sstevel@tonic-gate }
9747c478bd9Sstevel@tonic-gate 
9757c478bd9Sstevel@tonic-gate 
9767c478bd9Sstevel@tonic-gate /*******************************************************************************
9777c478bd9Sstevel@tonic-gate  *
9787c478bd9Sstevel@tonic-gate  * FUNCTION:    AcpiExInsertIntoField
9797c478bd9Sstevel@tonic-gate  *
9807c478bd9Sstevel@tonic-gate  * PARAMETERS:  ObjDesc             - Field to be written
9817c478bd9Sstevel@tonic-gate  *              Buffer              - Data to be written
9827c478bd9Sstevel@tonic-gate  *              BufferLength        - Length of Buffer
9837c478bd9Sstevel@tonic-gate  *
9847c478bd9Sstevel@tonic-gate  * RETURN:      Status
9857c478bd9Sstevel@tonic-gate  *
9867c478bd9Sstevel@tonic-gate  * DESCRIPTION: Store the Buffer contents into the given field
9877c478bd9Sstevel@tonic-gate  *
9887c478bd9Sstevel@tonic-gate  ******************************************************************************/
9897c478bd9Sstevel@tonic-gate 
9907c478bd9Sstevel@tonic-gate ACPI_STATUS
AcpiExInsertIntoField(ACPI_OPERAND_OBJECT * ObjDesc,void * Buffer,UINT32 BufferLength)9917c478bd9Sstevel@tonic-gate AcpiExInsertIntoField (
9927c478bd9Sstevel@tonic-gate     ACPI_OPERAND_OBJECT     *ObjDesc,
9937c478bd9Sstevel@tonic-gate     void                    *Buffer,
9947c478bd9Sstevel@tonic-gate     UINT32                  BufferLength)
9957c478bd9Sstevel@tonic-gate {
99626f3cdf0SGordon Ross     void                    *NewBuffer;
9977c478bd9Sstevel@tonic-gate     ACPI_STATUS             Status;
99826f3cdf0SGordon Ross     UINT64                  Mask;
99926f3cdf0SGordon Ross     UINT64                  WidthMask;
100026f3cdf0SGordon Ross     UINT64                  MergedDatum;
100126f3cdf0SGordon Ross     UINT64                  RawDatum = 0;
10027c478bd9Sstevel@tonic-gate     UINT32                  FieldOffset = 0;
10037c478bd9Sstevel@tonic-gate     UINT32                  BufferOffset = 0;
10047c478bd9Sstevel@tonic-gate     UINT32                  BufferTailBits;
10057c478bd9Sstevel@tonic-gate     UINT32                  DatumCount;
10067c478bd9Sstevel@tonic-gate     UINT32                  FieldDatumCount;
100726f3cdf0SGordon Ross     UINT32                  AccessBitWidth;
1008db2bae30SDana Myers     UINT32                  RequiredLength;
100926f3cdf0SGordon Ross     UINT32                  i;
10107c478bd9Sstevel@tonic-gate 
10117c478bd9Sstevel@tonic-gate 
101227f7c583Smyers     ACPI_FUNCTION_TRACE (ExInsertIntoField);
10137c478bd9Sstevel@tonic-gate 
10147c478bd9Sstevel@tonic-gate 
10157c478bd9Sstevel@tonic-gate     /* Validate input buffer */
10167c478bd9Sstevel@tonic-gate 
1017db2bae30SDana Myers     NewBuffer = NULL;
1018db2bae30SDana Myers     RequiredLength = ACPI_ROUND_BITS_UP_TO_BYTES (
10197b1019a6SJerry Jelinek         ObjDesc->CommonField.BitLength);
10207b1019a6SJerry Jelinek 
1021db2bae30SDana Myers     /*
1022db2bae30SDana Myers      * We must have a buffer that is at least as long as the field
10237b1019a6SJerry Jelinek      * we are writing to. This is because individual fields are
1024db2bae30SDana Myers      * indivisible and partial writes are not supported -- as per
1025db2bae30SDana Myers      * the ACPI specification.
1026db2bae30SDana Myers      */
1027db2bae30SDana Myers     if (BufferLength < RequiredLength)
10287c478bd9Sstevel@tonic-gate     {
1029db2bae30SDana Myers         /* We need to create a new buffer */
10307c478bd9Sstevel@tonic-gate 
1031db2bae30SDana Myers         NewBuffer = ACPI_ALLOCATE_ZEROED (RequiredLength);
1032db2bae30SDana Myers         if (!NewBuffer)
1033db2bae30SDana Myers         {
1034db2bae30SDana Myers             return_ACPI_STATUS (AE_NO_MEMORY);
1035db2bae30SDana Myers         }
1036db2bae30SDana Myers 
1037db2bae30SDana Myers         /*
1038db2bae30SDana Myers          * Copy the original data to the new buffer, starting
10397b1019a6SJerry Jelinek          * at Byte zero. All unused (upper) bytes of the
1040db2bae30SDana Myers          * buffer will be 0.
1041db2bae30SDana Myers          */
10427b1019a6SJerry Jelinek         memcpy ((char *) NewBuffer, (char *) Buffer, BufferLength);
1043db2bae30SDana Myers         Buffer = NewBuffer;
1044db2bae30SDana Myers         BufferLength = RequiredLength;
10457c478bd9Sstevel@tonic-gate     }
10467c478bd9Sstevel@tonic-gate 
104726f3cdf0SGordon Ross /* TBD: Move to common setup code */
104826f3cdf0SGordon Ross 
104926f3cdf0SGordon Ross     /* Algo is limited to sizeof(UINT64), so cut the AccessByteWidth */
105026f3cdf0SGordon Ross     if (ObjDesc->CommonField.AccessByteWidth > sizeof (UINT64))
105126f3cdf0SGordon Ross     {
105226f3cdf0SGordon Ross         ObjDesc->CommonField.AccessByteWidth = sizeof (UINT64);
105326f3cdf0SGordon Ross     }
105426f3cdf0SGordon Ross 
105526f3cdf0SGordon Ross     AccessBitWidth = ACPI_MUL_8 (ObjDesc->CommonField.AccessByteWidth);
105626f3cdf0SGordon Ross 
1057*35786f68SRobert Mustacchi     /* Create the bitmasks used for bit insertion */
105827f7c583Smyers 
1059*35786f68SRobert Mustacchi     WidthMask = ACPI_MASK_BITS_ABOVE_64 (AccessBitWidth);
106027f7c583Smyers     Mask = WidthMask &
106126f3cdf0SGordon Ross         ACPI_MASK_BITS_BELOW (ObjDesc->CommonField.StartFieldBitOffset);
106227f7c583Smyers 
10637c478bd9Sstevel@tonic-gate     /* Compute the number of datums (access width data items) */
10647c478bd9Sstevel@tonic-gate 
10657c478bd9Sstevel@tonic-gate     DatumCount = ACPI_ROUND_UP_TO (ObjDesc->CommonField.BitLength,
106626f3cdf0SGordon Ross         AccessBitWidth);
106727f7c583Smyers 
10687c478bd9Sstevel@tonic-gate     FieldDatumCount = ACPI_ROUND_UP_TO (ObjDesc->CommonField.BitLength +
106926f3cdf0SGordon Ross         ObjDesc->CommonField.StartFieldBitOffset,
107026f3cdf0SGordon Ross         AccessBitWidth);
10717c478bd9Sstevel@tonic-gate 
10727c478bd9Sstevel@tonic-gate     /* Get initial Datum from the input buffer */
10737c478bd9Sstevel@tonic-gate 
10747b1019a6SJerry Jelinek     memcpy (&RawDatum, Buffer,
10757c478bd9Sstevel@tonic-gate         ACPI_MIN(ObjDesc->CommonField.AccessByteWidth,
107627f7c583Smyers             BufferLength - BufferOffset));
10777c478bd9Sstevel@tonic-gate 
10787c478bd9Sstevel@tonic-gate     MergedDatum = RawDatum << ObjDesc->CommonField.StartFieldBitOffset;
10797c478bd9Sstevel@tonic-gate 
10807c478bd9Sstevel@tonic-gate     /* Write the entire field */
10817c478bd9Sstevel@tonic-gate 
10827c478bd9Sstevel@tonic-gate     for (i = 1; i < FieldDatumCount; i++)
10837c478bd9Sstevel@tonic-gate     {
10847c478bd9Sstevel@tonic-gate         /* Write merged datum to the target field */
10857c478bd9Sstevel@tonic-gate 
10867c478bd9Sstevel@tonic-gate         MergedDatum &= Mask;
10877b1019a6SJerry Jelinek         Status = AcpiExWriteWithUpdateRule (
10887b1019a6SJerry Jelinek             ObjDesc, Mask, MergedDatum, FieldOffset);
10897c478bd9Sstevel@tonic-gate         if (ACPI_FAILURE (Status))
10907c478bd9Sstevel@tonic-gate         {
1091db2bae30SDana Myers             goto Exit;
10927c478bd9Sstevel@tonic-gate         }
10937c478bd9Sstevel@tonic-gate 
10947c478bd9Sstevel@tonic-gate         FieldOffset += ObjDesc->CommonField.AccessByteWidth;
109527f7c583Smyers 
109627f7c583Smyers         /*
109727f7c583Smyers          * Start new output datum by merging with previous input datum
109827f7c583Smyers          * if necessary.
109927f7c583Smyers          *
110027f7c583Smyers          * Note: Before the shift, check if the shift value will be larger than
110127f7c583Smyers          * the integer size. If so, there is no need to perform the operation.
110227f7c583Smyers          * This avoids the differences in behavior between different compilers
110327f7c583Smyers          * concerning shift values larger than the target data width.
110427f7c583Smyers          */
110526f3cdf0SGordon Ross         if ((AccessBitWidth - ObjDesc->CommonField.StartFieldBitOffset) <
110626f3cdf0SGordon Ross             ACPI_INTEGER_BIT_SIZE)
110727f7c583Smyers         {
110827f7c583Smyers             MergedDatum = RawDatum >>
110926f3cdf0SGordon Ross                 (AccessBitWidth - ObjDesc->CommonField.StartFieldBitOffset);
111027f7c583Smyers         }
111127f7c583Smyers         else
111227f7c583Smyers         {
111327f7c583Smyers             MergedDatum = 0;
111427f7c583Smyers         }
111527f7c583Smyers 
111627f7c583Smyers         Mask = WidthMask;
11177c478bd9Sstevel@tonic-gate 
11187c478bd9Sstevel@tonic-gate         if (i == DatumCount)
11197c478bd9Sstevel@tonic-gate         {
11207c478bd9Sstevel@tonic-gate             break;
11217c478bd9Sstevel@tonic-gate         }
11227c478bd9Sstevel@tonic-gate 
11237c478bd9Sstevel@tonic-gate         /* Get the next input datum from the buffer */
11247c478bd9Sstevel@tonic-gate 
11257c478bd9Sstevel@tonic-gate         BufferOffset += ObjDesc->CommonField.AccessByteWidth;
11267b1019a6SJerry Jelinek         memcpy (&RawDatum, ((char *) Buffer) + BufferOffset,
11277c478bd9Sstevel@tonic-gate             ACPI_MIN(ObjDesc->CommonField.AccessByteWidth,
112826f3cdf0SGordon Ross                  BufferLength - BufferOffset));
112926f3cdf0SGordon Ross 
11307c478bd9Sstevel@tonic-gate         MergedDatum |= RawDatum << ObjDesc->CommonField.StartFieldBitOffset;
11317c478bd9Sstevel@tonic-gate     }
11327c478bd9Sstevel@tonic-gate 
11337c478bd9Sstevel@tonic-gate     /* Mask off any extra bits in the last datum */
11347c478bd9Sstevel@tonic-gate 
11357c478bd9Sstevel@tonic-gate     BufferTailBits = (ObjDesc->CommonField.BitLength +
113626f3cdf0SGordon Ross         ObjDesc->CommonField.StartFieldBitOffset) % AccessBitWidth;
11377c478bd9Sstevel@tonic-gate     if (BufferTailBits)
11387c478bd9Sstevel@tonic-gate     {
11397c478bd9Sstevel@tonic-gate         Mask &= ACPI_MASK_BITS_ABOVE (BufferTailBits);
11407c478bd9Sstevel@tonic-gate     }
11417c478bd9Sstevel@tonic-gate 
11427c478bd9Sstevel@tonic-gate     /* Write the last datum to the field */
11437c478bd9Sstevel@tonic-gate 
11447c478bd9Sstevel@tonic-gate     MergedDatum &= Mask;
11457b1019a6SJerry Jelinek     Status = AcpiExWriteWithUpdateRule (
11467b1019a6SJerry Jelinek         ObjDesc, Mask, MergedDatum, FieldOffset);
11477c478bd9Sstevel@tonic-gate 
1148db2bae30SDana Myers Exit:
1149db2bae30SDana Myers     /* Free temporary buffer if we used one */
1150db2bae30SDana Myers 
1151db2bae30SDana Myers     if (NewBuffer)
1152db2bae30SDana Myers     {
1153db2bae30SDana Myers         ACPI_FREE (NewBuffer);
1154db2bae30SDana Myers     }
11557c478bd9Sstevel@tonic-gate     return_ACPI_STATUS (Status);
11567c478bd9Sstevel@tonic-gate }
1157