2019-02-04 23:52:21 +00:00
|
|
|
|
// <copyright file="Bus.cs" company="Adrian Conlon">
|
|
|
|
|
// Copyright (c) Adrian Conlon. All rights reserved.
|
|
|
|
|
// </copyright>
|
|
|
|
|
|
|
|
|
|
namespace EightBit
|
2019-02-02 15:12:51 +00:00
|
|
|
|
{
|
|
|
|
|
using System;
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public abstract class Bus : IMapper
|
2019-02-02 15:12:51 +00:00
|
|
|
|
{
|
|
|
|
|
private byte data;
|
|
|
|
|
|
|
|
|
|
public event EventHandler<EventArgs> WritingByte;
|
2019-02-04 23:52:21 +00:00
|
|
|
|
|
2019-02-02 15:12:51 +00:00
|
|
|
|
public event EventHandler<EventArgs> WrittenByte;
|
|
|
|
|
|
|
|
|
|
public event EventHandler<EventArgs> ReadingByte;
|
2019-02-04 23:52:21 +00:00
|
|
|
|
|
2019-02-02 15:12:51 +00:00
|
|
|
|
public event EventHandler<EventArgs> ReadByte;
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public byte Data { get => this.data; set => this.data = value; }
|
|
|
|
|
|
2019-02-06 23:41:56 +00:00
|
|
|
|
public ushort Address { get; set; }
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-03 00:42:55 +00:00
|
|
|
|
public abstract MemoryMapping Mapping(ushort absolute);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public byte Peek() => this.Reference();
|
|
|
|
|
|
|
|
|
|
public byte Peek(ushort absolute) => this.Reference(absolute);
|
|
|
|
|
|
|
|
|
|
public byte Peek(byte low, byte high) => this.Reference(low, high);
|
|
|
|
|
|
|
|
|
|
public void Poke(byte value) => this.Reference() = value;
|
|
|
|
|
|
|
|
|
|
public byte Poke(ushort absolute, byte value) => this.Reference(absolute) = value;
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public byte Poke(byte low, byte high, byte value) => this.Reference(low, high) = value;
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
|
|
|
|
public byte Read()
|
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.OnReadingByte();
|
|
|
|
|
var returned = this.Data = this.Reference();
|
|
|
|
|
this.OnReadByte();
|
2019-02-02 15:12:51 +00:00
|
|
|
|
return returned;
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-03 00:42:55 +00:00
|
|
|
|
public byte Read(ushort absolute)
|
2019-02-02 15:12:51 +00:00
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.Address = absolute;
|
|
|
|
|
return this.Read();
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public byte Read(byte low, byte high) => this.Read(Chip.MakeWord(low, high));
|
2019-02-03 00:42:55 +00:00
|
|
|
|
|
2019-02-02 15:12:51 +00:00
|
|
|
|
public void Write()
|
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.OnWritingByte();
|
|
|
|
|
this.Reference() = this.Data;
|
|
|
|
|
this.OnWrittenByte();
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Write(byte value)
|
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.Data = value;
|
|
|
|
|
this.Write();
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-03 00:42:55 +00:00
|
|
|
|
public void Write(ushort absolute, byte value)
|
2019-02-02 15:12:51 +00:00
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.Address = absolute;
|
|
|
|
|
this.Write(value);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public void Write(byte low, byte high, byte value) => this.Write(Chip.MakeWord(low, high), value);
|
2019-02-03 00:42:55 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
public virtual void RaisePOWER()
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public virtual void LowerPOWER()
|
|
|
|
|
{
|
|
|
|
|
}
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
|
|
|
|
public abstract void Initialize();
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
protected virtual void OnWritingByte() => this.WritingByte?.Invoke(this, EventArgs.Empty);
|
|
|
|
|
|
|
|
|
|
protected virtual void OnWrittenByte() => this.WrittenByte?.Invoke(this, EventArgs.Empty);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
protected virtual void OnReadingByte() => this.ReadingByte?.Invoke(this, EventArgs.Empty);
|
|
|
|
|
|
|
|
|
|
protected virtual void OnReadByte() => this.ReadByte?.Invoke(this, EventArgs.Empty);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-03 00:42:55 +00:00
|
|
|
|
protected ref byte Reference(ushort absolute)
|
2019-02-02 15:12:51 +00:00
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
var mapped = this.Mapping(absolute);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
var offset = (ushort)((absolute - mapped.Begin) & mapped.Mask);
|
|
|
|
|
if (mapped.Access == AccessLevel.ReadOnly)
|
|
|
|
|
{
|
2019-02-04 23:52:21 +00:00
|
|
|
|
this.Data = mapped.Memory.Peek(offset);
|
|
|
|
|
return ref this.data;
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
2019-02-04 23:52:21 +00:00
|
|
|
|
|
2019-02-02 15:12:51 +00:00
|
|
|
|
return ref mapped.Memory.Reference(offset);
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
protected ref byte Reference() => ref this.Reference(this.Address);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
protected ref byte Reference(byte low, byte high) => ref this.Reference(Chip.MakeWord(low, high));
|
2019-02-02 15:12:51 +00:00
|
|
|
|
|
2019-02-04 23:52:21 +00:00
|
|
|
|
////[[nodiscard]] static std::map<uint16_t, std::vector<uint8_t>> parseHexFile(std::string path);
|
|
|
|
|
////void loadHexFile(std::string path);
|
2019-02-02 15:12:51 +00:00
|
|
|
|
}
|
|
|
|
|
}
|