-
Notifications
You must be signed in to change notification settings - Fork 424
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add low latency initialisation support on windows by directly interacting with WASAPI #6088
Conversation
135c201
to
1fa2610
Compare
871372e
to
973400c
Compare
5e12673
to
f3c39e8
Compare
From a code perspective, this is now ready for checking. Note that I haven't tested things still work after refactoring just yet. |
/// If a global mixer is being used, this will be the BASS handle for it. | ||
/// If non-null, all game mixers should be added to this mixer. | ||
/// </summary> | ||
internal readonly IBindable<int?> GlobalMixerHandle = new Bindable<int?>(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I get this may be a bit messy, but I don't want to over-complicate things. In the future we probably still want to revisit having a global mixer in all cases (#4915) at which point this will become the norm, not the "exception" for windows-only.
I could have added the global mixer in this PR but I felt that's probably a bad move to limit potential breakage.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding a global parent mixer is probably fine by itself, but what I worry about more is the changes in flags that the presence of this global mixer forces (BassFlags.Decode
, BassFlags.MixerChanBuffer | BassFlags.MixerChanNoRampin
).
I'd probably expect more commentary in xmldoc here, such that if the global mixer is present, then it basically takes over the duties of playback and all child mixers do not play on their own but rather expose audio data which the global pulls from as required. It seems like a major paradigm shift that isn't really documented anywhere (in fact I'm attempting to back-reason this currently using bass docs, but I'm not really completely sure if I read the intent of setting the aforementioned flags correctly).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
but I'm not really completely sure if I read the intent of setting the aforementioned flags correctly
the flags are required and it's all just bass nuances. i can document it but at the end of the day all i'll be doing is saying something like "this is done this way because it works". ie. without the decode flag it just breaks everything.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I thought there was some more nuance to it other than "make bass work in this configuration". If that is the case then I guess I can just not care.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Most of this stuff was trial and error in an isolated project until shit just worked 😅 💦
Co-authored-by: Bastian Pedersen <[email protected]>
internal static bool InitDevice(int deviceId) | ||
#region BASS Initialisation | ||
|
||
// TODO: All this bass init stuff should probably not be in this class. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure whether this is something to be actioned at this time or left for later but I most definitely agree. Should be in its own class, preferably behind a [SupportedOSPlatform("windows")]
attribute.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I could move it now, but was planning to leave for a follow-up effort, because I guarantee that will increase review overhead ten-fold.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, I'll look away for now then.
/// If a global mixer is being used, this will be the BASS handle for it. | ||
/// If non-null, all game mixers should be added to this mixer. | ||
/// </summary> | ||
internal readonly IBindable<int?> GlobalMixerHandle = new Bindable<int?>(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Adding a global parent mixer is probably fine by itself, but what I worry about more is the changes in flags that the presence of this global mixer forces (BassFlags.Decode
, BassFlags.MixerChanBuffer | BassFlags.MixerChanNoRampin
).
I'd probably expect more commentary in xmldoc here, such that if the global mixer is present, then it basically takes over the duties of playback and all child mixers do not play on their own but rather expose audio data which the global pulls from as required. It seems like a major paradigm shift that isn't really documented anywhere (in fact I'm attempting to back-reason this currently using bass docs, but I'm not really completely sure if I read the intent of setting the aforementioned flags correctly).
I've added some more commentary in the xmldoc, see what you think. |
…pleted" sync This fixes an issue identified with the WASAPI implementation in ppy/osu-framework#6088. It has no real effect on current `master`, but fixes a deadlock that occurs with the aforementioned framework branch when one lets a preview track play out to the end - at this point all audio will stop and an attempt to perform any synchronous BASS operation (playing another track, seeking) will result in a deadlock. It isn't terribly clear as to why this is happening precisely, but there does not appear to be any need to stop and seek at that point, so this feels like a decent workaround even if the actual issue is upstream (and will unblock pushing out WASAPI support to users).
Switching default output (either by audio settings, or by disconnecting / reconnecting a device) does not work by default for me. I have a rough version that makes it sorta work: diff --git a/osu.Framework/Audio/AudioManager.cs b/osu.Framework/Audio/AudioManager.cs
index 07e63484e..fd8f87f40 100644
--- a/osu.Framework/Audio/AudioManager.cs
+++ b/osu.Framework/Audio/AudioManager.cs
@@ -162,7 +162,8 @@ public AudioManager(AudioThread audioThread, ResourceStore<byte[]> trackStore, R
thread.RegisterManager(this);
- AudioDevice.ValueChanged += onDeviceChanged;
+ AudioDevice.ValueChanged += _ => onDeviceChanged();
+ GlobalMixerHandle.ValueChanged += _ => onDeviceChanged();
AddItem(TrackMixer = createAudioMixer(null, nameof(TrackMixer)));
AddItem(SampleMixer = createAudioMixer(null, nameof(SampleMixer)));
@@ -221,9 +222,9 @@ protected override void Dispose(bool disposing)
base.Dispose(disposing);
}
- private void onDeviceChanged(ValueChangedEvent<string> args)
+ private void onDeviceChanged()
{
- scheduler.Add(() => setAudioDevice(args.NewValue));
+ scheduler.Add(() => setAudioDevice(AudioDevice.Value));
}
private void onDevicesChanged()
diff --git a/osu.Framework/Threading/AudioThread.cs b/osu.Framework/Threading/AudioThread.cs
index 439cf4d16..7e02937ca 100644
--- a/osu.Framework/Threading/AudioThread.cs
+++ b/osu.Framework/Threading/AudioThread.cs
@@ -121,6 +121,7 @@ protected override void OnExit()
// TODO: All this bass init stuff should probably not be in this class.
private WasapiProcedure? wasapiProcedure;
+ private WasapiNotifyProcedure? wasapiNotifyProcedure;
/// <summary>
/// If a global mixer is being used, this will be the BASS handle for it.
@@ -212,7 +213,11 @@ private void attemptWasapiInitialisation()
// To keep things in a sane state let's only keep one device initialised via wasapi.
freeWasapi();
+ initWasapi(wasapiDevice);
+ }
+ private void initWasapi(int wasapiDevice)
+ {
// This is intentionally initialised inline and stored to a field.
// If we don't do this, it gets GC'd away.
wasapiProcedure = (buffer, length, _) =>
@@ -222,13 +227,30 @@ private void attemptWasapiInitialisation()
return Bass.ChannelGetData(globalMixerHandle.Value!.Value, buffer, length);
};
+ wasapiNotifyProcedure = (notify, device, _) =>
+ {
+ if (notify == WasapiNotificationType.DefaultOutput)
+ {
+ freeWasapi();
+ initWasapi(device);
+ }
+ };
+
+ bool initialised = BassWasapi.Init(wasapiDevice, Procedure: wasapiProcedure, Buffer: 0.02f, Period: 0.005f);
- if (BassWasapi.Init(wasapiDevice, Procedure: wasapiProcedure, Buffer: 0.02f, Period: 0.005f))
+ if (!initialised)
{
- BassWasapi.GetInfo(out var wasapiInfo);
- globalMixerHandle.Value = BassMix.CreateMixerStream(wasapiInfo.Frequency, wasapiInfo.Channels, BassFlags.MixerNonStop | BassFlags.Decode | BassFlags.Float);
- BassWasapi.Start();
+ if (Bass.LastError == Errors.Already)
+ BassWasapi.CurrentDevice = wasapiDevice;
+ else
+ return;
}
+
+ BassWasapi.GetInfo(out var wasapiInfo);
+ globalMixerHandle.Value = BassMix.CreateMixerStream(wasapiInfo.Frequency, wasapiInfo.Channels, BassFlags.MixerNonStop | BassFlags.Decode | BassFlags.Float);
+ BassWasapi.Start();
+
+ BassWasapi.SetNotify(wasapiNotifyProcedure);
}
private void freeWasapi()
@@ -237,7 +259,8 @@ private void freeWasapi()
// The mixer probably doesn't need to be recycled. Just keeping things sane for now.
Bass.StreamFree(globalMixerHandle.Value.Value);
- BassWasapi.Free();
+ BassWasapi.Stop();
+ //BassWasapi.Free();
globalMixerHandle.Value = null;
} but:
so I dunno. I'm leaving it there, this may need more serious restructuring. Definitely think that the lack of usage of |
I also got the following with my above broken patch once:
so if you're gonna use that for something, both of the bass callbacks should probably be created once ever. |
With this PR, opening the audio mixer visualiser (Ctrl+F9) makes the tracks and samples speed up really fast. Closing the mixer visualiser restores it back to normal. The "speed up" is tied to the audio thread update rate, eg. in single threaded, the audio goes faster as you cycle trough Vsync, 2x, 4x, 8x. Or if you unfocus the window. In Loud noise warning when you test this! |
Fwiw when testing osu!-side with this framework PR (on windows), I do need to set a universal offset of around |
@Morilli generally yes we know |
Fixed by 1d2f64c |
1d2f64c
to
28eb229
Compare
This reduces latency from perceivable (25-40ms) to non-perceivable (5-15ms). With this change, the latency on windows actually becomes limited by the update thread rate.
Pushing this as a draft for those interested in testing it ahead of time. I hope to clean up the single
static
usage.If you wish to test in the mean time, please checkout this PR, build osu! after running
UseLocalFramework.ps1
.