root/arch/csky/abiv2/fpu.c

/* [<][>][^][v][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. fpu_libc_helper
  2. fpu_fpe
  3. save_to_user_fp
  4. restore_from_user_fp

   1 // SPDX-License-Identifier: GPL-2.0
   2 // Copyright (C) 2018 Hangzhou C-SKY Microsystems co.,ltd.
   3 
   4 #include <linux/ptrace.h>
   5 #include <linux/uaccess.h>
   6 #include <abi/reg_ops.h>
   7 
   8 #define MTCR_MASK       0xFC00FFE0
   9 #define MFCR_MASK       0xFC00FFE0
  10 #define MTCR_DIST       0xC0006420
  11 #define MFCR_DIST       0xC0006020
  12 
  13 /*
  14  * fpu_libc_helper() is to help libc to excute:
  15  *  - mfcr %a, cr<1, 2>
  16  *  - mfcr %a, cr<2, 2>
  17  *  - mtcr %a, cr<1, 2>
  18  *  - mtcr %a, cr<2, 2>
  19  */
  20 int fpu_libc_helper(struct pt_regs *regs)
  21 {
  22         int fault;
  23         unsigned long instrptr, regx = 0;
  24         unsigned long index = 0, tmp = 0;
  25         unsigned long tinstr = 0;
  26         u16 instr_hi, instr_low;
  27 
  28         instrptr = instruction_pointer(regs);
  29         if (instrptr & 1)
  30                 return 0;
  31 
  32         fault = __get_user(instr_low, (u16 *)instrptr);
  33         if (fault)
  34                 return 0;
  35 
  36         fault = __get_user(instr_hi, (u16 *)(instrptr + 2));
  37         if (fault)
  38                 return 0;
  39 
  40         tinstr = instr_hi | ((unsigned long)instr_low << 16);
  41 
  42         if (((tinstr >> 21) & 0x1F) != 2)
  43                 return 0;
  44 
  45         if ((tinstr & MTCR_MASK) == MTCR_DIST) {
  46                 index = (tinstr >> 16) & 0x1F;
  47                 if (index > 13)
  48                         return 0;
  49 
  50                 tmp = tinstr & 0x1F;
  51                 if (tmp > 2)
  52                         return 0;
  53 
  54                 regx =  *(&regs->a0 + index);
  55 
  56                 if (tmp == 1)
  57                         mtcr("cr<1, 2>", regx);
  58                 else if (tmp == 2)
  59                         mtcr("cr<2, 2>", regx);
  60                 else
  61                         return 0;
  62 
  63                 regs->pc += 4;
  64                 return 1;
  65         }
  66 
  67         if ((tinstr & MFCR_MASK) == MFCR_DIST) {
  68                 index = tinstr & 0x1F;
  69                 if (index > 13)
  70                         return 0;
  71 
  72                 tmp = ((tinstr >> 16) & 0x1F);
  73                 if (tmp > 2)
  74                         return 0;
  75 
  76                 if (tmp == 1)
  77                         regx = mfcr("cr<1, 2>");
  78                 else if (tmp == 2)
  79                         regx = mfcr("cr<2, 2>");
  80                 else
  81                         return 0;
  82 
  83                 *(&regs->a0 + index) = regx;
  84 
  85                 regs->pc += 4;
  86                 return 1;
  87         }
  88 
  89         return 0;
  90 }
  91 
  92 void fpu_fpe(struct pt_regs *regs)
  93 {
  94         int sig, code;
  95         unsigned int fesr;
  96 
  97         fesr = mfcr("cr<2, 2>");
  98 
  99         sig = SIGFPE;
 100         code = FPE_FLTUNK;
 101 
 102         if (fesr & FPE_ILLE) {
 103                 sig = SIGILL;
 104                 code = ILL_ILLOPC;
 105         } else if (fesr & FPE_IDC) {
 106                 sig = SIGILL;
 107                 code = ILL_ILLOPN;
 108         } else if (fesr & FPE_FEC) {
 109                 sig = SIGFPE;
 110                 if (fesr & FPE_IOC)
 111                         code = FPE_FLTINV;
 112                 else if (fesr & FPE_DZC)
 113                         code = FPE_FLTDIV;
 114                 else if (fesr & FPE_UFC)
 115                         code = FPE_FLTUND;
 116                 else if (fesr & FPE_OFC)
 117                         code = FPE_FLTOVF;
 118                 else if (fesr & FPE_IXC)
 119                         code = FPE_FLTRES;
 120         }
 121 
 122         force_sig_fault(sig, code, (void __user *)regs->pc);
 123 }
 124 
 125 #define FMFVR_FPU_REGS(vrx, vry)        \
 126         "fmfvrl %0, "#vrx"\n"           \
 127         "fmfvrh %1, "#vrx"\n"           \
 128         "fmfvrl %2, "#vry"\n"           \
 129         "fmfvrh %3, "#vry"\n"
 130 
 131 #define FMTVR_FPU_REGS(vrx, vry)        \
 132         "fmtvrl "#vrx", %0\n"           \
 133         "fmtvrh "#vrx", %1\n"           \
 134         "fmtvrl "#vry", %2\n"           \
 135         "fmtvrh "#vry", %3\n"
 136 
 137 #define STW_FPU_REGS(a, b, c, d)        \
 138         "stw    %0, (%4, "#a")\n"       \
 139         "stw    %1, (%4, "#b")\n"       \
 140         "stw    %2, (%4, "#c")\n"       \
 141         "stw    %3, (%4, "#d")\n"
 142 
 143 #define LDW_FPU_REGS(a, b, c, d)        \
 144         "ldw    %0, (%4, "#a")\n"       \
 145         "ldw    %1, (%4, "#b")\n"       \
 146         "ldw    %2, (%4, "#c")\n"       \
 147         "ldw    %3, (%4, "#d")\n"
 148 
 149 void save_to_user_fp(struct user_fp *user_fp)
 150 {
 151         unsigned long flg;
 152         unsigned long tmp1, tmp2;
 153         unsigned long *fpregs;
 154 
 155         local_irq_save(flg);
 156 
 157         tmp1 = mfcr("cr<1, 2>");
 158         tmp2 = mfcr("cr<2, 2>");
 159 
 160         user_fp->fcr = tmp1;
 161         user_fp->fesr = tmp2;
 162 
 163         fpregs = &user_fp->vr[0];
 164 #ifdef CONFIG_CPU_HAS_FPUV2
 165 #ifdef CONFIG_CPU_HAS_VDSP
 166         asm volatile(
 167                 "vstmu.32    vr0-vr3,   (%0)\n"
 168                 "vstmu.32    vr4-vr7,   (%0)\n"
 169                 "vstmu.32    vr8-vr11,  (%0)\n"
 170                 "vstmu.32    vr12-vr15, (%0)\n"
 171                 "fstmu.64    vr16-vr31, (%0)\n"
 172                 : "+a"(fpregs)
 173                 ::"memory");
 174 #else
 175         asm volatile(
 176                 "fstmu.64    vr0-vr31,  (%0)\n"
 177                 : "+a"(fpregs)
 178                 ::"memory");
 179 #endif
 180 #else
 181         {
 182         unsigned long tmp3, tmp4;
 183 
 184         asm volatile(
 185                 FMFVR_FPU_REGS(vr0, vr1)
 186                 STW_FPU_REGS(0, 4, 16, 20)
 187                 FMFVR_FPU_REGS(vr2, vr3)
 188                 STW_FPU_REGS(32, 36, 48, 52)
 189                 FMFVR_FPU_REGS(vr4, vr5)
 190                 STW_FPU_REGS(64, 68, 80, 84)
 191                 FMFVR_FPU_REGS(vr6, vr7)
 192                 STW_FPU_REGS(96, 100, 112, 116)
 193                 "addi   %4, 128\n"
 194                 FMFVR_FPU_REGS(vr8, vr9)
 195                 STW_FPU_REGS(0, 4, 16, 20)
 196                 FMFVR_FPU_REGS(vr10, vr11)
 197                 STW_FPU_REGS(32, 36, 48, 52)
 198                 FMFVR_FPU_REGS(vr12, vr13)
 199                 STW_FPU_REGS(64, 68, 80, 84)
 200                 FMFVR_FPU_REGS(vr14, vr15)
 201                 STW_FPU_REGS(96, 100, 112, 116)
 202                 : "=a"(tmp1), "=a"(tmp2), "=a"(tmp3),
 203                   "=a"(tmp4), "+a"(fpregs)
 204                 ::"memory");
 205         }
 206 #endif
 207 
 208         local_irq_restore(flg);
 209 }
 210 
 211 void restore_from_user_fp(struct user_fp *user_fp)
 212 {
 213         unsigned long flg;
 214         unsigned long tmp1, tmp2;
 215         unsigned long *fpregs;
 216 
 217         local_irq_save(flg);
 218 
 219         tmp1 = user_fp->fcr;
 220         tmp2 = user_fp->fesr;
 221 
 222         mtcr("cr<1, 2>", tmp1);
 223         mtcr("cr<2, 2>", tmp2);
 224 
 225         fpregs = &user_fp->vr[0];
 226 #ifdef CONFIG_CPU_HAS_FPUV2
 227 #ifdef CONFIG_CPU_HAS_VDSP
 228         asm volatile(
 229                 "vldmu.32    vr0-vr3,   (%0)\n"
 230                 "vldmu.32    vr4-vr7,   (%0)\n"
 231                 "vldmu.32    vr8-vr11,  (%0)\n"
 232                 "vldmu.32    vr12-vr15, (%0)\n"
 233                 "fldmu.64    vr16-vr31, (%0)\n"
 234                 : "+a"(fpregs)
 235                 ::"memory");
 236 #else
 237         asm volatile(
 238                 "fldmu.64    vr0-vr31,  (%0)\n"
 239                 : "+a"(fpregs)
 240                 ::"memory");
 241 #endif
 242 #else
 243         {
 244         unsigned long tmp3, tmp4;
 245 
 246         asm volatile(
 247                 LDW_FPU_REGS(0, 4, 16, 20)
 248                 FMTVR_FPU_REGS(vr0, vr1)
 249                 LDW_FPU_REGS(32, 36, 48, 52)
 250                 FMTVR_FPU_REGS(vr2, vr3)
 251                 LDW_FPU_REGS(64, 68, 80, 84)
 252                 FMTVR_FPU_REGS(vr4, vr5)
 253                 LDW_FPU_REGS(96, 100, 112, 116)
 254                 FMTVR_FPU_REGS(vr6, vr7)
 255                 "addi   %4, 128\n"
 256                 LDW_FPU_REGS(0, 4, 16, 20)
 257                 FMTVR_FPU_REGS(vr8, vr9)
 258                 LDW_FPU_REGS(32, 36, 48, 52)
 259                 FMTVR_FPU_REGS(vr10, vr11)
 260                 LDW_FPU_REGS(64, 68, 80, 84)
 261                 FMTVR_FPU_REGS(vr12, vr13)
 262                 LDW_FPU_REGS(96, 100, 112, 116)
 263                 FMTVR_FPU_REGS(vr14, vr15)
 264                 : "=a"(tmp1), "=a"(tmp2), "=a"(tmp3),
 265                   "=a"(tmp4), "+a"(fpregs)
 266                 ::"memory");
 267         }
 268 #endif
 269         local_irq_restore(flg);
 270 }

/* [<][>][^][v][top][bottom][index][help] */