228 lines
7.6 KiB
C
228 lines
7.6 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 "libc/log/log.h"
|
|
#include "libc/macros.h"
|
|
#include "libc/runtime/runtime.h"
|
|
#include "tool/build/lib/alu.h"
|
|
#include "tool/build/lib/endian.h"
|
|
#include "tool/build/lib/flags.h"
|
|
#include "tool/build/lib/ioports.h"
|
|
#include "tool/build/lib/machine.h"
|
|
#include "tool/build/lib/memory.h"
|
|
#include "tool/build/lib/modrm.h"
|
|
#include "tool/build/lib/string.h"
|
|
#include "tool/build/lib/throw.h"
|
|
|
|
static uint64_t ReadInt(uint8_t p[8], unsigned long w) {
|
|
switch (w) {
|
|
case 0:
|
|
return Read8(p);
|
|
case 1:
|
|
return Read16(p);
|
|
case 2:
|
|
return Read32(p);
|
|
case 3:
|
|
return Read64(p);
|
|
default:
|
|
unreachable;
|
|
}
|
|
}
|
|
|
|
static void WriteInt(uint8_t p[8], uint64_t x, unsigned long w) {
|
|
switch (w) {
|
|
case 0:
|
|
Write8(p, x);
|
|
break;
|
|
case 1:
|
|
Write16(p, x);
|
|
break;
|
|
case 2:
|
|
Write64(p, x);
|
|
break;
|
|
case 3:
|
|
Write64(p, x);
|
|
break;
|
|
default:
|
|
unreachable;
|
|
}
|
|
}
|
|
|
|
void OpString(struct Machine *m, int op) {
|
|
void *p[2];
|
|
bool compare;
|
|
int64_t sgn, v;
|
|
uint8_t s[3][8];
|
|
unsigned n, lg2;
|
|
uint64_t asz, seg;
|
|
sgn = GetFlag(m->flags, FLAGS_DF) ? -1 : 1;
|
|
asz = Asz(m->xedd) ? 0xffffffff : 0xffffffffffffffff;
|
|
seg = GetSegment(m);
|
|
lg2 = RegLog2(m->xedd);
|
|
n = 1 << lg2;
|
|
for (;;) {
|
|
if (m->xedd->op.rep && !Read64(m->cx)) break;
|
|
v = 0;
|
|
*p = NULL;
|
|
compare = false;
|
|
switch (op) {
|
|
case STRING_CMPS:
|
|
Alu(lg2, ALU_SUB,
|
|
ReadInt(Load(m, (Read64(m->si) + seg) & asz, n, s[2]), lg2),
|
|
ReadInt(Load(m, Read64(m->di) & asz, n, s[1]), lg2), &m->flags);
|
|
Write64(m->di, (Read64(m->di) + sgn * n) & asz);
|
|
Write64(m->si, (Read64(m->si) + sgn * n) & asz);
|
|
compare = true;
|
|
break;
|
|
case STRING_MOVS:
|
|
memcpy(BeginStore(m, (v = Read64(m->di) & asz), n, p, s[0]),
|
|
Load(m, (Read64(m->si) + seg) & asz, n, s[1]), n);
|
|
Write64(m->di, (Read64(m->di) + sgn * n) & asz);
|
|
Write64(m->si, (Read64(m->si) + sgn * n) & asz);
|
|
break;
|
|
case STRING_STOS:
|
|
memcpy(BeginStore(m, (v = Read64(m->di) & asz), n, p, s[0]), m->ax, n);
|
|
Write64(m->di, (Read64(m->di) + sgn * n) & asz);
|
|
break;
|
|
case STRING_LODS:
|
|
memcpy(m->ax, Load(m, (Read64(m->si) + seg) & asz, n, s[1]), n);
|
|
Write64(m->si, (Read64(m->si) + sgn * n) & asz);
|
|
break;
|
|
case STRING_SCAS:
|
|
Alu(lg2, ALU_SUB, ReadInt(Load(m, Read64(m->di) & asz, n, s[1]), lg2),
|
|
ReadInt(m->ax, lg2), &m->flags);
|
|
Write64(m->di, (Read64(m->di) + sgn * n) & asz);
|
|
compare = true;
|
|
break;
|
|
case STRING_OUTS:
|
|
OpOut(m, Read16(m->dx),
|
|
ReadInt(Load(m, (Read64(m->si) + seg) & asz, n, s[1]), lg2));
|
|
Write64(m->si, (Read64(m->si) + sgn * n) & asz);
|
|
break;
|
|
case STRING_INS:
|
|
WriteInt(BeginStore(m, (v = Read64(m->di) & asz), n, p, s[0]),
|
|
OpIn(m, Read16(m->dx)), lg2);
|
|
Write64(m->di, (Read64(m->di) + sgn * n) & asz);
|
|
break;
|
|
default:
|
|
abort();
|
|
}
|
|
EndStore(m, v, n, p, s[0]);
|
|
if (!m->xedd->op.rep) break;
|
|
Write64(m->cx, Read64(m->cx) - 1);
|
|
if (compare) {
|
|
if (m->xedd->op.rep == 2 && GetFlag(m->flags, FLAGS_ZF)) break;
|
|
if (m->xedd->op.rep == 3 && !GetFlag(m->flags, FLAGS_ZF)) break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void OpRepMovsbEnhanced(struct Machine *m) {
|
|
bool failed;
|
|
uint8_t *direal, *sireal;
|
|
unsigned diremain, siremain, i, n;
|
|
uint64_t divirtual, sivirtual, diactual, siactual, failaddr, seg, asz, cx;
|
|
if (!(cx = Read64(m->cx))) return;
|
|
failed = false;
|
|
failaddr = 0;
|
|
seg = GetSegment(m);
|
|
asz = Asz(m->xedd) ? 0xffffffff : 0xffffffffffffffff;
|
|
divirtual = Read64(m->di) & asz;
|
|
sivirtual = Read64(m->si) & asz;
|
|
SetWriteAddr(m, (seg + divirtual) & asz, cx);
|
|
SetReadAddr(m, (seg + sivirtual) & asz, cx);
|
|
do {
|
|
diactual = (seg + divirtual) & asz;
|
|
siactual = (seg + sivirtual) & asz;
|
|
if (!(direal = FindReal(m, diactual))) {
|
|
failaddr = diactual;
|
|
failed = true;
|
|
break;
|
|
}
|
|
if (!(sireal = FindReal(m, siactual))) {
|
|
failaddr = siactual;
|
|
failed = true;
|
|
break;
|
|
}
|
|
diremain = 0x1000 - (divirtual & 0xfff);
|
|
siremain = 0x1000 - (sivirtual & 0xfff);
|
|
n = MIN(cx, MIN(diremain, siremain));
|
|
for (i = 0; i < n; ++i) {
|
|
direal[i] = sireal[i];
|
|
}
|
|
cx -= n;
|
|
divirtual = (divirtual + n) & asz;
|
|
sivirtual = (sivirtual + n) & asz;
|
|
} while (cx);
|
|
Write64(m->cx, cx);
|
|
Write64(m->di, divirtual);
|
|
Write64(m->si, sivirtual);
|
|
if (failed) ThrowSegmentationFault(m, failaddr);
|
|
}
|
|
|
|
void OpRepStosbEnhanced(struct Machine *m) {
|
|
bool failed;
|
|
uint8_t *direal, al;
|
|
unsigned diremain, i, n;
|
|
uint64_t divirtual, diactual, failaddr, seg, asz, cx;
|
|
if (!(cx = Read64(m->cx))) return;
|
|
failaddr = 0;
|
|
failed = false;
|
|
al = Read8(m->ax);
|
|
seg = GetSegment(m);
|
|
asz = Asz(m->xedd) ? 0xffffffff : 0xffffffffffffffff;
|
|
divirtual = Read64(m->di) & asz;
|
|
SetWriteAddr(m, (seg + divirtual) & asz, cx);
|
|
do {
|
|
diactual = (seg + divirtual) & asz;
|
|
if (!(direal = FindReal(m, diactual))) {
|
|
failaddr = diactual;
|
|
failed = true;
|
|
break;
|
|
}
|
|
diremain = 0x1000 - (divirtual & 0xfff);
|
|
n = MIN(cx, diremain);
|
|
for (i = 0; i < n; ++i) {
|
|
direal[i] = al;
|
|
}
|
|
cx -= n;
|
|
divirtual = (divirtual + n) & asz;
|
|
} while (cx);
|
|
Write64(m->cx, cx);
|
|
Write64(m->di, divirtual);
|
|
if (failed) ThrowSegmentationFault(m, failaddr);
|
|
}
|
|
|
|
void OpMovsb(struct Machine *m) {
|
|
if (m->xedd->op.rep && !GetFlag(m->flags, FLAGS_DF)) {
|
|
OpRepMovsbEnhanced(m);
|
|
} else {
|
|
OpString(m, STRING_MOVS);
|
|
}
|
|
}
|
|
|
|
void OpStosb(struct Machine *m) {
|
|
if (m->xedd->op.rep && !GetFlag(m->flags, FLAGS_DF)) {
|
|
OpRepStosbEnhanced(m);
|
|
} else {
|
|
OpString(m, STRING_STOS);
|
|
}
|
|
}
|