qemu/target/m68k/op_helper.c
<<
>>
Prefs
   1/*
   2 *  M68K helper routines
   3 *
   4 *  Copyright (c) 2007 CodeSourcery
   5 *
   6 * This library is free software; you can redistribute it and/or
   7 * modify it under the terms of the GNU Lesser General Public
   8 * License as published by the Free Software Foundation; either
   9 * version 2 of the License, or (at your option) any later version.
  10 *
  11 * This library is distributed in the hope that it will be useful,
  12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  14 * Lesser General Public License for more details.
  15 *
  16 * You should have received a copy of the GNU Lesser General Public
  17 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
  18 */
  19#include "qemu/osdep.h"
  20#include "cpu.h"
  21#include "exec/helper-proto.h"
  22#include "exec/exec-all.h"
  23#include "exec/cpu_ldst.h"
  24#include "exec/semihost.h"
  25
  26#if defined(CONFIG_USER_ONLY)
  27
  28void m68k_cpu_do_interrupt(CPUState *cs)
  29{
  30    cs->exception_index = -1;
  31}
  32
  33static inline void do_interrupt_m68k_hardirq(CPUM68KState *env)
  34{
  35}
  36
  37#else
  38
  39/* Try to fill the TLB and return an exception if error. If retaddr is
  40   NULL, it means that the function was called in C code (i.e. not
  41   from generated code or from helper.c) */
  42void tlb_fill(CPUState *cs, target_ulong addr, int size,
  43              MMUAccessType access_type, int mmu_idx, uintptr_t retaddr)
  44{
  45    int ret;
  46
  47    ret = m68k_cpu_handle_mmu_fault(cs, addr, size, access_type, mmu_idx);
  48    if (unlikely(ret)) {
  49        /* now we have a real cpu fault */
  50        cpu_loop_exit_restore(cs, retaddr);
  51    }
  52}
  53
  54static void cf_rte(CPUM68KState *env)
  55{
  56    uint32_t sp;
  57    uint32_t fmt;
  58
  59    sp = env->aregs[7];
  60    fmt = cpu_ldl_kernel(env, sp);
  61    env->pc = cpu_ldl_kernel(env, sp + 4);
  62    sp |= (fmt >> 28) & 3;
  63    env->aregs[7] = sp + 8;
  64
  65    cpu_m68k_set_sr(env, fmt);
  66}
  67
  68static void m68k_rte(CPUM68KState *env)
  69{
  70    uint32_t sp;
  71    uint16_t fmt;
  72    uint16_t sr;
  73
  74    sp = env->aregs[7];
  75throwaway:
  76    sr = cpu_lduw_kernel(env, sp);
  77    sp += 2;
  78    env->pc = cpu_ldl_kernel(env, sp);
  79    sp += 4;
  80    if (m68k_feature(env, M68K_FEATURE_QUAD_MULDIV)) {
  81        /*  all except 68000 */
  82        fmt = cpu_lduw_kernel(env, sp);
  83        sp += 2;
  84        switch (fmt >> 12) {
  85        case 0:
  86            break;
  87        case 1:
  88            env->aregs[7] = sp;
  89            cpu_m68k_set_sr(env, sr);
  90            goto throwaway;
  91        case 2:
  92        case 3:
  93            sp += 4;
  94            break;
  95        case 4:
  96            sp += 8;
  97            break;
  98        case 7:
  99            sp += 52;
 100            break;
 101        }
 102    }
 103    env->aregs[7] = sp;
 104    cpu_m68k_set_sr(env, sr);
 105}
 106
 107static const char *m68k_exception_name(int index)
 108{
 109    switch (index) {
 110    case EXCP_ACCESS:
 111        return "Access Fault";
 112    case EXCP_ADDRESS:
 113        return "Address Error";
 114    case EXCP_ILLEGAL:
 115        return "Illegal Instruction";
 116    case EXCP_DIV0:
 117        return "Divide by Zero";
 118    case EXCP_CHK:
 119        return "CHK/CHK2";
 120    case EXCP_TRAPCC:
 121        return "FTRAPcc, TRAPcc, TRAPV";
 122    case EXCP_PRIVILEGE:
 123        return "Privilege Violation";
 124    case EXCP_TRACE:
 125        return "Trace";
 126    case EXCP_LINEA:
 127        return "A-Line";
 128    case EXCP_LINEF:
 129        return "F-Line";
 130    case EXCP_DEBEGBP: /* 68020/030 only */
 131        return "Copro Protocol Violation";
 132    case EXCP_FORMAT:
 133        return "Format Error";
 134    case EXCP_UNINITIALIZED:
 135        return "Unitialized Interruot";
 136    case EXCP_SPURIOUS:
 137        return "Spurious Interrupt";
 138    case EXCP_INT_LEVEL_1:
 139        return "Level 1 Interrupt";
 140    case EXCP_INT_LEVEL_1 + 1:
 141        return "Level 2 Interrupt";
 142    case EXCP_INT_LEVEL_1 + 2:
 143        return "Level 3 Interrupt";
 144    case EXCP_INT_LEVEL_1 + 3:
 145        return "Level 4 Interrupt";
 146    case EXCP_INT_LEVEL_1 + 4:
 147        return "Level 5 Interrupt";
 148    case EXCP_INT_LEVEL_1 + 5:
 149        return "Level 6 Interrupt";
 150    case EXCP_INT_LEVEL_1 + 6:
 151        return "Level 7 Interrupt";
 152    case EXCP_TRAP0:
 153        return "TRAP #0";
 154    case EXCP_TRAP0 + 1:
 155        return "TRAP #1";
 156    case EXCP_TRAP0 + 2:
 157        return "TRAP #2";
 158    case EXCP_TRAP0 + 3:
 159        return "TRAP #3";
 160    case EXCP_TRAP0 + 4:
 161        return "TRAP #4";
 162    case EXCP_TRAP0 + 5:
 163        return "TRAP #5";
 164    case EXCP_TRAP0 + 6:
 165        return "TRAP #6";
 166    case EXCP_TRAP0 + 7:
 167        return "TRAP #7";
 168    case EXCP_TRAP0 + 8:
 169        return "TRAP #8";
 170    case EXCP_TRAP0 + 9:
 171        return "TRAP #9";
 172    case EXCP_TRAP0 + 10:
 173        return "TRAP #10";
 174    case EXCP_TRAP0 + 11:
 175        return "TRAP #11";
 176    case EXCP_TRAP0 + 12:
 177        return "TRAP #12";
 178    case EXCP_TRAP0 + 13:
 179        return "TRAP #13";
 180    case EXCP_TRAP0 + 14:
 181        return "TRAP #14";
 182    case EXCP_TRAP0 + 15:
 183        return "TRAP #15";
 184    case EXCP_FP_BSUN:
 185        return "FP Branch/Set on unordered condition";
 186    case EXCP_FP_INEX:
 187        return "FP Inexact Result";
 188    case EXCP_FP_DZ:
 189        return "FP Divide by Zero";
 190    case EXCP_FP_UNFL:
 191        return "FP Underflow";
 192    case EXCP_FP_OPERR:
 193        return "FP Operand Error";
 194    case EXCP_FP_OVFL:
 195        return "FP Overflow";
 196    case EXCP_FP_SNAN:
 197        return "FP Signaling NAN";
 198    case EXCP_FP_UNIMP:
 199        return "FP Unimplemented Data Type";
 200    case EXCP_MMU_CONF: /* 68030/68851 only */
 201        return "MMU Configuration Error";
 202    case EXCP_MMU_ILLEGAL: /* 68851 only */
 203        return "MMU Illegal Operation";
 204    case EXCP_MMU_ACCESS: /* 68851 only */
 205        return "MMU Access Level Violation";
 206    case 64 ... 255:
 207        return "User Defined Vector";
 208    }
 209    return "Unassigned";
 210}
 211
 212static void cf_interrupt_all(CPUM68KState *env, int is_hw)
 213{
 214    CPUState *cs = CPU(m68k_env_get_cpu(env));
 215    uint32_t sp;
 216    uint32_t sr;
 217    uint32_t fmt;
 218    uint32_t retaddr;
 219    uint32_t vector;
 220
 221    fmt = 0;
 222    retaddr = env->pc;
 223
 224    if (!is_hw) {
 225        switch (cs->exception_index) {
 226        case EXCP_RTE:
 227            /* Return from an exception.  */
 228            cf_rte(env);
 229            return;
 230        case EXCP_HALT_INSN:
 231            if (semihosting_enabled()
 232                    && (env->sr & SR_S) != 0
 233                    && (env->pc & 3) == 0
 234                    && cpu_lduw_code(env, env->pc - 4) == 0x4e71
 235                    && cpu_ldl_code(env, env->pc) == 0x4e7bf000) {
 236                env->pc += 4;
 237                do_m68k_semihosting(env, env->dregs[0]);
 238                return;
 239            }
 240            cs->halted = 1;
 241            cs->exception_index = EXCP_HLT;
 242            cpu_loop_exit(cs);
 243            return;
 244        }
 245        if (cs->exception_index >= EXCP_TRAP0
 246            && cs->exception_index <= EXCP_TRAP15) {
 247            /* Move the PC after the trap instruction.  */
 248            retaddr += 2;
 249        }
 250    }
 251
 252    vector = cs->exception_index << 2;
 253
 254    sr = env->sr | cpu_m68k_get_ccr(env);
 255    if (qemu_loglevel_mask(CPU_LOG_INT)) {
 256        static int count;
 257        qemu_log("INT %6d: %s(%#x) pc=%08x sp=%08x sr=%04x\n",
 258                 ++count, m68k_exception_name(cs->exception_index),
 259                 vector, env->pc, env->aregs[7], sr);
 260    }
 261
 262    fmt |= 0x40000000;
 263    fmt |= vector << 16;
 264    fmt |= sr;
 265
 266    env->sr |= SR_S;
 267    if (is_hw) {
 268        env->sr = (env->sr & ~SR_I) | (env->pending_level << SR_I_SHIFT);
 269        env->sr &= ~SR_M;
 270    }
 271    m68k_switch_sp(env);
 272    sp = env->aregs[7];
 273    fmt |= (sp & 3) << 28;
 274
 275    /* ??? This could cause MMU faults.  */
 276    sp &= ~3;
 277    sp -= 4;
 278    cpu_stl_kernel(env, sp, retaddr);
 279    sp -= 4;
 280    cpu_stl_kernel(env, sp, fmt);
 281    env->aregs[7] = sp;
 282    /* Jump to vector.  */
 283    env->pc = cpu_ldl_kernel(env, env->vbr + vector);
 284}
 285
 286static inline void do_stack_frame(CPUM68KState *env, uint32_t *sp,
 287                                  uint16_t format, uint16_t sr,
 288                                  uint32_t addr, uint32_t retaddr)
 289{
 290    if (m68k_feature(env, M68K_FEATURE_QUAD_MULDIV)) {
 291        /*  all except 68000 */
 292        CPUState *cs = CPU(m68k_env_get_cpu(env));
 293        switch (format) {
 294        case 4:
 295            *sp -= 4;
 296            cpu_stl_kernel(env, *sp, env->pc);
 297            *sp -= 4;
 298            cpu_stl_kernel(env, *sp, addr);
 299            break;
 300        case 3:
 301        case 2:
 302            *sp -= 4;
 303            cpu_stl_kernel(env, *sp, addr);
 304            break;
 305        }
 306        *sp -= 2;
 307        cpu_stw_kernel(env, *sp, (format << 12) + (cs->exception_index << 2));
 308    }
 309    *sp -= 4;
 310    cpu_stl_kernel(env, *sp, retaddr);
 311    *sp -= 2;
 312    cpu_stw_kernel(env, *sp, sr);
 313}
 314
 315static void m68k_interrupt_all(CPUM68KState *env, int is_hw)
 316{
 317    CPUState *cs = CPU(m68k_env_get_cpu(env));
 318    uint32_t sp;
 319    uint32_t retaddr;
 320    uint32_t vector;
 321    uint16_t sr, oldsr;
 322
 323    retaddr = env->pc;
 324
 325    if (!is_hw) {
 326        switch (cs->exception_index) {
 327        case EXCP_RTE:
 328            /* Return from an exception.  */
 329            m68k_rte(env);
 330            return;
 331        case EXCP_TRAP0 ...  EXCP_TRAP15:
 332            /* Move the PC after the trap instruction.  */
 333            retaddr += 2;
 334            break;
 335        }
 336    }
 337
 338    vector = cs->exception_index << 2;
 339
 340    sr = env->sr | cpu_m68k_get_ccr(env);
 341    if (qemu_loglevel_mask(CPU_LOG_INT)) {
 342        static int count;
 343        qemu_log("INT %6d: %s(%#x) pc=%08x sp=%08x sr=%04x\n",
 344                 ++count, m68k_exception_name(cs->exception_index),
 345                 vector, env->pc, env->aregs[7], sr);
 346    }
 347
 348    /*
 349     * MC68040UM/AD,  chapter 9.3.10
 350     */
 351
 352    /* "the processor first make an internal copy" */
 353    oldsr = sr;
 354    /* "set the mode to supervisor" */
 355    sr |= SR_S;
 356    /* "suppress tracing" */
 357    sr &= ~SR_T;
 358    /* "sets the processor interrupt mask" */
 359    if (is_hw) {
 360        sr |= (env->sr & ~SR_I) | (env->pending_level << SR_I_SHIFT);
 361    }
 362    cpu_m68k_set_sr(env, sr);
 363    sp = env->aregs[7];
 364
 365    sp &= ~1;
 366    if (cs->exception_index == EXCP_ACCESS) {
 367        if (env->mmu.fault) {
 368            cpu_abort(cs, "DOUBLE MMU FAULT\n");
 369        }
 370        env->mmu.fault = true;
 371        sp -= 4;
 372        cpu_stl_kernel(env, sp, 0); /* push data 3 */
 373        sp -= 4;
 374        cpu_stl_kernel(env, sp, 0); /* push data 2 */
 375        sp -= 4;
 376        cpu_stl_kernel(env, sp, 0); /* push data 1 */
 377        sp -= 4;
 378        cpu_stl_kernel(env, sp, 0); /* write back 1 / push data 0 */
 379        sp -= 4;
 380        cpu_stl_kernel(env, sp, 0); /* write back 1 address */
 381        sp -= 4;
 382        cpu_stl_kernel(env, sp, 0); /* write back 2 data */
 383        sp -= 4;
 384        cpu_stl_kernel(env, sp, 0); /* write back 2 address */
 385        sp -= 4;
 386        cpu_stl_kernel(env, sp, 0); /* write back 3 data */
 387        sp -= 4;
 388        cpu_stl_kernel(env, sp, env->mmu.ar); /* write back 3 address */
 389        sp -= 4;
 390        cpu_stl_kernel(env, sp, env->mmu.ar); /* fault address */
 391        sp -= 2;
 392        cpu_stw_kernel(env, sp, 0); /* write back 1 status */
 393        sp -= 2;
 394        cpu_stw_kernel(env, sp, 0); /* write back 2 status */
 395        sp -= 2;
 396        cpu_stw_kernel(env, sp, 0); /* write back 3 status */
 397        sp -= 2;
 398        cpu_stw_kernel(env, sp, env->mmu.ssw); /* special status word */
 399        sp -= 4;
 400        cpu_stl_kernel(env, sp, env->mmu.ar); /* effective address */
 401        do_stack_frame(env, &sp, 7, oldsr, 0, retaddr);
 402        env->mmu.fault = false;
 403        if (qemu_loglevel_mask(CPU_LOG_INT)) {
 404            qemu_log("            "
 405                     "ssw:  %08x ea:   %08x sfc:  %d    dfc: %d\n",
 406                     env->mmu.ssw, env->mmu.ar, env->sfc, env->dfc);
 407        }
 408    } else if (cs->exception_index == EXCP_ADDRESS) {
 409        do_stack_frame(env, &sp, 2, oldsr, 0, retaddr);
 410    } else if (cs->exception_index == EXCP_ILLEGAL ||
 411               cs->exception_index == EXCP_DIV0 ||
 412               cs->exception_index == EXCP_CHK ||
 413               cs->exception_index == EXCP_TRAPCC ||
 414               cs->exception_index == EXCP_TRACE) {
 415        /* FIXME: addr is not only env->pc */
 416        do_stack_frame(env, &sp, 2, oldsr, env->pc, retaddr);
 417    } else if (is_hw && oldsr & SR_M &&
 418               cs->exception_index >= EXCP_SPURIOUS &&
 419               cs->exception_index <= EXCP_INT_LEVEL_7) {
 420        do_stack_frame(env, &sp, 0, oldsr, 0, retaddr);
 421        oldsr = sr;
 422        env->aregs[7] = sp;
 423        cpu_m68k_set_sr(env, sr &= ~SR_M);
 424        sp = env->aregs[7] & ~1;
 425        do_stack_frame(env, &sp, 1, oldsr, 0, retaddr);
 426    } else {
 427        do_stack_frame(env, &sp, 0, oldsr, 0, retaddr);
 428    }
 429
 430    env->aregs[7] = sp;
 431    /* Jump to vector.  */
 432    env->pc = cpu_ldl_kernel(env, env->vbr + vector);
 433}
 434
 435static void do_interrupt_all(CPUM68KState *env, int is_hw)
 436{
 437    if (m68k_feature(env, M68K_FEATURE_M68000)) {
 438        m68k_interrupt_all(env, is_hw);
 439        return;
 440    }
 441    cf_interrupt_all(env, is_hw);
 442}
 443
 444void m68k_cpu_do_interrupt(CPUState *cs)
 445{
 446    M68kCPU *cpu = M68K_CPU(cs);
 447    CPUM68KState *env = &cpu->env;
 448
 449    do_interrupt_all(env, 0);
 450}
 451
 452static inline void do_interrupt_m68k_hardirq(CPUM68KState *env)
 453{
 454    do_interrupt_all(env, 1);
 455}
 456
 457void m68k_cpu_unassigned_access(CPUState *cs, hwaddr addr, bool is_write,
 458                                bool is_exec, int is_asi, unsigned size)
 459{
 460    M68kCPU *cpu = M68K_CPU(cs);
 461    CPUM68KState *env = &cpu->env;
 462#ifdef DEBUG_UNASSIGNED
 463    qemu_log_mask(CPU_LOG_INT, "Unassigned " TARGET_FMT_plx " wr=%d exe=%d\n",
 464             addr, is_write, is_exec);
 465#endif
 466    if (env == NULL) {
 467        /* when called from gdb, env is NULL */
 468        return;
 469    }
 470
 471    if (m68k_feature(env, M68K_FEATURE_M68040)) {
 472        env->mmu.mmusr = 0;
 473        env->mmu.ssw |= M68K_ATC_040;
 474        /* FIXME: manage MMU table access error */
 475        env->mmu.ssw &= ~M68K_TM_040;
 476        if (env->sr & SR_S) { /* SUPERVISOR */
 477            env->mmu.ssw |= M68K_TM_040_SUPER;
 478        }
 479        if (is_exec) { /* instruction or data */
 480            env->mmu.ssw |= M68K_TM_040_CODE;
 481        } else {
 482            env->mmu.ssw |= M68K_TM_040_DATA;
 483        }
 484        env->mmu.ssw &= ~M68K_BA_SIZE_MASK;
 485        switch (size) {
 486        case 1:
 487            env->mmu.ssw |= M68K_BA_SIZE_BYTE;
 488            break;
 489        case 2:
 490            env->mmu.ssw |= M68K_BA_SIZE_WORD;
 491            break;
 492        case 4:
 493            env->mmu.ssw |= M68K_BA_SIZE_LONG;
 494            break;
 495        }
 496
 497        if (!is_write) {
 498            env->mmu.ssw |= M68K_RW_040;
 499        }
 500
 501        env->mmu.ar = addr;
 502
 503        cs->exception_index = EXCP_ACCESS;
 504        cpu_loop_exit(cs);
 505    }
 506}
 507#endif
 508
 509bool m68k_cpu_exec_interrupt(CPUState *cs, int interrupt_request)
 510{
 511    M68kCPU *cpu = M68K_CPU(cs);
 512    CPUM68KState *env = &cpu->env;
 513
 514    if (interrupt_request & CPU_INTERRUPT_HARD
 515        && ((env->sr & SR_I) >> SR_I_SHIFT) < env->pending_level) {
 516        /* Real hardware gets the interrupt vector via an IACK cycle
 517           at this point.  Current emulated hardware doesn't rely on
 518           this, so we provide/save the vector when the interrupt is
 519           first signalled.  */
 520        cs->exception_index = env->pending_vector;
 521        do_interrupt_m68k_hardirq(env);
 522        return true;
 523    }
 524    return false;
 525}
 526
 527static void raise_exception_ra(CPUM68KState *env, int tt, uintptr_t raddr)
 528{
 529    CPUState *cs = CPU(m68k_env_get_cpu(env));
 530
 531    cs->exception_index = tt;
 532    cpu_loop_exit_restore(cs, raddr);
 533}
 534
 535static void raise_exception(CPUM68KState *env, int tt)
 536{
 537    raise_exception_ra(env, tt, 0);
 538}
 539
 540void HELPER(raise_exception)(CPUM68KState *env, uint32_t tt)
 541{
 542    raise_exception(env, tt);
 543}
 544
 545void HELPER(divuw)(CPUM68KState *env, int destr, uint32_t den)
 546{
 547    uint32_t num = env->dregs[destr];
 548    uint32_t quot, rem;
 549
 550    if (den == 0) {
 551        raise_exception_ra(env, EXCP_DIV0, GETPC());
 552    }
 553    quot = num / den;
 554    rem = num % den;
 555
 556    env->cc_c = 0; /* always cleared, even if overflow */
 557    if (quot > 0xffff) {
 558        env->cc_v = -1;
 559        /* real 68040 keeps N and unset Z on overflow,
 560         * whereas documentation says "undefined"
 561         */
 562        env->cc_z = 1;
 563        return;
 564    }
 565    env->dregs[destr] = deposit32(quot, 16, 16, rem);
 566    env->cc_z = (int16_t)quot;
 567    env->cc_n = (int16_t)quot;
 568    env->cc_v = 0;
 569}
 570
 571void HELPER(divsw)(CPUM68KState *env, int destr, int32_t den)
 572{
 573    int32_t num = env->dregs[destr];
 574    uint32_t quot, rem;
 575
 576    if (den == 0) {
 577        raise_exception_ra(env, EXCP_DIV0, GETPC());
 578    }
 579    quot = num / den;
 580    rem = num % den;
 581
 582    env->cc_c = 0; /* always cleared, even if overflow */
 583    if (quot != (int16_t)quot) {
 584        env->cc_v = -1;
 585        /* nothing else is modified */
 586        /* real 68040 keeps N and unset Z on overflow,
 587         * whereas documentation says "undefined"
 588         */
 589        env->cc_z = 1;
 590        return;
 591    }
 592    env->dregs[destr] = deposit32(quot, 16, 16, rem);
 593    env->cc_z = (int16_t)quot;
 594    env->cc_n = (int16_t)quot;
 595    env->cc_v = 0;
 596}
 597
 598void HELPER(divul)(CPUM68KState *env, int numr, int regr, uint32_t den)
 599{
 600    uint32_t num = env->dregs[numr];
 601    uint32_t quot, rem;
 602
 603    if (den == 0) {
 604        raise_exception_ra(env, EXCP_DIV0, GETPC());
 605    }
 606    quot = num / den;
 607    rem = num % den;
 608
 609    env->cc_c = 0;
 610    env->cc_z = quot;
 611    env->cc_n = quot;
 612    env->cc_v = 0;
 613
 614    if (m68k_feature(env, M68K_FEATURE_CF_ISA_A)) {
 615        if (numr == regr) {
 616            env->dregs[numr] = quot;
 617        } else {
 618            env->dregs[regr] = rem;
 619        }
 620    } else {
 621        env->dregs[regr] = rem;
 622        env->dregs[numr] = quot;
 623    }
 624}
 625
 626void HELPER(divsl)(CPUM68KState *env, int numr, int regr, int32_t den)
 627{
 628    int32_t num = env->dregs[numr];
 629    int32_t quot, rem;
 630
 631    if (den == 0) {
 632        raise_exception_ra(env, EXCP_DIV0, GETPC());
 633    }
 634    quot = num / den;
 635    rem = num % den;
 636
 637    env->cc_c = 0;
 638    env->cc_z = quot;
 639    env->cc_n = quot;
 640    env->cc_v = 0;
 641
 642    if (m68k_feature(env, M68K_FEATURE_CF_ISA_A)) {
 643        if (numr == regr) {
 644            env->dregs[numr] = quot;
 645        } else {
 646            env->dregs[regr] = rem;
 647        }
 648    } else {
 649        env->dregs[regr] = rem;
 650        env->dregs[numr] = quot;
 651    }
 652}
 653
 654void HELPER(divull)(CPUM68KState *env, int numr, int regr, uint32_t den)
 655{
 656    uint64_t num = deposit64(env->dregs[numr], 32, 32, env->dregs[regr]);
 657    uint64_t quot;
 658    uint32_t rem;
 659
 660    if (den == 0) {
 661        raise_exception_ra(env, EXCP_DIV0, GETPC());
 662    }
 663    quot = num / den;
 664    rem = num % den;
 665
 666    env->cc_c = 0; /* always cleared, even if overflow */
 667    if (quot > 0xffffffffULL) {
 668        env->cc_v = -1;
 669        /* real 68040 keeps N and unset Z on overflow,
 670         * whereas documentation says "undefined"
 671         */
 672        env->cc_z = 1;
 673        return;
 674    }
 675    env->cc_z = quot;
 676    env->cc_n = quot;
 677    env->cc_v = 0;
 678
 679    /*
 680     * If Dq and Dr are the same, the quotient is returned.
 681     * therefore we set Dq last.
 682     */
 683
 684    env->dregs[regr] = rem;
 685    env->dregs[numr] = quot;
 686}
 687
 688void HELPER(divsll)(CPUM68KState *env, int numr, int regr, int32_t den)
 689{
 690    int64_t num = deposit64(env->dregs[numr], 32, 32, env->dregs[regr]);
 691    int64_t quot;
 692    int32_t rem;
 693
 694    if (den == 0) {
 695        raise_exception_ra(env, EXCP_DIV0, GETPC());
 696    }
 697    quot = num / den;
 698    rem = num % den;
 699
 700    env->cc_c = 0; /* always cleared, even if overflow */
 701    if (quot != (int32_t)quot) {
 702        env->cc_v = -1;
 703        /* real 68040 keeps N and unset Z on overflow,
 704         * whereas documentation says "undefined"
 705         */
 706        env->cc_z = 1;
 707        return;
 708    }
 709    env->cc_z = quot;
 710    env->cc_n = quot;
 711    env->cc_v = 0;
 712
 713    /*
 714     * If Dq and Dr are the same, the quotient is returned.
 715     * therefore we set Dq last.
 716     */
 717
 718    env->dregs[regr] = rem;
 719    env->dregs[numr] = quot;
 720}
 721
 722/* We're executing in a serial context -- no need to be atomic.  */
 723void HELPER(cas2w)(CPUM68KState *env, uint32_t regs, uint32_t a1, uint32_t a2)
 724{
 725    uint32_t Dc1 = extract32(regs, 9, 3);
 726    uint32_t Dc2 = extract32(regs, 6, 3);
 727    uint32_t Du1 = extract32(regs, 3, 3);
 728    uint32_t Du2 = extract32(regs, 0, 3);
 729    int16_t c1 = env->dregs[Dc1];
 730    int16_t c2 = env->dregs[Dc2];
 731    int16_t u1 = env->dregs[Du1];
 732    int16_t u2 = env->dregs[Du2];
 733    int16_t l1, l2;
 734    uintptr_t ra = GETPC();
 735
 736    l1 = cpu_lduw_data_ra(env, a1, ra);
 737    l2 = cpu_lduw_data_ra(env, a2, ra);
 738    if (l1 == c1 && l2 == c2) {
 739        cpu_stw_data_ra(env, a1, u1, ra);
 740        cpu_stw_data_ra(env, a2, u2, ra);
 741    }
 742
 743    if (c1 != l1) {
 744        env->cc_n = l1;
 745        env->cc_v = c1;
 746    } else {
 747        env->cc_n = l2;
 748        env->cc_v = c2;
 749    }
 750    env->cc_op = CC_OP_CMPW;
 751    env->dregs[Dc1] = deposit32(env->dregs[Dc1], 0, 16, l1);
 752    env->dregs[Dc2] = deposit32(env->dregs[Dc2], 0, 16, l2);
 753}
 754
 755static void do_cas2l(CPUM68KState *env, uint32_t regs, uint32_t a1, uint32_t a2,
 756                     bool parallel)
 757{
 758    uint32_t Dc1 = extract32(regs, 9, 3);
 759    uint32_t Dc2 = extract32(regs, 6, 3);
 760    uint32_t Du1 = extract32(regs, 3, 3);
 761    uint32_t Du2 = extract32(regs, 0, 3);
 762    uint32_t c1 = env->dregs[Dc1];
 763    uint32_t c2 = env->dregs[Dc2];
 764    uint32_t u1 = env->dregs[Du1];
 765    uint32_t u2 = env->dregs[Du2];
 766    uint32_t l1, l2;
 767    uintptr_t ra = GETPC();
 768#if defined(CONFIG_ATOMIC64) && !defined(CONFIG_USER_ONLY)
 769    int mmu_idx = cpu_mmu_index(env, 0);
 770    TCGMemOpIdx oi;
 771#endif
 772
 773    if (parallel) {
 774        /* We're executing in a parallel context -- must be atomic.  */
 775#ifdef CONFIG_ATOMIC64
 776        uint64_t c, u, l;
 777        if ((a1 & 7) == 0 && a2 == a1 + 4) {
 778            c = deposit64(c2, 32, 32, c1);
 779            u = deposit64(u2, 32, 32, u1);
 780#ifdef CONFIG_USER_ONLY
 781            l = helper_atomic_cmpxchgq_be(env, a1, c, u);
 782#else
 783            oi = make_memop_idx(MO_BEQ, mmu_idx);
 784            l = helper_atomic_cmpxchgq_be_mmu(env, a1, c, u, oi, ra);
 785#endif
 786            l1 = l >> 32;
 787            l2 = l;
 788        } else if ((a2 & 7) == 0 && a1 == a2 + 4) {
 789            c = deposit64(c1, 32, 32, c2);
 790            u = deposit64(u1, 32, 32, u2);
 791#ifdef CONFIG_USER_ONLY
 792            l = helper_atomic_cmpxchgq_be(env, a2, c, u);
 793#else
 794            oi = make_memop_idx(MO_BEQ, mmu_idx);
 795            l = helper_atomic_cmpxchgq_be_mmu(env, a2, c, u, oi, ra);
 796#endif
 797            l2 = l >> 32;
 798            l1 = l;
 799        } else
 800#endif
 801        {
 802            /* Tell the main loop we need to serialize this insn.  */
 803            cpu_loop_exit_atomic(ENV_GET_CPU(env), ra);
 804        }
 805    } else {
 806        /* We're executing in a serial context -- no need to be atomic.  */
 807        l1 = cpu_ldl_data_ra(env, a1, ra);
 808        l2 = cpu_ldl_data_ra(env, a2, ra);
 809        if (l1 == c1 && l2 == c2) {
 810            cpu_stl_data_ra(env, a1, u1, ra);
 811            cpu_stl_data_ra(env, a2, u2, ra);
 812        }
 813    }
 814
 815    if (c1 != l1) {
 816        env->cc_n = l1;
 817        env->cc_v = c1;
 818    } else {
 819        env->cc_n = l2;
 820        env->cc_v = c2;
 821    }
 822    env->cc_op = CC_OP_CMPL;
 823    env->dregs[Dc1] = l1;
 824    env->dregs[Dc2] = l2;
 825}
 826
 827void HELPER(cas2l)(CPUM68KState *env, uint32_t regs, uint32_t a1, uint32_t a2)
 828{
 829    do_cas2l(env, regs, a1, a2, false);
 830}
 831
 832void HELPER(cas2l_parallel)(CPUM68KState *env, uint32_t regs, uint32_t a1,
 833                            uint32_t a2)
 834{
 835    do_cas2l(env, regs, a1, a2, true);
 836}
 837
 838struct bf_data {
 839    uint32_t addr;
 840    uint32_t bofs;
 841    uint32_t blen;
 842    uint32_t len;
 843};
 844
 845static struct bf_data bf_prep(uint32_t addr, int32_t ofs, uint32_t len)
 846{
 847    int bofs, blen;
 848
 849    /* Bound length; map 0 to 32.  */
 850    len = ((len - 1) & 31) + 1;
 851
 852    /* Note that ofs is signed.  */
 853    addr += ofs / 8;
 854    bofs = ofs % 8;
 855    if (bofs < 0) {
 856        bofs += 8;
 857        addr -= 1;
 858    }
 859
 860    /* Compute the number of bytes required (minus one) to
 861       satisfy the bitfield.  */
 862    blen = (bofs + len - 1) / 8;
 863
 864    /* Canonicalize the bit offset for data loaded into a 64-bit big-endian
 865       word.  For the cases where BLEN is not a power of 2, adjust ADDR so
 866       that we can use the next power of two sized load without crossing a
 867       page boundary, unless the field itself crosses the boundary.  */
 868    switch (blen) {
 869    case 0:
 870        bofs += 56;
 871        break;
 872    case 1:
 873        bofs += 48;
 874        break;
 875    case 2:
 876        if (addr & 1) {
 877            bofs += 8;
 878            addr -= 1;
 879        }
 880        /* fallthru */
 881    case 3:
 882        bofs += 32;
 883        break;
 884    case 4:
 885        if (addr & 3) {
 886            bofs += 8 * (addr & 3);
 887            addr &= -4;
 888        }
 889        break;
 890    default:
 891        g_assert_not_reached();
 892    }
 893
 894    return (struct bf_data){
 895        .addr = addr,
 896        .bofs = bofs,
 897        .blen = blen,
 898        .len = len,
 899    };
 900}
 901
 902static uint64_t bf_load(CPUM68KState *env, uint32_t addr, int blen,
 903                        uintptr_t ra)
 904{
 905    switch (blen) {
 906    case 0:
 907        return cpu_ldub_data_ra(env, addr, ra);
 908    case 1:
 909        return cpu_lduw_data_ra(env, addr, ra);
 910    case 2:
 911    case 3:
 912        return cpu_ldl_data_ra(env, addr, ra);
 913    case 4:
 914        return cpu_ldq_data_ra(env, addr, ra);
 915    default:
 916        g_assert_not_reached();
 917    }
 918}
 919
 920static void bf_store(CPUM68KState *env, uint32_t addr, int blen,
 921                     uint64_t data, uintptr_t ra)
 922{
 923    switch (blen) {
 924    case 0:
 925        cpu_stb_data_ra(env, addr, data, ra);
 926        break;
 927    case 1:
 928        cpu_stw_data_ra(env, addr, data, ra);
 929        break;
 930    case 2:
 931    case 3:
 932        cpu_stl_data_ra(env, addr, data, ra);
 933        break;
 934    case 4:
 935        cpu_stq_data_ra(env, addr, data, ra);
 936        break;
 937    default:
 938        g_assert_not_reached();
 939    }
 940}
 941
 942uint32_t HELPER(bfexts_mem)(CPUM68KState *env, uint32_t addr,
 943                            int32_t ofs, uint32_t len)
 944{
 945    uintptr_t ra = GETPC();
 946    struct bf_data d = bf_prep(addr, ofs, len);
 947    uint64_t data = bf_load(env, d.addr, d.blen, ra);
 948
 949    return (int64_t)(data << d.bofs) >> (64 - d.len);
 950}
 951
 952uint64_t HELPER(bfextu_mem)(CPUM68KState *env, uint32_t addr,
 953                            int32_t ofs, uint32_t len)
 954{
 955    uintptr_t ra = GETPC();
 956    struct bf_data d = bf_prep(addr, ofs, len);
 957    uint64_t data = bf_load(env, d.addr, d.blen, ra);
 958
 959    /* Put CC_N at the top of the high word; put the zero-extended value
 960       at the bottom of the low word.  */
 961    data <<= d.bofs;
 962    data >>= 64 - d.len;
 963    data |= data << (64 - d.len);
 964
 965    return data;
 966}
 967
 968uint32_t HELPER(bfins_mem)(CPUM68KState *env, uint32_t addr, uint32_t val,
 969                           int32_t ofs, uint32_t len)
 970{
 971    uintptr_t ra = GETPC();
 972    struct bf_data d = bf_prep(addr, ofs, len);
 973    uint64_t data = bf_load(env, d.addr, d.blen, ra);
 974    uint64_t mask = -1ull << (64 - d.len) >> d.bofs;
 975
 976    data = (data & ~mask) | (((uint64_t)val << (64 - d.len)) >> d.bofs);
 977
 978    bf_store(env, d.addr, d.blen, data, ra);
 979
 980    /* The field at the top of the word is also CC_N for CC_OP_LOGIC.  */
 981    return val << (32 - d.len);
 982}
 983
 984uint32_t HELPER(bfchg_mem)(CPUM68KState *env, uint32_t addr,
 985                           int32_t ofs, uint32_t len)
 986{
 987    uintptr_t ra = GETPC();
 988    struct bf_data d = bf_prep(addr, ofs, len);
 989    uint64_t data = bf_load(env, d.addr, d.blen, ra);
 990    uint64_t mask = -1ull << (64 - d.len) >> d.bofs;
 991
 992    bf_store(env, d.addr, d.blen, data ^ mask, ra);
 993
 994    return ((data & mask) << d.bofs) >> 32;
 995}
 996
 997uint32_t HELPER(bfclr_mem)(CPUM68KState *env, uint32_t addr,
 998                           int32_t ofs, uint32_t len)
 999{
1000    uintptr_t ra = GETPC();
1001    struct bf_data d = bf_prep(addr, ofs, len);
1002    uint64_t data = bf_load(env, d.addr, d.blen, ra);
1003    uint64_t mask = -1ull << (64 - d.len) >> d.bofs;
1004
1005    bf_store(env, d.addr, d.blen, data & ~mask, ra);
1006
1007    return ((data & mask) << d.bofs) >> 32;
1008}
1009
1010uint32_t HELPER(bfset_mem)(CPUM68KState *env, uint32_t addr,
1011                           int32_t ofs, uint32_t len)
1012{
1013    uintptr_t ra = GETPC();
1014    struct bf_data d = bf_prep(addr, ofs, len);
1015    uint64_t data = bf_load(env, d.addr, d.blen, ra);
1016    uint64_t mask = -1ull << (64 - d.len) >> d.bofs;
1017
1018    bf_store(env, d.addr, d.blen, data | mask, ra);
1019
1020    return ((data & mask) << d.bofs) >> 32;
1021}
1022
1023uint32_t HELPER(bfffo_reg)(uint32_t n, uint32_t ofs, uint32_t len)
1024{
1025    return (n ? clz32(n) : len) + ofs;
1026}
1027
1028uint64_t HELPER(bfffo_mem)(CPUM68KState *env, uint32_t addr,
1029                           int32_t ofs, uint32_t len)
1030{
1031    uintptr_t ra = GETPC();
1032    struct bf_data d = bf_prep(addr, ofs, len);
1033    uint64_t data = bf_load(env, d.addr, d.blen, ra);
1034    uint64_t mask = -1ull << (64 - d.len) >> d.bofs;
1035    uint64_t n = (data & mask) << d.bofs;
1036    uint32_t ffo = helper_bfffo_reg(n >> 32, ofs, d.len);
1037
1038    /* Return FFO in the low word and N in the high word.
1039       Note that because of MASK and the shift, the low word
1040       is already zero.  */
1041    return n | ffo;
1042}
1043
1044void HELPER(chk)(CPUM68KState *env, int32_t val, int32_t ub)
1045{
1046    /* From the specs:
1047     *   X: Not affected, C,V,Z: Undefined,
1048     *   N: Set if val < 0; cleared if val > ub, undefined otherwise
1049     * We implement here values found from a real MC68040:
1050     *   X,V,Z: Not affected
1051     *   N: Set if val < 0; cleared if val >= 0
1052     *   C: if 0 <= ub: set if val < 0 or val > ub, cleared otherwise
1053     *      if 0 > ub: set if val > ub and val < 0, cleared otherwise
1054     */
1055    env->cc_n = val;
1056    env->cc_c = 0 <= ub ? val < 0 || val > ub : val > ub && val < 0;
1057
1058    if (val < 0 || val > ub) {
1059        CPUState *cs = CPU(m68k_env_get_cpu(env));
1060
1061        /* Recover PC and CC_OP for the beginning of the insn.  */
1062        cpu_restore_state(cs, GETPC(), true);
1063
1064        /* flags have been modified by gen_flush_flags() */
1065        env->cc_op = CC_OP_FLAGS;
1066        /* Adjust PC to end of the insn.  */
1067        env->pc += 2;
1068
1069        cs->exception_index = EXCP_CHK;
1070        cpu_loop_exit(cs);
1071    }
1072}
1073
1074void HELPER(chk2)(CPUM68KState *env, int32_t val, int32_t lb, int32_t ub)
1075{
1076    /* From the specs:
1077     *   X: Not affected, N,V: Undefined,
1078     *   Z: Set if val is equal to lb or ub
1079     *   C: Set if val < lb or val > ub, cleared otherwise
1080     * We implement here values found from a real MC68040:
1081     *   X,N,V: Not affected
1082     *   Z: Set if val is equal to lb or ub
1083     *   C: if lb <= ub: set if val < lb or val > ub, cleared otherwise
1084     *      if lb > ub: set if val > ub and val < lb, cleared otherwise
1085     */
1086    env->cc_z = val != lb && val != ub;
1087    env->cc_c = lb <= ub ? val < lb || val > ub : val > ub && val < lb;
1088
1089    if (env->cc_c) {
1090        CPUState *cs = CPU(m68k_env_get_cpu(env));
1091
1092        /* Recover PC and CC_OP for the beginning of the insn.  */
1093        cpu_restore_state(cs, GETPC(), true);
1094
1095        /* flags have been modified by gen_flush_flags() */
1096        env->cc_op = CC_OP_FLAGS;
1097        /* Adjust PC to end of the insn.  */
1098        env->pc += 4;
1099
1100        cs->exception_index = EXCP_CHK;
1101        cpu_loop_exit(cs);
1102    }
1103}
1104