2011-12-07 11:07:42 +01:00

600 lines
12 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* 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; either version 2
* of the License, or (at your option) any later version.
* 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.
*
*/
#ifndef DREAMGEN_RUNTIME_H__
#define DREAMGEN_RUNTIME_H__
#include <assert.h>
#include "common/scummsys.h"
#include "common/array.h"
#include "common/debug.h"
#include "common/hashmap.h"
#include "common/list.h"
#include "common/ptr.h"
namespace DreamGen {
struct Register {
union {
uint16 _value;
uint8 _part[2];
};
inline Register(): _value() {}
inline Register& operator=(uint16 v) { _value = v; return *this; }
inline operator uint16&() { return _value; }
inline void cbw() {
if (_value & 0x80)
_value |= 0xff00;
else
_value &= 0x7f;
}
};
template<int kIndex> //from low to high
struct RegisterPart {
uint8 &_value;
explicit inline RegisterPart(Register &reg) : _value(reg._part[kIndex]) {}
inline operator uint8&() {
return _value;
}
inline RegisterPart& operator=(const RegisterPart& o) {
_value = o._value;
return *this;
}
inline RegisterPart& operator=(uint8 v) {
_value = v;
return *this;
}
};
#ifdef SCUMM_LITTLE_ENDIAN
typedef RegisterPart<0> LowPartOfRegister;
typedef RegisterPart<1> HighPartOfRegister;
#else
typedef RegisterPart<1> LowPartOfRegister;
typedef RegisterPart<0> HighPartOfRegister;
#endif
class WordRef {
uint8 *_data;
unsigned _index;
uint16 _value;
public:
inline WordRef(Common::Array<uint8> &data, unsigned index) : _data(data.begin() + index), _index(index) {
assert(index + 1 < data.size());
_value = _data[0] | (_data[1] << 8);
}
inline WordRef& operator=(const WordRef &ref) {
_value = ref._value;
return *this;
}
inline WordRef& operator=(uint16 v) {
_value = v;
return *this;
}
inline operator uint16&() {
return _value;
}
inline ~WordRef() {
_data[0] = _value & 0xff;
_data[1] = _value >> 8;
_value = _data[0] | (_data[1] << 8);
}
};
class Segment {
Common::Array<uint8> data;
public:
Segment(uint size = 0) {
if (size > 0)
data.resize(size);
}
inline void assign(const uint8 *b, const uint8 *e) {
data.assign(b, e);
}
inline uint8 &byte(unsigned index) {
assert(index < data.size());
return data[index];
}
inline WordRef word(unsigned index) {
return WordRef(data, index);
}
inline uint8 *ptr(unsigned index, unsigned size) {
assert(index + size <= data.size());
return data.begin() + index;
}
};
typedef Common::SharedPtr<Segment> SegmentPtr;
class Context;
class SegmentRef {
uint16 _value;
SegmentPtr _segment;
public:
SegmentRef(uint16 value = 0, SegmentPtr segment = SegmentPtr())
: _value(value), _segment(segment) {
}
inline operator uint16() const {
return _value;
}
SegmentPtr getSegmentPtr() const {
return _segment;
}
inline uint8 &byte(unsigned index) {
assert(_segment != 0);
return _segment->byte(index);
}
inline WordRef word(unsigned index) {
//debug(1, "getting word ref for %04x:%d", _value, index);
assert(_segment != 0);
return _segment->word(index);
}
inline void assign(const uint8 *b, const uint8 *e) {
assert(_segment != 0);
_segment->assign(b, e);
}
inline uint8 *ptr(unsigned index, unsigned size) {
assert(_segment != 0);
return _segment->ptr(index, size);
}
protected:
SegmentRef &operator=(const SegmentRef &seg) {
_value = seg._value;
_segment = seg._segment;
return *this;
}
};
class MutableSegmentRef : public SegmentRef {
protected:
Context *_context;
public:
MutableSegmentRef(Context *ctx, uint16 value = 0, SegmentPtr segment = SegmentPtr())
: _context(ctx), SegmentRef(value, segment) {
}
MutableSegmentRef(Context *ctx, SegmentRef seg)
: _context(ctx), SegmentRef(seg) {
}
inline MutableSegmentRef& operator=(const uint16 id);
};
struct Flags {
bool _z, _c, _s, _o;
inline Flags(): _z(true), _c(false), _s(false), _o(false) {}
inline bool z() const { return _z; }
inline bool c() const { return _c; }
inline bool s() const { return _s; }
inline bool l() const { return _o != _s; }
inline bool le() const { return _o != _s|| _z; }
inline void update_zs(uint8 v) {
_s = v & 0x80;
_z = v == 0;
}
inline void update_zs(uint16 v) {
_s = v & 0x8000;
_z = v == 0;
}
inline void update_o(uint8 v, uint8 a, uint8 b) {
uint8 s1 = a & 0x80, s2 = b & 0x80;
_o = (s1 == s2) && (v & 0x80) != s1;
}
inline void update_o(uint16 v, uint16 a, uint16 b) {
uint16 s1 = a & 0x8000, s2 = b & 0x8000;
_o = (s1 == s2) && (v & 0x8000) != s1;
}
};
class Context {
typedef Common::HashMap<uint16, SegmentPtr> SegmentMap;
SegmentMap _segments;
typedef Common::List<uint16> FreeSegmentList;
FreeSegmentList _freeSegments;
public:
enum { kDefaultDataSegment = 0x1000 };
Register ax, dx, bx, cx, si, di;
LowPartOfRegister al;
HighPartOfRegister ah;
LowPartOfRegister bl;
HighPartOfRegister bh;
LowPartOfRegister cl;
HighPartOfRegister ch;
LowPartOfRegister dl;
HighPartOfRegister dh;
SegmentRef cs;
MutableSegmentRef ds;
MutableSegmentRef es;
Flags flags;
Context(SegmentRef data): al(ax), ah(ax), bl(bx), bh(bx), cl(cx), ch(cx), dl(dx), dh(dx),
cs(data),
ds(this, data),
es(this, data) {
_segments[kDefaultDataSegment] = data.getSegmentPtr();
}
SegmentRef getSegment(uint16 value) {
SegmentMap::iterator i = _segments.find(value);
if (i != _segments.end())
return SegmentRef(value, i->_value);
else
return SegmentRef(value);
}
SegmentRef allocateSegment(uint size) {
unsigned id;
if (_freeSegments.empty())
id = kDefaultDataSegment + _segments.size();
else {
id = _freeSegments.front();
_freeSegments.pop_front();
}
assert(!_segments.contains(id));
SegmentPtr seg(new Segment(size));
_segments[id] = seg;
return SegmentRef(id, seg);
}
void deallocateSegment(uint16 id) {
SegmentMap::iterator i = _segments.find(id);
if(i != _segments.end()) {
_segments.erase(i);
_freeSegments.push_back(id);
} else {
debug("Deallocating non existent segment! Client code should be fixed.");
}
}
inline void _cmp(uint8 a, uint8 b) {
_sub(a, b);
}
inline void _cmp(uint16 a, uint16 b) {
_sub(a, b);
}
inline void _test(uint8 a, uint8 b) {
_and(a, b);
}
inline void _test(uint16 a, uint16 b) {
_and(a, b);
}
inline void _add(uint8 &dst, uint8 src) {
unsigned r = (unsigned)dst + src;
flags.update_o((uint8)r, dst, src);
flags._c = r >= 0x100;
dst = r;
flags.update_zs(dst);
}
inline void _add(uint16 &dst, uint16 src) {
unsigned r = (unsigned)dst + src;
flags.update_o((uint16)r, dst, src);
flags._c = r >= 0x10000;
dst = r;
flags.update_zs(dst);
}
inline void _sub(uint8 &dst, uint8 src) {
flags.update_o(uint8(dst - src), dst, (uint8)-src);
flags._c = dst < src;
dst -= src;
flags.update_zs(dst);
}
inline void _sub(uint16 &dst, uint16 src) {
flags.update_o(uint16(dst - src), dst, (uint16)-src);
flags._c = dst < src;
dst -= src;
flags.update_zs(dst);
}
inline void _inc(uint8 &dst) {
flags.update_o((uint8)(dst + 1), dst, 1);
++dst;
flags.update_zs(dst);
}
inline void _inc(uint16 &dst) {
flags.update_o((uint16)(dst + 1), dst, 1);
++dst;
flags.update_zs(dst);
}
inline void _dec(uint8 &dst) {
flags.update_o(uint8(dst - 1), dst, 1);
--dst;
flags.update_zs(dst);
}
inline void _dec(uint16 &dst) {
flags.update_o(uint16(dst - 1), dst, 1);
--dst;
flags.update_zs(dst);
}
inline void _and(uint8 &dst, uint8 src) {
dst &= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _and(uint16 &dst, uint16 src) {
dst &= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _or(uint8 &dst, uint8 src) {
dst |= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _or(uint16 &dst, uint16 src) {
dst |= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _xor(uint8 &dst, uint8 src) {
dst ^= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _xor(uint16 &dst, uint16 src) {
dst ^= src;
flags.update_zs(dst);
flags._c = flags._o = false;
}
inline void _shr(uint8 &dst, uint8 src) {
src &= 0x1f;
if (src > 0) {
dst >>= (src - 1);
flags._c = dst & 1;
dst >>= 1;
flags.update_zs(dst);
}
if (src == 1)
flags._o = dst & 0x80;
}
inline void _shr(uint16 &dst, uint8 src) {
src &= 0x1f;
if (src > 0) {
dst >>= (src - 1);
flags._c = dst & 1;
dst >>= 1;
flags.update_zs(dst);
}
if (src == 1)
flags._o = dst & 0x8000;
}
inline void _shl(uint8 &dst, uint8 src) {
src &= 0x1f;
if (src > 0) {
dst <<= (src - 1);
flags._c = dst & 0x80;
dst <<= 1;
flags.update_zs(dst);
}
if (src == 1)
flags._o = ((dst & 0x80) != 0) == flags._c;
}
inline void _shl(uint16 &dst, uint8 src) {
src &= 0x1f;
if (src > 0) {
dst <<= (src - 1);
flags._c = dst & 0x8000;
dst <<= 1;
flags.update_zs(dst);
}
if (src == 1)
flags._o = ((dst & 0x8000) != 0) == flags._c;
}
inline void _mul(uint8 src) {
unsigned r = unsigned(al) * src;
ax = (uint16)r;
flags._c = r >= 0x10000;
flags._z = r == 0;
flags._s = r & 0x8000;
flags._o = ah != 0;
}
inline void _mul(uint16 src) {
unsigned r = unsigned(ax) * src; //assuming here that we have at least 32 bits
dx = (r >> 16) & 0xffff;
ax = r & 0xffff;
flags._c = false;
flags._z = r == 0;
flags._s = r & 0x80000000;
flags._o = dx != 0;
}
inline void _neg(uint8 &src) {
uint8 r = 0;
_sub(r, src);
src = r;
}
inline void _neg(uint16 &src) {
uint16 r = 0;
_sub(r, src);
src = r;
}
inline void _lodsb() {
al = ds.byte(si++);
}
inline void _lodsw() {
ax = ds.word(si);
si += 2;
}
inline void _movsb() {
es.byte(di++) = ds.byte(si++);
}
inline void _movsb(uint size, bool clear_cx = false) {
assert(size != 0xffff);
//fixme: add overlap and segment boundary check and rewrite
while (size--)
_movsb();
if (clear_cx)
cx = 0;
}
inline void _movsw() {
_movsb();
_movsb();
}
inline void _movsw(uint size, bool clear_cx = false) {
assert(size != 0xffff);
_movsb(size * 2, clear_cx);
}
inline void _stosb() {
es.byte(di++) = al;
}
inline void _stosb(uint size, bool clear_cx = false) {
assert(size != 0xffff);
uint8 *dst = es.ptr(di, size);
memset(dst, al, size);
di += size;
if (clear_cx)
cx = 0;
}
inline void _stosw() {
es.byte(di++) = al;
es.byte(di++) = ah;
}
inline void _stosw(uint size, bool clear_cx = false) {
assert(size != 0xffff);
uint8 *dst = es.ptr(di, size * 2);
di += 2 * size;
while (size--) {
*dst++ = al;
*dst++ = ah;
}
if (clear_cx)
cx = 0;
}
inline void _xchg(uint16 &a, uint16 &b) {
uint16 x = a;
a = b;
b = x;
}
inline void _xchg(uint8 &a, uint8 &b) {
uint8 t = a;
a = b;
b = t;
}
Common::Array<uint16> stack;
inline void push(uint16 v) {
stack.push_back(v);
}
inline uint16 pop() {
assert(!stack.empty());
uint16 v = stack.back();
stack.pop_back();
return v;
}
};
inline MutableSegmentRef& MutableSegmentRef::operator=(const uint16 id) {
SegmentRef::operator=(_context->getSegment(id));
return *this;
}
class StackChecker {
const Context &_context;
const uint _stackDepth;
public:
StackChecker(const Context &context): _context(context), _stackDepth(context.stack.size()) {}
~StackChecker() { assert(_context.stack.size() == _stackDepth); }
};
#ifndef NDEBUG
# define STACK_CHECK StackChecker checker(*this)
#else
# define STACK_CHECK do {} while (0)
#endif
} // End of namespace DreamWeb
#endif