1 /* 2 * Microblaze helper routines. 3 * 4 * Copyright (c) 2009 Edgar E. Iglesias <edgar.iglesias@gmail.com>. 5 * Copyright (c) 2009-2012 PetaLogix Qld Pty Ltd. 6 * 7 * This library is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU Lesser General Public 9 * License as published by the Free Software Foundation; either 10 * version 2.1 of the License, or (at your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 * Lesser General Public License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public 18 * License along with this library; if not, see <http://www.gnu.org/licenses/>. 19 */ 20 21 #include "qemu/osdep.h" 22 #include "qemu/log.h" 23 #include "cpu.h" 24 #include "exec/helper-proto.h" 25 #include "qemu/host-utils.h" 26 #include "accel/tcg/cpu-ldst.h" 27 #include "fpu/softfloat.h" 28 29 void helper_put(uint32_t id, uint32_t ctrl, uint32_t data) 30 { 31 int test = ctrl & STREAM_TEST; 32 int atomic = ctrl & STREAM_ATOMIC; 33 int control = ctrl & STREAM_CONTROL; 34 int nonblock = ctrl & STREAM_NONBLOCK; 35 int exception = ctrl & STREAM_EXCEPTION; 36 37 qemu_log_mask(LOG_UNIMP, "Unhandled stream put to stream-id=%d data=%x %s%s%s%s%s\n", 38 id, data, 39 test ? "t" : "", 40 nonblock ? "n" : "", 41 exception ? "e" : "", 42 control ? "c" : "", 43 atomic ? "a" : ""); 44 } 45 46 uint32_t helper_get(uint32_t id, uint32_t ctrl) 47 { 48 int test = ctrl & STREAM_TEST; 49 int atomic = ctrl & STREAM_ATOMIC; 50 int control = ctrl & STREAM_CONTROL; 51 int nonblock = ctrl & STREAM_NONBLOCK; 52 int exception = ctrl & STREAM_EXCEPTION; 53 54 qemu_log_mask(LOG_UNIMP, "Unhandled stream get from stream-id=%d %s%s%s%s%s\n", 55 id, 56 test ? "t" : "", 57 nonblock ? "n" : "", 58 exception ? "e" : "", 59 control ? "c" : "", 60 atomic ? "a" : ""); 61 return 0xdead0000 | id; 62 } 63 64 void helper_raise_exception(CPUMBState *env, uint32_t index) 65 { 66 CPUState *cs = env_cpu(env); 67 68 cs->exception_index = index; 69 cpu_loop_exit(cs); 70 } 71 72 static bool check_divz(CPUMBState *env, uint32_t a, uint32_t b, uintptr_t ra) 73 { 74 if (unlikely(b == 0)) { 75 env->msr |= MSR_DZ; 76 77 if ((env->msr & MSR_EE) && 78 env_archcpu(env)->cfg.div_zero_exception) { 79 CPUState *cs = env_cpu(env); 80 81 env->esr = ESR_EC_DIVZERO; 82 cs->exception_index = EXCP_HW_EXCP; 83 cpu_loop_exit_restore(cs, ra); 84 } 85 return false; 86 } 87 return true; 88 } 89 90 uint32_t helper_divs(CPUMBState *env, uint32_t a, uint32_t b) 91 { 92 if (!check_divz(env, a, b, GETPC())) { 93 return 0; 94 } 95 return (int32_t)a / (int32_t)b; 96 } 97 98 uint32_t helper_divu(CPUMBState *env, uint32_t a, uint32_t b) 99 { 100 if (!check_divz(env, a, b, GETPC())) { 101 return 0; 102 } 103 return a / b; 104 } 105 106 /* raise FPU exception. */ 107 static void raise_fpu_exception(CPUMBState *env, uintptr_t ra) 108 { 109 CPUState *cs = env_cpu(env); 110 111 env->esr = ESR_EC_FPU; 112 cs->exception_index = EXCP_HW_EXCP; 113 cpu_loop_exit_restore(cs, ra); 114 } 115 116 static void update_fpu_flags(CPUMBState *env, int flags, uintptr_t ra) 117 { 118 int raise = 0; 119 120 if (flags & float_flag_invalid) { 121 env->fsr |= FSR_IO; 122 raise = 1; 123 } 124 if (flags & float_flag_divbyzero) { 125 env->fsr |= FSR_DZ; 126 raise = 1; 127 } 128 if (flags & float_flag_overflow) { 129 env->fsr |= FSR_OF; 130 raise = 1; 131 } 132 if (flags & float_flag_underflow) { 133 env->fsr |= FSR_UF; 134 raise = 1; 135 } 136 if (raise 137 && (env_archcpu(env)->cfg.pvr_regs[2] & PVR2_FPU_EXC_MASK) 138 && (env->msr & MSR_EE)) { 139 raise_fpu_exception(env, ra); 140 } 141 } 142 143 uint32_t helper_fadd(CPUMBState *env, uint32_t a, uint32_t b) 144 { 145 CPU_FloatU fd, fa, fb; 146 int flags; 147 148 set_float_exception_flags(0, &env->fp_status); 149 fa.l = a; 150 fb.l = b; 151 fd.f = float32_add(fa.f, fb.f, &env->fp_status); 152 153 flags = get_float_exception_flags(&env->fp_status); 154 update_fpu_flags(env, flags, GETPC()); 155 return fd.l; 156 } 157 158 uint32_t helper_frsub(CPUMBState *env, uint32_t a, uint32_t b) 159 { 160 CPU_FloatU fd, fa, fb; 161 int flags; 162 163 set_float_exception_flags(0, &env->fp_status); 164 fa.l = a; 165 fb.l = b; 166 fd.f = float32_sub(fb.f, fa.f, &env->fp_status); 167 flags = get_float_exception_flags(&env->fp_status); 168 update_fpu_flags(env, flags, GETPC()); 169 return fd.l; 170 } 171 172 uint32_t helper_fmul(CPUMBState *env, uint32_t a, uint32_t b) 173 { 174 CPU_FloatU fd, fa, fb; 175 int flags; 176 177 set_float_exception_flags(0, &env->fp_status); 178 fa.l = a; 179 fb.l = b; 180 fd.f = float32_mul(fa.f, fb.f, &env->fp_status); 181 flags = get_float_exception_flags(&env->fp_status); 182 update_fpu_flags(env, flags, GETPC()); 183 184 return fd.l; 185 } 186 187 uint32_t helper_fdiv(CPUMBState *env, uint32_t a, uint32_t b) 188 { 189 CPU_FloatU fd, fa, fb; 190 int flags; 191 192 set_float_exception_flags(0, &env->fp_status); 193 fa.l = a; 194 fb.l = b; 195 fd.f = float32_div(fb.f, fa.f, &env->fp_status); 196 flags = get_float_exception_flags(&env->fp_status); 197 update_fpu_flags(env, flags, GETPC()); 198 199 return fd.l; 200 } 201 202 uint32_t helper_fcmp_un(CPUMBState *env, uint32_t a, uint32_t b) 203 { 204 CPU_FloatU fa, fb; 205 uint32_t r = 0; 206 207 fa.l = a; 208 fb.l = b; 209 210 if (float32_is_signaling_nan(fa.f, &env->fp_status) || 211 float32_is_signaling_nan(fb.f, &env->fp_status)) { 212 update_fpu_flags(env, float_flag_invalid, GETPC()); 213 r = 1; 214 } 215 216 if (float32_is_quiet_nan(fa.f, &env->fp_status) || 217 float32_is_quiet_nan(fb.f, &env->fp_status)) { 218 r = 1; 219 } 220 221 return r; 222 } 223 224 uint32_t helper_fcmp_lt(CPUMBState *env, uint32_t a, uint32_t b) 225 { 226 CPU_FloatU fa, fb; 227 int r; 228 int flags; 229 230 set_float_exception_flags(0, &env->fp_status); 231 fa.l = a; 232 fb.l = b; 233 r = float32_lt(fb.f, fa.f, &env->fp_status); 234 flags = get_float_exception_flags(&env->fp_status); 235 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 236 237 return r; 238 } 239 240 uint32_t helper_fcmp_eq(CPUMBState *env, uint32_t a, uint32_t b) 241 { 242 CPU_FloatU fa, fb; 243 int flags; 244 int r; 245 246 set_float_exception_flags(0, &env->fp_status); 247 fa.l = a; 248 fb.l = b; 249 r = float32_eq_quiet(fa.f, fb.f, &env->fp_status); 250 flags = get_float_exception_flags(&env->fp_status); 251 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 252 253 return r; 254 } 255 256 uint32_t helper_fcmp_le(CPUMBState *env, uint32_t a, uint32_t b) 257 { 258 CPU_FloatU fa, fb; 259 int flags; 260 int r; 261 262 fa.l = a; 263 fb.l = b; 264 set_float_exception_flags(0, &env->fp_status); 265 r = float32_le(fa.f, fb.f, &env->fp_status); 266 flags = get_float_exception_flags(&env->fp_status); 267 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 268 269 270 return r; 271 } 272 273 uint32_t helper_fcmp_gt(CPUMBState *env, uint32_t a, uint32_t b) 274 { 275 CPU_FloatU fa, fb; 276 int flags, r; 277 278 fa.l = a; 279 fb.l = b; 280 set_float_exception_flags(0, &env->fp_status); 281 r = float32_lt(fa.f, fb.f, &env->fp_status); 282 flags = get_float_exception_flags(&env->fp_status); 283 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 284 return r; 285 } 286 287 uint32_t helper_fcmp_ne(CPUMBState *env, uint32_t a, uint32_t b) 288 { 289 CPU_FloatU fa, fb; 290 int flags, r; 291 292 fa.l = a; 293 fb.l = b; 294 set_float_exception_flags(0, &env->fp_status); 295 r = !float32_eq_quiet(fa.f, fb.f, &env->fp_status); 296 flags = get_float_exception_flags(&env->fp_status); 297 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 298 299 return r; 300 } 301 302 uint32_t helper_fcmp_ge(CPUMBState *env, uint32_t a, uint32_t b) 303 { 304 CPU_FloatU fa, fb; 305 int flags, r; 306 307 fa.l = a; 308 fb.l = b; 309 set_float_exception_flags(0, &env->fp_status); 310 r = !float32_lt(fa.f, fb.f, &env->fp_status); 311 flags = get_float_exception_flags(&env->fp_status); 312 update_fpu_flags(env, flags & float_flag_invalid, GETPC()); 313 314 return r; 315 } 316 317 uint32_t helper_flt(CPUMBState *env, uint32_t a) 318 { 319 CPU_FloatU fd, fa; 320 321 fa.l = a; 322 fd.f = int32_to_float32(fa.l, &env->fp_status); 323 return fd.l; 324 } 325 326 uint32_t helper_fint(CPUMBState *env, uint32_t a) 327 { 328 CPU_FloatU fa; 329 uint32_t r; 330 int flags; 331 332 set_float_exception_flags(0, &env->fp_status); 333 fa.l = a; 334 r = float32_to_int32(fa.f, &env->fp_status); 335 flags = get_float_exception_flags(&env->fp_status); 336 update_fpu_flags(env, flags, GETPC()); 337 338 return r; 339 } 340 341 uint32_t helper_fsqrt(CPUMBState *env, uint32_t a) 342 { 343 CPU_FloatU fd, fa; 344 int flags; 345 346 set_float_exception_flags(0, &env->fp_status); 347 fa.l = a; 348 fd.l = float32_sqrt(fa.f, &env->fp_status); 349 flags = get_float_exception_flags(&env->fp_status); 350 update_fpu_flags(env, flags, GETPC()); 351 352 return fd.l; 353 } 354 355 uint32_t helper_pcmpbf(uint32_t a, uint32_t b) 356 { 357 unsigned int i; 358 uint32_t mask = 0xff000000; 359 360 for (i = 0; i < 4; i++) { 361 if ((a & mask) == (b & mask)) 362 return i + 1; 363 mask >>= 8; 364 } 365 return 0; 366 } 367 368 void helper_stackprot(CPUMBState *env, target_ulong addr) 369 { 370 if (addr < env->slr || addr > env->shr) { 371 CPUState *cs = env_cpu(env); 372 373 qemu_log_mask(CPU_LOG_INT, "Stack protector violation at " 374 TARGET_FMT_lx " %x %x\n", 375 addr, env->slr, env->shr); 376 377 env->ear = addr; 378 env->esr = ESR_EC_STACKPROT; 379 cs->exception_index = EXCP_HW_EXCP; 380 cpu_loop_exit_restore(cs, GETPC()); 381 } 382 } 383 384 #if !defined(CONFIG_USER_ONLY) 385 #include "system/memory.h" 386 387 /* Writes/reads to the MMU's special regs end up here. */ 388 uint32_t helper_mmu_read(CPUMBState *env, uint32_t ext, uint32_t rn) 389 { 390 return mmu_read(env, ext, rn); 391 } 392 393 void helper_mmu_write(CPUMBState *env, uint32_t ext, uint32_t rn, uint32_t v) 394 { 395 mmu_write(env, ext, rn, v); 396 } 397 398 static void mb_transaction_failed_internal(CPUState *cs, hwaddr physaddr, 399 uint64_t addr, unsigned size, 400 MMUAccessType access_type, 401 uintptr_t retaddr) 402 { 403 CPUMBState *env = cpu_env(cs); 404 MicroBlazeCPU *cpu = env_archcpu(env); 405 const char *access_name = "INVALID"; 406 bool take = env->msr & MSR_EE; 407 uint32_t esr = ESR_EC_DATA_BUS; 408 409 switch (access_type) { 410 case MMU_INST_FETCH: 411 access_name = "INST_FETCH"; 412 esr = ESR_EC_INSN_BUS; 413 take &= cpu->cfg.iopb_bus_exception; 414 break; 415 case MMU_DATA_LOAD: 416 access_name = "DATA_LOAD"; 417 take &= cpu->cfg.dopb_bus_exception; 418 break; 419 case MMU_DATA_STORE: 420 access_name = "DATA_STORE"; 421 take &= cpu->cfg.dopb_bus_exception; 422 break; 423 } 424 425 qemu_log_mask(CPU_LOG_INT, "Transaction failed: addr 0x%" PRIx64 426 "physaddr 0x" HWADDR_FMT_plx " size %d access-type %s (%s)\n", 427 addr, physaddr, size, access_name, 428 take ? "TAKEN" : "DROPPED"); 429 430 if (take) { 431 env->esr = esr; 432 env->ear = addr; 433 cs->exception_index = EXCP_HW_EXCP; 434 cpu_loop_exit_restore(cs, retaddr); 435 } 436 } 437 438 void mb_cpu_transaction_failed(CPUState *cs, hwaddr physaddr, vaddr addr, 439 unsigned size, MMUAccessType access_type, 440 int mmu_idx, MemTxAttrs attrs, 441 MemTxResult response, uintptr_t retaddr) 442 { 443 mb_transaction_failed_internal(cs, physaddr, addr, size, 444 access_type, retaddr); 445 } 446 447 #define LD_EA(NAME, TYPE, FUNC) \ 448 uint32_t HELPER(NAME)(CPUMBState *env, uint64_t ea) \ 449 { \ 450 CPUState *cs = env_cpu(env); \ 451 MemTxResult txres; \ 452 TYPE ret = FUNC(cs->as, ea, MEMTXATTRS_UNSPECIFIED, &txres); \ 453 if (unlikely(txres != MEMTX_OK)) { \ 454 mb_transaction_failed_internal(cs, ea, ea, sizeof(TYPE), \ 455 MMU_DATA_LOAD, GETPC()); \ 456 } \ 457 return ret; \ 458 } 459 460 LD_EA(lbuea, uint8_t, address_space_ldub) 461 LD_EA(lhuea_be, uint16_t, address_space_lduw_be) 462 LD_EA(lhuea_le, uint16_t, address_space_lduw_le) 463 LD_EA(lwea_be, uint32_t, address_space_ldl_be) 464 LD_EA(lwea_le, uint32_t, address_space_ldl_le) 465 466 #define ST_EA(NAME, TYPE, FUNC) \ 467 void HELPER(NAME)(CPUMBState *env, uint32_t data, uint64_t ea) \ 468 { \ 469 CPUState *cs = env_cpu(env); \ 470 MemTxResult txres; \ 471 FUNC(cs->as, ea, data, MEMTXATTRS_UNSPECIFIED, &txres); \ 472 if (unlikely(txres != MEMTX_OK)) { \ 473 mb_transaction_failed_internal(cs, ea, ea, sizeof(TYPE), \ 474 MMU_DATA_STORE, GETPC()); \ 475 } \ 476 } 477 478 ST_EA(sbea, uint8_t, address_space_stb) 479 ST_EA(shea_be, uint16_t, address_space_stw_be) 480 ST_EA(shea_le, uint16_t, address_space_stw_le) 481 ST_EA(swea_be, uint32_t, address_space_stl_be) 482 ST_EA(swea_le, uint32_t, address_space_stl_le) 483 484 #endif 485