C64: Apply savestate changes to tape subsystem.
This commit is contained in:
parent
1e2c9e45f0
commit
70e1a2ca29
|
@ -1,57 +1,62 @@
|
|||
using System;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public sealed class CassettePort
|
||||
{
|
||||
public Func<bool> ReadDataOutput = () => true;
|
||||
public Func<bool> ReadMotor = () => true;
|
||||
private CassettePortDevice _device;
|
||||
private bool _connected;
|
||||
|
||||
public void HardReset()
|
||||
{
|
||||
if (_connected)
|
||||
{
|
||||
_device.HardReset();
|
||||
}
|
||||
}
|
||||
|
||||
public void ExecutePhase()
|
||||
{
|
||||
if (_connected)
|
||||
{
|
||||
_device.ExecutePhase2();
|
||||
}
|
||||
}
|
||||
|
||||
public bool ReadDataInputBuffer()
|
||||
{
|
||||
return !_connected || _device.ReadDataInputBuffer();
|
||||
}
|
||||
|
||||
public bool ReadSenseBuffer()
|
||||
{
|
||||
return !_connected || _device.ReadSenseBuffer();
|
||||
}
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
|
||||
public void Connect(CassettePortDevice device)
|
||||
{
|
||||
_connected = device != null;
|
||||
_device = device;
|
||||
if (_device == null)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
_device.ReadDataOutput = () => ReadDataOutput();
|
||||
_device.ReadMotor = () => ReadMotor();
|
||||
}
|
||||
}
|
||||
}
|
||||
using System;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public sealed class CassettePort
|
||||
{
|
||||
[SaveState.DoNotSave]
|
||||
public Func<bool> ReadDataOutput = () => true;
|
||||
[SaveState.DoNotSave]
|
||||
public Func<bool> ReadMotor = () => true;
|
||||
|
||||
[SaveState.SaveWithName("Device")]
|
||||
private CassettePortDevice _device;
|
||||
[SaveState.SaveWithName("Connected")]
|
||||
private bool _connected;
|
||||
|
||||
public void HardReset()
|
||||
{
|
||||
if (_connected)
|
||||
{
|
||||
_device.HardReset();
|
||||
}
|
||||
}
|
||||
|
||||
public void ExecutePhase()
|
||||
{
|
||||
if (_connected)
|
||||
{
|
||||
_device.ExecutePhase2();
|
||||
}
|
||||
}
|
||||
|
||||
public bool ReadDataInputBuffer()
|
||||
{
|
||||
return !_connected || _device.ReadDataInputBuffer();
|
||||
}
|
||||
|
||||
public bool ReadSenseBuffer()
|
||||
{
|
||||
return !_connected || _device.ReadSenseBuffer();
|
||||
}
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
|
||||
public void Connect(CassettePortDevice device)
|
||||
{
|
||||
_connected = device != null;
|
||||
_device = device;
|
||||
if (_device == null)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
_device.ReadDataOutput = () => ReadDataOutput();
|
||||
_device.ReadMotor = () => ReadMotor();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,34 +1,36 @@
|
|||
using System;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public abstract class CassettePortDevice
|
||||
{
|
||||
public Func<bool> ReadDataOutput;
|
||||
public Func<bool> ReadMotor;
|
||||
|
||||
public virtual void ExecutePhase2()
|
||||
{
|
||||
}
|
||||
|
||||
public virtual void HardReset()
|
||||
{
|
||||
}
|
||||
|
||||
public virtual bool ReadDataInputBuffer()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual bool ReadSenseBuffer()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
}
|
||||
}
|
||||
using System;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public abstract class CassettePortDevice
|
||||
{
|
||||
[SaveState.DoNotSave]
|
||||
public Func<bool> ReadDataOutput;
|
||||
[SaveState.DoNotSave]
|
||||
public Func<bool> ReadMotor;
|
||||
|
||||
public virtual void ExecutePhase2()
|
||||
{
|
||||
}
|
||||
|
||||
public virtual void HardReset()
|
||||
{
|
||||
}
|
||||
|
||||
public virtual bool ReadDataInputBuffer()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual bool ReadSenseBuffer()
|
||||
{
|
||||
return true;
|
||||
}
|
||||
|
||||
public virtual void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,40 +1,41 @@
|
|||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Cores.Computers.Commodore64.Media;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public class TapeDrive : CassettePortDevice
|
||||
{
|
||||
private Tape _tape;
|
||||
|
||||
public override void ExecutePhase2()
|
||||
{
|
||||
if (_tape != null && !ReadMotor()) _tape.ExecuteCycle();
|
||||
}
|
||||
|
||||
public override void HardReset()
|
||||
{
|
||||
if (_tape != null) _tape.Rewind();
|
||||
}
|
||||
|
||||
public override bool ReadDataInputBuffer()
|
||||
{
|
||||
return _tape == null || _tape.Read();
|
||||
}
|
||||
|
||||
public override bool ReadSenseBuffer()
|
||||
{
|
||||
return _tape == null;
|
||||
}
|
||||
|
||||
public override void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
|
||||
public void Insert(Tape tape)
|
||||
{
|
||||
_tape = tape;
|
||||
}
|
||||
}
|
||||
}
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Cores.Computers.Commodore64.Media;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Cassette
|
||||
{
|
||||
public class TapeDrive : CassettePortDevice
|
||||
{
|
||||
[SaveState.SaveWithName("Tape")]
|
||||
private Tape _tape;
|
||||
|
||||
public override void ExecutePhase2()
|
||||
{
|
||||
if (_tape != null && !ReadMotor()) _tape.ExecuteCycle();
|
||||
}
|
||||
|
||||
public override void HardReset()
|
||||
{
|
||||
if (_tape != null) _tape.Rewind();
|
||||
}
|
||||
|
||||
public override bool ReadDataInputBuffer()
|
||||
{
|
||||
return _tape == null || _tape.Read();
|
||||
}
|
||||
|
||||
public override bool ReadSenseBuffer()
|
||||
{
|
||||
return _tape == null;
|
||||
}
|
||||
|
||||
public override void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
|
||||
public void Insert(Tape tape)
|
||||
{
|
||||
_tape = tape;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,101 +1,103 @@
|
|||
using System;
|
||||
using System.Text;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media
|
||||
{
|
||||
/**
|
||||
* This class represents a tape. Only TAP-style tapes are supported for now.
|
||||
*/
|
||||
public class Tape
|
||||
{
|
||||
[SaveState.DoNotSave] private readonly byte[] _tapeData;
|
||||
[SaveState.DoNotSave] private readonly byte _version;
|
||||
[SaveState.DoNotSave] private readonly int _start;
|
||||
[SaveState.DoNotSave] private readonly int _end;
|
||||
private int _pos, _cycle;
|
||||
private bool _data;
|
||||
|
||||
public Tape(byte version, byte[] tapeData, int start, int end)
|
||||
{
|
||||
_version = version;
|
||||
_tapeData = tapeData;
|
||||
_start = start;
|
||||
_end = end;
|
||||
Rewind();
|
||||
}
|
||||
|
||||
public void ExecuteCycle()
|
||||
{
|
||||
if (_cycle == 0)
|
||||
{
|
||||
if (_pos >= _end)
|
||||
{
|
||||
_data = true;
|
||||
return;
|
||||
}
|
||||
|
||||
_cycle = _tapeData[_pos++] * 8;
|
||||
if (_cycle == 0)
|
||||
{
|
||||
if (_version == 0)
|
||||
{
|
||||
_cycle = 256 * 8; // unspecified overflow condition
|
||||
}
|
||||
else
|
||||
{
|
||||
_cycle = (int)(BitConverter.ToUInt32(_tapeData, _pos - 1) >> 8);
|
||||
_pos += 3;
|
||||
if (_cycle == 0)
|
||||
{
|
||||
throw new Exception("Bad tape data");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
_cycle++;
|
||||
}
|
||||
|
||||
// Send a single negative pulse at the end of a cycle
|
||||
_data = --_cycle != 0;
|
||||
}
|
||||
|
||||
// Rewinds the tape back to start
|
||||
public void Rewind()
|
||||
{
|
||||
_pos = _start;
|
||||
_cycle = 0;
|
||||
}
|
||||
|
||||
// Reads from tape, this will tell the caller if the flag pin should be raised
|
||||
public bool Read()
|
||||
{
|
||||
return _data;
|
||||
}
|
||||
|
||||
// Try to construct a tape file from file data. Returns null if not a tape file, throws exceptions for bad tape files.
|
||||
// (Note that some error conditions aren't caught right here.)
|
||||
public static Tape Load(byte[] tapeFile)
|
||||
{
|
||||
Tape result = null;
|
||||
|
||||
if (Encoding.ASCII.GetString(tapeFile, 0, 12) == "C64-TAPE-RAW")
|
||||
{
|
||||
var version = tapeFile[12];
|
||||
if (version > 1) throw new Exception("This tape has an unsupported version");
|
||||
var size = BitConverter.ToUInt32(tapeFile, 16);
|
||||
if (size + 20 != tapeFile.Length)
|
||||
{
|
||||
throw new Exception("Tape file header specifies a length that doesn't match the file size");
|
||||
}
|
||||
result = new Tape(version, tapeFile, 20, tapeFile.Length);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
}
|
||||
}
|
||||
using System;
|
||||
using System.Text;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.Commodore64.Media
|
||||
{
|
||||
public class Tape
|
||||
{
|
||||
[SaveState.DoNotSave] private readonly byte[] _tapeData;
|
||||
[SaveState.DoNotSave] private readonly byte _version;
|
||||
[SaveState.DoNotSave] private readonly int _start;
|
||||
[SaveState.DoNotSave] private readonly int _end;
|
||||
|
||||
[SaveState.SaveWithName("Position")]
|
||||
private int _pos;
|
||||
[SaveState.SaveWithName("Cycle")]
|
||||
private int _cycle;
|
||||
[SaveState.SaveWithName("Data")]
|
||||
private bool _data;
|
||||
|
||||
public Tape(byte version, byte[] tapeData, int start, int end)
|
||||
{
|
||||
_version = version;
|
||||
_tapeData = tapeData;
|
||||
_start = start;
|
||||
_end = end;
|
||||
Rewind();
|
||||
}
|
||||
|
||||
public void ExecuteCycle()
|
||||
{
|
||||
if (_cycle == 0)
|
||||
{
|
||||
if (_pos >= _end)
|
||||
{
|
||||
_data = true;
|
||||
return;
|
||||
}
|
||||
|
||||
_cycle = _tapeData[_pos++] * 8;
|
||||
if (_cycle == 0)
|
||||
{
|
||||
if (_version == 0)
|
||||
{
|
||||
_cycle = 256 * 8; // unspecified overflow condition
|
||||
}
|
||||
else
|
||||
{
|
||||
_cycle = (int)(BitConverter.ToUInt32(_tapeData, _pos - 1) >> 8);
|
||||
_pos += 3;
|
||||
if (_cycle == 0)
|
||||
{
|
||||
throw new Exception("Bad tape data");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
_cycle++;
|
||||
}
|
||||
|
||||
// Send a single negative pulse at the end of a cycle
|
||||
_data = --_cycle != 0;
|
||||
}
|
||||
|
||||
// Rewinds the tape back to start
|
||||
public void Rewind()
|
||||
{
|
||||
_pos = _start;
|
||||
_cycle = 0;
|
||||
}
|
||||
|
||||
// Reads from tape, this will tell the caller if the flag pin should be raised
|
||||
public bool Read()
|
||||
{
|
||||
return _data;
|
||||
}
|
||||
|
||||
// Try to construct a tape file from file data. Returns null if not a tape file, throws exceptions for bad tape files.
|
||||
// (Note that some error conditions aren't caught right here.)
|
||||
public static Tape Load(byte[] tapeFile)
|
||||
{
|
||||
Tape result = null;
|
||||
|
||||
if (Encoding.ASCII.GetString(tapeFile, 0, 12) == "C64-TAPE-RAW")
|
||||
{
|
||||
var version = tapeFile[12];
|
||||
if (version > 1) throw new Exception("This tape has an unsupported version");
|
||||
var size = BitConverter.ToUInt32(tapeFile, 16);
|
||||
if (size + 20 != tapeFile.Length)
|
||||
{
|
||||
throw new Exception("Tape file header specifies a length that doesn't match the file size");
|
||||
}
|
||||
result = new Tape(version, tapeFile, 20, tapeFile.Length);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
SaveState.SyncObject(ser, this);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue