mirror of https://github.com/stella-emu/stella.git
Sync time on bus snoops.
This commit is contained in:
parent
e672bb1f98
commit
7faa9c71b9
|
@ -363,78 +363,6 @@ uInt8 CartridgeELF::overdrivePoke(uInt16 address, uInt8 value)
|
||||||
return driveBus(address, value);
|
return driveBus(address, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::fetch16(
|
|
||||||
uInt32 address, uInt16& value, uInt8& op, CortexM0& cortex
|
|
||||||
) {
|
|
||||||
if (address == (RETURN_ADDR & ~1)) return CortexM0::errCustom(ERR_RETURN);
|
|
||||||
|
|
||||||
return handleError("fetch16", address, CortexM0::ERR_UNMAPPED_FETCH16, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read8(uInt32 address, uInt8& value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("read8", address, CortexM0::ERR_UNMAPPED_READ8, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read16(uInt32 address, uInt16& value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("read16", address, CortexM0::ERR_UNMAPPED_READ16, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read32(uInt32 address, uInt32& value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("read32", address, CortexM0::ERR_UNMAPPED_READ32, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write8(uInt32 address, uInt8 value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("write8", address, CortexM0::ERR_UNMAPPED_WRITE8, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write16(uInt32 address, uInt16 value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("write16", address, CortexM0::ERR_UNMAPPED_WRITE16, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write32(uInt32 address, uInt32 value, CortexM0& cortex)
|
|
||||||
{
|
|
||||||
return handleError("write32", address, CortexM0::ERR_UNMAPPED_WRITE32, cortex);
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
void CartridgeELF::BusFallbackDelegate::setErrorsAreFatal(bool fatal)
|
|
||||||
{
|
|
||||||
myErrorsAreFatal = fatal;
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
||||||
CortexM0::err_t CartridgeELF::BusFallbackDelegate::handleError(
|
|
||||||
string_view accessType, uInt32 address, CortexM0::err_t err, CortexM0& cortex
|
|
||||||
) const {
|
|
||||||
if (myErrorsAreFatal) return CortexM0::errIntrinsic(err, address);
|
|
||||||
|
|
||||||
stringstream s;
|
|
||||||
|
|
||||||
s
|
|
||||||
<< "invalid " << accessType << " access to 0x"
|
|
||||||
<< std::hex << std::setw(8) << std::setfill('0') << address
|
|
||||||
<< " (PC = 0x"
|
|
||||||
<< std::hex << std::setw(8) << std::setfill('0') << cortex.getRegister(15)
|
|
||||||
<< ")";
|
|
||||||
|
|
||||||
Logger::error(s.str());
|
|
||||||
|
|
||||||
return CortexM0::ERR_NONE;
|
|
||||||
}
|
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
inline uInt64 CartridgeELF::getArmCycles() const
|
inline uInt64 CartridgeELF::getArmCycles() const
|
||||||
{
|
{
|
||||||
|
@ -448,18 +376,24 @@ inline uInt8 CartridgeELF::driveBus(uInt16 address, uInt8 value)
|
||||||
mySystem->cycles() * myArmCyclesPer6502Cycle - myArmCyclesOffset);
|
mySystem->cycles() * myArmCyclesPer6502Cycle - myArmCyclesOffset);
|
||||||
if (nextTransaction) {
|
if (nextTransaction) {
|
||||||
nextTransaction->setBusState(myIsBusDriven, myDriveBusValue);
|
nextTransaction->setBusState(myIsBusDriven, myDriveBusValue);
|
||||||
myArmCyclesOffset = mySystem->cycles() * myArmCyclesPer6502Cycle - nextTransaction->timestamp;
|
syncArmTime(nextTransaction->timestamp);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (myIsBusDriven) value |= myDriveBusValue;
|
if (myIsBusDriven) value |= myDriveBusValue;
|
||||||
|
|
||||||
myVcsLib.updateBus(address, value);
|
if (myVcsLib.updateBus(address, value)) syncArmTime(myCortexEmu.getCycles());
|
||||||
|
|
||||||
runArm();
|
runArm();
|
||||||
|
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
inline void CartridgeELF::syncArmTime(uInt64 armCycles)
|
||||||
|
{
|
||||||
|
myArmCyclesOffset = mySystem->cycles() * myArmCyclesPer6502Cycle - armCycles;
|
||||||
|
}
|
||||||
|
|
||||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
void CartridgeELF::parseAndLinkElf()
|
void CartridgeELF::parseAndLinkElf()
|
||||||
{
|
{
|
||||||
|
@ -645,3 +579,75 @@ void CartridgeELF::runArm()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::fetch16(
|
||||||
|
uInt32 address, uInt16& value, uInt8& op, CortexM0& cortex
|
||||||
|
) {
|
||||||
|
if (address == (RETURN_ADDR & ~1)) return CortexM0::errCustom(ERR_RETURN);
|
||||||
|
|
||||||
|
return handleError("fetch16", address, CortexM0::ERR_UNMAPPED_FETCH16, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read8(uInt32 address, uInt8& value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("read8", address, CortexM0::ERR_UNMAPPED_READ8, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read16(uInt32 address, uInt16& value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("read16", address, CortexM0::ERR_UNMAPPED_READ16, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::read32(uInt32 address, uInt32& value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("read32", address, CortexM0::ERR_UNMAPPED_READ32, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write8(uInt32 address, uInt8 value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("write8", address, CortexM0::ERR_UNMAPPED_WRITE8, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write16(uInt32 address, uInt16 value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("write16", address, CortexM0::ERR_UNMAPPED_WRITE16, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::write32(uInt32 address, uInt32 value, CortexM0& cortex)
|
||||||
|
{
|
||||||
|
return handleError("write32", address, CortexM0::ERR_UNMAPPED_WRITE32, cortex);
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
void CartridgeELF::BusFallbackDelegate::setErrorsAreFatal(bool fatal)
|
||||||
|
{
|
||||||
|
myErrorsAreFatal = fatal;
|
||||||
|
}
|
||||||
|
|
||||||
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||||
|
CortexM0::err_t CartridgeELF::BusFallbackDelegate::handleError(
|
||||||
|
string_view accessType, uInt32 address, CortexM0::err_t err, CortexM0& cortex
|
||||||
|
) const {
|
||||||
|
if (myErrorsAreFatal) return CortexM0::errIntrinsic(err, address);
|
||||||
|
|
||||||
|
stringstream s;
|
||||||
|
|
||||||
|
s
|
||||||
|
<< "invalid " << accessType << " access to 0x"
|
||||||
|
<< std::hex << std::setw(8) << std::setfill('0') << address
|
||||||
|
<< " (PC = 0x"
|
||||||
|
<< std::hex << std::setw(8) << std::setfill('0') << cortex.getRegister(15)
|
||||||
|
<< ")";
|
||||||
|
|
||||||
|
Logger::error(s.str());
|
||||||
|
|
||||||
|
return CortexM0::ERR_NONE;
|
||||||
|
}
|
||||||
|
|
|
@ -31,7 +31,7 @@ class ElfLinker;
|
||||||
class CartridgeELF: public Cartridge {
|
class CartridgeELF: public Cartridge {
|
||||||
public:
|
public:
|
||||||
static constexpr uInt32 MIPS_MAX = 300;
|
static constexpr uInt32 MIPS_MAX = 300;
|
||||||
static constexpr uInt32 MIPS_MIN = 50;
|
static constexpr uInt32 MIPS_MIN = 50;
|
||||||
static constexpr uInt32 MIPS_DEF = 150;
|
static constexpr uInt32 MIPS_DEF = 150;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
@ -102,6 +102,7 @@ class CartridgeELF: public Cartridge {
|
||||||
uInt64 getArmCycles() const;
|
uInt64 getArmCycles() const;
|
||||||
|
|
||||||
uInt8 driveBus(uInt16 address, uInt8 value);
|
uInt8 driveBus(uInt16 address, uInt8 value);
|
||||||
|
void syncArmTime(uInt64 armCycles);
|
||||||
|
|
||||||
void parseAndLinkElf();
|
void parseAndLinkElf();
|
||||||
void allocationSections();
|
void allocationSections();
|
||||||
|
|
|
@ -32,9 +32,11 @@ class VcsLib: public CortexM0::BusTransactionDelegate {
|
||||||
|
|
||||||
CortexM0::err_t fetch16(uInt32 address, uInt16& value, uInt8& op, CortexM0& cortex) override;
|
CortexM0::err_t fetch16(uInt32 address, uInt16& value, uInt8& op, CortexM0& cortex) override;
|
||||||
|
|
||||||
void updateBus(uInt16 address, uInt8 value) {
|
bool updateBus(uInt16 address, uInt8 value) {
|
||||||
myCurrentAddress = address;
|
myCurrentAddress = address;
|
||||||
myCurrentValue = value;
|
myCurrentValue = value;
|
||||||
|
|
||||||
|
return myIsWaitingForRead && myTransactionQueue.size() == 0 && (myWaitingForReadAddress == myCurrentAddress);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isSuspended() const {
|
bool isSuspended() const {
|
||||||
|
|
Loading…
Reference in New Issue