2006-10-22 04:18:54 +04:00
|
|
|
/*
|
|
|
|
* m68k op helpers
|
2007-09-17 01:08:06 +04:00
|
|
|
*
|
2007-05-23 23:58:11 +04:00
|
|
|
* Copyright (c) 2006-2007 CodeSourcery
|
2006-10-22 04:18:54 +04:00
|
|
|
* Written by Paul Brook
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2009-07-17 00:47:01 +04:00
|
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
2006-10-22 04:18:54 +04:00
|
|
|
*/
|
|
|
|
|
2016-01-26 21:17:23 +03:00
|
|
|
#include "qemu/osdep.h"
|
2006-10-22 04:18:54 +04:00
|
|
|
#include "cpu.h"
|
2016-03-15 15:18:37 +03:00
|
|
|
#include "exec/exec-all.h"
|
2012-12-17 21:19:49 +04:00
|
|
|
#include "exec/gdbstub.h"
|
2006-10-22 04:18:54 +04:00
|
|
|
|
2014-04-08 09:31:41 +04:00
|
|
|
#include "exec/helper-proto.h"
|
2008-05-25 02:29:16 +04:00
|
|
|
|
|
|
|
#define SIGNBIT (1u << 31)
|
|
|
|
|
2012-04-15 05:30:10 +04:00
|
|
|
/* Sort alphabetically, except for "any". */
|
|
|
|
static gint m68k_cpu_list_compare(gconstpointer a, gconstpointer b)
|
2009-05-10 00:21:39 +04:00
|
|
|
{
|
2012-04-15 05:30:10 +04:00
|
|
|
ObjectClass *class_a = (ObjectClass *)a;
|
|
|
|
ObjectClass *class_b = (ObjectClass *)b;
|
|
|
|
const char *name_a, *name_b;
|
|
|
|
|
|
|
|
name_a = object_class_get_name(class_a);
|
|
|
|
name_b = object_class_get_name(class_b);
|
2013-01-27 23:16:17 +04:00
|
|
|
if (strcmp(name_a, "any-" TYPE_M68K_CPU) == 0) {
|
2012-04-15 05:30:10 +04:00
|
|
|
return 1;
|
2013-01-27 23:16:17 +04:00
|
|
|
} else if (strcmp(name_b, "any-" TYPE_M68K_CPU) == 0) {
|
2012-04-15 05:30:10 +04:00
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return strcasecmp(name_a, name_b);
|
2009-05-10 00:21:39 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-15 05:30:10 +04:00
|
|
|
static void m68k_cpu_list_entry(gpointer data, gpointer user_data)
|
|
|
|
{
|
|
|
|
ObjectClass *c = data;
|
2012-12-16 05:17:02 +04:00
|
|
|
CPUListState *s = user_data;
|
2013-01-27 23:16:17 +04:00
|
|
|
const char *typename;
|
|
|
|
char *name;
|
2012-04-15 05:30:10 +04:00
|
|
|
|
2013-01-27 23:16:17 +04:00
|
|
|
typename = object_class_get_name(c);
|
|
|
|
name = g_strndup(typename, strlen(typename) - strlen("-" TYPE_M68K_CPU));
|
2012-04-15 05:30:10 +04:00
|
|
|
(*s->cpu_fprintf)(s->file, "%s\n",
|
2013-01-27 23:16:17 +04:00
|
|
|
name);
|
|
|
|
g_free(name);
|
2012-04-15 05:30:10 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void m68k_cpu_list(FILE *f, fprintf_function cpu_fprintf)
|
|
|
|
{
|
2012-12-16 05:17:02 +04:00
|
|
|
CPUListState s = {
|
2012-04-15 05:30:10 +04:00
|
|
|
.file = f,
|
|
|
|
.cpu_fprintf = cpu_fprintf,
|
|
|
|
};
|
|
|
|
GSList *list;
|
|
|
|
|
|
|
|
list = object_class_get_list(TYPE_M68K_CPU, false);
|
|
|
|
list = g_slist_sort(list, m68k_cpu_list_compare);
|
|
|
|
g_slist_foreach(list, m68k_cpu_list_entry, &s);
|
|
|
|
g_slist_free(list);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
static int fpu_gdb_get_reg(CPUM68KState *env, uint8_t *mem_buf, int n)
|
2008-10-11 21:55:29 +04:00
|
|
|
{
|
|
|
|
if (n < 8) {
|
|
|
|
stfq_p(mem_buf, env->fregs[n]);
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
if (n < 11) {
|
|
|
|
/* FP control registers (not implemented) */
|
|
|
|
memset(mem_buf, 0, 4);
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
static int fpu_gdb_set_reg(CPUM68KState *env, uint8_t *mem_buf, int n)
|
2008-10-11 21:55:29 +04:00
|
|
|
{
|
|
|
|
if (n < 8) {
|
|
|
|
env->fregs[n] = ldfq_p(mem_buf);
|
|
|
|
return 8;
|
|
|
|
}
|
|
|
|
if (n < 11) {
|
|
|
|
/* FP control registers (not implemented) */
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-01-18 17:03:58 +04:00
|
|
|
M68kCPU *cpu_m68k_init(const char *cpu_model)
|
2007-11-10 18:15:54 +03:00
|
|
|
{
|
2012-04-15 02:35:50 +04:00
|
|
|
M68kCPU *cpu;
|
2007-11-10 18:15:54 +03:00
|
|
|
CPUM68KState *env;
|
2013-01-21 20:50:15 +04:00
|
|
|
ObjectClass *oc;
|
2007-11-10 18:15:54 +03:00
|
|
|
|
2013-01-21 20:50:15 +04:00
|
|
|
oc = cpu_class_by_name(TYPE_M68K_CPU, cpu_model);
|
|
|
|
if (oc == NULL) {
|
2012-04-15 05:30:10 +04:00
|
|
|
return NULL;
|
|
|
|
}
|
2013-01-21 20:50:15 +04:00
|
|
|
cpu = M68K_CPU(object_new(object_class_get_name(oc)));
|
2012-04-15 02:35:50 +04:00
|
|
|
env = &cpu->env;
|
2007-12-09 05:22:57 +03:00
|
|
|
|
2012-04-15 05:30:10 +04:00
|
|
|
register_m68k_insns(env);
|
2013-01-05 18:15:30 +04:00
|
|
|
|
|
|
|
object_property_set_bool(OBJECT(cpu), true, "realized", NULL);
|
|
|
|
|
2013-01-18 17:03:58 +04:00
|
|
|
return cpu;
|
2013-01-05 18:15:30 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
void m68k_cpu_init_gdb(M68kCPU *cpu)
|
|
|
|
{
|
2013-06-28 23:27:39 +04:00
|
|
|
CPUState *cs = CPU(cpu);
|
2013-01-05 18:15:30 +04:00
|
|
|
CPUM68KState *env = &cpu->env;
|
|
|
|
|
2012-04-15 05:30:10 +04:00
|
|
|
if (m68k_feature(env, M68K_FEATURE_CF_FPU)) {
|
2013-06-28 23:27:39 +04:00
|
|
|
gdb_register_coprocessor(cs, fpu_gdb_get_reg, fpu_gdb_set_reg,
|
2012-04-15 05:30:10 +04:00
|
|
|
11, "cf-fp.xml", 18);
|
2007-11-10 18:15:54 +03:00
|
|
|
}
|
2012-04-15 05:30:10 +04:00
|
|
|
/* TODO: Add [E]MAC registers. */
|
2007-11-10 18:15:54 +03:00
|
|
|
}
|
2007-05-26 20:52:21 +04:00
|
|
|
|
2006-10-22 04:18:54 +04:00
|
|
|
void cpu_m68k_flush_flags(CPUM68KState *env, int cc_op)
|
|
|
|
{
|
2013-09-03 19:38:47 +04:00
|
|
|
M68kCPU *cpu = m68k_env_get_cpu(env);
|
2006-10-22 04:18:54 +04:00
|
|
|
int flags;
|
|
|
|
uint32_t src;
|
|
|
|
uint32_t dest;
|
|
|
|
uint32_t tmp;
|
|
|
|
|
2015-08-09 03:41:40 +03:00
|
|
|
#define HIGHBIT(type) (1u << (sizeof(type) * 8 - 1))
|
2006-10-22 04:18:54 +04:00
|
|
|
|
2015-08-09 03:41:40 +03:00
|
|
|
#define SET_NZ(x, type) do { \
|
|
|
|
if ((type)(x) == 0) { \
|
|
|
|
flags |= CCF_Z; \
|
|
|
|
} else if ((type)(x) < 0) { \
|
|
|
|
flags |= CCF_N; \
|
|
|
|
} \
|
2006-10-22 04:18:54 +04:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define SET_FLAGS_SUB(type, utype) do { \
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_NZ(dest, type); \
|
|
|
|
tmp = dest + src; \
|
|
|
|
if ((utype) tmp < (utype) src) { \
|
|
|
|
flags |= CCF_C; \
|
|
|
|
} \
|
|
|
|
if (HIGHBIT(type) & (tmp ^ dest) & (tmp ^ src)) { \
|
|
|
|
flags |= CCF_V; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define SET_FLAGS_ADD(type, utype) do { \
|
|
|
|
SET_NZ(dest, type); \
|
|
|
|
if ((utype) dest < (utype) src) { \
|
|
|
|
flags |= CCF_C; \
|
|
|
|
} \
|
|
|
|
tmp = dest - src; \
|
|
|
|
if (HIGHBIT(type) & (src ^ dest) & ~(tmp ^ src)) { \
|
|
|
|
flags |= CCF_V; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define SET_FLAGS_ADDX(type, utype) do { \
|
|
|
|
SET_NZ(dest, type); \
|
|
|
|
if ((utype) dest <= (utype) src) { \
|
|
|
|
flags |= CCF_C; \
|
|
|
|
} \
|
|
|
|
tmp = dest - src - 1; \
|
|
|
|
if (HIGHBIT(type) & (src ^ dest) & ~(tmp ^ src)) { \
|
|
|
|
flags |= CCF_V; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define SET_FLAGS_SUBX(type, utype) do { \
|
|
|
|
SET_NZ(dest, type); \
|
|
|
|
tmp = dest + src + 1; \
|
|
|
|
if ((utype) tmp <= (utype) src) { \
|
|
|
|
flags |= CCF_C; \
|
|
|
|
} \
|
|
|
|
if (HIGHBIT(type) & (tmp ^ dest) & (tmp ^ src)) { \
|
|
|
|
flags |= CCF_V; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define SET_FLAGS_SHIFT(type) do { \
|
|
|
|
SET_NZ(dest, type); \
|
|
|
|
flags |= src; \
|
2006-10-22 04:18:54 +04:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
flags = 0;
|
|
|
|
src = env->cc_src;
|
|
|
|
dest = env->cc_dest;
|
|
|
|
switch (cc_op) {
|
|
|
|
case CC_OP_FLAGS:
|
|
|
|
flags = dest;
|
|
|
|
break;
|
2015-08-09 03:41:40 +03:00
|
|
|
case CC_OP_LOGICB:
|
|
|
|
SET_NZ(dest, int8_t);
|
|
|
|
goto set_x;
|
|
|
|
break;
|
|
|
|
case CC_OP_LOGICW:
|
|
|
|
SET_NZ(dest, int16_t);
|
|
|
|
goto set_x;
|
|
|
|
break;
|
2006-10-22 04:18:54 +04:00
|
|
|
case CC_OP_LOGIC:
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_NZ(dest, int32_t);
|
|
|
|
set_x:
|
|
|
|
if (!m68k_feature(env, M68K_FEATURE_M68000)) {
|
|
|
|
/* Unlike m68k, coldfire always clears the overflow bit. */
|
|
|
|
env->cc_x = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case CC_OP_ADDB:
|
|
|
|
SET_FLAGS_ADD(int8_t, uint8_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_ADDW:
|
|
|
|
SET_FLAGS_ADD(int16_t, uint16_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
|
|
|
case CC_OP_ADD:
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_FLAGS_ADD(int32_t, uint32_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SUBB:
|
|
|
|
SET_FLAGS_SUB(int8_t, uint8_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SUBW:
|
|
|
|
SET_FLAGS_SUB(int16_t, uint16_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
|
|
|
case CC_OP_SUB:
|
|
|
|
SET_FLAGS_SUB(int32_t, uint32_t);
|
|
|
|
break;
|
2015-08-09 03:41:40 +03:00
|
|
|
case CC_OP_ADDXB:
|
|
|
|
SET_FLAGS_ADDX(int8_t, uint8_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
2015-08-09 03:41:40 +03:00
|
|
|
case CC_OP_ADDXW:
|
|
|
|
SET_FLAGS_ADDX(int16_t, uint16_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
|
|
|
case CC_OP_ADDX:
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_FLAGS_ADDX(int32_t, uint32_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SUBXB:
|
|
|
|
SET_FLAGS_SUBX(int8_t, uint8_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SUBXW:
|
|
|
|
SET_FLAGS_SUBX(int16_t, uint16_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
|
|
|
case CC_OP_SUBX:
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_FLAGS_SUBX(int32_t, uint32_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SHIFTB:
|
|
|
|
SET_FLAGS_SHIFT(int8_t);
|
|
|
|
break;
|
|
|
|
case CC_OP_SHIFTW:
|
|
|
|
SET_FLAGS_SHIFT(int16_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
2008-05-25 02:29:16 +04:00
|
|
|
case CC_OP_SHIFT:
|
2015-08-09 03:41:40 +03:00
|
|
|
SET_FLAGS_SHIFT(int32_t);
|
2006-10-22 04:18:54 +04:00
|
|
|
break;
|
|
|
|
default:
|
2013-09-03 19:38:47 +04:00
|
|
|
cpu_abort(CPU(cpu), "Bad CC_OP %d", cc_op);
|
2006-10-22 04:18:54 +04:00
|
|
|
}
|
|
|
|
env->cc_op = CC_OP_FLAGS;
|
|
|
|
env->cc_dest = flags;
|
|
|
|
}
|
|
|
|
|
2008-05-25 02:29:16 +04:00
|
|
|
void HELPER(movec)(CPUM68KState *env, uint32_t reg, uint32_t val)
|
2007-05-23 23:58:11 +04:00
|
|
|
{
|
2013-09-03 19:38:47 +04:00
|
|
|
M68kCPU *cpu = m68k_env_get_cpu(env);
|
|
|
|
|
2007-05-23 23:58:11 +04:00
|
|
|
switch (reg) {
|
|
|
|
case 0x02: /* CACR */
|
2007-06-03 15:13:39 +04:00
|
|
|
env->cacr = val;
|
|
|
|
m68k_switch_sp(env);
|
|
|
|
break;
|
|
|
|
case 0x04: case 0x05: case 0x06: case 0x07: /* ACR[0-3] */
|
|
|
|
/* TODO: Implement Access Control Registers. */
|
2007-05-23 23:58:11 +04:00
|
|
|
break;
|
|
|
|
case 0x801: /* VBR */
|
|
|
|
env->vbr = val;
|
|
|
|
break;
|
|
|
|
/* TODO: Implement control registers. */
|
|
|
|
default:
|
2013-09-03 19:38:47 +04:00
|
|
|
cpu_abort(CPU(cpu), "Unimplemented control register write 0x%x = 0x%x\n",
|
2007-05-23 23:58:11 +04:00
|
|
|
reg, val);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-05-25 02:29:16 +04:00
|
|
|
void HELPER(set_macsr)(CPUM68KState *env, uint32_t val)
|
2007-05-29 18:57:59 +04:00
|
|
|
{
|
|
|
|
uint32_t acc;
|
|
|
|
int8_t exthigh;
|
|
|
|
uint8_t extlow;
|
|
|
|
uint64_t regval;
|
|
|
|
int i;
|
|
|
|
if ((env->macsr ^ val) & (MACSR_FI | MACSR_SU)) {
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
regval = env->macc[i];
|
|
|
|
exthigh = regval >> 40;
|
|
|
|
if (env->macsr & MACSR_FI) {
|
|
|
|
acc = regval >> 8;
|
|
|
|
extlow = regval;
|
|
|
|
} else {
|
|
|
|
acc = regval;
|
|
|
|
extlow = regval >> 32;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_FI) {
|
|
|
|
regval = (((uint64_t)acc) << 8) | extlow;
|
|
|
|
regval |= ((int64_t)exthigh) << 40;
|
|
|
|
} else if (env->macsr & MACSR_SU) {
|
|
|
|
regval = acc | (((int64_t)extlow) << 32);
|
|
|
|
regval |= ((int64_t)exthigh) << 40;
|
|
|
|
} else {
|
|
|
|
regval = acc | (((uint64_t)extlow) << 32);
|
|
|
|
regval |= ((uint64_t)(uint8_t)exthigh) << 40;
|
|
|
|
}
|
|
|
|
env->macc[i] = regval;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
env->macsr = val;
|
|
|
|
}
|
|
|
|
|
2007-06-03 15:13:39 +04:00
|
|
|
void m68k_switch_sp(CPUM68KState *env)
|
|
|
|
{
|
|
|
|
int new_sp;
|
|
|
|
|
|
|
|
env->sp[env->current_sp] = env->aregs[7];
|
|
|
|
new_sp = (env->sr & SR_S && env->cacr & M68K_CACR_EUSP)
|
|
|
|
? M68K_SSP : M68K_USP;
|
|
|
|
env->aregs[7] = env->sp[new_sp];
|
|
|
|
env->current_sp = new_sp;
|
|
|
|
}
|
|
|
|
|
2007-09-17 01:08:06 +04:00
|
|
|
#if defined(CONFIG_USER_ONLY)
|
2007-05-23 23:58:11 +04:00
|
|
|
|
2013-08-26 05:01:33 +04:00
|
|
|
int m68k_cpu_handle_mmu_fault(CPUState *cs, vaddr address, int rw,
|
|
|
|
int mmu_idx)
|
2007-05-23 23:58:11 +04:00
|
|
|
{
|
2013-08-26 05:01:33 +04:00
|
|
|
M68kCPU *cpu = M68K_CPU(cs);
|
|
|
|
|
2013-08-26 10:31:06 +04:00
|
|
|
cs->exception_index = EXCP_ACCESS;
|
2013-08-26 05:01:33 +04:00
|
|
|
cpu->env.mmu.ar = address;
|
2007-05-23 23:58:11 +04:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2010-03-01 06:46:18 +03:00
|
|
|
/* MMU */
|
|
|
|
|
|
|
|
/* TODO: This will need fixing once the MMU is implemented. */
|
2013-06-29 20:55:54 +04:00
|
|
|
hwaddr m68k_cpu_get_phys_page_debug(CPUState *cs, vaddr addr)
|
2010-03-01 06:46:18 +03:00
|
|
|
{
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
|
2013-08-26 05:01:33 +04:00
|
|
|
int m68k_cpu_handle_mmu_fault(CPUState *cs, vaddr address, int rw,
|
|
|
|
int mmu_idx)
|
2007-05-23 23:58:11 +04:00
|
|
|
{
|
|
|
|
int prot;
|
|
|
|
|
|
|
|
address &= TARGET_PAGE_MASK;
|
2010-03-17 05:14:28 +03:00
|
|
|
prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
|
2013-09-03 15:59:37 +04:00
|
|
|
tlb_set_page(cs, address, address, prot, mmu_idx, TARGET_PAGE_SIZE);
|
2010-03-17 05:14:28 +03:00
|
|
|
return 0;
|
2007-05-23 23:58:11 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Notify CPU of a pending interrupt. Prioritization and vectoring should
|
|
|
|
be handled by the interrupt controller. Real hardware only requests
|
|
|
|
the vector when the interrupt is acknowledged by the CPU. For
|
|
|
|
simplicitly we calculate it when the interrupt is signalled. */
|
2013-01-18 17:20:52 +04:00
|
|
|
void m68k_set_irq_level(M68kCPU *cpu, int level, uint8_t vector)
|
2007-05-23 23:58:11 +04:00
|
|
|
{
|
2013-01-18 01:30:20 +04:00
|
|
|
CPUState *cs = CPU(cpu);
|
2013-01-18 17:20:52 +04:00
|
|
|
CPUM68KState *env = &cpu->env;
|
|
|
|
|
2007-05-23 23:58:11 +04:00
|
|
|
env->pending_level = level;
|
|
|
|
env->pending_vector = vector;
|
2013-01-18 01:30:20 +04:00
|
|
|
if (level) {
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(cs, CPU_INTERRUPT_HARD);
|
2013-01-18 01:30:20 +04:00
|
|
|
} else {
|
|
|
|
cpu_reset_interrupt(cs, CPU_INTERRUPT_HARD);
|
|
|
|
}
|
2007-05-23 23:58:11 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2008-05-25 02:29:16 +04:00
|
|
|
|
|
|
|
uint32_t HELPER(bitrev)(uint32_t x)
|
|
|
|
{
|
|
|
|
x = ((x >> 1) & 0x55555555u) | ((x << 1) & 0xaaaaaaaau);
|
|
|
|
x = ((x >> 2) & 0x33333333u) | ((x << 2) & 0xccccccccu);
|
|
|
|
x = ((x >> 4) & 0x0f0f0f0fu) | ((x << 4) & 0xf0f0f0f0u);
|
|
|
|
return bswap32(x);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t HELPER(ff1)(uint32_t x)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
for (n = 32; x; n--)
|
|
|
|
x >>= 1;
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t HELPER(sats)(uint32_t val, uint32_t ccr)
|
|
|
|
{
|
|
|
|
/* The result has the opposite sign to the original value. */
|
|
|
|
if (ccr & CCF_V)
|
|
|
|
val = (((int32_t)val) >> 31) ^ SIGNBIT;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(subx_cc)(CPUM68KState *env, uint32_t op1, uint32_t op2)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t res;
|
|
|
|
uint32_t old_flags;
|
|
|
|
|
|
|
|
old_flags = env->cc_dest;
|
|
|
|
if (env->cc_x) {
|
|
|
|
env->cc_x = (op1 <= op2);
|
|
|
|
env->cc_op = CC_OP_SUBX;
|
|
|
|
res = op1 - (op2 + 1);
|
|
|
|
} else {
|
|
|
|
env->cc_x = (op1 < op2);
|
|
|
|
env->cc_op = CC_OP_SUB;
|
|
|
|
res = op1 - op2;
|
|
|
|
}
|
|
|
|
env->cc_dest = res;
|
|
|
|
env->cc_src = op2;
|
|
|
|
cpu_m68k_flush_flags(env, env->cc_op);
|
|
|
|
/* !Z is sticky. */
|
|
|
|
env->cc_dest &= (old_flags | ~CCF_Z);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(addx_cc)(CPUM68KState *env, uint32_t op1, uint32_t op2)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t res;
|
|
|
|
uint32_t old_flags;
|
|
|
|
|
|
|
|
old_flags = env->cc_dest;
|
|
|
|
if (env->cc_x) {
|
|
|
|
res = op1 + op2 + 1;
|
|
|
|
env->cc_x = (res <= op2);
|
|
|
|
env->cc_op = CC_OP_ADDX;
|
|
|
|
} else {
|
|
|
|
res = op1 + op2;
|
|
|
|
env->cc_x = (res < op2);
|
|
|
|
env->cc_op = CC_OP_ADD;
|
|
|
|
}
|
|
|
|
env->cc_dest = res;
|
|
|
|
env->cc_src = op2;
|
|
|
|
cpu_m68k_flush_flags(env, env->cc_op);
|
|
|
|
/* !Z is sticky. */
|
|
|
|
env->cc_dest &= (old_flags | ~CCF_Z);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(set_sr)(CPUM68KState *env, uint32_t val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
env->sr = val & 0xffff;
|
|
|
|
m68k_switch_sp(env);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(shl_cc)(CPUM68KState *env, uint32_t val, uint32_t shift)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t result;
|
|
|
|
uint32_t cf;
|
|
|
|
|
|
|
|
shift &= 63;
|
|
|
|
if (shift == 0) {
|
|
|
|
result = val;
|
|
|
|
cf = env->cc_src & CCF_C;
|
|
|
|
} else if (shift < 32) {
|
|
|
|
result = val << shift;
|
|
|
|
cf = (val >> (32 - shift)) & 1;
|
|
|
|
} else if (shift == 32) {
|
|
|
|
result = 0;
|
|
|
|
cf = val & 1;
|
|
|
|
} else /* shift > 32 */ {
|
|
|
|
result = 0;
|
|
|
|
cf = 0;
|
|
|
|
}
|
|
|
|
env->cc_src = cf;
|
|
|
|
env->cc_x = (cf != 0);
|
|
|
|
env->cc_dest = result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(shr_cc)(CPUM68KState *env, uint32_t val, uint32_t shift)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t result;
|
|
|
|
uint32_t cf;
|
|
|
|
|
|
|
|
shift &= 63;
|
|
|
|
if (shift == 0) {
|
|
|
|
result = val;
|
|
|
|
cf = env->cc_src & CCF_C;
|
|
|
|
} else if (shift < 32) {
|
|
|
|
result = val >> shift;
|
|
|
|
cf = (val >> (shift - 1)) & 1;
|
|
|
|
} else if (shift == 32) {
|
|
|
|
result = 0;
|
|
|
|
cf = val >> 31;
|
|
|
|
} else /* shift > 32 */ {
|
|
|
|
result = 0;
|
|
|
|
cf = 0;
|
|
|
|
}
|
|
|
|
env->cc_src = cf;
|
|
|
|
env->cc_x = (cf != 0);
|
|
|
|
env->cc_dest = result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(sar_cc)(CPUM68KState *env, uint32_t val, uint32_t shift)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t result;
|
|
|
|
uint32_t cf;
|
|
|
|
|
|
|
|
shift &= 63;
|
|
|
|
if (shift == 0) {
|
|
|
|
result = val;
|
|
|
|
cf = (env->cc_src & CCF_C) != 0;
|
|
|
|
} else if (shift < 32) {
|
|
|
|
result = (int32_t)val >> shift;
|
|
|
|
cf = (val >> (shift - 1)) & 1;
|
|
|
|
} else /* shift >= 32 */ {
|
|
|
|
result = (int32_t)val >> 31;
|
|
|
|
cf = val >> 31;
|
|
|
|
}
|
|
|
|
env->cc_src = cf;
|
|
|
|
env->cc_x = cf;
|
|
|
|
env->cc_dest = result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FPU helpers. */
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(f64_to_i32)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_to_int32(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float32 HELPER(f64_to_f32)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_to_float32(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(i32_to_f64)(CPUM68KState *env, uint32_t val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return int32_to_float64(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(f32_to_f64)(CPUM68KState *env, float32 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float32_to_float64(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(iround_f64)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_round_to_int(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(itrunc_f64)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_trunc_to_int(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(sqrt_f64)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_sqrt(val, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
float64 HELPER(abs_f64)(float64 val)
|
|
|
|
{
|
|
|
|
return float64_abs(val);
|
|
|
|
}
|
|
|
|
|
|
|
|
float64 HELPER(chs_f64)(float64 val)
|
|
|
|
{
|
|
|
|
return float64_chs(val);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(add_f64)(CPUM68KState *env, float64 a, float64 b)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_add(a, b, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(sub_f64)(CPUM68KState *env, float64 a, float64 b)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_sub(a, b, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(mul_f64)(CPUM68KState *env, float64 a, float64 b)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_mul(a, b, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(div_f64)(CPUM68KState *env, float64 a, float64 b)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_div(a, b, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
float64 HELPER(sub_cmp_f64)(CPUM68KState *env, float64 a, float64 b)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
/* ??? This may incorrectly raise exceptions. */
|
|
|
|
/* ??? Should flush denormals to zero. */
|
|
|
|
float64 res;
|
|
|
|
res = float64_sub(a, b, &env->fp_status);
|
softfloat: Implement run-time-configurable meaning of signaling NaN bit
This patch modifies SoftFloat library so that it can be configured in
run-time in relation to the meaning of signaling NaN bit, while, at the
same time, strictly preserving its behavior on all existing platforms.
Background:
In floating-point calculations, there is a need for denoting undefined or
unrepresentable values. This is achieved by defining certain floating-point
numerical values to be NaNs (which stands for "not a number"). For additional
reasons, virtually all modern floating-point unit implementations use two
kinds of NaNs: quiet and signaling. The binary representations of these two
kinds of NaNs, as a rule, differ only in one bit (that bit is, traditionally,
the first bit of mantissa).
Up to 2008, standards for floating-point did not specify all details about
binary representation of NaNs. More specifically, the meaning of the bit
that is used for distinguishing between signaling and quiet NaNs was not
strictly prescribed. (IEEE 754-2008 was the first floating-point standard
that defined that meaning clearly, see [1], p. 35) As a result, different
platforms took different approaches, and that presented considerable
challenge for multi-platform emulators like QEMU.
Mips platform represents the most complex case among QEMU-supported
platforms regarding signaling NaN bit. Up to the Release 6 of Mips
architecture, "1" in signaling NaN bit denoted signaling NaN, which is
opposite to IEEE 754-2008 standard. From Release 6 on, Mips architecture
adopted IEEE standard prescription, and "0" denotes signaling NaN. On top of
that, Mips architecture for SIMD (also known as MSA, or vector instructions)
also specifies signaling bit in accordance to IEEE standard. MSA unit can be
implemented with both pre-Release 6 and Release 6 main processor units.
QEMU uses SoftFloat library to implement various floating-point-related
instructions on all platforms. The current QEMU implementation allows for
defining meaning of signaling NaN bit during build time, and is implemented
via preprocessor macro called SNAN_BIT_IS_ONE.
On the other hand, the change in this patch enables SoftFloat library to be
configured in run-time. This configuration is meant to occur during CPU
initialization, at the moment when it is definitely known what desired
behavior for particular CPU (or any additional FPUs) is.
The change is implemented so that it is consistent with existing
implementation of similar cases. This means that structure float_status is
used for passing the information about desired signaling NaN bit on each
invocation of SoftFloat functions. The additional field in float_status is
called snan_bit_is_one, which supersedes macro SNAN_BIT_IS_ONE.
IMPORTANT:
This change is not meant to create any change in emulator behavior or
functionality on any platform. It just provides the means for SoftFloat
library to be used in a more flexible way - in other words, it will just
prepare SoftFloat library for usage related to Mips platform and its
specifics regarding signaling bit meaning, which is done in some of
subsequent patches from this series.
Further break down of changes:
1) Added field snan_bit_is_one to the structure float_status, and
correspondent setter function set_snan_bit_is_one().
2) Constants <float16|float32|float64|floatx80|float128>_default_nan
(used both internally and externally) converted to functions
<float16|float32|float64|floatx80|float128>_default_nan(float_status*).
This is necessary since they are dependent on signaling bit meaning.
At the same time, for the sake of code cleanup and simplicity, constants
<floatx80|float128>_default_nan_<low|high> (used only internally within
SoftFloat library) are removed, as not needed.
3) Added a float_status* argument to SoftFloat library functions
XXX_is_quiet_nan(XXX a_), XXX_is_signaling_nan(XXX a_),
XXX_maybe_silence_nan(XXX a_). This argument must be present in
order to enable correct invocation of new version of functions
XXX_default_nan(). (XXX is <float16|float32|float64|floatx80|float128>
here)
4) Updated code for all platforms to reflect changes in SoftFloat library.
This change is twofolds: it includes modifications of SoftFloat library
functions invocations, and an addition of invocation of function
set_snan_bit_is_one() during CPU initialization, with arguments that
are appropriate for each particular platform. It was established that
all platforms zero their main CPU data structures, so snan_bit_is_one(0)
in appropriate places is not added, as it is not needed.
[1] "IEEE Standard for Floating-Point Arithmetic",
IEEE Computer Society, August 29, 2008.
Signed-off-by: Thomas Schwinge <thomas@codesourcery.com>
Signed-off-by: Maciej W. Rozycki <macro@codesourcery.com>
Signed-off-by: Aleksandar Markovic <aleksandar.markovic@imgtec.com>
Tested-by: Bastian Koppelmann <kbastian@mail.uni-paderborn.de>
Reviewed-by: Leon Alrae <leon.alrae@imgtec.com>
Tested-by: Leon Alrae <leon.alrae@imgtec.com>
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
[leon.alrae@imgtec.com:
* cherry-picked 2 chunks from patch #2 to fix compilation warnings]
Signed-off-by: Leon Alrae <leon.alrae@imgtec.com>
2016-06-10 12:57:28 +03:00
|
|
|
if (float64_is_quiet_nan(res, &env->fp_status)) {
|
2008-05-25 02:29:16 +04:00
|
|
|
/* +/-inf compares equal against itself, but sub returns nan. */
|
softfloat: Implement run-time-configurable meaning of signaling NaN bit
This patch modifies SoftFloat library so that it can be configured in
run-time in relation to the meaning of signaling NaN bit, while, at the
same time, strictly preserving its behavior on all existing platforms.
Background:
In floating-point calculations, there is a need for denoting undefined or
unrepresentable values. This is achieved by defining certain floating-point
numerical values to be NaNs (which stands for "not a number"). For additional
reasons, virtually all modern floating-point unit implementations use two
kinds of NaNs: quiet and signaling. The binary representations of these two
kinds of NaNs, as a rule, differ only in one bit (that bit is, traditionally,
the first bit of mantissa).
Up to 2008, standards for floating-point did not specify all details about
binary representation of NaNs. More specifically, the meaning of the bit
that is used for distinguishing between signaling and quiet NaNs was not
strictly prescribed. (IEEE 754-2008 was the first floating-point standard
that defined that meaning clearly, see [1], p. 35) As a result, different
platforms took different approaches, and that presented considerable
challenge for multi-platform emulators like QEMU.
Mips platform represents the most complex case among QEMU-supported
platforms regarding signaling NaN bit. Up to the Release 6 of Mips
architecture, "1" in signaling NaN bit denoted signaling NaN, which is
opposite to IEEE 754-2008 standard. From Release 6 on, Mips architecture
adopted IEEE standard prescription, and "0" denotes signaling NaN. On top of
that, Mips architecture for SIMD (also known as MSA, or vector instructions)
also specifies signaling bit in accordance to IEEE standard. MSA unit can be
implemented with both pre-Release 6 and Release 6 main processor units.
QEMU uses SoftFloat library to implement various floating-point-related
instructions on all platforms. The current QEMU implementation allows for
defining meaning of signaling NaN bit during build time, and is implemented
via preprocessor macro called SNAN_BIT_IS_ONE.
On the other hand, the change in this patch enables SoftFloat library to be
configured in run-time. This configuration is meant to occur during CPU
initialization, at the moment when it is definitely known what desired
behavior for particular CPU (or any additional FPUs) is.
The change is implemented so that it is consistent with existing
implementation of similar cases. This means that structure float_status is
used for passing the information about desired signaling NaN bit on each
invocation of SoftFloat functions. The additional field in float_status is
called snan_bit_is_one, which supersedes macro SNAN_BIT_IS_ONE.
IMPORTANT:
This change is not meant to create any change in emulator behavior or
functionality on any platform. It just provides the means for SoftFloat
library to be used in a more flexible way - in other words, it will just
prepare SoftFloat library for usage related to Mips platform and its
specifics regarding signaling bit meaning, which is done in some of
subsequent patches from this series.
Further break down of changes:
1) Added field snan_bit_is_one to the structure float_status, and
correspondent setter function set_snan_bit_is_one().
2) Constants <float16|float32|float64|floatx80|float128>_default_nan
(used both internally and externally) converted to functions
<float16|float32|float64|floatx80|float128>_default_nan(float_status*).
This is necessary since they are dependent on signaling bit meaning.
At the same time, for the sake of code cleanup and simplicity, constants
<floatx80|float128>_default_nan_<low|high> (used only internally within
SoftFloat library) are removed, as not needed.
3) Added a float_status* argument to SoftFloat library functions
XXX_is_quiet_nan(XXX a_), XXX_is_signaling_nan(XXX a_),
XXX_maybe_silence_nan(XXX a_). This argument must be present in
order to enable correct invocation of new version of functions
XXX_default_nan(). (XXX is <float16|float32|float64|floatx80|float128>
here)
4) Updated code for all platforms to reflect changes in SoftFloat library.
This change is twofolds: it includes modifications of SoftFloat library
functions invocations, and an addition of invocation of function
set_snan_bit_is_one() during CPU initialization, with arguments that
are appropriate for each particular platform. It was established that
all platforms zero their main CPU data structures, so snan_bit_is_one(0)
in appropriate places is not added, as it is not needed.
[1] "IEEE Standard for Floating-Point Arithmetic",
IEEE Computer Society, August 29, 2008.
Signed-off-by: Thomas Schwinge <thomas@codesourcery.com>
Signed-off-by: Maciej W. Rozycki <macro@codesourcery.com>
Signed-off-by: Aleksandar Markovic <aleksandar.markovic@imgtec.com>
Tested-by: Bastian Koppelmann <kbastian@mail.uni-paderborn.de>
Reviewed-by: Leon Alrae <leon.alrae@imgtec.com>
Tested-by: Leon Alrae <leon.alrae@imgtec.com>
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
[leon.alrae@imgtec.com:
* cherry-picked 2 chunks from patch #2 to fix compilation warnings]
Signed-off-by: Leon Alrae <leon.alrae@imgtec.com>
2016-06-10 12:57:28 +03:00
|
|
|
if (!float64_is_quiet_nan(a, &env->fp_status)
|
|
|
|
&& !float64_is_quiet_nan(b, &env->fp_status)) {
|
2008-05-25 02:29:16 +04:00
|
|
|
res = float64_zero;
|
|
|
|
if (float64_lt_quiet(a, res, &env->fp_status))
|
|
|
|
res = float64_chs(res);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(compare_f64)(CPUM68KState *env, float64 val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
return float64_compare_quiet(val, float64_zero, &env->fp_status);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* MAC unit. */
|
|
|
|
/* FIXME: The MAC unit implementation is a bit of a mess. Some helpers
|
|
|
|
take values, others take register numbers and manipulate the contents
|
|
|
|
in-place. */
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(mac_move)(CPUM68KState *env, uint32_t dest, uint32_t src)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t mask;
|
|
|
|
env->macc[dest] = env->macc[src];
|
|
|
|
mask = MACSR_PAV0 << dest;
|
|
|
|
if (env->macsr & (MACSR_PAV0 << src))
|
|
|
|
env->macsr |= mask;
|
|
|
|
else
|
|
|
|
env->macsr &= ~mask;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint64_t HELPER(macmuls)(CPUM68KState *env, uint32_t op1, uint32_t op2)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int64_t product;
|
|
|
|
int64_t res;
|
|
|
|
|
|
|
|
product = (uint64_t)op1 * op2;
|
|
|
|
res = (product << 24) >> 24;
|
|
|
|
if (res != product) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
|
|
|
/* Make sure the accumulate operation overflows. */
|
|
|
|
if (product < 0)
|
|
|
|
res = ~(1ll << 50);
|
|
|
|
else
|
|
|
|
res = 1ll << 50;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint64_t HELPER(macmulu)(CPUM68KState *env, uint32_t op1, uint32_t op2)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint64_t product;
|
|
|
|
|
|
|
|
product = (uint64_t)op1 * op2;
|
|
|
|
if (product & (0xffffffull << 40)) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
|
|
|
/* Make sure the accumulate operation overflows. */
|
|
|
|
product = 1ll << 50;
|
|
|
|
} else {
|
|
|
|
product &= ((1ull << 40) - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return product;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint64_t HELPER(macmulf)(CPUM68KState *env, uint32_t op1, uint32_t op2)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint64_t product;
|
|
|
|
uint32_t remainder;
|
|
|
|
|
|
|
|
product = (uint64_t)op1 * op2;
|
|
|
|
if (env->macsr & MACSR_RT) {
|
|
|
|
remainder = product & 0xffffff;
|
|
|
|
product >>= 24;
|
|
|
|
if (remainder > 0x800000)
|
|
|
|
product++;
|
|
|
|
else if (remainder == 0x800000)
|
|
|
|
product += (product & 1);
|
|
|
|
} else {
|
|
|
|
product >>= 24;
|
|
|
|
}
|
|
|
|
return product;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(macsats)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int64_t tmp;
|
|
|
|
int64_t result;
|
|
|
|
tmp = env->macc[acc];
|
|
|
|
result = ((tmp << 16) >> 16);
|
|
|
|
if (result != tmp) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_V) {
|
|
|
|
env->macsr |= MACSR_PAV0 << acc;
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
2011-04-28 19:20:38 +04:00
|
|
|
/* The result is saturated to 32 bits, despite overflow occurring
|
2008-05-25 02:29:16 +04:00
|
|
|
at 48 bits. Seems weird, but that's what the hardware docs
|
|
|
|
say. */
|
|
|
|
result = (result >> 63) ^ 0x7fffffff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
env->macc[acc] = result;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(macsatu)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint64_t val;
|
|
|
|
|
|
|
|
val = env->macc[acc];
|
|
|
|
if (val & (0xffffull << 48)) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_V) {
|
|
|
|
env->macsr |= MACSR_PAV0 << acc;
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
|
|
|
if (val > (1ull << 53))
|
|
|
|
val = 0;
|
|
|
|
else
|
|
|
|
val = (1ull << 48) - 1;
|
|
|
|
} else {
|
|
|
|
val &= ((1ull << 48) - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
env->macc[acc] = val;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(macsatf)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int64_t sum;
|
|
|
|
int64_t result;
|
|
|
|
|
|
|
|
sum = env->macc[acc];
|
|
|
|
result = (sum << 16) >> 16;
|
|
|
|
if (result != sum) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_V) {
|
|
|
|
env->macsr |= MACSR_PAV0 << acc;
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
|
|
|
result = (result >> 63) ^ 0x7fffffffffffll;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
env->macc[acc] = result;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(mac_set_flags)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint64_t val;
|
|
|
|
val = env->macc[acc];
|
2010-04-23 23:22:22 +04:00
|
|
|
if (val == 0) {
|
2008-05-25 02:29:16 +04:00
|
|
|
env->macsr |= MACSR_Z;
|
2010-04-23 23:22:22 +04:00
|
|
|
} else if (val & (1ull << 47)) {
|
2008-05-25 02:29:16 +04:00
|
|
|
env->macsr |= MACSR_N;
|
2010-04-23 23:22:22 +04:00
|
|
|
}
|
2008-05-25 02:29:16 +04:00
|
|
|
if (env->macsr & (MACSR_PAV0 << acc)) {
|
|
|
|
env->macsr |= MACSR_V;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_FI) {
|
|
|
|
val = ((int64_t)val) >> 40;
|
|
|
|
if (val != 0 && val != -1)
|
|
|
|
env->macsr |= MACSR_EV;
|
|
|
|
} else if (env->macsr & MACSR_SU) {
|
|
|
|
val = ((int64_t)val) >> 32;
|
|
|
|
if (val != 0 && val != -1)
|
|
|
|
env->macsr |= MACSR_EV;
|
|
|
|
} else {
|
|
|
|
if ((val >> 32) != 0)
|
|
|
|
env->macsr |= MACSR_EV;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(flush_flags)(CPUM68KState *env, uint32_t cc_op)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
cpu_m68k_flush_flags(env, cc_op);
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(get_macf)(CPUM68KState *env, uint64_t val)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int rem;
|
|
|
|
uint32_t result;
|
|
|
|
|
|
|
|
if (env->macsr & MACSR_SU) {
|
|
|
|
/* 16-bit rounding. */
|
|
|
|
rem = val & 0xffffff;
|
|
|
|
val = (val >> 24) & 0xffffu;
|
|
|
|
if (rem > 0x800000)
|
|
|
|
val++;
|
|
|
|
else if (rem == 0x800000)
|
|
|
|
val += (val & 1);
|
|
|
|
} else if (env->macsr & MACSR_RT) {
|
|
|
|
/* 32-bit rounding. */
|
|
|
|
rem = val & 0xff;
|
|
|
|
val >>= 8;
|
|
|
|
if (rem > 0x80)
|
|
|
|
val++;
|
|
|
|
else if (rem == 0x80)
|
|
|
|
val += (val & 1);
|
|
|
|
} else {
|
|
|
|
/* No rounding. */
|
|
|
|
val >>= 8;
|
|
|
|
}
|
|
|
|
if (env->macsr & MACSR_OMC) {
|
|
|
|
/* Saturate. */
|
|
|
|
if (env->macsr & MACSR_SU) {
|
|
|
|
if (val != (uint16_t) val) {
|
|
|
|
result = ((val >> 63) ^ 0x7fff) & 0xffff;
|
|
|
|
} else {
|
|
|
|
result = val & 0xffff;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (val != (uint32_t)val) {
|
|
|
|
result = ((uint32_t)(val >> 63) & 0x7fffffff);
|
|
|
|
} else {
|
|
|
|
result = (uint32_t)val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* No saturation. */
|
|
|
|
if (env->macsr & MACSR_SU) {
|
|
|
|
result = val & 0xffff;
|
|
|
|
} else {
|
|
|
|
result = (uint32_t)val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t HELPER(get_macs)(uint64_t val)
|
|
|
|
{
|
|
|
|
if (val == (int32_t)val) {
|
|
|
|
return (int32_t)val;
|
|
|
|
} else {
|
|
|
|
return (val >> 61) ^ ~SIGNBIT;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t HELPER(get_macu)(uint64_t val)
|
|
|
|
{
|
|
|
|
if ((val >> 32) == 0) {
|
|
|
|
return (uint32_t)val;
|
|
|
|
} else {
|
|
|
|
return 0xffffffffu;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(get_mac_extf)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t val;
|
|
|
|
val = env->macc[acc] & 0x00ff;
|
2016-07-15 18:29:22 +03:00
|
|
|
val |= (env->macc[acc] >> 32) & 0xff00;
|
2008-05-25 02:29:16 +04:00
|
|
|
val |= (env->macc[acc + 1] << 16) & 0x00ff0000;
|
|
|
|
val |= (env->macc[acc + 1] >> 16) & 0xff000000;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
uint32_t HELPER(get_mac_exti)(CPUM68KState *env, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint32_t val;
|
|
|
|
val = (env->macc[acc] >> 32) & 0xffff;
|
|
|
|
val |= (env->macc[acc + 1] >> 16) & 0xffff0000;
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(set_mac_extf)(CPUM68KState *env, uint32_t val, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int64_t res;
|
|
|
|
int32_t tmp;
|
|
|
|
res = env->macc[acc] & 0xffffffff00ull;
|
|
|
|
tmp = (int16_t)(val & 0xff00);
|
|
|
|
res |= ((int64_t)tmp) << 32;
|
|
|
|
res |= val & 0xff;
|
|
|
|
env->macc[acc] = res;
|
|
|
|
res = env->macc[acc + 1] & 0xffffffff00ull;
|
|
|
|
tmp = (val & 0xff000000);
|
|
|
|
res |= ((int64_t)tmp) << 16;
|
|
|
|
res |= (val >> 16) & 0xff;
|
|
|
|
env->macc[acc + 1] = res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(set_mac_exts)(CPUM68KState *env, uint32_t val, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
int64_t res;
|
|
|
|
int32_t tmp;
|
|
|
|
res = (uint32_t)env->macc[acc];
|
|
|
|
tmp = (int16_t)val;
|
|
|
|
res |= ((int64_t)tmp) << 32;
|
|
|
|
env->macc[acc] = res;
|
|
|
|
res = (uint32_t)env->macc[acc + 1];
|
|
|
|
tmp = val & 0xffff0000;
|
|
|
|
res |= (int64_t)tmp << 16;
|
|
|
|
env->macc[acc + 1] = res;
|
|
|
|
}
|
|
|
|
|
2012-03-14 04:38:22 +04:00
|
|
|
void HELPER(set_mac_extu)(CPUM68KState *env, uint32_t val, uint32_t acc)
|
2008-05-25 02:29:16 +04:00
|
|
|
{
|
|
|
|
uint64_t res;
|
|
|
|
res = (uint32_t)env->macc[acc];
|
|
|
|
res |= ((uint64_t)(val & 0xffff)) << 32;
|
|
|
|
env->macc[acc] = res;
|
|
|
|
res = (uint32_t)env->macc[acc + 1];
|
|
|
|
res |= (uint64_t)(val & 0xffff0000) << 16;
|
|
|
|
env->macc[acc + 1] = res;
|
|
|
|
}
|