-
Notifications
You must be signed in to change notification settings - Fork 78
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
Crash in PortAudioDebugRedirector #230
Comments
This looks like an attempt to create two separate instances of the FlexASIO driver class in the same process, i.e. initializing FlexASIO twice in a row (without destroying the first instance first). FlexASIO doesn't support that - it only supports up to 1 driver instance at any given time in a given process. This is just a quick guess just from looking at the stack trace though - I haven't taken a close look at the user code. Do you think that's what may be happening in the offending app? It may be useful to gather a log to shed more light into what may be going on - if FlexASIO is being initialized twice, that will definitely be visible in the log. |
That's exactly what it was, thanks! Any particular reason for this design decision? |
It's not really a "design decision", per se, just a consequence of FlexASIO using global state in a few places, mostly because PortAudio itself uses global state. Which is sad, but that's the way it is. That being said, IMHO from the perspective of the driver this behavior is perfectly reasonable and cannot be considered a driver bug. Indeed, the ASIO SDK does not allow more than one driver instance to exist at any given time - it uses a global variable to store the instance and the client API is very strongly designed to revolve around a single global instance (as evidenced by the fact that ASIO client API functions do not even take a driver instance as a parameter). The ASIO SDK defines the ASIO spec. Therefore, if the ASIO SDK makes it impossible to have more than one driver instance at any given time, then it logically follows that a spec-compliant ASIO driver is allowed to assume it will not be instantiated more than once at any given time in a given process. For this reason, I do not see this as a bug in FlexASIO - it is strictly abiding by the ASIO spec. The reason why you're seeing this problem is because your code does not go through the ASIO SDK client library code - it bypasses it and interacts with the driver directly. If you were going through the ASIO SDK, this issue would be impossible by construction. Now, if you want to bypass the ASIO SDK and talk to the driver directly, then that's fine, but drivers will still expect you to abide by ASIO SDK rules, so you need to make sure that your code never interacts with an ASIO driver in a way that contradicts the ASIO SDK. I could consider a request to make FlexASIO go "above and beyond" the ASIO spec and tolerate this particular usage pattern, but to be clear that would be considered a feature request, not a bugfix. (It may also not be a trivial one - in particular I'd be worried about the thread safety implications, given PortAudio itself uses global state and is infamously thread-hostile.) But that wouldn't really "fix" the broader problem, which is that your code makes unwarranted assumptions about what ASIO drivers will tolerate. I can "fix" FlexASIO, but that would still leave open the possibility that other ASIO drivers have the same "issue", so you may still face compatibility issues with other drivers. For this reason, a better approach would be for you to fix it on your side, i.e. make it so that your library will never attempt to create multiple concurrent instances of an ASIO driver. This way, you can guarantee compatibility with ASIO-compliant drivers, as opposed to hoping they will accept non-compliant usage. |
Just to be clear, I'm with you on this. You play by the rules, and I don't. Which is exactly the reason I bypass the SDK and go for IASIO directly. << as evidenced by the fact that ASIO client API functions do not even take a driver instance as a parameter << I could consider a request to make FlexASIO go "above and beyond" the ASIO spec << For this reason, a better approach would be for you to fix it on your side Thanks for your very detailed response, and I think this is a fun discussion really :) And to be very clear, xt-audio does not require to instantiate multiple IASIO drivers simultaneously. Users can very much code against it while still adhering to the asio spec. Just open one driver at a time. It's just the gui sample application that wants to open multiple of the same drivers (https://sjoerdvankreel.github.io/xt-audio/img/gui.png). The xt-audio library itself is fine (depending on usage pattern of course). Anyway, all of the multi-instance stuff turned out to be a red herring. I wrote up a sample program over here sjoerdvankreel/xt-audio#26 that uses FlexASIO no problem (single instance of course). For me, at least. User of my library takes a STATUS_STACK_BUFFER_OVERRUN on that sample code. I'm hoping for them to run a debug build of xt-audio to narrow down the problem. I'm also trying to get my hands on a debug FlexASIO build using git clone --recursive, run the cmake build, compile with visual studio. But no luck, see below. Any clue?
|
For the record, I will definitely agree that many aspects of ASIO can be… problematic. This reminds me of this fun little experience I had when writing the first version of the driver: FlexASIO/src/flexasio/FlexASIO/cflexasio.cpp Lines 22 to 25 in 57e14e8
There is also the fact that the ASIO SDK docs around buffer management are so confusing that I ended up having to write my own doc to clarify what I think they meant. (You may find that doc of interest to you, by the way. Feedback is welcome.)
Just FYI, loading multiple ASIO drivers at the same time can also cause its own problems - drivers can interfere with each other, especially if multiple drivers try to use the same hardware at the same time (e.g. a universal audio driver such as FlexASIO or ASIO4ALL interfering with a device's own bespoke ASIO driver). See also #86. That being said, if the set of ASIO drivers being loaded is under the control of the end user, then I guess it's up to them to make sure they don't select a problematic combination of drivers.
Oh wow that's… impressive, I'll give you that!
My suggestion would be to still allow your users to instantiate different drivers at the same time (as that can indeed be very useful, and is unlikely to cause problems that the end user wouldn't know how to fix), but maybe return an error if the user attempts to instantiate the same driver multiple times? This would compel your users to structure their code in such a way that this doesn't happen. Of course, it may also make sense to simply keep things as is and just document the issue so that your users are aware. I can think of only one "clean" way to have multiple concurrent instances of one ASIO driver in a single process: by somehow tricking the Windows DLL loader to load the same DLL multiple times at different points of the memory address space, such that the various "instances" of the ASIO driver DLL wouldn't share the same global state. (FlexASIO would definitely work in that case.) I suspect this may be possible, perhaps using activation contexts, or somehow loading the same DLL under different file names, or something like that. I thought maybe you'd like to give it a try, given you seem to have a penchant for impressively clever hacks… By the way, another reason to refrain from initializing ASIO drivers multiple times (concurrently or otherwise) is that some of them can actually take a while to initialize. Sadly that includes FlexASIO, which can take a while because PortAudio itself can take a while, especially if there are many audio devices to enumerate.
I don't recognize that error. I get the impression you are trying to generate a Visual Studio project from CMake? That's never going to work - the project must be built using Visual Studio's native CMake support (i.e. "Open folder", then build using CMake inside VS). You can also take inspiration from the CI workflow: FlexASIO/.github/workflows/continuous-integration.yml Lines 30 to 32 in 6376094
That being said, it's been a while since I've worked on FlexASIO, I haven't tried to build it recently. I just gave it a go and the only real breakage right now is the broken ASIO SDK download hash in ASIOUtil (dechamps/ASIOUtil@b445f3f), which I just fixed on the |
<< This reminds me of this fun little experience I had when writing the first version of the driver: Oh man that's nasty. Glad I never got to experience that, being "on the other side". << There is also the fact that the ASIO SDK docs around buffer management are so confusing that I ended up having to write my own doc to clarify what I think they meant. (You may find that doc of interest to you, by the way. Feedback is welcome.) Doc seems good to me. Reminds me of me fighting ALSA's hw-direct memory-mapped buffer mode. Eventually i got a good enough understanding of it to help someone out, see https://stackoverflow.com/questions/36906981/is-mmap-what-i-need-from-alsa-to-play-simultaneous-immediate-sounds-in-my-game/37309825#37309825. But honestly I don't expect any modern day ASIO driver to come at you with directProcess=false. That's for really low IRQ levels in case you cannot process the buffer in-place. I mean I do understand their API design but you gotta remember this is 1990-s level stuff. Honestly I cannot envision any present day ASIO driver doing that. Everything these days is threaded. In fact I don't even support the !directProcess case in xt-audio, never had a complaint about it. See https://github.com/sjoerdvankreel/xt-audio/blob/d34abe36d3b789bdc3f30ce92179afa33d32d382/src/core/xt/xt/backend/asio/Stream.cpp#L70. << Just FYI, loading multiple ASIO drivers at the same time can also cause its own problems - drivers can interfere with each other, especially if multiple drivers try to use the same hardware at the same time << Oh wow that's… impressive, I'll give you that! << My suggestion would be to still allow your users to instantiate different drivers at the same time (as that can indeed be very useful, and is unlikely to cause problems that the end user wouldn't know how to fix), but maybe return an error if the user attempts to instantiate the same driver multiple times? This makes sense, but it's also not a panacea. For example my old e-mu card came with a dedicated asio driver with a full-blown mixer GUI allowing to route various in-out with per-channel volume controls etc. And it worked perfectly with multiple processes hitting the same asio driver, so i see no reason why it wouldn't work with a single process hitting that same driver. I'm 100% convinced it has an internal audio mixer and only hits the device after that. << Of course, it may also make sense to simply keep things as is and just document the issue so that your users are aware. This is my most likely course of action :) This problem is really not confined to ASIO. "Once upon a time", DirectSound was hardware-direct. Same for ASIO. Same for Linux ALSA. Any modern distro, the default ALSA driver feeds back into the PulseAudio sound mixer server which in turn feeds back into the first ALSA hardware device. That's even counting out ALSA's own distinction between HW:xyz and DMix:xyz. And WASAPI's exclusive/shared distinction. Today you cannot really tell what driver will "take over the device" just by looking at the API being used. << I can think of only one "clean" way to have multiple concurrent instances of one ASIO driver in a single process I think, stuff is complicated enough already as it is :) Mind you, apart from the XtGui sample (in which I just dropped the ball, honestly), there's no real need to fire up the same driver twice. << By the way, another reason to refrain from initializing ASIO drivers multiple times (concurrently or otherwise) is that some of them can actually take a while to initialize This is not limited to ASIO. WASAPI exhibits the same problem. Especially with USB drivers. All in all, I don't see an immediate course of action for xt-audio here. Audio API landscape has changed quite a bit since i wrote that library. For one, i cannot assume that ASIO=exclusive mode. And for that reason I also cannot assume 1 ASIO driver per process. It would benefit some use cases and hurt some others. And like you, "That being said, it's been a while since I've worked on FlexASIO", same goes for xt-audio. << I get the impression you are trying to generate a Visual Studio project from CMake? That's never going to work - the project must be built using Visual Studio's native CMake support Again spot-on, that's exactly what I did. I'll try to give it another go sometime soon. Also please note like I said above, this whole multi-instance thing is NOT the root cause of my user's issue! I thinks it's fun and all to talk about it, but please don't feel obliged in any way. The current issue is some sort of memory corruption. Can be either xt-audio of FlexASIO. Once I figure it out, I'll get back to you (or not, in case it's xt-audio). |
Well, sure, but the consequences are different. Typically, if you try to open a device that is already opened in exclusive mode somewhere else, you'll almost certainly get a clean error code back. In contrast, if you create multiple instances of the same ASIO driver in the same process, technically that's undefined behavior, so the driver is allowed to do anything in response, including crashing the entire process (which is what FlexASIO does). I think I'll actually take an action item here, and that would be to at least make FlexASIO return a clean error with a nice error message on
Understood. We can look at the other issue separately, and feel free to file another ticket if it turns out the root cause is in FlexASIO. |
<< Well, sure, but the consequences are different. Typically, if you try to open a device that is already opened in exclusive mode somewhere else, you'll almost certainly get a clean error code back. In contrast, if you create multiple instances of the same ASIO driver in the same process, technically that's undefined behavior, so the driver is allowed to do anything in response, including crashing the entire process (which is what FlexASIO does). Agreed. Actually, if you open multiple instances of a different driver in the same process, that's already allowed to crash per-the-spec. But as I said, multiple different drivers is my main use case. Although I'm still not sure if i should actively prohibit multiple of the same drivers. Some can handle it, some don't. It's a "nice to have" when the driver let's you do that. OTOH, it leads to github tickets like these when it doesn't. To be honest, I probably won't do anything about it untill I take a github issue that's actually caused by this. But this one ain't it (meaning I don't find the gui sample demo important enough to go fixing stuff, seeing as the library proper can be used perfectly fine, within bounds). Closing this issue, if the other one turns out to be in FlexASIO, i'll open a new one. And congrats on this nice piece of work, btw. I'm very much supportive of anything audio-related :) |
Let me just keep this open - I'm using this issue to remind me to make FlexASIO degrade more gracefully when the host attempts to instantiate multiple times (i.e. return an error instead of crashing the process). |
So, this is hilarious, but while doing some cleaning up of old issues, I stumbled upon #183 which is exactly the same issue (multiple FlexASIO instances being created)… and I even already wrote the code to fix it on the @sjoerdvankreel You'll be pleased to know that the new code actually makes FlexASIO support multiple instances - as in, it actually works, as opposed to what I wrote earlier when I said I was planning to make it return an error on initialization. Looks like I took a closer look back then and concluded that it was actually fine to allow multiple instances. I still wouldn't recommend doing that though, especially in concurrent code (see #185). |
This is fixed in FlexASIO 1.10, which now allows multiple FlexASIO instances to exist simultaneously in the same process. |
Hi!
I'm trying to track down a problem of a user of my library xt-audio over here: sjoerdvankreel/xt-audio#26.
That project is pretty much like PortAudio, i.e. a single API with multiple backends. When FlexASIO is plugged in as an ASIO backend,
i get a crash caused by an abort() over here:
FlexASIO/src/flexasio/FlexASIOUtil/portaudio.cpp
Line 43 in 57e14e8
Any clue as to what may be wrong here? See stacktrace below. This is for a .NET/winforms app, but I doubt if that's the problem.
The text was updated successfully, but these errors were encountered: