From 701d340e0001d7102dafac716274a72b45c2bca5 Mon Sep 17 00:00:00 2001 From: feos Date: Sat, 8 May 2021 21:01:29 +0300 Subject: [PATCH] mame: probably fix audio --- .../Arcades/MAME/MAME.ISoundProvider.cs | 31 +++++-------------- .../Arcades/MAME/MAME.IStatable.cs | 2 ++ .../Arcades/MAME/MAME.cs | 5 ++- 3 files changed, 11 insertions(+), 27 deletions(-) diff --git a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.ISoundProvider.cs b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.ISoundProvider.cs index 29db6cbf8f..a7ea4e92a1 100644 --- a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.ISoundProvider.cs +++ b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.ISoundProvider.cs @@ -11,9 +11,8 @@ namespace BizHawk.Emulation.Cores.Arcades.MAME public SyncSoundMode SyncMode => SyncSoundMode.Sync; private readonly Queue _audioSamples = new Queue(); - private decimal _dAudioSamples = 0; private readonly int _sampleRate = 44100; - private int _numSamples = 0; + private long _soundRemainder = 0; public void SetSyncMode(SyncSoundMode mode) { @@ -34,31 +33,14 @@ namespace BizHawk.Emulation.Cores.Arcades.MAME * I'm doing my own logic here for now. I grab MAME's audio buffer * whenever it's filled (MAMESoundCallback()) and enqueue it. * - * Whenever Hawk wants new audio, I dequeue it, but with a little quirk. - * Since sample count per frame may not align with frame duration, I - * subtract the entire decimal fraction of "required" samples from total - * samples. I check if the fractional reminder of total samples is > 0.5 - * by rounding it. I invert it to see what number I should add to the - * integer representation of "required" samples, to compensate for - * misalignment between fractional and integral "required" samples. - * - * TODO: Figure out how MAME does this and maybe use their method instead. + * Whenever Hawk wants new audio, I dequeue it, while preserving the + * fractinal part of the sample count, to use it later. */ public void GetSamplesSync(out short[] samples, out int nsamp) { - decimal dSamplesPerFrame = (decimal)_sampleRate * VsyncDenominator / VsyncNumerator; - - if (_audioSamples.Any()) - { - _dAudioSamples -= dSamplesPerFrame; - int remainder = (int)Math.Round(_dAudioSamples - Math.Truncate(_dAudioSamples)) ^ 1; - nsamp = (int)Math.Round(dSamplesPerFrame) + remainder; - } - else - { - nsamp = (int)Math.Round(dSamplesPerFrame); - } - + long nSampNumerator = _sampleRate * (long)VsyncDenominator + _soundRemainder; + nsamp = (int)(nSampNumerator / VsyncNumerator); + _soundRemainder = nSampNumerator % VsyncNumerator; // exactly remember fractional parts of an audio sample samples = new short[nsamp * 2]; for (int i = 0; i < nsamp * 2; i++) @@ -81,6 +63,7 @@ namespace BizHawk.Emulation.Cores.Arcades.MAME public void DiscardSamples() { + _soundRemainder = 0; _audioSamples.Clear(); } } diff --git a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.IStatable.cs b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.IStatable.cs index 10ba6dec80..e36093fa00 100644 --- a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.IStatable.cs +++ b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.IStatable.cs @@ -51,6 +51,8 @@ namespace BizHawk.Emulation.Cores.Arcades.MAME Frame = reader.ReadInt32(); LagCount = reader.ReadInt32(); IsLagFrame = reader.ReadBoolean(); + + DiscardSamples(); } } } \ No newline at end of file diff --git a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.cs b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.cs index adcd588cb3..ce8a139de3 100644 --- a/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.cs +++ b/src/BizHawk.Emulation.Cores/Arcades/MAME/MAME.cs @@ -258,15 +258,14 @@ namespace BizHawk.Emulation.Cores.Arcades.MAME return; } - _numSamples = lengthInBytes / bytesPerSample; + int numSamples = lengthInBytes / bytesPerSample; unsafe { short* pSample = (short*)ptr.ToPointer(); - for (int i = 0; i < _numSamples; i++) + for (int i = 0; i < numSamples; i++) { _audioSamples.Enqueue(*(pSample + i)); - _dAudioSamples++; } }