-
Notifications
You must be signed in to change notification settings - Fork 20
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 lifetimes to objects that contain callbacks #30
Conversation
BoxedCallback now takes a lifetime parameter, which means that the callback is no longer required to be 'static. Lifetime parameters are added to Client, InputPort, and VirtualDestination.
@depp thank you very much for the contribution. I'm not very proficient with lifetimes and I'd like other users to have a look into this too. @Boddlnagg, @jasongrlicky I'd love if you could have a look and help me asses the consequences of this change. I thought that the |
Pinging @kangalioo, because this sounds similar to Boddlnagg/midir#44 |
The problem is that Rust allows leaking memory in safe code. Users can |
While not explicitly prohibited, this would make it impossible to free memory used by callbacks. A quick test adding sleep to the callback confirms this—when you destroy the port, Core MIDI will block until the callback finishes.
I guess this change is no-go, if there is no guarantee that destructors are ever called. If we created the threads, we could at least choose to create them with something like crossbeam, but as it stands I have no idea how to get my app to work unless I start throwing |
Closing because I see no way to do this safely. |
@depp if it helps, I have never needed to do something like what you show in your example, because MIDI callbacks happen in a real-time thread, which gives you very few margin to do things (no blocking calls at all, no allocation, no OS calls, no mutexes, ...). Then the only code that I write in the callback is for parsing the message and sending it through a ring buffer, for another thread to process it. Please see my code here. In that specific example, I pass the ownership of a handler to my MIDI driver, but the handler basically does what I told you, parsing and pushing to the ring buffer, then the data can be used somewhere else. I hope it helps. |
@Boddlnagg, @kangalioo thank you very much for your help. |
The example code is just illustrative. The problem is that it means that any callback has to communicate with the rest of the program through globals or unsafe pointer casts. But I understand now that this is a fundamental limitation of Rust's type system and just something you have to deal with. |
This is probably a breaking change, so it needs a version bump.
Consider the following code:
With the existing code,
callback
must beFnMut + 'static
, but this is not really necessary. We just needcallback
to outlive theInputPort
.With the new code,
callback
can beFnMut + 'a
and give you anInputPort<'a>
. The same change is made to other types that contain callbacks.