1 /*******************************************************************************
2  *
3  * Module Name: hwregs - Read/write access functions for the various ACPI
4  *                       control and status registers.
5  *
6  ******************************************************************************/
7 
8 /*
9  * Copyright (C) 2000 - 2022, Intel Corp.
10  * All rights reserved.
11  *
12  * Redistribution and use in source and binary forms, with or without
13  * modification, are permitted provided that the following conditions
14  * are met:
15  * 1. Redistributions of source code must retain the above copyright
16  *    notice, this list of conditions, and the following disclaimer,
17  *    without modification.
18  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
19  *    substantially similar to the "NO WARRANTY" disclaimer below
20  *    ("Disclaimer") and any redistribution must be conditioned upon
21  *    including a substantially similar Disclaimer requirement for further
22  *    binary redistribution.
23  * 3. Neither the names of the above-listed copyright holders nor the names
24  *    of any contributors may be used to endorse or promote products derived
25  *    from this software without specific prior written permission.
26  *
27  * Alternatively, this software may be distributed under the terms of the
28  * GNU General Public License ("GPL") version 2 as published by the Free
29  * Software Foundation.
30  *
31  * NO WARRANTY
32  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
33  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
34  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
35  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
36  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
37  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
38  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
39  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
40  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
41  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
42  * POSSIBILITY OF SUCH DAMAGES.
43  */
44 
45 #include "acpi.h"
46 #include "accommon.h"
47 #include "acevents.h"
48 
49 #define _COMPONENT          ACPI_HARDWARE
50         ACPI_MODULE_NAME    ("hwregs")
51 
52 
53 #if (!ACPI_REDUCED_HARDWARE)
54 
55 /* Local Prototypes */
56 
57 static UINT8
58 AcpiHwGetAccessBitWidth (
59     UINT64                  Address,
60     ACPI_GENERIC_ADDRESS    *Reg,
61     UINT8                   MaxBitWidth);
62 
63 static ACPI_STATUS
64 AcpiHwReadMultiple (
65     UINT32                  *Value,
66     ACPI_GENERIC_ADDRESS    *RegisterA,
67     ACPI_GENERIC_ADDRESS    *RegisterB);
68 
69 static ACPI_STATUS
70 AcpiHwWriteMultiple (
71     UINT32                  Value,
72     ACPI_GENERIC_ADDRESS    *RegisterA,
73     ACPI_GENERIC_ADDRESS    *RegisterB);
74 
75 #endif /* !ACPI_REDUCED_HARDWARE */
76 
77 
78 /******************************************************************************
79  *
80  * FUNCTION:    AcpiHwGetAccessBitWidth
81  *
82  * PARAMETERS:  Address             - GAS register address
83  *              Reg                 - GAS register structure
84  *              MaxBitWidth         - Max BitWidth supported (32 or 64)
85  *
86  * RETURN:      Status
87  *
88  * DESCRIPTION: Obtain optimal access bit width
89  *
90  ******************************************************************************/
91 
92 static UINT8
93 AcpiHwGetAccessBitWidth (
94     UINT64                  Address,
95     ACPI_GENERIC_ADDRESS    *Reg,
96     UINT8                   MaxBitWidth)
97 {
98     UINT8                   AccessBitWidth;
99 
100 
101     /*
102      * GAS format "register", used by FADT:
103      *  1. Detected if BitOffset is 0 and BitWidth is 8/16/32/64;
104      *  2. AccessSize field is ignored and BitWidth field is used for
105      *     determining the boundary of the IO accesses.
106      * GAS format "region", used by APEI registers:
107      *  1. Detected if BitOffset is not 0 or BitWidth is not 8/16/32/64;
108      *  2. AccessSize field is used for determining the boundary of the
109      *     IO accesses;
110      *  3. BitOffset/BitWidth fields are used to describe the "region".
111      *
112      * Note: This algorithm assumes that the "Address" fields should always
113      *       contain aligned values.
114      */
115     if (!Reg->BitOffset && Reg->BitWidth &&
116         ACPI_IS_POWER_OF_TWO (Reg->BitWidth) &&
117         ACPI_IS_ALIGNED (Reg->BitWidth, 8))
118     {
119         AccessBitWidth = Reg->BitWidth;
120     }
121     else if (Reg->AccessWidth)
122     {
123         AccessBitWidth = ACPI_ACCESS_BIT_WIDTH (Reg->AccessWidth);
124     }
125     else
126     {
127         AccessBitWidth = ACPI_ROUND_UP_POWER_OF_TWO_8 (
128             Reg->BitOffset + Reg->BitWidth);
129         if (AccessBitWidth <= 8)
130         {
131             AccessBitWidth = 8;
132         }
133         else
134         {
135             while (!ACPI_IS_ALIGNED (Address, AccessBitWidth >> 3))
136             {
137                 AccessBitWidth >>= 1;
138             }
139         }
140     }
141 
142     /* Maximum IO port access bit width is 32 */
143 
144     if (Reg->SpaceId == ACPI_ADR_SPACE_SYSTEM_IO)
145     {
146         MaxBitWidth = 32;
147     }
148 
149     /*
150      * Return access width according to the requested maximum access bit width,
151      * as the caller should know the format of the register and may enforce
152      * a 32-bit accesses.
153      */
154     if (AccessBitWidth < MaxBitWidth)
155     {
156         return (AccessBitWidth);
157     }
158     return (MaxBitWidth);
159 }
160 
161 
162 /******************************************************************************
163  *
164  * FUNCTION:    AcpiHwValidateRegister
165  *
166  * PARAMETERS:  Reg                 - GAS register structure
167  *              MaxBitWidth         - Max BitWidth supported (32 or 64)
168  *              Address             - Pointer to where the gas->address
169  *                                    is returned
170  *
171  * RETURN:      Status
172  *
173  * DESCRIPTION: Validate the contents of a GAS register. Checks the GAS
174  *              pointer, Address, SpaceId, BitWidth, and BitOffset.
175  *
176  ******************************************************************************/
177 
178 ACPI_STATUS
179 AcpiHwValidateRegister (
180     ACPI_GENERIC_ADDRESS    *Reg,
181     UINT8                   MaxBitWidth,
182     UINT64                  *Address)
183 {
184     UINT8                   BitWidth;
185     UINT8                   AccessWidth;
186 
187 
188     /* Must have a valid pointer to a GAS structure */
189 
190     if (!Reg)
191     {
192         return (AE_BAD_PARAMETER);
193     }
194 
195     /*
196      * Copy the target address. This handles possible alignment issues.
197      * Address must not be null. A null address also indicates an optional
198      * ACPI register that is not supported, so no error message.
199      */
200     ACPI_MOVE_64_TO_64 (Address, &Reg->Address);
201     if (!(*Address))
202     {
203         return (AE_BAD_ADDRESS);
204     }
205 
206     /* Validate the SpaceID */
207 
208     if ((Reg->SpaceId != ACPI_ADR_SPACE_SYSTEM_MEMORY) &&
209         (Reg->SpaceId != ACPI_ADR_SPACE_SYSTEM_IO))
210     {
211         ACPI_ERROR ((AE_INFO,
212             "Unsupported address space: 0x%X", Reg->SpaceId));
213         return (AE_SUPPORT);
214     }
215 
216     /* Validate the AccessWidth */
217 
218     if (Reg->AccessWidth > 4)
219     {
220         ACPI_ERROR ((AE_INFO,
221             "Unsupported register access width: 0x%X", Reg->AccessWidth));
222         return (AE_SUPPORT);
223     }
224 
225     /* Validate the BitWidth, convert AccessWidth into number of bits */
226 
227     AccessWidth = AcpiHwGetAccessBitWidth (*Address, Reg, MaxBitWidth);
228     BitWidth = ACPI_ROUND_UP (Reg->BitOffset + Reg->BitWidth, AccessWidth);
229     if (MaxBitWidth < BitWidth)
230     {
231         ACPI_WARNING ((AE_INFO,
232             "Requested bit width 0x%X is smaller than register bit width 0x%X",
233             MaxBitWidth, BitWidth));
234         return (AE_SUPPORT);
235     }
236 
237     return (AE_OK);
238 }
239 
240 
241 /******************************************************************************
242  *
243  * FUNCTION:    AcpiHwRead
244  *
245  * PARAMETERS:  Value               - Where the value is returned
246  *              Reg                 - GAS register structure
247  *
248  * RETURN:      Status
249  *
250  * DESCRIPTION: Read from either memory or IO space. This is a 64-bit max
251  *              version of AcpiRead.
252  *
253  * LIMITATIONS: <These limitations also apply to AcpiHwWrite>
254  *      SpaceID must be SystemMemory or SystemIO.
255  *
256  ******************************************************************************/
257 
258 ACPI_STATUS
259 AcpiHwRead (
260     UINT64                  *Value,
261     ACPI_GENERIC_ADDRESS    *Reg)
262 {
263     UINT64                  Address;
264     UINT8                   AccessWidth;
265     UINT32                  BitWidth;
266     UINT8                   BitOffset;
267     UINT64                  Value64;
268     UINT32                  Value32;
269     UINT8                   Index;
270     ACPI_STATUS             Status;
271 
272 
273     ACPI_FUNCTION_NAME (HwRead);
274 
275 
276     /* Validate contents of the GAS register */
277 
278     Status = AcpiHwValidateRegister (Reg, 64, &Address);
279     if (ACPI_FAILURE (Status))
280     {
281         return (Status);
282     }
283 
284     /*
285      * Initialize entire 64-bit return value to zero, convert AccessWidth
286      * into number of bits based
287      */
288     *Value = 0;
289     AccessWidth = AcpiHwGetAccessBitWidth (Address, Reg, 64);
290     BitWidth = Reg->BitOffset + Reg->BitWidth;
291     BitOffset = Reg->BitOffset;
292 
293     /*
294      * Two address spaces supported: Memory or IO. PCI_Config is
295      * not supported here because the GAS structure is insufficient
296      */
297     Index = 0;
298     while (BitWidth)
299     {
300         if (BitOffset >= AccessWidth)
301         {
302             Value64 = 0;
303             BitOffset -= AccessWidth;
304         }
305         else
306         {
307             if (Reg->SpaceId == ACPI_ADR_SPACE_SYSTEM_MEMORY)
308             {
309                 Status = AcpiOsReadMemory ((ACPI_PHYSICAL_ADDRESS)
310                     Address + Index * ACPI_DIV_8 (AccessWidth),
311                     &Value64, AccessWidth);
312             }
313             else /* ACPI_ADR_SPACE_SYSTEM_IO, validated earlier */
314             {
315                 Status = AcpiHwReadPort ((ACPI_IO_ADDRESS)
316                     Address + Index * ACPI_DIV_8 (AccessWidth),
317                     &Value32, AccessWidth);
318                 Value64 = (UINT64) Value32;
319             }
320         }
321 
322         /*
323          * Use offset style bit writes because "Index * AccessWidth" is
324          * ensured to be less than 64-bits by AcpiHwValidateRegister().
325          */
326         ACPI_SET_BITS (Value, Index * AccessWidth,
327             ACPI_MASK_BITS_ABOVE_64 (AccessWidth), Value64);
328 
329         BitWidth -= BitWidth > AccessWidth ? AccessWidth : BitWidth;
330         Index++;
331     }
332 
333     ACPI_DEBUG_PRINT ((ACPI_DB_IO,
334         "Read:  %8.8X%8.8X width %2d from %8.8X%8.8X (%s)\n",
335         ACPI_FORMAT_UINT64 (*Value), AccessWidth,
336         ACPI_FORMAT_UINT64 (Address), AcpiUtGetRegionName (Reg->SpaceId)));
337 
338     return (Status);
339 }
340 
341 
342 /******************************************************************************
343  *
344  * FUNCTION:    AcpiHwWrite
345  *
346  * PARAMETERS:  Value               - Value to be written
347  *              Reg                 - GAS register structure
348  *
349  * RETURN:      Status
350  *
351  * DESCRIPTION: Write to either memory or IO space. This is a 64-bit max
352  *              version of AcpiWrite.
353  *
354  ******************************************************************************/
355 
356 ACPI_STATUS
357 AcpiHwWrite (
358     UINT64                  Value,
359     ACPI_GENERIC_ADDRESS    *Reg)
360 {
361     UINT64                  Address;
362     UINT8                   AccessWidth;
363     UINT32                  BitWidth;
364     UINT8                   BitOffset;
365     UINT64                  Value64;
366     UINT8                   Index;
367     ACPI_STATUS             Status;
368 
369 
370     ACPI_FUNCTION_NAME (HwWrite);
371 
372 
373     /* Validate contents of the GAS register */
374 
375     Status = AcpiHwValidateRegister (Reg, 64, &Address);
376     if (ACPI_FAILURE (Status))
377     {
378         return (Status);
379     }
380 
381     /* Convert AccessWidth into number of bits based */
382 
383     AccessWidth = AcpiHwGetAccessBitWidth (Address, Reg, 64);
384     BitWidth = Reg->BitOffset + Reg->BitWidth;
385     BitOffset = Reg->BitOffset;
386 
387     /*
388      * Two address spaces supported: Memory or IO. PCI_Config is
389      * not supported here because the GAS structure is insufficient
390      */
391     Index = 0;
392     while (BitWidth)
393     {
394         /*
395          * Use offset style bit reads because "Index * AccessWidth" is
396          * ensured to be less than 64-bits by AcpiHwValidateRegister().
397          */
398         Value64 = ACPI_GET_BITS (&Value, Index * AccessWidth,
399             ACPI_MASK_BITS_ABOVE_64 (AccessWidth));
400 
401         if (BitOffset >= AccessWidth)
402         {
403             BitOffset -= AccessWidth;
404         }
405         else
406         {
407             if (Reg->SpaceId == ACPI_ADR_SPACE_SYSTEM_MEMORY)
408             {
409                 Status = AcpiOsWriteMemory ((ACPI_PHYSICAL_ADDRESS)
410                     Address + Index * ACPI_DIV_8 (AccessWidth),
411                     Value64, AccessWidth);
412             }
413             else /* ACPI_ADR_SPACE_SYSTEM_IO, validated earlier */
414             {
415                 Status = AcpiHwWritePort ((ACPI_IO_ADDRESS)
416                     Address + Index * ACPI_DIV_8 (AccessWidth),
417                     (UINT32) Value64, AccessWidth);
418             }
419         }
420 
421         /*
422          * Index * AccessWidth is ensured to be less than 32-bits by
423          * AcpiHwValidateRegister().
424          */
425         BitWidth -= BitWidth > AccessWidth ? AccessWidth : BitWidth;
426         Index++;
427     }
428 
429     ACPI_DEBUG_PRINT ((ACPI_DB_IO,
430         "Wrote: %8.8X%8.8X width %2d   to %8.8X%8.8X (%s)\n",
431         ACPI_FORMAT_UINT64 (Value), AccessWidth,
432         ACPI_FORMAT_UINT64 (Address), AcpiUtGetRegionName (Reg->SpaceId)));
433 
434     return (Status);
435 }
436 
437 
438 #if (!ACPI_REDUCED_HARDWARE)
439 /*******************************************************************************
440  *
441  * FUNCTION:    AcpiHwClearAcpiStatus
442  *
443  * PARAMETERS:  None
444  *
445  * RETURN:      Status
446  *
447  * DESCRIPTION: Clears all fixed and general purpose status bits
448  *
449  ******************************************************************************/
450 
451 ACPI_STATUS
452 AcpiHwClearAcpiStatus (
453     void)
454 {
455     ACPI_STATUS             Status;
456     ACPI_CPU_FLAGS          LockFlags = 0;
457 
458 
459     ACPI_FUNCTION_TRACE (HwClearAcpiStatus);
460 
461 
462     ACPI_DEBUG_PRINT ((ACPI_DB_IO, "About to write %04X to %8.8X%8.8X\n",
463         ACPI_BITMASK_ALL_FIXED_STATUS,
464         ACPI_FORMAT_UINT64 (AcpiGbl_XPm1aStatus.Address)));
465 
466     LockFlags = AcpiOsAcquireLock (AcpiGbl_HardwareLock);
467 
468     /* Clear the fixed events in PM1 A/B */
469 
470     Status = AcpiHwRegisterWrite (ACPI_REGISTER_PM1_STATUS,
471         ACPI_BITMASK_ALL_FIXED_STATUS);
472 
473     AcpiOsReleaseLock (AcpiGbl_HardwareLock, LockFlags);
474 
475     if (ACPI_FAILURE (Status))
476     {
477         goto Exit;
478     }
479 
480     /* Clear the GPE Bits in all GPE registers in all GPE blocks */
481 
482     Status = AcpiEvWalkGpeList (AcpiHwClearGpeBlock, NULL);
483 
484 Exit:
485     return_ACPI_STATUS (Status);
486 }
487 
488 
489 /*******************************************************************************
490  *
491  * FUNCTION:    AcpiHwGetBitRegisterInfo
492  *
493  * PARAMETERS:  RegisterId          - Index of ACPI Register to access
494  *
495  * RETURN:      The bitmask to be used when accessing the register
496  *
497  * DESCRIPTION: Map RegisterId into a register bitmask.
498  *
499  ******************************************************************************/
500 
501 ACPI_BIT_REGISTER_INFO *
502 AcpiHwGetBitRegisterInfo (
503     UINT32                  RegisterId)
504 {
505     ACPI_FUNCTION_ENTRY ();
506 
507 
508     if (RegisterId > ACPI_BITREG_MAX)
509     {
510         ACPI_ERROR ((AE_INFO, "Invalid BitRegister ID: 0x%X", RegisterId));
511         return (NULL);
512     }
513 
514     return (&AcpiGbl_BitRegisterInfo[RegisterId]);
515 }
516 
517 
518 /******************************************************************************
519  *
520  * FUNCTION:    AcpiHwWritePm1Control
521  *
522  * PARAMETERS:  Pm1aControl         - Value to be written to PM1A control
523  *              Pm1bControl         - Value to be written to PM1B control
524  *
525  * RETURN:      Status
526  *
527  * DESCRIPTION: Write the PM1 A/B control registers. These registers are
528  *              different than the PM1 A/B status and enable registers
529  *              in that different values can be written to the A/B registers.
530  *              Most notably, the SLP_TYP bits can be different, as per the
531  *              values returned from the _Sx predefined methods.
532  *
533  ******************************************************************************/
534 
535 ACPI_STATUS
536 AcpiHwWritePm1Control (
537     UINT32                  Pm1aControl,
538     UINT32                  Pm1bControl)
539 {
540     ACPI_STATUS             Status;
541 
542 
543     ACPI_FUNCTION_TRACE (HwWritePm1Control);
544 
545 
546     Status = AcpiHwWrite (Pm1aControl, &AcpiGbl_FADT.XPm1aControlBlock);
547     if (ACPI_FAILURE (Status))
548     {
549         return_ACPI_STATUS (Status);
550     }
551 
552     if (AcpiGbl_FADT.XPm1bControlBlock.Address)
553     {
554         Status = AcpiHwWrite (Pm1bControl, &AcpiGbl_FADT.XPm1bControlBlock);
555     }
556     return_ACPI_STATUS (Status);
557 }
558 
559 
560 /******************************************************************************
561  *
562  * FUNCTION:    AcpiHwRegisterRead
563  *
564  * PARAMETERS:  RegisterId          - ACPI Register ID
565  *              ReturnValue         - Where the register value is returned
566  *
567  * RETURN:      Status and the value read.
568  *
569  * DESCRIPTION: Read from the specified ACPI register
570  *
571  ******************************************************************************/
572 
573 ACPI_STATUS
574 AcpiHwRegisterRead (
575     UINT32                  RegisterId,
576     UINT32                  *ReturnValue)
577 {
578     UINT32                  Value = 0;
579     UINT64                  Value64;
580     ACPI_STATUS             Status;
581 
582 
583     ACPI_FUNCTION_TRACE (HwRegisterRead);
584 
585 
586     switch (RegisterId)
587     {
588     case ACPI_REGISTER_PM1_STATUS:           /* PM1 A/B: 16-bit access each */
589 
590         Status = AcpiHwReadMultiple (&Value,
591             &AcpiGbl_XPm1aStatus,
592             &AcpiGbl_XPm1bStatus);
593         break;
594 
595     case ACPI_REGISTER_PM1_ENABLE:           /* PM1 A/B: 16-bit access each */
596 
597         Status = AcpiHwReadMultiple (&Value,
598             &AcpiGbl_XPm1aEnable,
599             &AcpiGbl_XPm1bEnable);
600         break;
601 
602     case ACPI_REGISTER_PM1_CONTROL:          /* PM1 A/B: 16-bit access each */
603 
604         Status = AcpiHwReadMultiple (&Value,
605             &AcpiGbl_FADT.XPm1aControlBlock,
606             &AcpiGbl_FADT.XPm1bControlBlock);
607 
608         /*
609          * Zero the write-only bits. From the ACPI specification, "Hardware
610          * Write-Only Bits": "Upon reads to registers with write-only bits,
611          * software masks out all write-only bits."
612          */
613         Value &= ~ACPI_PM1_CONTROL_WRITEONLY_BITS;
614         break;
615 
616     case ACPI_REGISTER_PM2_CONTROL:          /* 8-bit access */
617 
618         Status = AcpiHwRead (&Value64, &AcpiGbl_FADT.XPm2ControlBlock);
619         if (ACPI_SUCCESS (Status))
620         {
621             Value = (UINT32) Value64;
622         }
623         break;
624 
625     case ACPI_REGISTER_PM_TIMER:             /* 32-bit access */
626 
627         Status = AcpiHwRead (&Value64, &AcpiGbl_FADT.XPmTimerBlock);
628         if (ACPI_SUCCESS (Status))
629         {
630             Value = (UINT32) Value64;
631         }
632 
633         break;
634 
635     case ACPI_REGISTER_SMI_COMMAND_BLOCK:    /* 8-bit access */
636 
637         Status = AcpiHwReadPort (AcpiGbl_FADT.SmiCommand, &Value, 8);
638         break;
639 
640     default:
641 
642         ACPI_ERROR ((AE_INFO, "Unknown Register ID: 0x%X",
643             RegisterId));
644         Status = AE_BAD_PARAMETER;
645         break;
646     }
647 
648     if (ACPI_SUCCESS (Status))
649     {
650         *ReturnValue = (UINT32) Value;
651     }
652 
653     return_ACPI_STATUS (Status);
654 }
655 
656 
657 /******************************************************************************
658  *
659  * FUNCTION:    AcpiHwRegisterWrite
660  *
661  * PARAMETERS:  RegisterId          - ACPI Register ID
662  *              Value               - The value to write
663  *
664  * RETURN:      Status
665  *
666  * DESCRIPTION: Write to the specified ACPI register
667  *
668  * NOTE: In accordance with the ACPI specification, this function automatically
669  * preserves the value of the following bits, meaning that these bits cannot be
670  * changed via this interface:
671  *
672  * PM1_CONTROL[0] = SCI_EN
673  * PM1_CONTROL[9]
674  * PM1_STATUS[11]
675  *
676  * ACPI References:
677  * 1) Hardware Ignored Bits: When software writes to a register with ignored
678  *      bit fields, it preserves the ignored bit fields
679  * 2) SCI_EN: OSPM always preserves this bit position
680  *
681  ******************************************************************************/
682 
683 ACPI_STATUS
684 AcpiHwRegisterWrite (
685     UINT32                  RegisterId,
686     UINT32                  Value)
687 {
688     ACPI_STATUS             Status;
689     UINT32                  ReadValue;
690     UINT64                  ReadValue64;
691 
692 
693     ACPI_FUNCTION_TRACE (HwRegisterWrite);
694 
695 
696     switch (RegisterId)
697     {
698     case ACPI_REGISTER_PM1_STATUS:           /* PM1 A/B: 16-bit access each */
699         /*
700          * Handle the "ignored" bit in PM1 Status. According to the ACPI
701          * specification, ignored bits are to be preserved when writing.
702          * Normally, this would mean a read/modify/write sequence. However,
703          * preserving a bit in the status register is different. Writing a
704          * one clears the status, and writing a zero preserves the status.
705          * Therefore, we must always write zero to the ignored bit.
706          *
707          * This behavior is clarified in the ACPI 4.0 specification.
708          */
709         Value &= ~ACPI_PM1_STATUS_PRESERVED_BITS;
710 
711         Status = AcpiHwWriteMultiple (Value,
712             &AcpiGbl_XPm1aStatus,
713             &AcpiGbl_XPm1bStatus);
714         break;
715 
716     case ACPI_REGISTER_PM1_ENABLE:           /* PM1 A/B: 16-bit access each */
717 
718         Status = AcpiHwWriteMultiple (Value,
719             &AcpiGbl_XPm1aEnable,
720             &AcpiGbl_XPm1bEnable);
721         break;
722 
723     case ACPI_REGISTER_PM1_CONTROL:          /* PM1 A/B: 16-bit access each */
724         /*
725          * Perform a read first to preserve certain bits (per ACPI spec)
726          * Note: This includes SCI_EN, we never want to change this bit
727          */
728         Status = AcpiHwReadMultiple (&ReadValue,
729             &AcpiGbl_FADT.XPm1aControlBlock,
730             &AcpiGbl_FADT.XPm1bControlBlock);
731         if (ACPI_FAILURE (Status))
732         {
733             goto Exit;
734         }
735 
736         /* Insert the bits to be preserved */
737 
738         ACPI_INSERT_BITS (Value, ACPI_PM1_CONTROL_PRESERVED_BITS, ReadValue);
739 
740         /* Now we can write the data */
741 
742         Status = AcpiHwWriteMultiple (Value,
743             &AcpiGbl_FADT.XPm1aControlBlock,
744             &AcpiGbl_FADT.XPm1bControlBlock);
745         break;
746 
747     case ACPI_REGISTER_PM2_CONTROL:          /* 8-bit access */
748         /*
749          * For control registers, all reserved bits must be preserved,
750          * as per the ACPI spec.
751          */
752         Status = AcpiHwRead (&ReadValue64, &AcpiGbl_FADT.XPm2ControlBlock);
753         if (ACPI_FAILURE (Status))
754         {
755             goto Exit;
756         }
757         ReadValue = (UINT32) ReadValue64;
758 
759         /* Insert the bits to be preserved */
760 
761         ACPI_INSERT_BITS (Value, ACPI_PM2_CONTROL_PRESERVED_BITS, ReadValue);
762 
763         Status = AcpiHwWrite (Value, &AcpiGbl_FADT.XPm2ControlBlock);
764         break;
765 
766     case ACPI_REGISTER_PM_TIMER:             /* 32-bit access */
767 
768         Status = AcpiHwWrite (Value, &AcpiGbl_FADT.XPmTimerBlock);
769         break;
770 
771     case ACPI_REGISTER_SMI_COMMAND_BLOCK:    /* 8-bit access */
772 
773         /* SMI_CMD is currently always in IO space */
774 
775         Status = AcpiHwWritePort (AcpiGbl_FADT.SmiCommand, Value, 8);
776         break;
777 
778     default:
779 
780         ACPI_ERROR ((AE_INFO, "Unknown Register ID: 0x%X",
781             RegisterId));
782         Status = AE_BAD_PARAMETER;
783         break;
784     }
785 
786 Exit:
787     return_ACPI_STATUS (Status);
788 }
789 
790 
791 /******************************************************************************
792  *
793  * FUNCTION:    AcpiHwReadMultiple
794  *
795  * PARAMETERS:  Value               - Where the register value is returned
796  *              RegisterA           - First ACPI register (required)
797  *              RegisterB           - Second ACPI register (optional)
798  *
799  * RETURN:      Status
800  *
801  * DESCRIPTION: Read from the specified two-part ACPI register (such as PM1 A/B)
802  *
803  ******************************************************************************/
804 
805 static ACPI_STATUS
806 AcpiHwReadMultiple (
807     UINT32                  *Value,
808     ACPI_GENERIC_ADDRESS    *RegisterA,
809     ACPI_GENERIC_ADDRESS    *RegisterB)
810 {
811     UINT32                  ValueA = 0;
812     UINT32                  ValueB = 0;
813     UINT64                  Value64;
814     ACPI_STATUS             Status;
815 
816 
817     /* The first register is always required */
818 
819     Status = AcpiHwRead (&Value64, RegisterA);
820     if (ACPI_FAILURE (Status))
821     {
822         return (Status);
823     }
824     ValueA = (UINT32) Value64;
825 
826     /* Second register is optional */
827 
828     if (RegisterB->Address)
829     {
830         Status = AcpiHwRead (&Value64, RegisterB);
831         if (ACPI_FAILURE (Status))
832         {
833             return (Status);
834         }
835         ValueB = (UINT32) Value64;
836     }
837 
838     /*
839      * OR the two return values together. No shifting or masking is necessary,
840      * because of how the PM1 registers are defined in the ACPI specification:
841      *
842      * "Although the bits can be split between the two register blocks (each
843      * register block has a unique pointer within the FADT), the bit positions
844      * are maintained. The register block with unimplemented bits (that is,
845      * those implemented in the other register block) always returns zeros,
846      * and writes have no side effects"
847      */
848     *Value = (ValueA | ValueB);
849     return (AE_OK);
850 }
851 
852 
853 /******************************************************************************
854  *
855  * FUNCTION:    AcpiHwWriteMultiple
856  *
857  * PARAMETERS:  Value               - The value to write
858  *              RegisterA           - First ACPI register (required)
859  *              RegisterB           - Second ACPI register (optional)
860  *
861  * RETURN:      Status
862  *
863  * DESCRIPTION: Write to the specified two-part ACPI register (such as PM1 A/B)
864  *
865  ******************************************************************************/
866 
867 static ACPI_STATUS
868 AcpiHwWriteMultiple (
869     UINT32                  Value,
870     ACPI_GENERIC_ADDRESS    *RegisterA,
871     ACPI_GENERIC_ADDRESS    *RegisterB)
872 {
873     ACPI_STATUS             Status;
874 
875 
876     /* The first register is always required */
877 
878     Status = AcpiHwWrite (Value, RegisterA);
879     if (ACPI_FAILURE (Status))
880     {
881         return (Status);
882     }
883 
884     /*
885      * Second register is optional
886      *
887      * No bit shifting or clearing is necessary, because of how the PM1
888      * registers are defined in the ACPI specification:
889      *
890      * "Although the bits can be split between the two register blocks (each
891      * register block has a unique pointer within the FADT), the bit positions
892      * are maintained. The register block with unimplemented bits (that is,
893      * those implemented in the other register block) always returns zeros,
894      * and writes have no side effects"
895      */
896     if (RegisterB->Address)
897     {
898         Status = AcpiHwWrite (Value, RegisterB);
899     }
900 
901     return (Status);
902 }
903 
904 #endif /* !ACPI_REDUCED_HARDWARE */
905