272 lines
6.6 KiB
C#
272 lines
6.6 KiB
C#
using System;
|
|
using System.IO;
|
|
using System.Collections.Generic;
|
|
using System.Windows.Forms;
|
|
|
|
using BizHawk.Client.Common;
|
|
using BizHawk.Common.BufferExtensions;
|
|
using BizHawk.Emulation.Cores.PCEngine;
|
|
using BizHawk.Emulation.Common;
|
|
|
|
using ICSharpCode.SharpZipLib.Zip;
|
|
|
|
namespace BizHawk.Client.EmuHawk
|
|
{
|
|
public partial class PCESoundDebugger : Form, IToolFormAutoConfig
|
|
{
|
|
[RequiredService]
|
|
private PCEngine PCE { get; set; }
|
|
|
|
public PCESoundDebugger()
|
|
{
|
|
InitializeComponent();
|
|
|
|
SetStyle(ControlStyles.AllPaintingInWmPaint, true);
|
|
SetStyle(ControlStyles.UserPaint, true);
|
|
SetStyle(ControlStyles.OptimizedDoubleBuffer, true);
|
|
}
|
|
|
|
private readonly byte[] _waveformTemp = new byte[32 * 2];
|
|
|
|
protected override void OnShown(EventArgs e)
|
|
{
|
|
for (int i = 0; i < lvChEn.Items.Count; i++)
|
|
{
|
|
lvChEn.Items[i].Checked = true;
|
|
}
|
|
|
|
base.OnShown(e);
|
|
}
|
|
|
|
public void NewUpdate(ToolFormUpdateType type) { }
|
|
|
|
public void UpdateValues()
|
|
{
|
|
foreach (var entry in _psgEntries)
|
|
{
|
|
entry.WasActive = entry.Active;
|
|
entry.Active = false;
|
|
}
|
|
|
|
bool sync = false;
|
|
lvPsgWaveforms.BeginUpdate();
|
|
lvChannels.BeginUpdate();
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
{
|
|
var ch = PCE.PSG.Channels[i];
|
|
|
|
// these conditions mean a sample isn't playing
|
|
if (!ch.Enabled)
|
|
{
|
|
lvChannels.Items[i].SubItems[1].Text = "-";
|
|
lvChannels.Items[i].SubItems[2].Text = "-";
|
|
lvChannels.Items[i].SubItems[3].Text = "(disabled)";
|
|
goto DEAD;
|
|
}
|
|
if (ch.DDA)
|
|
{
|
|
lvChannels.Items[i].SubItems[1].Text = "-";
|
|
lvChannels.Items[i].SubItems[2].Text = "-";
|
|
lvChannels.Items[i].SubItems[3].Text = "(DDA)";
|
|
goto DEAD;
|
|
}
|
|
lvChannels.Items[i].SubItems[1].Text = ch.Volume.ToString();
|
|
lvChannels.Items[i].SubItems[2].Text = ch.Frequency.ToString();
|
|
if (ch.NoiseChannel)
|
|
{
|
|
lvChannels.Items[i].SubItems[3].Text = "(noise)";
|
|
goto DEAD;
|
|
}
|
|
|
|
if (ch.Volume == 0) goto DEAD;
|
|
|
|
lvChannels.Items[i].SubItems[3].Text = "-";
|
|
|
|
// ok, a sample is playing. copy out the waveform
|
|
short[] waveform = (short[])ch.Wave.Clone();
|
|
|
|
// hash it
|
|
var ms = new MemoryStream(_waveformTemp);
|
|
var bw = new BinaryWriter(ms);
|
|
foreach (var s in waveform)
|
|
{
|
|
bw.Write(s);
|
|
}
|
|
|
|
bw.Flush();
|
|
string md5 = _waveformTemp.HashMD5();
|
|
|
|
if (!_psgEntryTable.ContainsKey(md5))
|
|
{
|
|
var entry = new PsgEntry
|
|
{
|
|
Name = md5,
|
|
WaveForm = waveform,
|
|
Active = true,
|
|
HitCount = 1,
|
|
Index = _psgEntries.Count
|
|
};
|
|
_psgEntries.Add(entry);
|
|
_psgEntryTable[md5] = entry;
|
|
sync = true;
|
|
_lastSamples[i] = entry;
|
|
}
|
|
else
|
|
{
|
|
PsgEntry entry = _psgEntryTable[md5];
|
|
entry.Active = true;
|
|
|
|
// are we playing the same sample as before?
|
|
if (_lastSamples[i] != entry)
|
|
{
|
|
_lastSamples[i] = entry;
|
|
entry.HitCount++;
|
|
if (entry.Index < lvPsgWaveforms.Items.Count)
|
|
{
|
|
lvPsgWaveforms.Items[entry.Index].SubItems[1].Text = entry.HitCount.ToString();
|
|
}
|
|
else
|
|
{
|
|
sync = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
lvChannels.Items[i].SubItems[3].Text = _psgEntryTable[md5].Name;
|
|
|
|
continue;
|
|
|
|
DEAD:
|
|
_lastSamples[i] = null;
|
|
}
|
|
|
|
if (sync)
|
|
SyncLists();
|
|
lvPsgWaveforms.EndUpdate();
|
|
lvChannels.EndUpdate();
|
|
}
|
|
|
|
public void FastUpdate()
|
|
{
|
|
// Todo
|
|
}
|
|
|
|
private class PsgEntry
|
|
{
|
|
public int Index { get; set; }
|
|
public bool Active { get; set; }
|
|
public bool WasActive { get; set; }
|
|
public int HitCount { get; set; }
|
|
public string Name { get; set; }
|
|
public short[] WaveForm { get; set; }
|
|
}
|
|
|
|
private readonly PsgEntry[] _lastSamples = new PsgEntry[8];
|
|
private readonly List<PsgEntry> _psgEntries = new List<PsgEntry>();
|
|
private readonly Dictionary<string, PsgEntry> _psgEntryTable = new Dictionary<string, PsgEntry>();
|
|
|
|
public void Restart()
|
|
{
|
|
}
|
|
|
|
public bool AskSaveChanges() => true;
|
|
|
|
public bool UpdateBefore => false;
|
|
|
|
|
|
// 32*16 samples, 16bit, mono, 8khz (but we'll change the sample rate)
|
|
private static readonly byte[] EmptyWav = {
|
|
0x52, 0x49, 0x46, 0x46, 0x24, 0x04, 0x00, 0x00, 0x57, 0x41, 0x56, 0x45, 0x66, 0x6D, 0x74, 0x20,
|
|
0x10, 0x00, 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0xE0, 0x2E, 0x00, 0x00, 0xC0, 0x5D, 0x00, 0x00,
|
|
0x02, 0x00, 0x10, 0x00, 0x64, 0x61, 0x74, 0x61, 0x00, 0x04, 0x00, 0x00,
|
|
};
|
|
|
|
|
|
private void BtnExport_Click(object sender, EventArgs e)
|
|
{
|
|
string tmpFilename = $"{Path.GetTempFileName()}.zip";
|
|
using (var stream = new FileStream(tmpFilename, FileMode.Create, FileAccess.Write, FileShare.Read))
|
|
{
|
|
var zip = new ZipOutputStream(stream)
|
|
{
|
|
IsStreamOwner = false,
|
|
UseZip64 = UseZip64.Off
|
|
};
|
|
|
|
foreach (var entry in _psgEntries)
|
|
{
|
|
var ze = new ZipEntry($"{entry.Name}.wav") { CompressionMethod = CompressionMethod.Deflated };
|
|
zip.PutNextEntry(ze);
|
|
var ms = new MemoryStream();
|
|
var bw = new BinaryWriter(ms);
|
|
bw.Write(EmptyWav, 0, EmptyWav.Length);
|
|
ms.Position = 0x18; // samplerate and avgbytespersecond
|
|
bw.Write(20000);
|
|
bw.Write(20000 * 2);
|
|
bw.Flush();
|
|
ms.Position = 0x2C;
|
|
for (int i = 0; i < 32; i++)
|
|
{
|
|
for (int j = 0; j < 16; j++)
|
|
{
|
|
bw.Write(entry.WaveForm[i]);
|
|
}
|
|
}
|
|
|
|
bw.Flush();
|
|
var buf = ms.GetBuffer();
|
|
zip.Write(buf, 0, (int)ms.Length);
|
|
zip.Flush();
|
|
zip.CloseEntry();
|
|
}
|
|
|
|
zip.Close();
|
|
stream.Flush();
|
|
}
|
|
System.Diagnostics.Process.Start(tmpFilename);
|
|
}
|
|
|
|
private void BtnReset_Click(object sender, EventArgs e)
|
|
{
|
|
_psgEntryTable.Clear();
|
|
_psgEntries.Clear();
|
|
for (int i = 0; i < 8; i++) _lastSamples[i] = null;
|
|
SyncLists();
|
|
}
|
|
|
|
private void SyncLists()
|
|
{
|
|
lvPsgWaveforms.Items.Clear();
|
|
foreach (var entry in _psgEntries)
|
|
{
|
|
var lvi = new ListViewItem(entry.Name);
|
|
lvi.SubItems.Add(entry.HitCount.ToString());
|
|
lvPsgWaveforms.Items.Add(lvi);
|
|
}
|
|
}
|
|
|
|
private void lvPsgWaveforms_KeyDown(object sender, KeyEventArgs e)
|
|
{
|
|
if (e.KeyCode == Keys.F2 && lvPsgWaveforms.SelectedItems.Count > 0)
|
|
{
|
|
lvPsgWaveforms.SelectedItems[0].BeginEdit();
|
|
}
|
|
}
|
|
|
|
private void lvPsgWaveforms_AfterLabelEdit(object sender, LabelEditEventArgs e)
|
|
{
|
|
var entry = _psgEntries[e.Item];
|
|
entry.Name = e.Label;
|
|
}
|
|
|
|
private void lvChEn_ItemChecked(object sender, ItemCheckedEventArgs e)
|
|
{
|
|
for (int i = 0; i < 6; i++)
|
|
{
|
|
PCE.PSG.UserMute[i] = !lvChEn.Items[i].Checked;
|
|
}
|
|
}
|
|
}
|
|
}
|