-
Notifications
You must be signed in to change notification settings - Fork 531
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
Lazily initialize JCW delegates #9306
Comments
I like the idea in general, however there is one thing that has to be measured in a larger example is the impact of adding another |
It would be interesting to see results of those tests on a real device. Marshal methods are, by design, 100% lazy - there is no overhead other than Java VM doing a symbol lookup the first time the method is called. 55ms sounds a bit much, but I guess it's Java VM calling |
I'm not at all sure how to do that. The problem is that android/src/Mono.Android/Android.Runtime/AndroidRuntime.cs Lines 547 to 549 in b667af5
The only straightforward way to "automatically add" a layer of indirection would be for I think this needs to be done in bindings. What I would prefer is: dotnet/runtime#108211 If/when we have "proper" dotnet/runtime construct that doesn't require System.Reflection.Emit, then we can update (We would also need such a construct for eventual NativeAOT support. Exception handling must be baked into the binding assemblies, not delegated off to |
Building upon my earlier comment, I believe that this would be better, if "we" can get all of our proverbial ducks in a row: dotnet/java-interop#1258 |
Note: all performance numbers mentioned are run on Android Emulator on a DevBox, so they are somewhat inflated.
Imagine we have a Java type that contains a lot of virtual methods:
Now imagine we bind it, and override all of those methods:
Then we create an instance of this object in C#:
new MyDelegateTester()
.For a
Release
config with marshal methods turned off, this object creation takes ~694ms. (As an aside, this case really shines with marshal methods.)The issue is that we call
mono.android.Runtime.register
from thestatic
constructor of the generated Java peer type, and this calls the C# generatedGet*Handler
immediately for every overridden method (in this case 100) even if they aren't immediately used. If we could make these calls lazy instead it would likely improve app startup and navigating to a new activity, since these are places where objects are often created but are not used until later.Ideally, this could be done in the Java(?) layer as it could then improve all existing bindings. However it could also be done at the binding level.
Instead of:
We could add an extra level of indirection:
Switching to a test case of calling a single method instead of 100, we can see that
CreateDelegate
cost has moved from the constructor to the first invoke of the method:Note: The cost is abnormally high in this example because it is the first call that uses S.R.E, causing the overhead of initializing those APIs.
The text was updated successfully, but these errors were encountered: