Skip to content

Commit e323d1b

Browse files
committed
[Java.Interop] CreatePeer() must satisfy targetType
Context: dotnet/android#9747 Context: https://discord.com/channels/732297728826277939/732297837953679412/1339638847864176640 Context: https://discord.com/channels/732297728826277939/732297837953679412/1340011105510101063 While trying to get a MAUI sample running atop NativeAOT, we're observing the following crash: E AndroidRuntime: net.dot.jni.internal.JavaProxyThrowable: System.InvalidCastException: Arg_InvalidCastException E AndroidRuntime: at Java.Lang.Object._GetObject[T](IntPtr, JniHandleOwnership) + 0x64 E AndroidRuntime: at Microsoft.Maui.WindowOverlay.Initialize() + 0x168 Further investigation shows that the crash is from accessing the `Activity.WindowManager` property: namespace Android.App; partial class Activity { public virtual unsafe Android.Views.IWindowManager? WindowManager { // Metadata.xml XPath method reference: path="/api/package[@name='android.app']/class[@name='Activity']/method[@name='getWindowManager' and count(parameter)=0]" [Register ("getWindowManager", "()Landroid/view/WindowManager;", "GetGetWindowManagerHandler")] get { const string __id = "getWindowManager.()Landroid/view/WindowManager;"; try { var __rm = _members.InstanceMethods.InvokeVirtualObjectMethod (__id, this, null); return global::Java.Lang.Object.GetObject<Android.Views.IWindowManager> (__rm.Handle, JniHandleOwnership.TransferLocalRef); } finally { } } } } `Object.GetObject<T>()` is now a wrapper around `JniRuntime.JniValueManager.GetPeer()`, so the problem, rephrased, is that this: var peer = JniEnvironment.Runtime.ValueManager.GetPeer ( ref h, JniObjectReferenceOptions.CopyAndDispose, targetType:typeof (IWindowManager)); returns a value which *does not implement* `IWindowManager`. It was, in fact, returning a `Java.Lang.Object` instance (!). The cause of the bug is that `JniRuntime.JniValueManager.CreatePeer()` was not checking that the `Type` returned form `Runtime.TypeManager.GetType(JniTypeSignature)` was compatible with `targetType`; instead, it returned the first type in the inheritance chain that had an activation constructor. This was `Java.Lang.Object`. Later, when `_GetObject<T>()` tried to cast the return value of `JniRuntime.JniValueManager.GetPeer()` to `IWindowManager`, it failed. Fix this by updating `CreatePeer()` to check that the `Type` from `JniRuntime.JniTypeManager.GetType(JniTypeSignature)` is assignable to `targetType`. This ensures that we *don't* return a `Java.Lang.Object` instance, allowing the cast to succeed. Update `JniRuntimeJniValueManagerContract` to test these new semantics.
1 parent d62008d commit e323d1b

File tree

5 files changed

+76
-3
lines changed

5 files changed

+76
-3
lines changed

src/Java.Interop/Java.Interop/JniRuntime.JniValueManager.cs

