2017-06-04 21:38:34 +01:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <array>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <string>
|
|
|
|
|
2017-06-04 23:37:25 +01:00
|
|
|
#if defined(_M_X64) || defined(_M_IX86 )
|
|
|
|
# define HOST_LITTLE_ENDIAN
|
|
|
|
#else
|
|
|
|
# define HOST_BIG_ENDIAN
|
|
|
|
#endif
|
|
|
|
|
2017-06-04 21:38:34 +01:00
|
|
|
namespace EightBit {
|
|
|
|
|
|
|
|
typedef union {
|
|
|
|
struct {
|
|
|
|
#ifdef HOST_LITTLE_ENDIAN
|
|
|
|
uint8_t low;
|
|
|
|
uint8_t high;
|
|
|
|
#endif
|
|
|
|
#ifdef HOST_BIG_ENDIAN
|
|
|
|
uint8_t high;
|
|
|
|
uint8_t low;
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
uint16_t word;
|
|
|
|
} register16_t;
|
|
|
|
|
|
|
|
class Memory {
|
|
|
|
public:
|
2017-06-09 10:23:51 +01:00
|
|
|
|
|
|
|
static uint8_t highByte(uint16_t value) {
|
|
|
|
return value >> 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t lowByte(uint16_t value) {
|
|
|
|
return value & 0xff;
|
|
|
|
}
|
|
|
|
|
2017-06-04 21:38:34 +01:00
|
|
|
Memory(uint16_t addressMask);
|
|
|
|
|
2017-06-05 22:39:15 +01:00
|
|
|
virtual register16_t& ADDRESS() { return m_address; }
|
2017-06-04 21:38:34 +01:00
|
|
|
virtual uint8_t& DATA() { return *m_data; }
|
|
|
|
|
|
|
|
virtual uint8_t& placeDATA(uint8_t value) {
|
|
|
|
m_temporary = value;
|
|
|
|
m_data = &m_temporary;
|
|
|
|
return DATA();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uint8_t& referenceDATA(uint8_t& value) {
|
|
|
|
m_data = &value;
|
|
|
|
return DATA();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual uint8_t peek(uint16_t address) const;
|
|
|
|
virtual uint16_t peekWord(uint16_t address) const;
|
|
|
|
|
|
|
|
virtual uint8_t get(uint16_t address) {
|
2017-06-05 22:39:15 +01:00
|
|
|
ADDRESS().word = address;
|
2017-06-04 21:38:34 +01:00
|
|
|
return reference();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual register16_t getWord(uint16_t address);
|
|
|
|
|
|
|
|
virtual void set(uint16_t address, uint8_t value) {
|
2017-06-05 22:39:15 +01:00
|
|
|
ADDRESS().word = address;
|
2017-06-04 21:38:34 +01:00
|
|
|
reference() = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void setWord(uint16_t address, register16_t value);
|
|
|
|
|
|
|
|
virtual uint8_t& reference() {
|
2017-06-09 10:23:51 +01:00
|
|
|
auto effective = effectiveAddress(ADDRESS().word);
|
2017-06-04 21:38:34 +01:00
|
|
|
return m_locked[effective] ? placeDATA(m_bus[effective]) : referenceDATA(m_bus[effective]);
|
|
|
|
}
|
|
|
|
|
2017-06-09 10:23:51 +01:00
|
|
|
virtual uint16_t effectiveAddress(uint16_t address) const {
|
|
|
|
return address & m_addressMask;
|
|
|
|
}
|
|
|
|
|
2017-06-04 21:38:34 +01:00
|
|
|
void clear();
|
|
|
|
void loadRom(const std::string& path, uint16_t offset);
|
|
|
|
void loadRam(const std::string& path, uint16_t offset);
|
|
|
|
|
|
|
|
protected:
|
|
|
|
std::array<uint8_t, 0x10000> m_bus;
|
|
|
|
std::array<bool, 0x10000> m_locked;
|
|
|
|
|
|
|
|
uint16_t m_addressMask; // Mirror
|
|
|
|
uint8_t m_temporary; // Used to simulate ROM
|
2017-06-05 22:39:15 +01:00
|
|
|
register16_t m_address;
|
2017-06-04 21:38:34 +01:00
|
|
|
uint8_t* m_data;
|
|
|
|
|
|
|
|
int loadMemory(const std::string& path, uint16_t offset);
|
|
|
|
};
|
|
|
|
}
|