kvisc/pc/instrs/arith.c

82 lines
1.1 KiB
C
Raw Normal View History

2019-05-30 12:44:56 +02:00
// The OS/K Team licenses this file to you under the MIT license.
2019-05-29 16:57:22 +02:00
// See the LICENSE file in the project root for more information.
#include "instrs.h"
#include "arch_i.h"
2019-05-30 20:23:27 +02:00
IMPL_COND(inc);
IMPL_COND(dec);
IMPL_COND(add);
IMPL_COND(sub);
IMPL_COND(mul);
IMPL_COND(div);
2019-05-31 21:25:56 +02:00
IMPL_COND(sgn);
2019-05-30 20:23:27 +02:00
2019-05-29 16:57:22 +02:00
//
// Unsigned arithmetic instructions
//
IMPL_START_2(add)
{
v1 += v2;
}
IMPL_OUT;
IMPL_START_2(sub)
{
v1 -= v2;
}
IMPL_OUT;
IMPL_START_1(mul)
{
// Adapted from www.codeproject.com/Tips/618570/UInt-Multiplication-Squaring
ulong v2 = v1;
2019-06-02 16:33:28 +02:00
v1 = rax;
2019-05-29 16:57:22 +02:00
ulong u1 = (v1 & 0xffffffff);
ulong u2 = (v2 & 0xffffffff);
ulong t = (u1 * u2);
ulong w3 = (t & 0xffffffff);
ulong k = (t >> 32);
2019-05-30 12:44:56 +02:00
2019-05-29 16:57:22 +02:00
v1 >>= 32;
t = (v1 * u2) + k;
k = (t & 0xffffffff);
ulong w1 = (t >> 32);
2019-05-30 12:44:56 +02:00
2019-05-29 16:57:22 +02:00
v2 >>= 32;
t = (u1 * v2) + k;
k = (t >> 32);
2019-05-30 12:44:56 +02:00
2019-06-02 16:33:28 +02:00
rdx = (v1 * v2) + w1 + k;
rax = (t << 32) + w3;
2019-05-29 16:57:22 +02:00
}
IMPL_END;
IMPL_START_1(div)
{
2019-06-02 16:33:28 +02:00
rdx = rax % v1;
rax = rax / v1;
2019-05-29 16:57:22 +02:00
}
IMPL_END;
IMPL_START_1(inc)
{
v1++;
}
IMPL_OUT;
IMPL_START_1(dec)
{
v1--;
}
IMPL_OUT;
2019-05-31 21:25:56 +02:00
IMPL_START_2(sgn)
{
v1 = (v2 ? ((long)v2 > 0 ? 1 : (ulong)-1L) : 0);
}
IMPL_OUT;