+3-3
Original file line numberDiff line numberDiff line change
@@ -348,7 +348,7 @@ static Type GetPeerType ([DynamicallyAccessedMembers (Constructors)] Type type)
348348
IJavaPeerable? CreatePeerInstance (
349349
ref JniObjectReference klass,
350350
[DynamicallyAccessedMembers (Constructors)]
351-
Type fallbackType,
351+
Type targetType,
352352
ref JniObjectReference reference,
353353
JniObjectReferenceOptions transfer)
354354
{
@@ -362,7 +362,7 @@ static Type GetPeerType ([DynamicallyAccessedMembers (Constructors)] Type type)
362362

363363
type = Runtime.TypeManager.GetType (sig);
364364

365-
if (type != null) {
365+
if (type != null && targetType.IsAssignableFrom (type)) {
366366
var peer = TryCreatePeerInstance (ref reference, transfer, type);
367367

368368
if (peer != null) {
@@ -381,7 +381,7 @@ static Type GetPeerType ([DynamicallyAccessedMembers (Constructors)] Type type)
381381
}
382382
JniObjectReference.Dispose (ref klass, JniObjectReferenceOptions.CopyAndDispose);
383383

384-
return TryCreatePeerInstance (ref reference, transfer, fallbackType);
384+
return TryCreatePeerInstance (ref reference, transfer, targetType);
385385
}
386386

387387
IJavaPeerable? TryCreatePeerInstance (

tests/Java.Interop-Tests/Java.Interop-Tests.csproj

+1
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
</ItemGroup>
3737

3838
<ItemGroup>
39+
<JavaInteropTestJar Include="$(MSBuildThisFileDirectory)java\net\dot\jni\test\AnotherJavaInterfaceImpl.java" />
3940
<JavaInteropTestJar Include="$(MSBuildThisFileDirectory)java\net\dot\jni\test\CrossReferenceBridge.java" />
4041
<JavaInteropTestJar Include="$(MSBuildThisFileDirectory)java\net\dot\jni\test\CallNonvirtualBase.java" />
4142
<JavaInteropTestJar Include="$(MSBuildThisFileDirectory)java\net\dot\jni\test\CallNonvirtualDerived.java" />

tests/Java.Interop-Tests/Java.Interop/JavaVMFixture.cs

+1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@ class JavaVMFixtureTypeManager : JniRuntime.JniTypeManager {
3737
[GenericHolder<int>.JniTypeName] = typeof (GenericHolder<>),
3838
[RenameClassBase.JniTypeName] = typeof (RenameClassBase),
3939
[RenameClassDerived.JniTypeName] = typeof (RenameClassDerived),
40+
[AnotherJavaInterfaceImpl.JniTypeName] = typeof (AnotherJavaInterfaceImpl),
4041
[CallVirtualFromConstructorBase.JniTypeName] = typeof (CallVirtualFromConstructorBase),
4142
[CallVirtualFromConstructorDerived.JniTypeName] = typeof (CallVirtualFromConstructorDerived),
4243
[CrossReferenceBridge.JniTypeName] = typeof (CrossReferenceBridge),

tests/Java.Interop-Tests/Java.Interop/JniRuntimeJniValueManagerContract.cs

+58
Original file line numberDiff line numberDiff line change
@@ -133,6 +133,30 @@ public void CollectPeers ()
133133
// TODO
134134
}
135135

136+
[Test]
137+
public void CreatePeer_InvalidHandleReturnsNull ()
138+
{
139+
var r = new JniObjectReference ();
140+
var o = valueManager.CreatePeer (ref r, JniObjectReferenceOptions.Copy, null);
141+
Assert.IsNull (o);
142+
}
143+
144+
[Test]
145+
public unsafe void CreatePeer_UsesFallbackType ()
146+
{
147+
using var t = new JniType (AnotherJavaInterfaceImpl.JniTypeName);
148+
149+
var ctor = t.GetConstructor ("()V");
150+
var lref = t.NewObject (ctor, null);
151+
152+
using var p = valueManager.CreatePeer (ref lref, JniObjectReferenceOptions.CopyAndDispose, typeof (IJavaInterface));
153+
154+
Assert.IsFalse (lref.IsValid); // .CopyAndDispose disposes
155+
156+
Assert.IsNotNull (p);
157+
Assert.AreSame (typeof (IJavaInterfaceInvoker), p!.GetType ());
158+
}
159+
136160
[Test]
137161
public void CreateValue ()
138162
{
@@ -294,4 +318,38 @@ public class JniRuntimeJniValueManagerContract_NoGCIntegration : JniRuntimeJniVa
294318
protected override Type ValueManagerType => ManagedValueManagerType;
295319
}
296320
#endif // !__ANDROID__
321+
322+
// Note: Java side implements JavaInterface, while managed binding DOES NOT.
323+
// This is so that `CreatePeer(…, typeof(IJavaInterface))` tests don't use an existing AnotherJavaInterfaceImpl instance.
324+
//
325+
// This is mostly identical to MyJavaInterfaceImpl; the important difference is that
326+
// it contains an activation constructor, while MyJavaInterfaceImpl does not.
327+
// MyJavaInterfaceImpl can't have one, as that's what provokes the NotSupportedException in the JavaAs() tests.
328+
//
329+
// We want one here so that in "bad" `CreatePeer()` implementations, we'll find this peer and construct it
330+
// before verifying that it satisfies the targetType requirement.
331+
[JniTypeSignature (JniTypeName, GenerateJavaPeer=false)]
332+
public class AnotherJavaInterfaceImpl : JavaObject {
333+
internal const string JniTypeName = "net/dot/jni/test/AnotherJavaInterfaceImpl";
334+
335+
internal static readonly JniPeerMembers _members = new JniPeerMembers (JniTypeName, typeof (AnotherJavaInterfaceImpl));
336+
337+
public override JniPeerMembers JniPeerMembers {
338+
get {return _members;}
339+
}
340+
341+
AnotherJavaInterfaceImpl (ref JniObjectReference reference, JniObjectReferenceOptions options)
342+
: base (ref reference, options)
343+
{
344+
}
345+
346+
public unsafe AnotherJavaInterfaceImpl ()
347+
: base (ref *InvalidJniObjectReference, JniObjectReferenceOptions.None)
348+
{
349+
const string id = "()V";
350+
var peer = _members.InstanceMethods.StartCreateInstance (id, GetType (), null);
351+
Construct (ref peer, JniObjectReferenceOptions.CopyAndDispose);
352+
_members.InstanceMethods.FinishCreateInstance (id, this, null);
353+
}
354+
}
297355
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
package net.dot.jni.test;
2+
3+
public class AnotherJavaInterfaceImpl
4+
implements JavaInterface, Cloneable
5+
{
6+
public String getValue() {
7+
return "Another hello from Java!";
8+
}
9+
10+
public Object clone() {
11+
return this;
12+
}
13+
}

0 commit comments

Comments
 (0)