ZXHawk: Added wav tape image support
This commit is contained in:
parent
e2b58cfb98
commit
625f063861
|
@ -66,7 +66,8 @@ namespace BizHawk.Client.Common
|
|||
{
|
||||
RomData = FileData;
|
||||
}
|
||||
else if (file.Extension == ".DSK" || file.Extension == ".TAP" || file.Extension == ".TZX" || file.Extension == ".PZX" || file.Extension == ".CSW")
|
||||
else if (file.Extension == ".DSK" || file.Extension == ".TAP" || file.Extension == ".TZX" ||
|
||||
file.Extension == ".PZX" || file.Extension == ".CSW" || file.Extension == ".WAV")
|
||||
{
|
||||
// these are not roms. unforunately if treated as such there are certain edge-cases
|
||||
// where a header offset is detected. This should mitigate this issue until a cleaner solution is found
|
||||
|
|
|
@ -51,7 +51,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
return new[]
|
||||
{
|
||||
".NES", ".FDS", ".UNF", ".SMS", ".GG", ".SG", ".GB", ".GBC", ".GBA", ".PCE", ".SGX", ".BIN", ".SMD", ".GEN", ".MD", ".SMC", ".SFC", ".A26", ".A78", ".LNX", ".COL", ".ROM", ".M3U", ".CUE", ".CCD", ".SGB", ".Z64", ".V64", ".N64", ".WS", ".WSC", ".XML", ".DSK", ".DO", ".PO", ".PSF", ".MINIPSF", ".NSF",
|
||||
".EXE", ".PRG", ".D64", "*G64", ".CRT", ".TAP", ".32X", ".MDS", ".TZX", ".PZX", ".CSW"
|
||||
".EXE", ".PRG", ".D64", "*G64", ".CRT", ".TAP", ".32X", ".MDS", ".TZX", ".PZX", ".CSW", ".WAV"
|
||||
};
|
||||
}
|
||||
|
||||
|
|
|
@ -2081,7 +2081,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
if (VersionInfo.DeveloperBuild)
|
||||
{
|
||||
return FormatFilter(
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.pce;*.sgx;*.bin;*.smd;*.rom;*.a26;*.a78;*.lnx;*.m3u;*.cue;*.ccd;*.mds;*.exe;*.gb;*.gbc;*.gba;*.gen;*.md;*.32x;*.col;*.int;*.smc;*.sfc;*.prg;*.d64;*.g64;*.crt;*.tap;*.sgb;*.xml;*.z64;*.v64;*.n64;*.ws;*.wsc;*.dsk;*.do;*.po;*.vb;*.ngp;*.ngc;*.psf;*.minipsf;*.nsf;*.tzx;*.pzx;*.csw;%ARCH%",
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.pce;*.sgx;*.bin;*.smd;*.rom;*.a26;*.a78;*.lnx;*.m3u;*.cue;*.ccd;*.mds;*.exe;*.gb;*.gbc;*.gba;*.gen;*.md;*.32x;*.col;*.int;*.smc;*.sfc;*.prg;*.d64;*.g64;*.crt;*.tap;*.sgb;*.xml;*.z64;*.v64;*.n64;*.ws;*.wsc;*.dsk;*.do;*.po;*.vb;*.ngp;*.ngc;*.psf;*.minipsf;*.nsf;*.tzx;*.pzx;*.csw;*.wav;%ARCH%",
|
||||
"Music Files", "*.psf;*.minipsf;*.sid;*.nsf",
|
||||
"Disc Images", "*.cue;*.ccd;*.mds;*.m3u",
|
||||
"NES", "*.nes;*.fds;*.unf;*.nsf;%ARCH%",
|
||||
|
@ -2109,7 +2109,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
"Apple II", "*.dsk;*.do;*.po;%ARCH%",
|
||||
"Virtual Boy", "*.vb;%ARCH%",
|
||||
"Neo Geo Pocket", "*.ngp;*.ngc;%ARCH%",
|
||||
"Sinclair ZX Spectrum", "*.tzx;*.tap;*.dsk;*.pzx;*.csw;%ARCH%",
|
||||
"Sinclair ZX Spectrum", "*.tzx;*.tap;*.dsk;*.pzx;*.csw;*.wav;%ARCH%",
|
||||
"All Files", "*.*");
|
||||
}
|
||||
|
||||
|
|
|
@ -306,11 +306,11 @@ namespace BizHawk.Emulation.Common
|
|||
case ".TZX":
|
||||
case ".PZX":
|
||||
case ".CSW":
|
||||
case ".WAV":
|
||||
game.System = "ZXSpectrum";
|
||||
break;
|
||||
|
||||
case ".TAP":
|
||||
|
||||
case ".TAP":
|
||||
byte[] head = romData.Take(8).ToArray();
|
||||
if (System.Text.Encoding.Default.GetString(head).Contains("C64-TAPE"))
|
||||
game.System = "C64";
|
||||
|
|
|
@ -299,6 +299,10 @@
|
|||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\TapeDataBlock.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\TAP\TapConverter.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\TZX\TzxConverter.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\WAV\StreamHelper.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\WAV\WavConverter.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\WAV\WavHeader.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Media\Tape\WAV\WavStreamReader.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\SoundProviderMixer.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Machine\MachineType.cs" />
|
||||
<Compile Include="Computers\SinclairSpectrum\Machine\SpectrumBase.cs" />
|
||||
|
|
|
@ -325,6 +325,7 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
TapConverter tapSer = new TapConverter(this);
|
||||
PzxConverter pzxSer = new PzxConverter(this);
|
||||
CswConverter cswSer = new CswConverter(this);
|
||||
WavConverter wavSer = new WavConverter(this);
|
||||
|
||||
// TZX
|
||||
if (tzxSer.CheckType(tapeData))
|
||||
|
@ -386,6 +387,26 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
}
|
||||
}
|
||||
|
||||
// WAV
|
||||
else if (wavSer.CheckType(tapeData))
|
||||
{
|
||||
// this file has a csw header - attempt serialization
|
||||
try
|
||||
{
|
||||
wavSer.Read(tapeData);
|
||||
// reset block index
|
||||
CurrentDataBlockIndex = 0;
|
||||
return;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
// exception during operation
|
||||
var e = ex;
|
||||
throw new Exception(this.GetType().ToString() +
|
||||
"\n\nTape image file has a valid WAV header, but threw an exception whilst data was being parsed.\n\n" + e.ToString());
|
||||
}
|
||||
}
|
||||
|
||||
// Assume TAP
|
||||
else
|
||||
{
|
||||
|
@ -838,7 +859,9 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
{
|
||||
if (_tapeIsPlaying && _autoPlay)
|
||||
{
|
||||
if (DataBlocks.Count > 1 || _dataBlocks[_currentDataBlockIndex].BlockDescription != BlockType.CSW_Recording)
|
||||
if (DataBlocks.Count > 1 ||
|
||||
(_dataBlocks[_currentDataBlockIndex].BlockDescription != BlockType.CSW_Recording &&
|
||||
_dataBlocks[_currentDataBlockIndex].BlockDescription != BlockType.WAV_Recording))
|
||||
{
|
||||
// we should only stop the tape when there are multiple blocks
|
||||
// if we just have one big block (maybe a CSW or WAV) then auto stopping will cock things up
|
||||
|
@ -882,8 +905,12 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
return;
|
||||
|
||||
// dont autostop if there is only 1 block
|
||||
if (DataBlocks.Count > 2 || _dataBlocks[_currentDataBlockIndex].BlockDescription == BlockType.CSW_Recording)
|
||||
if (DataBlocks.Count == 1 || _dataBlocks[_currentDataBlockIndex].BlockDescription == BlockType.CSW_Recording ||
|
||||
_dataBlocks[_currentDataBlockIndex].BlockDescription == BlockType.WAV_Recording
|
||||
)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
if (diff >= timeout * 2)
|
||||
{
|
||||
|
|
|
@ -207,6 +207,11 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
// spectrum .tzx tape file
|
||||
return SpectrumMediaType.Tape;
|
||||
}
|
||||
if (hdr.ToUpper().Contains("WAVE"))
|
||||
{
|
||||
// spectrum .tzx tape file
|
||||
return SpectrumMediaType.Tape;
|
||||
}
|
||||
|
||||
// if we get this far, assume a .tap file
|
||||
return SpectrumMediaType.Tape;
|
||||
|
|
|
@ -11,6 +11,7 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
TAP,
|
||||
PZX,
|
||||
CSW,
|
||||
WAV,
|
||||
DSK
|
||||
}
|
||||
}
|
||||
|
|
|
@ -242,6 +242,8 @@ namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
|||
|
||||
// zxhawk proprietry
|
||||
PAUSE_BLOCK,
|
||||
|
||||
WAV_Recording
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -0,0 +1,104 @@
|
|||
using System;
|
||||
using System.IO;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
||||
{
|
||||
/// <summary>
|
||||
/// From https://archive.codeplex.com/?p=zxmak2
|
||||
/// </summary>
|
||||
public static class StreamHelper
|
||||
{
|
||||
public static void Write(Stream stream, Int32 value)
|
||||
{
|
||||
byte[] data = BitConverter.GetBytes(value);
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, UInt32 value)
|
||||
{
|
||||
byte[] data = BitConverter.GetBytes(value);
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, Int16 value)
|
||||
{
|
||||
byte[] data = BitConverter.GetBytes(value);
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, UInt16 value)
|
||||
{
|
||||
byte[] data = BitConverter.GetBytes(value);
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, Byte value)
|
||||
{
|
||||
byte[] data = new byte[1];
|
||||
data[0] = value;
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, SByte value)
|
||||
{
|
||||
byte[] data = new byte[1];
|
||||
data[0] = (byte)value;
|
||||
stream.Write(data, 0, data.Length);
|
||||
}
|
||||
|
||||
public static void Write(Stream stream, byte[] value)
|
||||
{
|
||||
stream.Write(value, 0, value.Length);
|
||||
}
|
||||
|
||||
|
||||
public static void Read(Stream stream, out Int32 value)
|
||||
{
|
||||
byte[] data = new byte[4];
|
||||
stream.Read(data, 0, data.Length);
|
||||
//if (!BitConverter.IsLittleEndian)
|
||||
// Array.Reverse(data);
|
||||
value = BitConverter.ToInt32(data, 0);
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, out UInt32 value)
|
||||
{
|
||||
byte[] data = new byte[4];
|
||||
stream.Read(data, 0, data.Length);
|
||||
value = BitConverter.ToUInt32(data, 0);
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, out Int16 value)
|
||||
{
|
||||
byte[] data = new byte[2];
|
||||
stream.Read(data, 0, data.Length);
|
||||
value = BitConverter.ToInt16(data, 0);
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, out UInt16 value)
|
||||
{
|
||||
byte[] data = new byte[2];
|
||||
stream.Read(data, 0, data.Length);
|
||||
value = BitConverter.ToUInt16(data, 0);
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, out Byte value)
|
||||
{
|
||||
byte[] data = new byte[1];
|
||||
stream.Read(data, 0, data.Length);
|
||||
value = data[0];
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, out SByte value)
|
||||
{
|
||||
byte[] data = new byte[1];
|
||||
stream.Read(data, 0, data.Length);
|
||||
value = (sbyte)data[0];
|
||||
}
|
||||
|
||||
public static void Read(Stream stream, byte[] value)
|
||||
{
|
||||
stream.Read(value, 0, value.Length);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,130 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
||||
{
|
||||
/// <summary>
|
||||
/// Reponsible for WAV format conversion
|
||||
/// Based heavily on code from zxmak2: https://archive.codeplex.com/?p=zxmak2
|
||||
/// </summary>
|
||||
public class WavConverter : MediaConverter
|
||||
{
|
||||
/// <summary>
|
||||
/// The type of serializer
|
||||
/// </summary>
|
||||
private MediaConverterType _formatType = MediaConverterType.WAV;
|
||||
public override MediaConverterType FormatType
|
||||
{
|
||||
get
|
||||
{
|
||||
return _formatType;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Signs whether this class can be used to read the data format
|
||||
/// </summary>
|
||||
public override bool IsReader { get { return true; } }
|
||||
|
||||
/// <summary>
|
||||
/// Signs whether this class can be used to write the data format
|
||||
/// </summary>
|
||||
public override bool IsWriter { get { return false; } }
|
||||
|
||||
/// <summary>
|
||||
/// Position counter
|
||||
/// </summary>
|
||||
private int _position = 0;
|
||||
|
||||
#region Construction
|
||||
|
||||
private DatacorderDevice _datacorder;
|
||||
|
||||
public WavConverter(DatacorderDevice _tapeDevice)
|
||||
{
|
||||
_datacorder = _tapeDevice;
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
/// <summary>
|
||||
/// Returns TRUE if pzx header is detected
|
||||
/// </summary>
|
||||
/// <param name="data"></param>
|
||||
public override bool CheckType(byte[] data)
|
||||
{
|
||||
// WAV Header
|
||||
|
||||
// check whether this is a valid wav format file by looking at the identifier in the header
|
||||
string ident = Encoding.ASCII.GetString(data, 8, 4);
|
||||
|
||||
if (ident.ToUpper() != "WAVE")
|
||||
{
|
||||
// this is not a valid WAV format file
|
||||
return false;
|
||||
}
|
||||
else
|
||||
{
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// DeSerialization method
|
||||
/// </summary>
|
||||
/// <param name="data"></param>
|
||||
public override void Read(byte[] data)
|
||||
{
|
||||
// clear existing tape blocks
|
||||
_datacorder.DataBlocks.Clear();
|
||||
|
||||
// check whether this is a valid pzx format file by looking at the identifier in the header block
|
||||
string ident = Encoding.ASCII.GetString(data, 8, 4);
|
||||
|
||||
if (ident.ToUpper() != "WAVE")
|
||||
{
|
||||
// this is not a valid TZX format file
|
||||
throw new Exception(this.GetType().ToString() +
|
||||
"This is not a valid WAV format file");
|
||||
}
|
||||
|
||||
_position = 0;
|
||||
|
||||
MemoryStream stream = new MemoryStream();
|
||||
stream.Write(data, 0, data.Length);
|
||||
stream.Position = 0;
|
||||
|
||||
WavStreamReader reader = new WavStreamReader(stream);
|
||||
|
||||
int rate = (69888 * 50) / reader.Header.sampleRate;
|
||||
int smpCounter = 0;
|
||||
int state = reader.ReadNext();
|
||||
|
||||
// create the single tape block
|
||||
TapeDataBlock t = new TapeDataBlock();
|
||||
t.BlockDescription = BlockType.WAV_Recording;
|
||||
t.BlockID = 0;
|
||||
t.DataPeriods = new List<int>();
|
||||
|
||||
for (int i = 0; i < reader.Count; i++)
|
||||
{
|
||||
int sample = reader.ReadNext();
|
||||
smpCounter++;
|
||||
if ((state < 0 && sample < 0) || (state >= 0 && sample >= 0))
|
||||
continue;
|
||||
t.DataPeriods.Add(smpCounter * rate);
|
||||
smpCounter = 0;
|
||||
state = sample;
|
||||
}
|
||||
|
||||
// add closing period
|
||||
t.DataPeriods.Add((69888 * 50) / 10);
|
||||
|
||||
// add to datacorder
|
||||
_datacorder.DataBlocks.Add(t);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,107 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
||||
{
|
||||
/// <summary>
|
||||
/// From https://archive.codeplex.com/?p=zxmak2
|
||||
/// </summary>
|
||||
public class WavHeader
|
||||
{
|
||||
// RIFF chunk (12 bytes)
|
||||
public Int32 chunkID; // "RIFF"
|
||||
public Int32 fileSize;
|
||||
public Int32 riffType; // "WAVE"
|
||||
|
||||
// Format chunk (24 bytes)
|
||||
public Int32 fmtID; // "fmt "
|
||||
public Int32 fmtSize;
|
||||
public Int16 fmtCode;
|
||||
public Int16 channels;
|
||||
public Int32 sampleRate;
|
||||
public Int32 fmtAvgBPS;
|
||||
public Int16 fmtBlockAlign;
|
||||
public Int16 bitDepth;
|
||||
public Int16 fmtExtraSize;
|
||||
|
||||
// Data chunk
|
||||
public Int32 dataID; // "data"
|
||||
public Int32 dataSize; // The data size should be file size - 36 bytes.
|
||||
|
||||
|
||||
public void Deserialize(Stream stream)
|
||||
{
|
||||
StreamHelper.Read(stream, out chunkID);
|
||||
StreamHelper.Read(stream, out fileSize);
|
||||
StreamHelper.Read(stream, out riffType);
|
||||
if (chunkID != BitConverter.ToInt32(Encoding.ASCII.GetBytes("RIFF"), 0))
|
||||
{
|
||||
throw new FormatException("Invalid WAV file header");
|
||||
}
|
||||
if (riffType != BitConverter.ToInt32(Encoding.ASCII.GetBytes("WAVE"), 0))
|
||||
{
|
||||
throw new FormatException(string.Format(
|
||||
"Not supported RIFF type: '{0}'",
|
||||
Encoding.ASCII.GetString(BitConverter.GetBytes(riffType))));
|
||||
}
|
||||
Int32 chunkId;
|
||||
Int32 chunkSize;
|
||||
while (stream.Position < stream.Length)
|
||||
{
|
||||
StreamHelper.Read(stream, out chunkId);
|
||||
StreamHelper.Read(stream, out chunkSize);
|
||||
string strChunkId = Encoding.ASCII.GetString(
|
||||
BitConverter.GetBytes(chunkId));
|
||||
if (strChunkId == "fmt ")
|
||||
{
|
||||
read_fmt(stream, chunkId, chunkSize);
|
||||
}
|
||||
else if (strChunkId == "data")
|
||||
{
|
||||
read_data(stream, chunkId, chunkSize);
|
||||
break;
|
||||
}
|
||||
else
|
||||
{
|
||||
stream.Seek(chunkSize, SeekOrigin.Current);
|
||||
}
|
||||
}
|
||||
if (fmtID != BitConverter.ToInt32(Encoding.ASCII.GetBytes("fmt "), 0))
|
||||
{
|
||||
throw new FormatException("WAV format chunk not found");
|
||||
}
|
||||
if (dataID != BitConverter.ToInt32(Encoding.ASCII.GetBytes("data"), 0))
|
||||
{
|
||||
throw new FormatException("WAV data chunk not found");
|
||||
}
|
||||
}
|
||||
|
||||
private void read_data(Stream stream, int chunkId, int chunkSize)
|
||||
{
|
||||
dataID = chunkId;
|
||||
dataSize = chunkSize;
|
||||
}
|
||||
|
||||
private void read_fmt(Stream stream, int chunkId, int chunkSize)
|
||||
{
|
||||
fmtID = chunkId;
|
||||
fmtSize = chunkSize;
|
||||
StreamHelper.Read(stream, out fmtCode);
|
||||
StreamHelper.Read(stream, out channels);
|
||||
StreamHelper.Read(stream, out sampleRate);
|
||||
StreamHelper.Read(stream, out fmtAvgBPS);
|
||||
StreamHelper.Read(stream, out fmtBlockAlign);
|
||||
StreamHelper.Read(stream, out bitDepth);
|
||||
if (fmtSize == 18)
|
||||
{
|
||||
// Read any extra values
|
||||
StreamHelper.Read(stream, out fmtExtraSize);
|
||||
stream.Seek(fmtExtraSize, SeekOrigin.Current);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,117 @@
|
|||
using System;
|
||||
using System.IO;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Computers.SinclairSpectrum
|
||||
{
|
||||
/// <summary>
|
||||
/// From https://archive.codeplex.com/?p=zxmak2
|
||||
/// </summary>
|
||||
public class WavStreamReader
|
||||
{
|
||||
private Stream m_stream;
|
||||
private WavHeader m_header = new WavHeader();
|
||||
|
||||
public WavStreamReader(Stream stream)
|
||||
{
|
||||
m_stream = stream;
|
||||
m_header.Deserialize(stream);
|
||||
}
|
||||
|
||||
public WavHeader Header { get { return m_header; } }
|
||||
|
||||
public int Count { get { return m_header.dataSize / m_header.fmtBlockAlign; } }
|
||||
|
||||
public Int32 ReadNext()
|
||||
{
|
||||
// check - sample should be in PCM format
|
||||
if (m_header.fmtCode != WAVE_FORMAT_PCM &&
|
||||
m_header.fmtCode != WAVE_FORMAT_IEEE_FLOAT)
|
||||
{
|
||||
throw new FormatException(string.Format(
|
||||
"Not supported audio format: fmtCode={0}, bitDepth={1}",
|
||||
m_header.fmtCode,
|
||||
m_header.bitDepth));
|
||||
}
|
||||
byte[] data = new byte[m_header.fmtBlockAlign];
|
||||
m_stream.Read(data, 0, data.Length);
|
||||
if (m_header.fmtCode == WAVE_FORMAT_PCM)
|
||||
{
|
||||
// use first channel only
|
||||
if (m_header.bitDepth == 8)
|
||||
return getSamplePcm8(data, 0, 0);
|
||||
if (m_header.bitDepth == 16)
|
||||
return getSamplePcm16(data, 0, 0);
|
||||
if (m_header.bitDepth == 24)
|
||||
return getSamplePcm24(data, 0, 0);
|
||||
if (m_header.bitDepth == 32)
|
||||
return getSamplePcm32(data, 0, 0);
|
||||
}
|
||||
else if (m_header.fmtCode == WAVE_FORMAT_IEEE_FLOAT)
|
||||
{
|
||||
// use first channel only
|
||||
if (m_header.bitDepth == 32)
|
||||
return getSampleFloat32(data, 0, 0);
|
||||
if (m_header.bitDepth == 64)
|
||||
return getSampleFloat64(data, 0, 0);
|
||||
}
|
||||
throw new NotSupportedException(string.Format(
|
||||
"Not supported audio format ({0}/{1} bit)",
|
||||
m_header.fmtCode == WAVE_FORMAT_PCM ? "PCM" : "FLOAT",
|
||||
m_header.bitDepth));
|
||||
}
|
||||
|
||||
private Int32 getSamplePcm8(byte[] bufferRaw, int offset, int channel)
|
||||
{
|
||||
return bufferRaw[offset + channel] - 128;
|
||||
}
|
||||
|
||||
private Int32 getSamplePcm16(byte[] bufferRaw, int offset, int channel)
|
||||
{
|
||||
return BitConverter.ToInt16(bufferRaw, offset + 2 * channel);
|
||||
}
|
||||
|
||||
private Int32 getSamplePcm24(byte[] bufferRaw, int offset, int channel)
|
||||
{
|
||||
Int32 result;
|
||||
int subOffset = offset + channel * 3;
|
||||
if (BitConverter.IsLittleEndian)
|
||||
{
|
||||
result = ((sbyte)bufferRaw[2 + subOffset]) * 0x10000;
|
||||
result |= bufferRaw[1 + subOffset] * 0x100;
|
||||
result |= bufferRaw[0 + subOffset];
|
||||
}
|
||||
else
|
||||
{
|
||||
result = ((sbyte)bufferRaw[0 + subOffset]) * 0x10000;
|
||||
result |= bufferRaw[1 + subOffset] * 0x100;
|
||||
result |= bufferRaw[2 + subOffset];
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
private Int32 getSamplePcm32(byte[] bufferRaw, int offset, int channel)
|
||||
{
|
||||
return BitConverter.ToInt32(bufferRaw, offset + 4 * channel);
|
||||
}
|
||||
|
||||
private Int32 getSampleFloat32(byte[] data, int offset, int channel)
|
||||
{
|
||||
float fSample = BitConverter.ToSingle(data, offset + 4 * channel);
|
||||
// convert to 32 bit integer
|
||||
return (Int32)(fSample * Int32.MaxValue);
|
||||
}
|
||||
|
||||
private Int32 getSampleFloat64(byte[] data, int offset, int channel)
|
||||
{
|
||||
double fSample = BitConverter.ToDouble(data, offset + 8 * channel);
|
||||
// convert to 32 bit integer
|
||||
return (Int32)(fSample * Int32.MaxValue);
|
||||
}
|
||||
|
||||
private const int WAVE_FORMAT_PCM = 1; /* PCM */
|
||||
private const int WAVE_FORMAT_IEEE_FLOAT = 3; /* IEEE float */
|
||||
private const int WAVE_FORMAT_ALAW = 6; /* 8-bit ITU-T G.711 A-law */
|
||||
private const int WAVE_FORMAT_MULAW = 7; /* 8-bit ITU-T G.711 µ-law */
|
||||
private const int WAVE_FORMAT_EXTENSIBLE = 0xFFFE; /* Determined by SubFormat */
|
||||
}
|
||||
}
|
|
@ -11,7 +11,7 @@ ZXHawk is still in dev but is potentially nearing a release state.
|
|||
* Full beeper and AY-3-3912 sound emulation
|
||||
* Tape device (datacorder) emulation
|
||||
* Internal 3" disk drive emulation (found in the +3 model)
|
||||
* Currently supports the following tape image formats: *.tzx, *.tap
|
||||
* Currently supports the following tape image formats: *.tzx, *.tap, *.pzx, *.csw, *.wav
|
||||
* Currently supports the following disk image formats (+3 only): *.dsk
|
||||
* Fully integrated into the Bizhawk ecosystem
|
||||
* See the ZXSpectrum menu for all available configuration options
|
||||
|
|
Loading…
Reference in New Issue