LCOV - code coverage report
Current view: top level - include - processor.h (source / functions) Coverage Total Hit
Test: skiboot.info Lines: 33.3 % 12 4
Test Date: 2025-10-14 13:37:15 Functions: 50.0 % 2 1
Branches: - 0 0

             Branch data     Line data    Source code
       1                 :             : // SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
       2                 :             : /* Copyright 2013-2019 IBM Corp. */
       3                 :             : 
       4                 :             : #ifndef __PROCESSOR_H
       5                 :             : #define __PROCESSOR_H
       6                 :             : 
       7                 :             : #include <bitutils.h>
       8                 :             : 
       9                 :             : /* MSR bits */
      10                 :             : #define MSR_SF          PPC_BIT(0)      /* 64-bit mode */
      11                 :             : #define MSR_HV          PPC_BIT(3)      /* Hypervisor mode */
      12                 :             : #define MSR_VEC         PPC_BIT(38)     /* VMX enable */
      13                 :             : #define MSR_VSX         PPC_BIT(40)     /* VSX enable */
      14                 :             : #define MSR_S           PPC_BIT(41)     /* Secure mode */
      15                 :             : #define MSR_EE          PPC_BIT(48)     /* External Int. Enable */
      16                 :             : #define MSR_PR          PPC_BIT(49)             /* Problem state */
      17                 :             : #define MSR_FP          PPC_BIT(50)     /* Floating Point Enable */
      18                 :             : #define MSR_ME          PPC_BIT(51)     /* Machine Check Enable */
      19                 :             : #define MSR_FE0         PPC_BIT(52)     /* FP Exception 0 */
      20                 :             : #define MSR_SE          PPC_BIT(53)     /* Step enable */
      21                 :             : #define MSR_BE          PPC_BIT(54)     /* Branch trace enable */
      22                 :             : #define MSR_FE1         PPC_BIT(55)     /* FP Exception 1 */
      23                 :             : #define MSR_IR          PPC_BIT(58)     /* Instructions reloc */
      24                 :             : #define MSR_DR          PPC_BIT(59)     /* Data reloc */
      25                 :             : #define MSR_PMM         PPC_BIT(61)     /* Perf Monitor */
      26                 :             : #define MSR_RI          PPC_BIT(62)     /* Recoverable Interrupt */
      27                 :             : #define MSR_LE          PPC_BIT(63)     /* Little Endian */
      28                 :             : 
      29                 :             : /* PIR */
      30                 :             : #define SPR_PIR_P10_MASK        0x7fff  /* Mask of implemented bits */
      31                 :             : #define SPR_PIR_P9_MASK         0x7fff  /* Mask of implemented bits */
      32                 :             : #define SPR_PIR_P8_MASK         0x1fff  /* Mask of implemented bits */
      33                 :             : 
      34                 :             : /* SPR register definitions */
      35                 :             : #define SPR_DSCR        0x011   /* RW: Data stream control reg */
      36                 :             : #define SPR_DSISR       0x012   /* RW: Data storage interrupt status reg */
      37                 :             : #define SPR_DAR         0x013   /* RW: Data address reg */
      38                 :             : #define SPR_DEC         0x016   /* RW: Decrement Register */
      39                 :             : #define SPR_SDR1        0x019
      40                 :             : #define SPR_SRR0        0x01a   /* RW: Exception save/restore reg 0 */
      41                 :             : #define SPR_SRR1        0x01b   /* RW: Exception save/restore reg 1 */
      42                 :             : #define SPR_CFAR        0x01c   /* RW: Come From Address Register */
      43                 :             : #define SPR_AMR         0x01d   /* RW: Authority Mask Register */
      44                 :             : #define SPR_IAMR        0x03d   /* RW: Instruction Authority Mask Register */
      45                 :             : #define SPR_RPR         0x0ba   /* RW: Relative Priority Register */
      46                 :             : #define SPR_TBRL        0x10c   /* RO: Timebase low */
      47                 :             : #define SPR_TBRU        0x10d   /* RO: Timebase high */
      48                 :             : #define SPR_SPRC        0x114   /* RW: Access to uArch SPRs (ex SCOMC) */
      49                 :             : #define SPR_SPRD        0x115   /* RW: Access to uArch SPRs (ex SCOMD) */
      50                 :             : #define SPR_SCOMC       0x114   /* RW: SCOM Control - old name of SPRC */
      51                 :             : #define SPR_SCOMD       0x115   /* RW: SCOM Data    - old name of SPRD */
      52                 :             : #define SPR_TBWL        0x11c   /* RW: Timebase low */
      53                 :             : #define SPR_TBWU        0x11d   /* RW: Timebase high */
      54                 :             : #define SPR_TBU40       0x11e   /* RW: Timebase Upper 40 bit */
      55                 :             : #define SPR_PVR         0x11f   /* RO: Processor version register */
      56                 :             : #define SPR_HSPRG0      0x130   /* RW: Hypervisor scratch 0 */
      57                 :             : #define SPR_HSPRG1      0x131   /* RW: Hypervisor scratch 1 */
      58                 :             : #define SPR_SPURR       0x134   /* RW: Scaled Processor Utilization Resource */
      59                 :             : #define SPR_PURR        0x135   /* RW: Processor Utilization Resource reg */
      60                 :             : #define SPR_HDEC        0x136   /* RW: Hypervisor Decrementer */
      61                 :             : #define SPR_HRMOR       0x139   /* RW: Hypervisor Real Mode Offset reg */
      62                 :             : #define SPR_HSRR0       0x13a   /* RW: HV Exception save/restore reg 0 */
      63                 :             : #define SPR_HSRR1       0x13b   /* RW: HV Exception save/restore reg 1 */
      64                 :             : #define SPR_TFMR        0x13d
      65                 :             : #define SPR_LPCR        0x13e
      66                 :             : #define SPR_HMER        0x150   /* Hypervisor Maintenance Exception */
      67                 :             : #define SPR_HMEER       0x151   /* HMER interrupt enable mask */
      68                 :             : #define SPR_PCR         0x152
      69                 :             : #define SPR_AMOR        0x15d
      70                 :             : #define SPR_USRR0       0x1fa   /* RW: Ultravisor Save/Restore Register 0 */
      71                 :             : #define SPR_USRR1       0x1fb   /* RW: Ultravisor Save/Restore Register 1 */
      72                 :             : #define SPR_SMFCTRL     0x1ff   /* RW: Secure Memory Facility Control */
      73                 :             : #define SPR_PSSCR       0x357   /* RW: Stop status and control (ISA 3) */
      74                 :             : #define SPR_PPR32       0x382
      75                 :             : #define SPR_TSCR        0x399
      76                 :             : #define SPR_HID0        0x3f0
      77                 :             : #define SPR_HID1        0x3f1
      78                 :             : #define SPR_HID2        0x3f8
      79                 :             : #define SPR_HID4        0x3f4
      80                 :             : #define SPR_HID5        0x3f6
      81                 :             : #define SPR_PIR         0x3ff   /* RO: Processor Identification */
      82                 :             : 
      83                 :             : /* Bits in SRR1 */
      84                 :             : 
      85                 :             : #define SPR_SRR1_PM_WAKE_MASK   0x3c0000        /* PM wake reason for P8/9 */
      86                 :             : #define SPR_SRR1_PM_WAKE_SRESET 0x100000
      87                 :             : #define SPR_SRR1_PM_WAKE_MCE    0x3c0000        /* Use reserved value for MCE */
      88                 :             : 
      89                 :             : /* Bits in LPCR */
      90                 :             : 
      91                 :             : /* Powersave Exit Cause Enable is different on each generation */
      92                 :             : #define SPR_LPCR_P8_PECE        PPC_BITMASK(47,51)
      93                 :             : #define SPR_LPCR_P8_PECE0       PPC_BIT(47)   /* Wake on priv doorbell */
      94                 :             : #define SPR_LPCR_P8_PECE1       PPC_BIT(48)   /* Wake on hv doorbell */
      95                 :             : #define SPR_LPCR_P8_PECE2       PPC_BIT(49)   /* Wake on external interrupts */
      96                 :             : #define SPR_LPCR_P8_PECE3       PPC_BIT(50)   /* Wake on decrementer */
      97                 :             : #define SPR_LPCR_P8_PECE4       PPC_BIT(51)   /* Wake on MCs, HMIs, etc... */
      98                 :             : 
      99                 :             : #define SPR_LPCR_P9_PECE        (PPC_BITMASK(47,51) | PPC_BITMASK(17,17))
     100                 :             : #define SPR_LPCR_P9_PECEU0      PPC_BIT(17)   /* Wake on HVI */
     101                 :             : #define SPR_LPCR_P9_PECEL0      PPC_BIT(47)   /* Wake on priv doorbell */
     102                 :             : #define SPR_LPCR_P9_PECEL1      PPC_BIT(48)   /* Wake on hv doorbell */
     103                 :             : #define SPR_LPCR_P9_PECEL2      PPC_BIT(49)   /* Wake on external interrupts */
     104                 :             : #define SPR_LPCR_P9_PECEL3      PPC_BIT(50)   /* Wake on decrementer */
     105                 :             : #define SPR_LPCR_P9_PECEL4      PPC_BIT(51)   /* Wake on MCs, HMIs, etc... */
     106                 :             : #define SPR_LPCR_P9_LD          PPC_BIT(46)   /* Large decrementer mode bit */
     107                 :             : 
     108                 :             : 
     109                 :             : /* Bits in TFMR - control bits */
     110                 :             : #define SPR_TFMR_MAX_CYC_BET_STEPS      PPC_BITMASK(0,7)
     111                 :             : #define SPR_TFMR_N_CLKS_PER_STEP        PPC_BITMASK(8,9)
     112                 :             : #define SPR_TFMR_MASK_HMI               PPC_BIT(10)
     113                 :             : #define SPR_TFMR_SYNC_BIT_SEL           PPC_BITMASK(11,13)
     114                 :             : #define SPR_TFMR_TB_ECLIPZ              PPC_BIT(14)
     115                 :             : #define SPR_TFMR_LOAD_TOD_MOD           PPC_BIT(16)
     116                 :             : #define SPR_TFMR_MOVE_CHIP_TOD_TO_TB    PPC_BIT(18)
     117                 :             : #define SPR_TFMR_CLEAR_TB_ERRORS        PPC_BIT(24)
     118                 :             : /* Bits in TFMR - thread indep. status bits */
     119                 :             : #define SPR_TFMR_TFAC_XFER_ERROR        PPC_BIT(25)
     120                 :             : #define SPR_TFMR_HDEC_PARITY_ERROR      PPC_BIT(26)
     121                 :             : #define SPR_TFMR_TBST_CORRUPT           PPC_BIT(27)
     122                 :             : #define SPR_TFMR_TBST_ENCODED           PPC_BITMASK(28,31)
     123                 :             : #define SPR_TFMR_TBST_LAST              PPC_BITMASK(32,35)
     124                 :             : #define SPR_TFMR_TB_ENABLED             PPC_BIT(40)
     125                 :             : #define SPR_TFMR_TB_VALID               PPC_BIT(41)
     126                 :             : #define SPR_TFMR_TB_SYNC_OCCURED        PPC_BIT(42)
     127                 :             : #define SPR_TFMR_TB_MISSING_SYNC        PPC_BIT(43)
     128                 :             : #define SPR_TFMR_TB_MISSING_STEP        PPC_BIT(44)
     129                 :             : #define SPR_TFMR_TB_RESIDUE_ERR         PPC_BIT(45)
     130                 :             : #define SPR_TFMR_FW_CONTROL_ERR         PPC_BIT(46)
     131                 :             : #define SPR_TFMR_CHIP_TOD_STATUS        PPC_BITMASK(47,50)
     132                 :             : #define SPR_TFMR_CHIP_TOD_INTERRUPT     PPC_BIT(51)
     133                 :             : #define SPR_TFMR_CHIP_TOD_TIMEOUT       PPC_BIT(54)
     134                 :             : #define SPR_TFMR_CHIP_TOD_PARITY_ERR    PPC_BIT(56)
     135                 :             : /* Bits in TFMR - thread specific. status bits */
     136                 :             : #define SPR_TFMR_PURR_PARITY_ERR        PPC_BIT(57)
     137                 :             : #define SPR_TFMR_SPURR_PARITY_ERR       PPC_BIT(58)
     138                 :             : #define SPR_TFMR_DEC_PARITY_ERR         PPC_BIT(59)
     139                 :             : #define SPR_TFMR_TFMR_CORRUPT           PPC_BIT(60)
     140                 :             : #define SPR_TFMR_PURR_OVERFLOW          PPC_BIT(61)
     141                 :             : #define SPR_TFMR_SPURR_OVERFLOW         PPC_BIT(62)
     142                 :             : 
     143                 :             : /* Bits in HMER/HMEER */
     144                 :             : #define SPR_HMER_MALFUNCTION_ALERT      PPC_BIT(0)
     145                 :             : #define SPR_HMER_PROC_RECV_DONE         PPC_BIT(2)
     146                 :             : #define SPR_HMER_PROC_RECV_ERROR_MASKED PPC_BIT(3) /* Not P10 */
     147                 :             : #define SPR_HMER_TFAC_ERROR             PPC_BIT(4)
     148                 :             : #define SPR_HMER_TFMR_PARITY_ERROR      PPC_BIT(5) /* P9 */
     149                 :             : #define SPR_HMER_TFAC_SHADOW_XFER_ERROR PPC_BIT(5) /* P10 */
     150                 :             : #define SPR_HMER_SPURR_SCALE_LIMIT      PPC_BIT(6) /* P10 */
     151                 :             : #define SPR_HMER_XSCOM_FAIL             PPC_BIT(8)
     152                 :             : #define SPR_HMER_XSCOM_DONE             PPC_BIT(9)
     153                 :             : #define SPR_HMER_PROC_RECV_AGAIN        PPC_BIT(11)
     154                 :             : #define SPR_HMER_WARN_RISE              PPC_BIT(14) /* Not P10 */
     155                 :             : #define SPR_HMER_WARN_FALL              PPC_BIT(15) /* Not P10 */
     156                 :             : #define SPR_HMER_SCOM_FIR_HMI           PPC_BIT(16)
     157                 :             : #define SPR_HMER_TRIG_FIR_HMI           PPC_BIT(17) /* Not P10 */
     158                 :             : #define SPR_HMER_THD_WAKE_BLOCKED_TM_SUSPEND    PPC_BIT(17) /* Not P10 */
     159                 :             : #define SPR_HMER_P10_TRIG_FIR_HMI       PPC_BIT(18)
     160                 :             : #define SPR_HMER_HYP_RESOURCE_ERR       PPC_BIT(20) /* Not P10 */
     161                 :             : #define SPR_HMER_XSCOM_STATUS           PPC_BITMASK(21,23)
     162                 :             : 
     163                 :             : /*
     164                 :             :  * HMEER: initial bits for HMI interrupt enable mask.
     165                 :             :  * Per Dave Larson, never enable 8,9,21-23
     166                 :             :  */
     167                 :             : #define SPR_HMEER_HMI_ENABLE_MASK       (SPR_HMER_MALFUNCTION_ALERT |\
     168                 :             :                                          SPR_HMER_HYP_RESOURCE_ERR |\
     169                 :             :                                          SPR_HMER_PROC_RECV_DONE |\
     170                 :             :                                          SPR_HMER_PROC_RECV_ERROR_MASKED |\
     171                 :             :                                          SPR_HMER_TFAC_ERROR |\
     172                 :             :                                          SPR_HMER_TFMR_PARITY_ERROR |\
     173                 :             :                                          SPR_HMER_PROC_RECV_AGAIN)
     174                 :             : 
     175                 :             : #define SPR_HMEER_P10_HMI_ENABLE_MASK   (SPR_HMER_MALFUNCTION_ALERT |\
     176                 :             :                                          SPR_HMER_PROC_RECV_DONE |\
     177                 :             :                                          SPR_HMER_TFAC_ERROR |\
     178                 :             :                                          SPR_HMER_TFAC_SHADOW_XFER_ERROR |\
     179                 :             :                                          SPR_HMER_SPURR_SCALE_LIMIT |\
     180                 :             :                                          SPR_HMER_PROC_RECV_AGAIN)
     181                 :             : 
     182                 :             : /* Bits in HID0 */
     183                 :             : #define SPR_HID0_POWER8_4LPARMODE       PPC_BIT(2)
     184                 :             : #define SPR_HID0_POWER8_2LPARMODE       PPC_BIT(6)
     185                 :             : #define SPR_HID0_POWER8_DYNLPARDIS      PPC_BIT(15)
     186                 :             : #define SPR_HID0_POWER8_HILE            PPC_BIT(19)
     187                 :             : #define SPR_HID0_POWER9_HILE            PPC_BIT(4)
     188                 :             : #define SPR_HID0_POWER10_HILE           PPC_BIT(4)
     189                 :             : #define SPR_HID0_POWER8_ENABLE_ATTN     PPC_BIT(31)
     190                 :             : #define SPR_HID0_POWER9_ENABLE_ATTN     PPC_BIT(3)
     191                 :             : #define SPR_HID0_POWER10_ENABLE_ATTN    PPC_BIT(3)
     192                 :             : #define SPR_HID0_POWER9_FLUSH_ICACHE    PPC_BIT(2)
     193                 :             : #define SPR_HID0_POWER10_FLUSH_ICACHE   PPC_BIT(2)
     194                 :             : #define SPR_HID0_POWER9_RADIX           PPC_BIT(8)
     195                 :             : 
     196                 :             : /* PVR bits */
     197                 :             : #define SPR_PVR_TYPE                    0xffff0000
     198                 :             : #define SPR_PVR_CHIP_TYPE               0x0000f000
     199                 :             : #define SPR_PVR_VERS_MAJ                0x00000f00
     200                 :             : #define SPR_PVR_VERS_MIN                0x000000ff
     201                 :             : 
     202                 :             : #define PVR_TYPE(_pvr)          GETFIELD(SPR_PVR_TYPE, _pvr)
     203                 :             : #define PVR_CHIP_TYPE(_pvr)     GETFIELD(SPR_PVR_CHIP_TYPE, _pvr)
     204                 :             : #define PVR_VERS_MAJ(_pvr)      GETFIELD(SPR_PVR_VERS_MAJ, _pvr)
     205                 :             : #define PVR_VERS_MIN(_pvr)      GETFIELD(SPR_PVR_VERS_MIN, _pvr)
     206                 :             : 
     207                 :             : /* PVR definitions */
     208                 :             : #define PVR_TYPE_P8E    0x004b /* Murano */
     209                 :             : #define PVR_TYPE_P8     0x004d /* Venice */
     210                 :             : #define PVR_TYPE_P8NVL  0x004c /* Naples */
     211                 :             : #define PVR_TYPE_P9     0x004e
     212                 :             : #define PVR_TYPE_P9P    0x004f /* Axone */
     213                 :             : #define PVR_TYPE_P10    0x0080
     214                 :             : #define PVR_TYPE_P11    0x0082
     215                 :             : 
     216                 :             : #ifdef __ASSEMBLY__
     217                 :             : 
     218                 :             : /* Thread priority control opcodes */
     219                 :             : #define smt_low         or 1,1,1
     220                 :             : #define smt_medium      or 2,2,2
     221                 :             : #define smt_high        or 3,3,3
     222                 :             : #define smt_medium_high or 5,5,5
     223                 :             : #define smt_medium_low  or 6,6,6
     224                 :             : #define smt_extra_high  or 7,7,7
     225                 :             : #define smt_very_low    or 31,31,31
     226                 :             : #define smt_lowest      smt_low ; smt_very_low
     227                 :             : 
     228                 :             : #else /* __ASSEMBLY__ */
     229                 :             : 
     230                 :             : #include <ccan/str/str.h>
     231                 :             : #include <compiler.h>
     232                 :             : #include <stdbool.h>
     233                 :             : #include <stdint.h>
     234                 :             : 
     235                 :             : #define PPC_INST_NOP    0x60000000UL
     236                 :             : #define PPC_INST_TRAP   0x7fe00008UL
     237                 :             : 
     238                 :             : #define RB(b)           (((b) & 0x1f) << 11)
     239                 :             : #define MSGSND(b)       stringify(.long 0x7c00019c | RB(b))
     240                 :             : #define MSGCLR(b)       stringify(.long 0x7c0001dc | RB(b))
     241                 :             : #define MSGSYNC         stringify(.long 0x7c0006ec)
     242                 :             : 
     243                 :           4 : static inline bool is_power9n(uint32_t version)
     244                 :             : {
     245                 :           4 :         if (PVR_TYPE(version) != PVR_TYPE_P9)
     246                 :           0 :                 return false;
     247                 :             :         /*
     248                 :             :          * Bit 13 tells us:
     249                 :             :          *   0 = Scale out (aka Nimbus)
     250                 :             :          *   1 = Scale up  (aka Cumulus)
     251                 :             :          */
     252                 :           4 :         if ((version >> 13) & 1)
     253                 :           0 :                 return false;
     254                 :           4 :         return true;
     255                 :             : }
     256                 :             : 
     257                 :             : static inline bool is_fused_core(uint32_t version)
     258                 :             : {
     259                 :             :         if (PVR_TYPE(version) == PVR_TYPE_P9) {
     260                 :             :                 switch(PVR_CHIP_TYPE(version)) {
     261                 :             :                         case 0:
     262                 :             :                         case 2:
     263                 :             :                                 return true;
     264                 :             :                         default:
     265                 :             :                                 return false;
     266                 :             :                 }
     267                 :             : 
     268                 :             :         } else if(PVR_TYPE(version) == PVR_TYPE_P10) {
     269                 :             :                 if(PVR_CHIP_TYPE(version) & 0x01)
     270                 :             :                         return false;
     271                 :             :                 else
     272                 :             :                         return true;
     273                 :             :         } else if (PVR_TYPE(version) == PVR_TYPE_P11) {
     274                 :             :                 if (PVR_CHIP_TYPE(version) & 0x01)
     275                 :             :                         return false;
     276                 :             :                 else
     277                 :             :                         return true;
     278                 :             :         } else
     279                 :             :                 return false;
     280                 :             : }
     281                 :             : 
     282                 :           0 : static inline bool is_power9c(uint32_t version) 
     283                 :             : {
     284                 :             : 
     285                 :           0 :         if (PVR_TYPE(version) != PVR_TYPE_P9)
     286                 :           0 :                 return false;
     287                 :             :         /*
     288                 :             :          * Bit 13 tells us:
     289                 :             :          *   0 = Scale out (aka Nimbus)
     290                 :             :          *   1 = Scale up  (aka Cumulus)
     291                 :             :          */
     292                 :           0 :         if (!((version >> 13) & 1))
     293                 :           0 :                 return false;
     294                 :           0 :         return true;
     295                 :             : }
     296                 :             : 
     297                 :             : #ifndef __TEST__
     298                 :             : 
     299                 :             : /* POWER9 and above only */
     300                 :             : static inline void flush_erat(void)
     301                 :             : {
     302                 :             :         asm volatile("slbia        7");
     303                 :             : }
     304                 :             : 
     305                 :             : /*
     306                 :             :  * SMT priority
     307                 :             :  */
     308                 :             : 
     309                 :             : static inline void smt_low(void)        { asm volatile("or 1,1,1");   }
     310                 :             : static inline void smt_medium(void)     { asm volatile("or 2,2,2");   }
     311                 :             : static inline void smt_high(void)       { asm volatile("or 3,3,3");   }
     312                 :             : static inline void smt_medium_high(void){ asm volatile("or 5,5,5");   }
     313                 :             : static inline void smt_medium_low(void) { asm volatile("or 6,6,6");   }
     314                 :             : static inline void smt_extra_high(void) { asm volatile("or 7,7,7");   }
     315                 :             : static inline void smt_very_low(void)   { asm volatile("or 31,31,31");        }
     316                 :             : static inline void smt_lowest(void)     { smt_low(); smt_very_low();    }
     317                 :             : 
     318                 :             : /*
     319                 :             :  * SPR access functions
     320                 :             :  */
     321                 :             : 
     322                 :             : static inline unsigned long mfmsr(void)
     323                 :             : {
     324                 :             :         unsigned long val;
     325                 :             :         
     326                 :             :         asm volatile("mfmsr %0" : "=r"(val) : : "memory");
     327                 :             :         return val;
     328                 :             : }
     329                 :             : 
     330                 :             : static inline void mtmsr(unsigned long val)
     331                 :             : {
     332                 :             :         asm volatile("mtmsr %0" : : "r"(val) : "memory");
     333                 :             : }
     334                 :             : 
     335                 :             : static inline void mtmsrd(unsigned long val, int l)
     336                 :             : {
     337                 :             :         asm volatile("mtmsrd %0,%1" : : "r"(val), "i"(l) : "memory");
     338                 :             : }
     339                 :             : 
     340                 :             : static inline __attribute__((always_inline))
     341                 :             : unsigned long mfspr(const unsigned int spr)
     342                 :             : {
     343                 :             :         unsigned long val;
     344                 :             : 
     345                 :             :         asm volatile("mfspr %0,%1" : "=r"(val) : "i"(spr) : "memory");
     346                 :             :         return val;
     347                 :             : }
     348                 :             : 
     349                 :             : static inline __attribute__((always_inline))
     350                 :             : void mtspr(const unsigned int spr, unsigned long val)
     351                 :             : {
     352                 :             :         asm volatile("mtspr %0,%1" : : "i"(spr), "r"(val) : "memory");
     353                 :             : }
     354                 :             : 
     355                 :             : /* Helpers for special sequences needed by some registers */
     356                 :             : extern void set_hid0(unsigned long hid0);
     357                 :             : extern void trigger_attn(void);
     358                 :             : 
     359                 :             : /*
     360                 :             :  * Barriers
     361                 :             :  */
     362                 :             : 
     363                 :             : static inline void eieio(void)
     364                 :             : {
     365                 :             :         asm volatile("eieio" : : : "memory");
     366                 :             : }
     367                 :             : 
     368                 :             : static inline void sync(void)
     369                 :             : {
     370                 :             :         asm volatile("sync" : : : "memory");
     371                 :             : }
     372                 :             : 
     373                 :             : static inline void lwsync(void)
     374                 :             : {
     375                 :             :         asm volatile("lwsync" : : : "memory");
     376                 :             : }
     377                 :             : 
     378                 :             : static inline void isync(void)
     379                 :             : {
     380                 :             :         asm volatile("isync" : : : "memory");
     381                 :             : }
     382                 :             : 
     383                 :             : 
     384                 :             : /*
     385                 :             :  * Cache sync
     386                 :             :  */
     387                 :             : static inline void sync_icache(void)
     388                 :             : {
     389                 :             :         asm volatile("sync; icbi 0,%0; sync; isync" : : "r" (0) : "memory");
     390                 :             : }
     391                 :             : 
     392                 :             : /*
     393                 :             :  * Doorbells
     394                 :             :  */
     395                 :             : static inline void msgclr(void)
     396                 :             : {
     397                 :             :         uint64_t rb = (0x05 << (63-36));
     398                 :             :         asm volatile(MSGCLR(%0) : : "r"(rb));
     399                 :             : }
     400                 :             : 
     401                 :             : static inline void p9_dbell_receive(void)
     402                 :             : {
     403                 :             :         uint64_t rb = (0x05 << (63-36));
     404                 :             :         asm volatile(MSGCLR(%0) ";"
     405                 :             :                      MSGSYNC    ";"
     406                 :             :                      "lwsync"
     407                 :             :                      : : "r"(rb));
     408                 :             : }
     409                 :             : 
     410                 :             : static inline void p9_dbell_send(uint32_t pir)
     411                 :             : {
     412                 :             :         uint64_t rb = (0x05 << (63-36)) | pir;
     413                 :             :         asm volatile("sync ;"
     414                 :             :                      MSGSND(%0)
     415                 :             :                      : : "r"(rb));
     416                 :             : }
     417                 :             : 
     418                 :             : /*
     419                 :             :  * Byteswap load/stores
     420                 :             :  */
     421                 :             : 
     422                 :             : static inline uint16_t ld_le16(const uint16_t *addr)
     423                 :             : {
     424                 :             :         uint16_t val;
     425                 :             :         asm volatile("lhbrx %0,0,%1" : "=r"(val) : "r"(addr), "m"(*addr));
     426                 :             :         return val;
     427                 :             : }
     428                 :             : 
     429                 :             : static inline uint32_t ld_le32(const uint32_t *addr)
     430                 :             : {
     431                 :             :         uint32_t val;
     432                 :             :         asm volatile("lwbrx %0,0,%1" : "=r"(val) : "r"(addr), "m"(*addr));
     433                 :             :         return val;
     434                 :             : }
     435                 :             : 
     436                 :             : static inline void st_le16(uint16_t *addr, uint16_t val)
     437                 :             : {
     438                 :             :         asm volatile("sthbrx %0,0,%1" : : "r"(val), "r"(addr), "m"(*addr));
     439                 :             : }
     440                 :             : 
     441                 :             : static inline void st_le32(uint32_t *addr, uint32_t val)
     442                 :             : {
     443                 :             :         asm volatile("stwbrx %0,0,%1" : : "r"(val), "r"(addr), "m"(*addr));
     444                 :             : }
     445                 :             : 
     446                 :             : #endif /* __TEST__ */
     447                 :             : 
     448                 :             : #endif /* __ASSEMBLY__ */
     449                 :             : 
     450                 :             : #endif /* __PROCESSOR_H */
        

Generated by: LCOV version 2.0-1