Refactor memory related code a little to make the hierarchy of classes a little easier to understand.

Signed-off-by: Adrian Conlon <Adrian.conlon@gmail.com>
This commit is contained in:
Adrian Conlon 2018-11-04 16:38:57 +00:00
parent 003cea0d64
commit e688411cb9
18 changed files with 112 additions and 106 deletions

View File

@ -2,8 +2,6 @@
#include "Disassembler.h" #include "Disassembler.h"
#include "Intel8080.h" #include "Intel8080.h"
#include <Memory.h>
#include <sstream> #include <sstream>
#include <iomanip> #include <iomanip>
#include <bitset> #include <bitset>

View File

@ -15,6 +15,5 @@
#include <boost/format.hpp> #include <boost/format.hpp>
#include <Memory.h>
#include <IntelProcessor.h> #include <IntelProcessor.h>
#include <InputOutput.h> #include <InputOutput.h>

View File

@ -5,6 +5,7 @@
#include <string> #include <string>
#include <Ram.h> #include <Ram.h>
#include <Rom.h>
#include <Bus.h> #include <Bus.h>
#include <Register.h> #include <Register.h>
#include <UnusedMemory.h> #include <UnusedMemory.h>

View File

@ -5,8 +5,6 @@
#include <iomanip> #include <iomanip>
#include <bitset> #include <bitset>
#include <Memory.h>
#include "LR35902.h" #include "LR35902.h"
#include "IoRegisters.h" #include "IoRegisters.h"

View File

@ -25,12 +25,12 @@
#include <boost/format.hpp> #include <boost/format.hpp>
#include <Memory.h>
#include <Processor.h> #include <Processor.h>
#include <IntelProcessor.h> #include <IntelProcessor.h>
#include <Register.h> #include <Register.h>
#include <Signal.h> #include <Signal.h>
#include <Ram.h> #include <Ram.h>
#include <Rom.h>
#include <Bus.h> #include <Bus.h>
#ifdef _MSC_VER #ifdef _MSC_VER

View File

@ -5,6 +5,7 @@
#include <string> #include <string>
#include <Ram.h> #include <Ram.h>
#include <Rom.h>
#include <Bus.h> #include <Bus.h>
#include <mc6809.h> #include <mc6809.h>
#include <Disassembly.h> #include <Disassembly.h>

View File

@ -17,6 +17,7 @@
#endif #endif
#include <Ram.h> #include <Ram.h>
#include <Rom.h>
#include <Bus.h> #include <Bus.h>
#include <mc6809.h> #include <mc6809.h>
#include <TestHarness.h> #include <TestHarness.h>

View File

@ -6,8 +6,6 @@
#include <bitset> #include <bitset>
#include <iostream> #include <iostream>
#include <Memory.h>
#include "Z80.h" #include "Z80.h"
EightBit::Disassembler::Disassembler(Bus& bus) noexcept EightBit::Disassembler::Disassembler(Bus& bus) noexcept

View File

@ -16,7 +16,6 @@
#include <boost/format.hpp> #include <boost/format.hpp>
#include <Memory.h>
#include <IntelProcessor.h> #include <IntelProcessor.h>
#include <InputOutput.h> #include <InputOutput.h>
#include <Signal.h> #include <Signal.h>

View File

