82 lines
3.2 KiB
C
82 lines
3.2 KiB
C
/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│
|
|
│vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│
|
|
╞══════════════════════════════════════════════════════════════════════════════╡
|
|
│ Copyright 2020 Justine Alexandra Roberts Tunney │
|
|
│ │
|
|
│ This program is free software; you can redistribute it and/or modify │
|
|
│ it under the terms of the GNU General Public License as published by │
|
|
│ the Free Software Foundation; version 2 of the License. │
|
|
│ │
|
|
│ This program 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 General Public License │
|
|
│ along with this program; if not, write to the Free Software │
|
|
│ Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA │
|
|
│ 02110-1301 USA │
|
|
╚─────────────────────────────────────────────────────────────────────────────*/
|
|
#include "libc/assert.h"
|
|
#include "tool/build/lib/alu.h"
|
|
#include "tool/build/lib/flags.h"
|
|
|
|
/**
|
|
* NexGen32e Arithmetic Unit.
|
|
*/
|
|
int64_t Alu(int w, int h, uint64_t x, uint64_t y, uint32_t *flags) {
|
|
bool zf, sf, c, o, cf;
|
|
uint64_t t, z, s, m, k;
|
|
assert(w < 4);
|
|
k = 8;
|
|
k <<= w;
|
|
s = 1;
|
|
s <<= k - 1;
|
|
m = s;
|
|
m |= s - 1;
|
|
t = x;
|
|
c = 0;
|
|
o = 0;
|
|
cf = GetFlag(*flags, FLAGS_CF);
|
|
switch (h & 7) {
|
|
case ALU_OR:
|
|
z = x | y;
|
|
break;
|
|
case ALU_AND:
|
|
z = x & y;
|
|
break;
|
|
case ALU_XOR:
|
|
z = x ^ y;
|
|
break;
|
|
case ALU_CMP:
|
|
h |= 8;
|
|
cf = 0;
|
|
case ALU_SBB:
|
|
t = (x & m) - cf;
|
|
c = (x & m) < (t & m);
|
|
case ALU_SUB:
|
|
z = (t & m) - (y & m);
|
|
c |= (t & m) < (z & m);
|
|
o = !!((z ^ x) & (x ^ y) & s);
|
|
break;
|
|
case ALU_ADC:
|
|
t = (x & m) + cf;
|
|
c = (t & m) < (x & m);
|
|
case ALU_ADD:
|
|
z = (t & m) + (y & m);
|
|
c |= (z & m) < (y & m);
|
|
o = !!((z ^ x) & (z ^ y) & s);
|
|
break;
|
|
default:
|
|
unreachable;
|
|
}
|
|
z &= m;
|
|
zf = !z;
|
|
sf = !!(z & s);
|
|
*flags &= ~(1 << FLAGS_CF | 1 << FLAGS_ZF | 1 << FLAGS_SF | 1 << FLAGS_OF);
|
|
*flags |= c << FLAGS_CF | zf << FLAGS_ZF | sf << FLAGS_SF | o << FLAGS_OF;
|
|
*flags = SetLazyParityByte(*flags, x);
|
|
if (h & ALU_TEST) z = x;
|
|
return z;
|
|
}
|