aboutsummaryrefslogtreecommitdiff
path: root/sim/v850/simops.c
diff options
context:
space:
mode:
authorJeff Law <law@redhat.com>1996-08-30 03:07:24 +0000
committerJeff Law <law@redhat.com>1996-08-30 03:07:24 +0000
commitaabce0f46c338ee715d0b74887fe15b1acd3449c (patch)
tree9f950d0a6003dc26cbd9644c5c580dab2968a269 /sim/v850/simops.c
parent0ef0eba580c6ef4450878098fdbe71681de42bb4 (diff)
downloadgdb-aabce0f46c338ee715d0b74887fe15b1acd3449c.zip
gdb-aabce0f46c338ee715d0b74887fe15b1acd3449c.tar.gz
gdb-aabce0f46c338ee715d0b74887fe15b1acd3449c.tar.bz2
* Makefile.in: Fix typo.
* simops.c: Add condition code handling to "sub" "subr" and "divh" instructions.
Diffstat (limited to 'sim/v850/simops.c')
-rw-r--r--sim/v850/simops.c94
1 files changed, 75 insertions, 19 deletions
diff --git a/sim/v850/simops.c b/sim/v850/simops.c
index 1a36dc1..0bcdabe 100644
--- a/sim/v850/simops.c
+++ b/sim/v850/simops.c
@@ -204,27 +204,58 @@ OP_600 ()
| (cy ? PSW_CY : 0) | (ov ? PSW_OV : 0));
}
-/* sub reg1, reg2
-
- XXX condition codes */
+/* sub reg1, reg2 */
void
OP_1A0 ()
{
- State.regs[OP[1]] -= State.regs[OP[0]];
-}
+ unsigned int op0, op1, result, z, s, cy, ov;
+
+ /* Compute the result. */
+ op0 = State.regs[OP[0]];
+ op1 = State.regs[OP[1]];
+ result = op1 - op0;
-/* subr reg1, reg2
+ /* Compute the condition codes. */
+ z = (result == 0);
+ s = (result & 0x80000000);
+ cy = (result < -op0);
+ ov = ((op1 & 0x80000000) != (op0 & 0x80000000)
+ && (op1 & 0x80000000) != (result & 0x80000000));
- XXX condition codes */
+ /* Store the result and condition codes. */
+ State.regs[OP[1]] = result;
+ State.psw &= ~(PSW_Z | PSW_S | PSW_CY | PSW_OV);
+ State.psw |= ((z ? PSW_Z : 0) | (s ? PSW_S : 0)
+ | (cy ? PSW_CY : 0) | (ov ? PSW_OV : 0));
+ State.regs[OP[1]] = State.regs[OP[0]];
+}
+
+/* subr reg1, reg2 */
void
OP_180 ()
{
- State.regs[OP[1]] = State.regs[OP[0]] - State.regs[OP[1]];
-}
+ unsigned int op0, op1, result, z, s, cy, ov;
+
+ /* Compute the result. */
+ op0 = State.regs[OP[0]];
+ op1 = State.regs[OP[1]];
+ result = op0 - op1;
+
+ /* Compute the condition codes. */
+ z = (result == 0);
+ s = (result & 0x80000000);
+ cy = (result < -op1);
+ ov = ((op0 & 0x80000000) != (op1 & 0x80000000)
+ && (op0 & 0x80000000) != (result & 0x80000000));
-/* mulh reg1, reg2
+ /* Store the result and condition codes. */
+ State.regs[OP[1]] = result;
+ State.psw &= ~(PSW_Z | PSW_S | PSW_CY | PSW_OV);
+ State.psw |= ((z ? PSW_Z : 0) | (s ? PSW_S : 0)
+ | (cy ? PSW_CY : 0) | (ov ? PSW_OV : 0));
+}
- XXX condition codes */
+/* mulh reg1, reg2 */
void
OP_E0 ()
{
@@ -245,9 +276,7 @@ OP_2E0 ()
State.regs[OP[1]] = (State.regs[OP[1]] & 0xffff) * value;
}
-/* mulhi imm16, reg1, reg2
-
- XXX condition codes */
+/* mulhi imm16, reg1, reg2 */
void
OP_6E0 ()
{
@@ -258,14 +287,41 @@ OP_6E0 ()
State.regs[OP[2]] = (State.regs[OP[1]] & 0xffff) * value;
}
-/* divh reg1, reg2
-
- XXX condition codes.
- XXX Is this signed or unsigned? */
+/* divh reg1, reg2 */
void
OP_40 ()
{
- State.regs[OP[1]] /= (State.regs[OP[0]] & 0xffff);
+ unsigned int op0, op1, result, z, s, cy, ov;
+ int temp;
+
+ /* Compute the result. */
+ temp = State.regs[OP[0]] & 0xffff;
+ temp = (temp << 16) >> 16;
+ op0 = temp;
+ op1 = State.regs[OP[1]];
+
+ if (op0 == 0xffffffff && op1 == 0x80000000)
+ {
+ result = 0x80000000;
+ ov = 1;
+ }
+ else if (op0 != 0)
+ {
+ result = op1 / op0;
+ ov = 0;
+ }
+ else
+ ov = 1;
+
+ /* Compute the condition codes. */
+ z = (result == 0);
+ s = (result & 0x80000000);
+
+ /* Store the result and condition codes. */
+ State.regs[OP[1]] = result;
+ State.psw &= ~(PSW_Z | PSW_S | PSW_OV);
+ State.psw |= ((z ? PSW_Z : 0) | (s ? PSW_S : 0)
+ | (ov ? PSW_OV : 0));
}
void