@ -1,56 +1,34 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint>
#include <vector>
#include <string>
#include <fstream> #include <fstream>
#include <stdexcept>
#include "MemoryInterface.h" #include <string>
#include <vector>
namespace EightBit { namespace EightBit {
class Memory : public MemoryInterface { // Memory is:
private: // *) Definitely has a size
std::vector<uint8_t> m_bytes; // *) Definitely 'peek'able (although you might not like the answer you get!)
// *) Probably 'load'able (i.e. able to be externally initialised)
// *) At the implementation level, probably 'poke'able (although may not be exposed to users)
// *) Possibly 'reference'able (Very likely if you've exposed 'poke')
class Memory {
public:
virtual ~Memory() = default;
virtual size_t size() const = 0;
virtual uint8_t peek(uint16_t address) const = 0;
virtual uint8_t& reference(uint16_t) {
throw new std::logic_error("Reference operation not allowed.");
}
virtual int load(std::ifstream& file, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
virtual int load(const std::string& path, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
virtual int load(const std::vector<uint8_t>& bytes, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
protected: protected:
const auto& BYTES() const { return m_bytes; } virtual void poke(uint16_t address, uint8_t value) = 0;
auto& BYTES() { return m_bytes; }
virtual void poke(const uint16_t address, const uint8_t value) override {
BYTES()[address] = value;
}
public:
static int load(std::ifstream& file, std::vector<uint8_t>& output, int writeOffset = 0, int readOffset = 0, int limit = -1, int maximumSize = -1);
static int load(const std::string& path, std::vector<uint8_t>& output, int writeOffset = 0, int readOffset = 0, int limit = -1, int maximumSize = -1);
Memory(const size_t size = 0) noexcept
: m_bytes(size) {
}
virtual size_t size() const final { return m_bytes.size(); }
virtual int load(std::ifstream& file, const int writeOffset = 0, const int readOffset = 0, const int limit = -1) final {
const auto maximumSize = (int)size() - writeOffset;
return load(file, m_bytes, writeOffset, readOffset, limit, maximumSize);
}
virtual int load(const std::string& path, const int writeOffset = 0, const int readOffset = 0, const int limit = -1) final {
const auto maximumSize = (int)size() - writeOffset;
return load(path, m_bytes, writeOffset, readOffset, limit, maximumSize);
}
virtual int load(const std::vector<uint8_t>& bytes, const int writeOffset = 0, const int readOffset = 0, int limit = -1) final {
if (limit < 0)
limit = (int)bytes.size() - readOffset;
std::copy(bytes.cbegin() + readOffset, bytes.cbegin() + limit, m_bytes.begin() + writeOffset);
return limit;
}
virtual uint8_t peek(const uint16_t address) const final {
return BYTES()[address];
}
}; };
typedef Memory Rom;
} }

View File

@ -1,26 +0,0 @@
#pragma once
#include <cstdint>
#include <fstream>
#include <stdexcept>
#include <string>
#include <vector>
namespace EightBit {
class MemoryInterface {
public:
virtual size_t size() const = 0;
virtual uint8_t peek(uint16_t address) const = 0;
virtual uint8_t& reference(uint16_t) {
throw new std::logic_error("Reference operation not allowed.");
}
virtual int load(std::ifstream& file, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
virtual int load(const std::string& path, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
virtual int load(const std::vector<uint8_t>& bytes, int writeOffset = 0, int readOffset = 0, int limit = -1) = 0;
protected:
virtual void poke(uint16_t address, uint8_t value) = 0;
};
}

View File

@ -4,13 +4,13 @@
namespace EightBit { namespace EightBit {
class MemoryInterface; class Memory;
struct MemoryMapping { struct MemoryMapping {
enum AccessLevel { Unknown, ReadOnly, ReadWrite, }; enum AccessLevel { Unknown, ReadOnly, ReadWrite, };
MemoryInterface& memory; Memory& memory;
uint16_t begin = 0xffff; uint16_t begin = 0xffff;
uint16_t mask = 0U; uint16_t mask = 0U;
AccessLevel access = Unknown; AccessLevel access = Unknown;

View File

@ -1,13 +1,16 @@
#pragma once #pragma once
#include <cstdint> #include <cstdint>
#include "Memory.h"
#include "Rom.h"
namespace EightBit { namespace EightBit {
class Ram : public Memory { // The RAM class is everything the ROM class is, plus
// it's externally 'reference'able and 'poke'able.
class Ram : public Rom {
public: public:
Ram(const size_t size = 0) noexcept Ram(const size_t size = 0) noexcept
: Memory(size) { : Rom(size) {
} }
virtual uint8_t& reference(const uint16_t address) final { virtual uint8_t& reference(const uint16_t address) final {
@ -15,7 +18,7 @@ namespace EightBit {
} }
virtual void poke(const uint16_t address, const uint8_t value) final { virtual void poke(const uint16_t address, const uint8_t value) final {
Memory::poke(address, value); Rom::poke(address, value);
} }
}; };
} }

57
inc/Rom.h Normal file
View File

@ -0,0 +1,57 @@
#pragma once
#include <cstdint>
#include <vector>
#include <string>
#include <fstream>
#include "Memory.h"
namespace EightBit {
// ROM is a basic implementation of the Memory interface.
// Nothing over and above the interface is exposed to users
// of the ROM class.
class Rom : public Memory {
private:
std::vector<uint8_t> m_bytes;
protected:
const auto& BYTES() const { return m_bytes; }
auto& BYTES() { return m_bytes; }
virtual void poke(const uint16_t address, const uint8_t value) override {
BYTES()[address] = value;
}
public:
static int load(std::ifstream& file, std::vector<uint8_t>& output, int writeOffset = 0, int readOffset = 0, int limit = -1, int maximumSize = -1);
static int load(const std::string& path, std::vector<uint8_t>& output, int writeOffset = 0, int readOffset = 0, int limit = -1, int maximumSize = -1);
Rom(const size_t size = 0) noexcept
: m_bytes(size) {
}
virtual size_t size() const final { return m_bytes.size(); }
virtual int load(std::ifstream& file, const int writeOffset = 0, const int readOffset = 0, const int limit = -1) final {
const auto maximumSize = (int)size() - writeOffset;
return load(file, m_bytes, writeOffset, readOffset, limit, maximumSize);
}
virtual int load(const std::string& path, const int writeOffset = 0, const int readOffset = 0, const int limit = -1) final {
const auto maximumSize = (int)size() - writeOffset;
return load(path, m_bytes, writeOffset, readOffset, limit, maximumSize);
}
virtual int load(const std::vector<uint8_t>& bytes, const int writeOffset = 0, const int readOffset = 0, int limit = -1) final {
if (limit < 0)
limit = (int)bytes.size() - readOffset;
std::copy(bytes.cbegin() + readOffset, bytes.cbegin() + limit, m_bytes.begin() + writeOffset);
return limit;
}
virtual uint8_t peek(const uint16_t address) const final {
return BYTES()[address];
}
};
}

View File

@ -1,9 +1,12 @@
#pragma once #pragma once
#include "MemoryInterface.h" #include "Memory.h"
namespace EightBit { namespace EightBit {
class UnusedMemory final : public MemoryInterface { // A read-only Memory implementation that has a fixed size and will
// *always* returns the same value, from whichever location
// is being read.
class UnusedMemory final : public Memory {
public: public:
UnusedMemory(const size_t size, const uint8_t value) UnusedMemory(const size_t size, const uint8_t value)
: m_size(size), m_value(value) {} : m_size(size), m_value(value) {}
@ -12,10 +15,6 @@ namespace EightBit {
virtual size_t size() const final { return m_size; } virtual size_t size() const final { return m_size; }
virtual uint8_t peek(uint16_t address) const final { return m_value; } virtual uint8_t peek(uint16_t address) const final { return m_value; }
virtual uint8_t& reference(uint16_t) {
throw new std::logic_error("Reference operation not allowed.");
}
virtual int load(std::ifstream& file, int writeOffset = 0, int readOffset = 0, int limit = -1) final { virtual int load(std::ifstream& file, int writeOffset = 0, int readOffset = 0, int limit = -1) final {
throw new std::logic_error("load operation not allowed."); throw new std::logic_error("load operation not allowed.");
} }

View File

@ -148,8 +148,8 @@
<ClInclude Include="..\inc\InputOutput.h" /> <ClInclude Include="..\inc\InputOutput.h" />
<ClInclude Include="..\inc\IntelProcessor.h" /> <ClInclude Include="..\inc\IntelProcessor.h" />
<ClInclude Include="..\inc\LittleEndianProcessor.h" /> <ClInclude Include="..\inc\LittleEndianProcessor.h" />
<ClInclude Include="..\inc\Rom.h" />
<ClInclude Include="..\inc\Memory.h" /> <ClInclude Include="..\inc\Memory.h" />
<ClInclude Include="..\inc\MemoryInterface.h" />
<ClInclude Include="..\inc\MemoryMapping.h" /> <ClInclude Include="..\inc\MemoryMapping.h" />
<ClInclude Include="..\inc\Processor.h" /> <ClInclude Include="..\inc\Processor.h" />
<ClInclude Include="..\inc\Ram.h" /> <ClInclude Include="..\inc\Ram.h" />
@ -167,7 +167,7 @@
<ClCompile Include="InputOutput.cpp" /> <ClCompile Include="InputOutput.cpp" />
<ClCompile Include="IntelProcessor.cpp" /> <ClCompile Include="IntelProcessor.cpp" />
<ClCompile Include="LittleEndianProcessor.cpp" /> <ClCompile Include="LittleEndianProcessor.cpp" />
<ClCompile Include="Memory.cpp" /> <ClCompile Include="Rom.cpp" />
<ClCompile Include="Processor.cpp" /> <ClCompile Include="Processor.cpp" />
<ClCompile Include="stdafx.cpp"> <ClCompile Include="stdafx.cpp">
<PrecompiledHeader Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">Create</PrecompiledHeader> <PrecompiledHeader Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">Create</PrecompiledHeader>

View File

@ -17,9 +17,6 @@
<ClInclude Include="..\inc\EventArgs.h"> <ClInclude Include="..\inc\EventArgs.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="..\inc\Memory.h">
<Filter>Header Files</Filter>
</ClInclude>
<ClInclude Include="..\inc\Processor.h"> <ClInclude Include="..\inc\Processor.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
@ -59,10 +56,13 @@
<ClInclude Include="..\inc\Chip.h"> <ClInclude Include="..\inc\Chip.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="..\inc\MemoryInterface.h"> <ClInclude Include="..\inc\UnusedMemory.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
<ClInclude Include="..\inc\UnusedMemory.h"> <ClInclude Include="..\inc\Rom.h">
<Filter>Header Files</Filter>
</ClInclude>
<ClInclude Include="..\inc\Memory.h">
<Filter>Header Files</Filter> <Filter>Header Files</Filter>
</ClInclude> </ClInclude>
</ItemGroup> </ItemGroup>
@ -73,9 +73,6 @@
<ClCompile Include="EventArgs.cpp"> <ClCompile Include="EventArgs.cpp">
<Filter>Source Files</Filter> <Filter>Source Files</Filter>
</ClCompile> </ClCompile>
<ClCompile Include="Memory.cpp">
<Filter>Source Files</Filter>
</ClCompile>
<ClCompile Include="Processor.cpp"> <ClCompile Include="Processor.cpp">
<Filter>Source Files</Filter> <Filter>Source Files</Filter>
</ClCompile> </ClCompile>
@ -97,5 +94,8 @@
<ClCompile Include="Chip.cpp"> <ClCompile Include="Chip.cpp">
<Filter>Source Files</Filter> <Filter>Source Files</Filter>
</ClCompile> </ClCompile>
<ClCompile Include="Rom.cpp">
<Filter>Source Files</Filter>
</ClCompile>
</ItemGroup> </ItemGroup>
</Project> </Project>

View File

@ -1,9 +1,9 @@
#include "stdafx.h" #include "stdafx.h"
#include "Memory.h" #include "Rom.h"
#include <iostream> #include <iostream>
int EightBit::Memory::load(std::ifstream& file, std::vector<uint8_t>& output, const int writeOffset, const int readOffset, int limit, const int maximumSize) { int EightBit::Rom::load(std::ifstream& file, std::vector<uint8_t>& output, const int writeOffset, const int readOffset, int limit, const int maximumSize) {
file.seekg(0, std::ios::end); file.seekg(0, std::ios::end);
@ -25,7 +25,7 @@ int EightBit::Memory::load(std::ifstream& file, std::vector<uint8_t>& output, co
return size; return size;
} }
int EightBit::Memory::load(const std::string& path, std::vector<uint8_t>& output, const int writeOffset, const int readOffset, const int limit, const int maximumSize) { int EightBit::Rom::load(const std::string& path, std::vector<uint8_t>& output, const int writeOffset, const int readOffset, const int limit, const int maximumSize) {
std::ifstream file; std::ifstream file;
file.exceptions(std::ios::failbit | std::ios::badbit); file.exceptions(std::ios::failbit | std::ios::badbit);