-
Notifications
You must be signed in to change notification settings - Fork 56
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
Figure out and document (a) pattern(s) for handling #if
#1589
Comments
👍 To the above. Somewhat related: If the variability per OS is mappable to one type, we could generate ABI specific types. We already support that for integers, but not yet for structs (dart-lang/sdk#42816). There will definitely be cases which don't map to types such as different arguments to a function or different functions altogether.
In general, the variation can be arbitrary, behind arbitrary if-defs. In practice, I've mostly seen variation based on target OS. Maybe we should generate bindings per ifdef config. I believe @liamappelbe had some ideas about this? But it would be good if everything that's not behind an ifdef is shared. But then every type refering to such type cannot be shared anymore either. I think it would be good to start accumulating examples so that we can try to make an informed decision. @stuartmorgan would you mind linking the various examples that you find in this issue?
Yeah that probably doesn't scale. 😄
Do you mean a wrapper in native code and then running FFIgen on that wrapper? Or did you mean something else? |
I am only going to personally come across a vanishingly small fraction of cases in my own development. Since the goal is to provide access to the entire OS SDK surface, the way to make a reasonably fully informed decision (for Obj-C/Swift) seems like it would be to use tooling to audit all use of
Yes, a native wrapper that does not itself expose any structural platform |
If the platform differences are that great, my suggestion would be to generate 2 different sets of ffigen bindings, and conditionally import them. If you only use a shared subset of the methods, that would just work. If you need to use methods that only exist on one platform then you'd need to also put that code behind a conditional import. But at least that would let you write the compatibility layer in Dart. It'd be nice if Dart had something like C++'s |
Based on what condition? Last I checked conditional import by platform was impossible. Was there a language change? (Edit: The public docs still seem to indicate that this is impossible.) |
Yeah, you're right. I had a vague memory that this was possible, but I was wrong. I think @sstrickl was working on it, but I haven't heard any updates in a while: dart-lang/sdk#50473. If this sort of conditional import is important for iOS/macOS interop, we should reprioritize that bug. It was P1 for a while. |
We should probably figure out if there's a better option before reprioritizing anything, because this:
is still significantly worse than the native development experience. |
Currently packages that are only used within guarded code blocks like, for Dart: import 'dart:io';
import 'pkg:for_windows' as for_windows;
...
if (Platform.isWindows) {
... direct or indirect use of for_windows package...
}
... or for Flutter, import 'package:flutter/foundation.dart';
import 'pkg:for_windows' as for_windows;
...
if (defaultTargetPlatform == TargetPlatform.windows) {
... direct or indirect use of for_windows package...
}
... should end up with that package tree-shaken out on non-Windows platforms, since For cases where there's an interface and separate implementation classes for each platform, something like this example from dart-lang/language#2133 (comment) should work nowadays with only the platform-specific implementation being kept in the final program: import "dart:io";
class Client { }
class WindowsClient extends Client { }
class MacClient extends Client { }
class AndroidClient extends Client { }
class IPhoneClient extends Client { }
Client getClient() =>
Platform.isWindows ? WindowsClient() :
Platform.isMacOS ? MacClient() :
Platform.isAndroid ? AndroidClient() :
Platform.isIOS ? IPhoneClient() :
throw "Unrecognized platform"; So perhaps one of these approaches would be useful for what |
I'm not seeing how either of those approaches would address case 2 at all; could you give an example? |
Sadly, thinking about it, I don't think they do. Initial thoughts below. It'd be nice if you could just, say, In addition, the constant evaluation of if (Platform.iOS) import 'wkwebview_ios.dart' as wkwebview;
if (Platform.macOS) import 'wkwebview_macos.dart' as wkwebview; and the front end allowed multiple conditional imports that bind the same name ( I guess the TL;DR is if you're not able to make a common interface that supports what's on both platforms, then I don't see a way of supporting that in our current implementation, since code that's tree-shaken is still type-checked, and so simply using platform-based guards in your code isn't enough if there are unrelated types involved in those guarded bits of code, even if they're consistent if you only consider a specific condition for the guards (e.g., compiling for iOS). We'd need to sink the idea of the target platform much deeper into the CFE so it could determine which import is "active", so to speak, for a given compilation run, and then we'd also start running into the question of "how does/should this work with modular compliation", where now intermediate kernel files may be platform-dependent, not just the final tree-shaken kernel file used to create the snapshot. (Perhaps that is what we'd need to do to support this use case, or perhaps there's another way around it I haven't considered yet. I'll continue to think on this.) |
This is a high-level issue for general tracking, since it's a topic I'm sure we'll need at least some solution to, even if it's docs. Maybe we will identify specific use cases that we can split out into sub-issues to address in specific ways, but we will need some way of answering a user with the question "What do I do about this
#if
/#ifdef
when I'm trying to use FFI?"There are lots of potential use cases; here are a couple of concrete ones to provide a starting point:
I have a plugin that is 95+% shared iOS/macOS code, and I'm trying to convert it from Obj-C to FFI. The code that's not shared is behind a platform
#if
check. What's the right way to handle that in Dart? For now, I changed it to aPlatform
runtime check, but that's objectively worse: in the native version, if I accidentally try to use an iOS API in the macOS codepath, the code won't compile. In the Dart version, my IDE will happily auto-complete APIs that I can't actually use, without even a warning, and the code will crash at runtime, and only (I think?) once I hit that code. Can we do better?WKWebView
has the following definition:What should
ffigen
even create here? Is there any good option? The general case of this can get arbitrarily complicated; e.g., if the answer to the first question is that it inherits from some franken-class that is a mashup ofNSView
andUIView
so that I can get any method I need, what happens if there's a collision between methods (e.g., each has a method with the same name but a different return type—a plausible scenario since methods calledview
that return the platform view type are common, for example). Or do we recommend putting everything behind a more FFI-friendly native facade in a case like this?(#1469 touched on this slightly, but was specifically about the generation of code that did not compile, rather than the overall use case.)
The text was updated successfully, but these errors were encountered: