-
Notifications
You must be signed in to change notification settings - Fork 28.3k
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
Expose Flutter Engine address to be used with FFI #163430
Comments
@HosseinYousefi Would this help with dart-lang/native#1865 (comment) ? Also @stuartmorgan might have ideas about the interaction between FFI code and the engine. |
Yes, that's exactly what I need as well! |
#110353 has some earlier related discussion, but is focused on the plugin API surface specifically. There's been some discussion that wasn't captured there about how exactly we would vend things, given that the types are platform-specific. I like this handle proposal; it sidesteps the problems of what types we would use, how we would do generation of shared objects without adding framework dependencies on the |
Capturing from Discord, Chinmay pointed out that there will be some inherent danger for anyone using this system. E.g., if someone writes a plugin right now against the plugin registry APIs, and then those change, then someone trying to use the plugin with a new version of Flutter will get compilation errors. If they write equivalent code using FFI generation of the Flutter plugin API surface, getting to the instance via this handle, then if the API surface changes it will instead explode at runtime. The only way I see to avoid that would be if we did FFI generation as part of the framework, but then we get back to all of the same problems mentioned in my previous comment. |
@stuartmorgan, I think the danger mentioned was mostly for my original proposal which would be providing raw engine pointer directly instead of handle. That was probably not the best idea. Using handle instead means that you have to call actual embedder API to get the engine pointer. And if you manage to do that, chances are you are able to call the other engine method safely as well. Our embedder API is so far has been pretty stable. I would not expect this to change much. Once #163476 and #162935 land (and the new threading model gets enabled by default, which would be the next PR) it should be possible to write a file picker plugin completely in Dart. |
How would one follow from the other? The fact that the static get-engine-from-handle function still exists doesn't mean that the entire public embedding API on that platform is unchanged since the package did FFI generation.
Sure, but "pretty stable" and formally stable are not the same thing. For plugin-visible APIs in particular we try to be very conservative with breaking changes, because they are potentially very disruptive to the ecosystem, but they do still happen. Android's removal of the v1 embedding APIs, for instance.
And doing so on will (on several platforms) involve a call to a method that we will be deprecating for multi-view support. A native plugin could get deprecation warnings from annotations, and if we do ever remove that deprecated method (unlikely in this particular case, but possible) a native plugin would get a compile-time error. A Dart version, on the other hand, would never get any deprecation warning, and would fail at runtime if we removed the method. I'm not saying this is a blocker, but it is the case that the Dart version of this code would have pitfalls that the native version does not, and in providing this API we need to recognize that we will be creating new failures modes that will not be obvious to most developers, and we need to consider the consequences. |
We should maintain and ship the generated bindings for Flutter APIs either as a separate package or as part of the SDK to avoid this problem. The |
Well, yes, that is a downside with FFI and something to keep in mind. Fortunately the API surface is not huge. And for plugins written completely in Dart we would probably want to have official Dart bindings for engine. |
That creates a different set of problems, because either way it requires the SDK to exactly pin any dependencies ( |
I'm not sure I understand why shipping a flutter engine bindings package separately would require the SDK to have dependencies to jni/objective_c? |
We could expose a |
Whether it's technically a separate package or not is basically irrelevant here. Packages cannot have upper bounds on the Flutter SDK, so a Flutter-team-maintained bindings package only addresses the stability problem if the Flutter SDK depends on, and pins the exact version of, that package. Otherwise we just move the stability problem from one package to another.
That doesn't address the issue I'm talking about. Any package that uses FFI generation from the non-stable Flutter embedding APIs is, inherently, tied to that version of Flutter (or a compatible version, which we can't predict in advance). Either we enforce that tie, which can only be done with a dependency from the SDK, or we don't in which case a breaking change to the embedding API looks like one of the following We follow semver in the bindings packages:
We don't follow semver in the bindings packages:
|
I see. How much of an issue is this in practice though? The biggest change coming is multiwindow, where we're going to add API to get a FlutterView from the engine by viewId. However the original method that assumes single view will still be implemented, return a valid result and possibly print a warning message about being deprecated. The API surface we need to expose to plugins is pretty small (get engine, get view by Id, and possibly get texture registry). I don't see much potential for major braking changes, at least not in a way where we couldn't maintain compatibility while providing deprecation warnings at least at runtime). |
OK, this explains it. |
Hard to say. I don't expect it to be common certainly, but I don't know how much more disruptive it will be when it does happen due to the failure mode being harder to understand. The compile-time version of this kind of thing is very easy to find matching errors for, for instance, while I expect the runtime version to be much more opaque to the vast majority of developers.
Only because this conversation is happening after the Android v1 embedding change, where we had to remove symbols. I'm not saying these things are common, but we have a recent example of a worst-case kind of change happening, so I don't think we should assume they will never happen again.
If we provide official bindings packages and steer people very strongly toward using them, then the surface is as small as the plugin API surface (which is definitely bigger than what you listed; there are a ton of delegation methods on iOS, activity lifecycle and activity handler result callbacks on Android, etc.) If we don't, and most developers are doing their own FFI generation, then the effective surface will be the entire embedding API surface, because nothing will indicate to anyone that it shouldn't be. |
There are some things (life cycle, platform views) that require writing an actual plugin currently. I don't think we'll be able to implement those in pure dart for a while (and without same changes in the architecture). But there is quite a bit of utility that can be obtained, especially on desktop, with modest API and hopefully reasonably small potential for breaking. More importantly, getting the engine pointer from FFI is something that's already possible and being done, just with tradeoffs where it either requres asynchronous API or fails with multiple engines. |
We have addressed this issue in Dart with having
Our thinking for the Dart C API is to ship it as an SDK package (so pinned). dart-lang/native#839 (comment) This package should both have the Dart generated bindings, the C header file, and a version file ( The failure mode for users using the Dart API should be a Dart compilation error on a breaking change. The failure mode for users using the C API and the native assets feature depends on what the build hook does:
I believe ordinary plugins (not using the native assets feature) should also have compilation errors. Is there a case I'm overlooking?
I am not sure I understand this. Do you mean that the stability issue shows up as a compile-time error rather than resolution forcing the right version? But pinning Or do you mean something else? @jonasfj Has proposed being able specify package incompatibilities after the fact between an SDK and a package if the SDK introduced a breaking change. That would move the compile-time error to a pub resolution error for packages that we own and know are incompatible. (Basically adding a Flutter or Dart SDK upper bound to a package after the fact.) |
It is, but the step from "it is possible" to "we actively encourage it by creating an API just to do that thing" is an important one in terms of what expectations we are setting, and how much responsibility we have to guide people away from pitfalls that the new API might otherwise encourage. |
I'm not sure what this question means exactly, given that it's at the end of your description of a proposal to use a pinned package, and the context of the quote you are asking about is that I was explaining why an unpinned package would be problematic. To summarize my position again:
|
Right, counterpoint to that is that if somebody can figure out how to call native methods on engine through FFI, they will likely also figure out how to get engine instance, just in a less robust way than what we can provide. So you end up with plugins that fail with multiple engines for example (like package:jni right now). |
I have requested this in the past, but I haven't seen any indication that it's likely to happen in the foreseeable future; unless that changes we can't plan around it. |
I don't see this as a counterpoint, personally. Making something work better will encourage use relative to the status quo, just as making it easier will encourage use. |
The way I see it - it's a hypothetical problem (we might break the subset of engine API used in FFI plugins in future in a way where we can't provide backwards compatibility or build failure) versus an actual problem (plugins built against package:jni won't work with multiple engines). In any case, it seems that this
would solve the issue? If, with bit of luck, it happens before the API break. |
Thanks for the clarification @stuartmorgan!
Unpinned could be a working model if we have have the retroactive package incompatibilities feature on pub.
I am not sure I understand this. Wouldn't |
I think this is in the case where we choose to expose say a Edit: I read the original comment again. Yeah, if we're not exposing jni/objc types then there is no need to have dependencies to the packages, we just use |
Before you opening this issue, I planned to have a map from isolate group ids to contexts since each engine will create a different isolate group but it seems like we don't have a |
Is that the case? I assumed In any case, when the plugin is initialized the isolate is not running (thread local is not set), is it? So I'm not sure how you'd pair to either isolate or isolate group. |
I thought that was the case. Not super familiar with Flutter engine, so you are likely right.
Good point. So it seems that we really need the Flutter engine address. |
You were talking about the Dart C API in your comment. Almost all of the previous discussion in this issue, including the original issue description and including the comment of mine that you pulled out in the first place, is about the Flutter embedding APIs (not If you take my comments about a proposal to make packages to wrap Obj-C and Java API surfaces, and place them out of context in a different discussion about wrapping a pure C API, then yes, they don't make sense.
Let's make this concrete. This and this comprise the macOS embedding API surface, including This comment suggests having third-party package developers calling those APIs from Dart, in their packages. I pointed out that this would be somewhat dangerous, because they would be shipping generated code for an API surface that is not fully stable, and that resulting failures would be runtime (vs. the traditional plugin model, where the failures are compile time). This comment proposed that the solution to that could be that "We should maintain and ship the generated bindings for Flutter APIs either as a separate package or as part of the SDK". An example of a Flutter API that we would need to expose in the proposed Flutter-team-maintained bindings package in order to enable the example of a Dart-only file picker package while solving the runtime stability concern is this method, which returns an |
I don't agree that that's an accurate description of the actual problem. As you said in the initial issue description: "One options is to write a Flutter plugin, that [...] uses platform channels to let the isolate access engine address and then possibly use it through FFI calls." I have built a plugin that does that; it works fine. I would say that the problem you are trying to solve is that it is currently not possible to build a package that is multi-engine aware using only Dart, and only synchronous calls. It would be nice to enable that, but that's very different than "plugins won't work". And when we are talking about adding APIs to enable new functionality, we need to consider the downsides of the functionality we are creating. ETA: Since I feel like we're going around in circles on this bit, let me reframe my view on this as a question: what is the use case (not approach to writing plugins, but actual app client user journey) that cannot be addressed with the current technology, and is so time critical that we cannot spend more than a couple of days discussing entirely foreseeable, long term implications of adding this new API--and how we might mitigate them--before landing this? |
I don't think I quite get the point. Everything this proposal describes is possible today, is already done today, but with the cost of requiring asynchronous call, which objectively makes the resulting API worse. All the downsides that you are describing already exist. The only change here is a net benefit - allowing to provide synchronous API and not relying on a plugin to capture engine instance. Yes, if we make this more convenient, chances are more people will use it, but if we don't, people will do it anyway, just badly (like assuming a single engine). This is not to say that it's not worth addressing the downsides that you are describing.
To fix package:jni without this proposal would require making a breaking change and would make the API objectively worse. Also writing dart-only plugins was an example. It's not the only purpose of the proposal, or even a primary one for us. For example I'm currently working on an experimental multi-window support in the macOS embedder that does not rely on platform channels. And there are other parts of embedder that would greatly benefit from moving away from platform channels. Though for this usecase the |
Or, if we don't, the vast majority of people will continue to use the technology that we have had for years and that we currently recommend, and which doesn't have that problem. Meanwhile, we spend some time considering what the actual developer journey we want to encourage is, and what else me may need to build in order to have that in place so that when people start using the new thing, they are using it in a way that we think will be good for the long-term health of the ecosystem. I have a responsibility to consider the long-term impacts of new APIs we expose for the express use of package developers. You don't appear to consider the difference between "sort of possible" and "actively supported and encouraged by the SDK developers" to be a critical one, but I do, and I'm not going to rush approval of a new public API.
I'm not sure what this is referring to; could you link to the issue you are talking about?
I'm fine with a private version of this landing while we consider the bigger picture of what exposing it publicly should look like. |
This is the issue. dart-lang/native#1865 The only way to fix it right now (I can think of) would be to make |
For the jni issue, it's also enough if we expose a unique ID for the engine as well (so it's not even a pointer that has access to the engine and no concerns with regards to the API breaking, etc). This way I can keep a map from the engine ID to the respective |
I see, I wrongly applied my Dart C API knowledge where the API to build embedders is a superset of APIs developers can use with FFI. And in Dart these APIs are C only, no other languages, so they work with
Right, so using JNIgen and FFIgen to generate bindings for these will create a dependency on
Let me see if I understand this correctly. The failures show up at runtime because if you remove a method the Pinning the After discussing dart-lang/native#1865 with @HosseinYousefi, it seems that if we a solution for #110353, access to the Context should be in that place, and should be removed from However, postponing a solution until I wonder if we shouldn't consider providing some stability guarantees/versioning for the Java/ObjC APIs. Like Apple does with their version constraints on OS APIs and sqlite does. Plus a runtime version method. That seems to be the default way of dealing with version skew between bindings an API. As @stuartmorgan points out this leads to runtime failures. Be it really infrequent for sqlite or Apple APIs, so maybe it's unlikely as @knopp points out. (Edit: avoiding runtime failures is essentially what things like the minimum NDK version does. So having a Flutter Pluggin API version would be a good thing.) An alternative solution only for dart-lang/native#1865 is to have Flutter provide a C API that gives an engine ID (not even a pointer that you can interact with). Then we could track the context per engine. (Having a C API avoids having to have a dependency on |
For Android, we could make |
I see; that's very unfortunate.
It seems like taking something that was possible before (writing a plugin shim, incorrectly assuming the plugin was a singleton, and exposing an API based on that), but packaging it into an easy-to-use, officially (quasi-)endorsed public API had problematic long-term effects that are difficult to unwind 😉 |
Apple doesn't solve the problem we're discussion here with versioning (which only works backwards, not forwards), they solve it by essentially never removing anything. They just deprecate things and leave them around until some major break point allows them to do some cleanup (e.g., Carbon was kept in zombie mode for many years, and only "removed" in that they didn't allow linking against it in 64-bit apps.) We could do that, but it would be a policy change that would require a design document and sign off from leadership across the org. That policy would have prevented the recent Android v1 embedder cleanup, which the engine team felt very strongly was critical. |
Unfortunately somehow this was one of the first things written into
In order for |
Context getter requires Flutter plugin, and would thus imply Flutter dependency, no? What exactly is the use case for removing dependency on Flutter? |
Calling it in Dart-standalone could return Right now the fact that jni doesn't actually depend on any APIs from the Flutter framework means that you can technically still use it in a Dart-standalone application as long as you don't call Flutter specific stuff like the context getter. This is not an ideal design though, hence the issue to separate the packages. |
My point is why having |
Yeah that's what I tried to say. We should move it to |
Fixes #163430. This PR adds `engineId` field to `PlatformDispatcher`. When provided by the engine, this can be used to retrieve the engine instance from native code. Dart code: ```dart final identifier = PlatformDispatcher.instance.engineId!; ``` macOS, iOS: ```objc FlutterEngine *engine = [FlutterEngine engineForIdentifier: identifier]; ``` Android: ```java FlutterEngine engine = FlutterEngine.engineForId(identifier); ``` Linux ```cpp FlEngine *engine = fl_engine_for_id(identifier); ``` Windows ```cpp FlutterDesktopEngineRef engine = FlutterDesktopEngineForId(identifier); ``` *If you had to change anything in the [flutter/tests] repo, include a link to the migration guide as per the [breaking change policy].* ## Pre-launch Checklist - [X] I read the [Contributor Guide] and followed the process outlined there for submitting PRs. - [X] I read the [Tree Hygiene] wiki page, which explains my responsibilities. - [X] I read and followed the [Flutter Style Guide], including [Features we expect every widget to implement]. - [X] I signed the [CLA]. - [X] I listed at least one issue that this PR fixes in the description above. - [X] I updated/added relevant documentation (doc comments with `///`). - [X] I added new tests to check the change I am making, or this PR is [test-exempt]. - [X] I followed the [breaking change policy] and added [Data Driven Fixes] where supported. - [X] All existing and new tests are passing. If you need help, consider asking for advice on the #hackers-new channel on [Discord]. <!-- Links --> [Contributor Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#overview [Tree Hygiene]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md [test-exempt]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#tests [Flutter Style Guide]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md [Features we expect every widget to implement]: https://github.com/flutter/flutter/blob/main/docs/contributing/Style-guide-for-Flutter-repo.md#features-we-expect-every-widget-to-implement [CLA]: https://cla.developers.google.com/ [flutter/tests]: https://github.com/flutter/tests [breaking change policy]: https://github.com/flutter/flutter/blob/main/docs/contributing/Tree-hygiene.md#handling-breaking-changes [Discord]: https://github.com/flutter/flutter/blob/main/docs/contributing/Chat.md [Data Driven Fixes]: https://github.com/flutter/flutter/blob/main/docs/contributing/Data-driven-Fixes.md
This thread has been automatically locked since there has not been any recent activity after it was closed. If you are still experiencing a similar issue, please open a new bug, including the output of |
Currently, when writing FFI plugins there is no good way to interact with the engine. One options is to write a Flutter plugin, that either stores reference to the engine statically to be used with FFI, or uses platform channels to let the isolate access engine address and then possibly use it through FFI calls. Both of these are problematic, because a) storing reference to engine statically assumes that there's only one engine running b) using platform channels requires asynchronous call to get engine address.
Neither of these workarounds would help us if we want to use FFI inside our embedders, which is something we might want to do in future, i.e. for #150460 or multi-window.
Better solution would be to provide engine handle to the root isolate at startup. Dart code can then pass this value to FFI calls which could then obtain the Flutter engine.
How exactly the engine is obtained depends on the platform.
[FlutterEngine engineForHandle:handle]
.FlutterDesktopEngineForHandle(handle)
fl_engine_for_handle(handle)
.io.flutter.embedding.engine.FlutterEngine.forHandle(handle)
(static method).On dart side this could be accessed from
PlatformDispatcher
, i.e.PlatformDispatcher.instance.engineHandle
. The value would benull
for web and custom embedders that don't set the value.The value would be set through embedder API by the engine and available from isolate start.
The text was updated successfully, but these errors were encountered: