2011-01-16 21:23:03 +00:00
|
|
|
//============================================================================
|
|
|
|
//
|
|
|
|
// SSSS tt lll lll
|
|
|
|
// SS SS tt ll ll
|
|
|
|
// SS tttttt eeee ll ll aaaa
|
|
|
|
// SSSS tt ee ee ll ll aa
|
|
|
|
// SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator"
|
|
|
|
// SS SS tt ee ll ll aa aa
|
|
|
|
// SSSS ttt eeeee llll llll aaaaa
|
|
|
|
//
|
2019-12-31 17:18:56 +00:00
|
|
|
// Copyright (c) 1995-2020 by Bradford W. Mott, Stephen Anthony
|
2011-01-16 21:23:03 +00:00
|
|
|
// and the Stella Team
|
|
|
|
//
|
|
|
|
// See the file "License.txt" for information on usage and redistribution of
|
|
|
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
|
|
|
//============================================================================
|
|
|
|
|
|
|
|
//============================================================================
|
|
|
|
// This class provides Thumb emulation code ("Thumbulator")
|
|
|
|
// by David Welch (dwelch@dwelch.com)
|
|
|
|
// Modified by Fred Quimby
|
|
|
|
// Code is public domain and used with the author's consent
|
|
|
|
//============================================================================
|
|
|
|
|
2016-01-23 22:56:16 +00:00
|
|
|
#ifndef THUMBULATOR_HXX
|
|
|
|
#define THUMBULATOR_HXX
|
|
|
|
|
2017-11-20 18:57:05 +00:00
|
|
|
class Cartridge;
|
|
|
|
|
2011-01-16 21:23:03 +00:00
|
|
|
#include "bspf.hxx"
|
2017-03-21 02:55:27 +00:00
|
|
|
#include "Console.hxx"
|
2011-01-16 21:23:03 +00:00
|
|
|
|
2019-02-23 15:09:00 +00:00
|
|
|
#ifdef RETRON77
|
|
|
|
#define UNSAFE_OPTIMIZATIONS
|
|
|
|
#define NO_THUMB_STATS
|
|
|
|
#endif
|
|
|
|
|
2011-01-16 21:23:03 +00:00
|
|
|
#define ROMADDMASK 0x7FFF
|
|
|
|
#define RAMADDMASK 0x1FFF
|
|
|
|
|
|
|
|
#define ROMSIZE (ROMADDMASK+1)
|
|
|
|
#define RAMSIZE (RAMADDMASK+1)
|
|
|
|
|
2018-09-23 01:28:57 +00:00
|
|
|
#define CPSR_N (1u<<31)
|
|
|
|
#define CPSR_Z (1u<<30)
|
|
|
|
#define CPSR_C (1u<<29)
|
|
|
|
#define CPSR_V (1u<<28)
|
2011-01-16 21:23:03 +00:00
|
|
|
|
|
|
|
class Thumbulator
|
|
|
|
{
|
|
|
|
public:
|
2017-03-24 02:25:33 +00:00
|
|
|
// control cartridge specific features of the Thumbulator class,
|
|
|
|
// such as the start location for calling custom code
|
2019-03-19 20:08:35 +00:00
|
|
|
enum class ConfigureFor {
|
2017-03-24 02:25:33 +00:00
|
|
|
BUS, // cartridges of type BUS
|
|
|
|
CDF, // cartridges of type CDF
|
2017-08-14 11:41:54 +00:00
|
|
|
CDF1, // cartridges of type CDF version 1
|
2019-03-19 20:08:35 +00:00
|
|
|
CDFJ, // cartrdiges of type CDFJ
|
2017-03-24 02:25:33 +00:00
|
|
|
DPCplus // cartridges of type DPC+
|
|
|
|
};
|
2017-03-24 20:32:08 +00:00
|
|
|
|
2019-03-16 18:54:24 +00:00
|
|
|
Thumbulator(const uInt16* rom_ptr, uInt16* ram_ptr, uInt16 rom_size,
|
2019-03-02 15:36:08 +00:00
|
|
|
bool traponfatal, Thumbulator::ConfigureFor configurefor,
|
|
|
|
Cartridge* cartridge);
|
2011-01-16 21:23:03 +00:00
|
|
|
|
2011-05-24 16:04:48 +00:00
|
|
|
/**
|
2015-06-12 20:44:09 +00:00
|
|
|
Run the ARM code, and return when finished. A runtime_error exception is
|
2011-11-07 22:50:23 +00:00
|
|
|
thrown in case of any fatal errors/aborts (if enabled), containing the
|
|
|
|
actual error, and the contents of the registers at that point in time.
|
2011-05-24 16:04:48 +00:00
|
|
|
|
|
|
|
@return The results of any debugging output (if enabled),
|
|
|
|
otherwise an empty string
|
|
|
|
*/
|
2011-07-03 21:52:33 +00:00
|
|
|
string run();
|
2017-03-21 02:55:27 +00:00
|
|
|
string run(uInt32 cycles);
|
2011-01-16 21:23:03 +00:00
|
|
|
|
2019-02-23 15:09:00 +00:00
|
|
|
#ifndef UNSAFE_OPTIMIZATIONS
|
2011-11-07 22:50:23 +00:00
|
|
|
/**
|
|
|
|
Normally when a fatal error is encountered, the ARM emulation
|
|
|
|
immediately throws an exception and exits. This method allows execution
|
|
|
|
to continue, and simply log the error.
|
|
|
|
|
|
|
|
Note that this is meant for developers only, and should normally be
|
|
|
|
always enabled. It can be used to temporarily ignore illegal reads
|
|
|
|
and writes, but a ROM which consistently performs these operations
|
|
|
|
should be fixed, as it can cause crashes on real hardware.
|
|
|
|
|
|
|
|
@param enable Enable (the default) or disable exceptions on fatal errors
|
|
|
|
*/
|
|
|
|
static void trapFatalErrors(bool enable) { trapOnFatal = enable; }
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
2017-03-24 20:32:08 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
Inform the Thumbulator class about the console currently in use,
|
|
|
|
which is used to accurately determine how many 6507 cycles have
|
|
|
|
passed while ARM code is being executed.
|
|
|
|
*/
|
|
|
|
void setConsoleTiming(ConsoleTiming timing);
|
2011-11-07 22:50:23 +00:00
|
|
|
|
2019-02-19 23:32:48 +00:00
|
|
|
private:
|
|
|
|
|
|
|
|
enum class Op : uInt8 {
|
|
|
|
invalid,
|
|
|
|
adc,
|
|
|
|
add1, add2, add3, add4, add5, add6, add7,
|
|
|
|
and_,
|
|
|
|
asr1, asr2,
|
|
|
|
b1, b2,
|
|
|
|
bic,
|
|
|
|
bkpt,
|
|
|
|
blx1, blx2,
|
|
|
|
bx,
|
|
|
|
cmn,
|
|
|
|
cmp1, cmp2, cmp3,
|
|
|
|
cps,
|
|
|
|
cpy,
|
|
|
|
eor,
|
|
|
|
ldmia,
|
|
|
|
ldr1, ldr2, ldr3, ldr4,
|
|
|
|
ldrb1, ldrb2,
|
|
|
|
ldrh1, ldrh2,
|
|
|
|
ldrsb,
|
|
|
|
ldrsh,
|
|
|
|
lsl1, lsl2,
|
|
|
|
lsr1, lsr2,
|
|
|
|
mov1, mov2, mov3,
|
|
|
|
mul,
|
|
|
|
mvn,
|
|
|
|
neg,
|
|
|
|
orr,
|
|
|
|
pop,
|
|
|
|
push,
|
|
|
|
rev,
|
|
|
|
rev16,
|
|
|
|
revsh,
|
|
|
|
ror,
|
|
|
|
sbc,
|
|
|
|
setend,
|
|
|
|
stmia,
|
|
|
|
str1, str2, str3,
|
|
|
|
strb1, strb2,
|
|
|
|
strh1, strh2,
|
|
|
|
sub1, sub2, sub3, sub4,
|
|
|
|
swi,
|
|
|
|
sxtb,
|
|
|
|
sxth,
|
|
|
|
tst,
|
|
|
|
uxtb,
|
|
|
|
uxth
|
|
|
|
};
|
|
|
|
|
2011-01-16 21:23:03 +00:00
|
|
|
private:
|
2016-02-27 19:58:20 +00:00
|
|
|
uInt32 read_register(uInt32 reg);
|
|
|
|
void write_register(uInt32 reg, uInt32 data);
|
|
|
|
uInt32 fetch16(uInt32 addr);
|
|
|
|
uInt32 read16(uInt32 addr);
|
|
|
|
uInt32 read32(uInt32 addr);
|
2019-02-23 15:09:00 +00:00
|
|
|
#ifndef UNSAFE_OPTIMIZATIONS
|
2017-12-05 19:28:41 +00:00
|
|
|
bool isProtected(uInt32 addr);
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
2016-02-27 19:58:20 +00:00
|
|
|
void write16(uInt32 addr, uInt32 data);
|
|
|
|
void write32(uInt32 addr, uInt32 data);
|
2017-03-21 02:55:27 +00:00
|
|
|
void updateTimer(uInt32 cycles);
|
2016-02-27 19:58:20 +00:00
|
|
|
|
2019-02-22 19:11:19 +00:00
|
|
|
static Op decodeInstructionWord(uint16_t inst);
|
2019-02-19 23:32:48 +00:00
|
|
|
|
2016-02-27 19:58:20 +00:00
|
|
|
void do_zflag(uInt32 x);
|
|
|
|
void do_nflag(uInt32 x);
|
|
|
|
void do_cflag(uInt32 a, uInt32 b, uInt32 c);
|
|
|
|
void do_vflag(uInt32 a, uInt32 b, uInt32 c);
|
|
|
|
void do_cflag_bit(uInt32 x);
|
|
|
|
void do_vflag_bit(uInt32 x);
|
2011-01-16 21:23:03 +00:00
|
|
|
|
2019-02-23 15:09:00 +00:00
|
|
|
#ifndef UNSAFE_OPTIMIZATIONS
|
2015-06-12 20:44:09 +00:00
|
|
|
// Throw a runtime_error exception containing an error referencing the
|
|
|
|
// given message and variables
|
2011-07-03 21:52:33 +00:00
|
|
|
// Note that the return value is never used in these methods
|
|
|
|
int fatalError(const char* opcode, uInt32 v1, const char* msg);
|
|
|
|
int fatalError(const char* opcode, uInt32 v1, uInt32 v2, const char* msg);
|
|
|
|
|
2016-02-27 19:58:20 +00:00
|
|
|
void dump_counters();
|
|
|
|
void dump_regs();
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
2016-02-27 19:58:20 +00:00
|
|
|
int execute();
|
|
|
|
int reset();
|
2011-01-16 21:23:03 +00:00
|
|
|
|
|
|
|
private:
|
2019-12-29 22:06:56 +00:00
|
|
|
const uInt16* rom{nullptr};
|
|
|
|
uInt16 romSize{0};
|
2019-12-27 22:52:09 +00:00
|
|
|
const unique_ptr<Op[]> decodedRom; // NOLINT
|
2019-12-29 22:06:56 +00:00
|
|
|
uInt16* ram{nullptr};
|
2016-02-27 19:58:20 +00:00
|
|
|
|
2019-12-27 22:52:09 +00:00
|
|
|
std::array<uInt32, 16> reg_norm; // normal execution mode, do not have a thread mode
|
2019-12-29 22:06:56 +00:00
|
|
|
uInt32 cpsr{0}, mamcr{0};
|
|
|
|
bool handler_mode{false};
|
|
|
|
uInt32 systick_ctrl{0}, systick_reload{0}, systick_count{0}, systick_calibrate{0};
|
2019-02-23 15:09:00 +00:00
|
|
|
#ifndef UNSAFE_OPTIMIZATIONS
|
2019-12-29 22:06:56 +00:00
|
|
|
uInt64 instructions{0};
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
|
|
|
#ifndef NO_THUMB_STATS
|
2019-12-29 22:06:56 +00:00
|
|
|
uInt64 fetches{0}, reads{0}, writes{0};
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
2017-03-24 20:32:08 +00:00
|
|
|
|
2017-03-21 02:55:27 +00:00
|
|
|
// For emulation of LPC2103's timer 1, used for NTSC/PAL/SECAM detection.
|
|
|
|
// Register names from documentation:
|
|
|
|
// http://www.nxp.com/documents/user_manual/UM10161.pdf
|
2019-12-29 22:06:56 +00:00
|
|
|
uInt32 T1TCR{0}; // Timer 1 Timer Control Register
|
|
|
|
uInt32 T1TC{0}; // Timer 1 Timer Counter
|
|
|
|
double timing_factor{0.0};
|
2011-01-16 21:23:03 +00:00
|
|
|
|
2019-02-24 08:23:57 +00:00
|
|
|
#ifndef UNSAFE_OPTIMIZATIONS
|
2011-05-24 16:04:48 +00:00
|
|
|
ostringstream statusMsg;
|
2011-11-07 22:50:23 +00:00
|
|
|
|
|
|
|
static bool trapOnFatal;
|
2019-02-23 15:09:00 +00:00
|
|
|
#endif
|
2017-03-24 20:32:08 +00:00
|
|
|
|
2017-03-24 02:25:33 +00:00
|
|
|
ConfigureFor configuration;
|
2017-03-24 20:32:08 +00:00
|
|
|
|
2017-03-24 02:25:33 +00:00
|
|
|
Cartridge* myCartridge;
|
2015-04-26 19:02:42 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
// Following constructors and assignment operators not supported
|
|
|
|
Thumbulator() = delete;
|
|
|
|
Thumbulator(const Thumbulator&) = delete;
|
|
|
|
Thumbulator(Thumbulator&&) = delete;
|
|
|
|
Thumbulator& operator=(const Thumbulator&) = delete;
|
|
|
|
Thumbulator& operator=(Thumbulator&&) = delete;
|
2011-01-16 21:23:03 +00:00
|
|
|
};
|
2011-04-16 16:53:41 +00:00
|
|
|
|
2016-01-23 22:56:16 +00:00
|
|
|
#endif // THUMBULATOR_HXX
|