Skip to content

Commit

Permalink
[Xamarin.Android.Build.Tasks] fix <CilStrip/> for Hybrid AOT (#4850)
Browse files Browse the repository at this point in the history
Fixes: #4818

Use of Hybrid AOT currently results in assemblies that have not been
CIL-stripped.

In 77ab240, a change was made to fix how `Mono.Android.dll` was
treated during AOT.  `Mono.Android.dll` used to be the only assembly
copied to the `shrunk` directory, and we had two copies of the this
assembly passed to the AOT compiler.  This change unintentionally
made `<CilStrip/>` strip the wrong set of assemblies...

The initial fix would be:

	<CilStrip
	    ...
	-   ResolvedAssemblies="@(_ResolvedAssemblies)">
	+   ResolvedAssemblies="@(_ShrunkAssemblies)">

Unfortunately, this causes a crash:

	E/AndroidRuntime(18806): java.lang.UnsatisfiedLinkError: No implementation found for void mono.android.TypeManager.n_activate(java.lang.String, java.lang.String, java.lang.Object, java.lang.Object[]) (tried Java_mono_android_TypeManager_n_1activate and Java_mono_android_TypeManager_n_1activate__Ljava_lang_String_2Ljava_lang_String_2Ljava_lang_Object_2_3Ljava_lang_Object_2)
	E/AndroidRuntime(18806): 	at mono.android.TypeManager.n_activate(Native Method)
	E/AndroidRuntime(18806): 	at mono.android.TypeManager.Activate(:7)
	E/AndroidRuntime(18806): 	at crc64446c24ccf511bf5f.SplashScreenActivity.<init>(:25)
	E/AndroidRuntime(18806): 	at java.lang.Class.newInstance(Native Method)
	E/AndroidRuntime(18806): 	at android.app.Instrumentation.newActivity(Instrumentation.java:1086)
	E/AndroidRuntime(18806): 	at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2809)
	E/AndroidRuntime(18806): 	at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2988)
	E/AndroidRuntime(18806): 	at android.app.ActivityThread.-wrap14(ActivityThread.java)
	E/AndroidRuntime(18806): 	at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1631)
	E/AndroidRuntime(18806): 	at android.os.Handler.dispatchMessage(Handler.java:102)
	E/AndroidRuntime(18806): 	at android.os.Looper.loop(Looper.java:154)
	E/AndroidRuntime(18806): 	at android.app.ActivityThread.main(ActivityThread.java:6682)
	E/AndroidRuntime(18806): 	at java.lang.reflect.Method.invoke(Native Method)
	E/AndroidRuntime(18806): 	at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1520)
	E/AndroidRuntime(18806): 	at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1410)

Thinking about how things "used to work", we weren't stripping the
*correct* `Mono.Android.dll`.  So we could run `<CilStrip/>` on every
assembly *besides* `Mono.Android.dll`?

I think this could be improved further if we could strip
`Mono.Android.dll`, but this at least gets things back to working the
way they used to.

I updated the `BuildTest.BuildIncrementalAot()` test that had a
`//TODO` comment, as it works properly now.  I also added a new test
to check that method bodies are stripped.
  • Loading branch information
jonathanpeppers authored Jun 23, 2020
1 parent 77464dc commit 88215f9
Show file tree
Hide file tree
Showing 4 changed files with 42 additions and 6 deletions.
5 changes: 5 additions & 0 deletions Documentation/release-notes/4818.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
#### Application and library build and deployment

* [GitHub 4818](https://github.com/xamarin/xamarin-android/issues/4818):
Applications using `AndroidAotMode=Hybrid` did not properly strip
away the IL from the resulting .NET assemblies.
Original file line number Diff line number Diff line change
Expand Up @@ -4331,5 +4331,37 @@ public void XA4310 ([Values ("apk", "aab")] string packageFormat)

}
}

[Test]
public void HybridAOT ()
{
var proj = new XamarinAndroidApplicationProject () {
IsRelease = true,
AotAssemblies = true,
};
proj.SetProperty ("AndroidAotMode", "Hybrid");
// So we can use Mono.Cecil to open assemblies directly
proj.SetProperty ("AndroidEnableAssemblyCompression", "False");

using (var b = CreateApkBuilder ()) {
b.Build (proj);

var apk = Path.Combine (Root, b.ProjectDirectory, proj.OutputPath, $"{proj.PackageName}.apk");
FileAssert.Exists (apk);
using (var zip = ZipHelper.OpenZip (apk)) {
var entry = zip.ReadEntry ($"assemblies/{proj.ProjectName}.dll");
Assert.IsNotNull (entry, $"{proj.ProjectName}.dll should exist in apk!");
using (var stream = new MemoryStream ()) {
entry.Extract (stream);
stream.Position = 0;
using (var assembly = AssemblyDefinition.ReadAssembly (stream)) {
var type = assembly.MainModule.GetType ($"{proj.ProjectName}.MainActivity");
var method = type.Methods.First (m => m.Name == "OnCreate");
Assert.LessOrEqual (method.Body.Instructions.Count, 1, "OnCreate should have stripped method bodies!");
}
}
}
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -929,11 +929,6 @@ public void BuildIncrementalAot (string supportedAbis, string androidAotMode, bo
Assert.IsFalse (b.Output.IsTargetSkipped (target), $"`{target}` should *not* be skipped on first build!");
}

if (androidAotMode == "Hybrid") {
// FIXME: with Hybrid AOT, <CilStrip/> modifies assemblies in-place
Assert.Ignore ("Ignoring, Hybrid AOT triggers _BuildApkEmbed.");
}

b.BuildLogFile = "second.log";
b.CleanupAfterSuccessfulBuild = false;
b.CleanupOnDispose = false;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2151,13 +2151,17 @@ because xbuild doesn't support framework reference assemblies.
<Output TaskParameter="NativeLibrariesReferences" ItemName="_AdditionalNativeLibraryReferences" />
</Aot>

<ItemGroup Condition=" '$(AndroidAotMode)' == 'Hybrid' And '$(AotAssemblies)' == 'True' ">
<_CilStripAssemblies Include="@(_ShrunkAssemblies)" Condition=" '%(FileName)' != 'Mono.Android' " />
</ItemGroup>

<!-- Strip the IL code of the resolved managed assemblies -->
<CilStrip
Condition=" '$(AndroidAotMode)' == 'Hybrid' And '$(AotAssemblies)' == 'True' "
AndroidAotMode="$(AndroidAotMode)"
ToolPath="$(_MonoAndroidToolsDirectory)"
ApkOutputPath="$(_BuildApkEmbedOutputs)"
ResolvedAssemblies="@(_ResolvedAssemblies)">
ResolvedAssemblies="@(_CilStripAssemblies)">
</CilStrip>

<!-- Bundle the assemblies into native libraries in the apk -->
Expand Down

0 comments on commit 88215f9

Please sign in to comment.