forked from mozilla/uniffi-rs
-
Notifications
You must be signed in to change notification settings - Fork 1
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
The initial motivation for this was cancellation. PR#1697 made it so if an async function was cancelled we would eventually release the resources. However, it would be better if we could immedately release the resources. In order to implement that, I realized I needed to change the future FFI quite a bit. The new FFI is simpler overall and supports cancel and drop operations. Cancel ensures that the foreign code will resume and break out of its async code. Drop ensures that all resources from the wrapped future are relased. The new code does not use ForeignExecutor anymore, so that code is in a state of limbo. I hope to repurpose it for foreign dispatch queues (mozilla#1734). If that doesn't work out, we can just delete it. The FFI calls need some care since we pass a type-erased handle to the foreign code, while RustFuture is generic over an anonymous Future type: - The concrete RustFuture type implements the RustFutureFFI. - We give the foreign side a raw Box<Arc<dyn RustFutureFFI>>>. This makes it easy to call the FFI from the scaffolding code. The extra Box turns the wide pointer into a normal sized pointer, which simplifies the foreign code. - The last bit of complexity is completing a Rust future, since the future can output any of the FFI types that we return from sync functions. Handled this with a brute-force monomorphization. The scaffolding code defines a completion function for each FFI type and the bindings code has to call the correct one for the future. More changes: - Updated the futures fixture tests for this to hold on to the mutex longer in the initial call. This makes it so they will fail unless the future is dropped while the mutex is still locked. Before they would only succeed as long as the mutex was dropped once the timeout expired. - Updated `RustCallStatus.code` field to be an enum. Added `Cancelled` as one of the variants. `Cancelled` is only used for async functions. - Removed the FutureCallback and invoke_future_callback from `FfiConverter`. - New syncronization handling code in RustFuture that's hopefully clearer, more correct, and more understandable than the old stuff. - Updated `UNIFFI_CONTRACT_VERSION` since this is an ABI change - Removed the `RustCallStatus` param from async scaffolding functions. These functions can't fail, so there's no need. - Added is_async() to the Callable trait.
- Loading branch information
Showing
52 changed files
with
1,305 additions
and
1,056 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
// Async return type handlers | ||
|
||
internal const val UNIFFI_RUST_FUTURE_POLL_READY = 0.toShort() | ||
internal const val UNIFFI_RUST_FUTURE_POLL_MAYBE_READY = 1.toShort() | ||
|
||
internal val uniffiContinuationHandleMap = UniFfiHandleMap<CancellableContinuation<Short>>() | ||
|
||
// FFI type for Rust future continuations | ||
internal object uniffiRustFutureContinuation: UniFffiRustFutureContinutationType { | ||
override fun callback(continuationHandle: USize, pollResult: Short) { | ||
uniffiContinuationHandleMap.remove(continuationHandle)?.resume(pollResult) | ||
} | ||
} | ||
|
||
internal suspend fun<T, F, E: Exception> uniffiRustCallAsync( | ||
rustFuture: Pointer, | ||
completeFunc: (Pointer, RustCallStatus) -> F, | ||
liftFunc: (F) -> T, | ||
errorHandler: CallStatusErrorHandler<E> | ||
): T { | ||
try { | ||
do { | ||
val pollResult = suspendCancellableCoroutine<Short> { continuation -> | ||
_UniFFILib.INSTANCE.{{ ci.ffi_rust_future_poll().name() }}( | ||
rustFuture, | ||
uniffiRustFutureContinuation, | ||
uniffiContinuationHandleMap.insert(continuation) | ||
) | ||
} | ||
} while (pollResult != UNIFFI_RUST_FUTURE_POLL_READY); | ||
|
||
return liftFunc( | ||
rustCallWithError(errorHandler, { status -> completeFunc(rustFuture, status) }) | ||
) | ||
} finally { | ||
_UniFFILib.INSTANCE.{{ ci.ffi_rust_future_free().name() }}(rustFuture) | ||
} | ||
} |
Oops, something went wrong.