Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 20fa0aa

Browse files
committedFeb 10, 2023
[Java.Interop.Tools.Expressions] Add Java.Interop.Tools.Expressions
Fixes: dotnet#616 Context: dotnet#14 Context: ff4053c Context: da5d1b8 Context: 4787e01 Context: 41ba348 Remember `jnimarshalmethod-gen` (176240d)? And it's crazy idea to use the `System.Linq.Expressions`-based custom marshaling infrastructure (ff4053c, da5d1b8) to generate JNI marshal methods at build/packaging time. And how we had to back burner it because it depended upon `System.Reflection.Emit` being able to write assemblies to disk, which is a feature that never made it to .NET Core, and is still lacking as of .NET 7? Add `src/Java.Interop.Tools.Expressions`, which contains code which uses Mono.Cecil to compile `Expression<T>` expressions to IL. Then update `jnimarshalmethod-gen` to use it! ~~ Usage ~~ % dotnet bin/Debug-net7.0/jnimarshalmethod-gen.dll \ bin/TestDebug-net7.0/Java.Interop.Export-Tests.dll \ -v -v --keeptemp \ --jvm /Library/Java/JavaVirtualMachines/microsoft-11.jdk/Contents/Home/lib/jli/libjli.dylib \ -o _x \ -L bin/TestDebug-net7.0 \ -L /usr/local/share/dotnet/shared/Microsoft.NETCore.App/7.0.0 First param is assembly to process; `Java.Interop.Export-Tests.dll` is handy because that's what the `run-test-jnimarshal` target in `Makefile` processed. * `-v -v` is *really* verbose output * `--keeptemp` is keep temporary files, in this case `_x/Java.Interop.Export-Tests.dll.cecil`. * `--jvm PATH` is the path to the JVM library to load+use. * `-o DIR` is where to place output files; this will create `_x/Java.Interop.Export-Tests.dll`. * `-L DIR` adds `DIR` to library resolution paths; this adds `bin/TestDebug/net7.0` (dependencies of `Java.Interop.Export-Tests.dll`) and `Microsoft.NETCore.App/7.0.0-rc.1.22422.12` (net7 libs). By default the directories containing input assemblies and the directory containing `System.Private.CoreLib.dll` are part of the default `-L` list. When running in-tree, e.g. AzDO pipeline execution, `--jvm PATH` will attempt to read the path in `bin/Build*/JdkInfo.props` a'la `TestJVM` (002dea4). This allows an in-place update in `core-tests.yaml` which does: dotnet bin/Debug-net7.0/jnimarshalmethod-gen.dll \ bin/TestDebug-net7.0/Java.Interop.Export-Tests.dll \ -v -v --keeptemp -o bin/TestDebug-net7.0 ~~ Using `jnimarshalmethod-gen` output ~~ What does `jnimarshalmethod-gen` *do*? % ikdasm bin/TestDebug-net7.0/Java.Interop.Export-Tests.dll > beg.il % ikdasm _x/Java.Interop.Export-Tests.dll > end.il % git diff --no-index beg.il end.il # https://gist.github.com/jonpryor/b8233444f2e51043732bea922f6afc81 is a ~1KB diff which shows, paraphrasing greatly: public partial class ExportTest { partial class '__<$>_jni_marshal_methods' { static IntPtr funcIJavaObject (IntPtr jnienv, IntPtr this) => … // … [JniAddNativeMethodRegistration] static void __RegisterNativeMembers (JniNativeMethodRegistrationArguments args) => … } } internal delegate long _JniMarshal_PP_L (IntPtr jnienv, IntPtr self); // … wherein `ExportTest._<$>_jni_marshal_methods` and the `_JniMarshal*` delegate types are added to the assembly. This also unblocks the desire stated in 4787e01: > For `Java.Base`, @jonpryor wants to support the custom marshaling > infrastructure introduced in 77a6bf8. This would allow types to > participate in JNI marshal method ("connector method") generation > *at runtime*, allowing specialization based on the current set of > types and assemblies. What can we do with this `jnimarshalmethod-gen` output? Use it! First, make sure the tests work: % dotnet test --logger "console;verbosity=detailed" bin/TestDebug-net7.0/Java.Interop.Export-Tests.dll … Passed! - Failed: 0, Passed: 17, Skipped: 0, Total: 17, Duration: 103 ms - Java.Interop.Export-Tests.dll (net7.0) Then update/replace the unit test assembly with `jnimarshalmethod-gen` output: % \cp _x/Java.Interop.Export-Tests.dll bin/TestDebug-net7.0 % dotnet test --logger "console;verbosity=detailed" bin/TestDebug-net7.0/Java.Interop.Export-Tests.dll … Total tests: 17 Passed: 17 `core-tests.yaml` has been updated to do this. ~~ One-Off Tests ~~ One-off tests: ensure that the generated assembly can be decompiled: % ikdasm bin/TestDebug-net7.0/Java.Interop.Tools.Expressions-Tests-ExpressionAssemblyBuilderTests.dll % monodis bin/TestDebug-net7.0/Java.Interop.Tools.Expressions-Tests-ExpressionAssemblyBuilderTests.dll % ikdasm _x/Java.Interop.Export-Tests.dll % monodis _x/Java.Interop.Export-Tests.dll # which currently fails :-() Re-enable most of `Java.Interop.Export-Tests.dll` for .NET 7; see 41ba348, which disabled those tests. To verify the generated IL, use the [dotnet-ilverify][0] tool: dotnet tool install --global dotnet-ilverify Usage of which is "weird": $HOME/.dotnet/tools/ilverify _x/Java.Interop.Export-Tests.dll \ --tokens --system-module System.Private.CoreLib \ -r 'bin/TestDebug-net7.0/*.dll' \ -r '/usr/local/share/dotnet/shared/Microsoft.NETCore.App/7.0.0/*.dll' All Classes and Methods in /Volumes/Xamarin-Work/src/xamarin/Java.Interop/_x/Java.Interop.Export-Tests.dll Verified. # no errors! where: * `--tokens`: Include metadata tokens in error messages. * `--system-module NAME`: set the "System module name". Defaults to `mscorlib`, which is wrong for .NET 5+, so this must be set to `System.Private.CoreLib` (no `.dll` suffix!). * `-r FILE-GLOB`: Where to resolve assembly references for the input assembly. Fortunately file globs are supported… ~~ Removing `System.Private.CoreLib` ~~ `System.Private.CoreLib.dll` is *private*; it's not part of the public assembly surface area, so you can't use `csc -r:System.Private.CoreLib …` and expect it to work. This makes things interesting because *at runtime* everything "important" is in `System.Private.CoreLib.dll`, like `System.Object`. Specifically, if we do the "obvious" thing and do: newTypeDefinition.BaseType = assemblyDefinition.MainModule .ImportReference (typeof (object)); you're gonna have a bad type, because the resulting IL for `newTypeDefinition` will have a base class of `[System.Private.CoreLib]System.Object`, which isn't usable. Fix this by: 1. Writing the assembly to a `Stream`. 2. Reading the `Stream` from (1) 3. Fixing all member references and assembly references so that `System.Private.CoreLib` is not referenced. If `jnimarshalmethod-gen.dll --keeptemp` is used, then a `.cecil` file is created with the contents of (1). Additionally, and unexpectedly -- [jbevain/cecil#895][1] -- Mono.Cecil adds a reference to the assembly being modified. Remove the declaring assembly from `AssemblyReferences`. [0]: https://www.nuget.org/packages/dotnet-ilverify [1]: jbevain/cecil#895
1 parent 5fa7ac4 commit 20fa0aa

File tree

16 files changed

+2138
-129
lines changed

16 files changed

+2138
-129
lines changed
 

‎Java.Interop.sln

+14
Original file line numberDiff line numberDiff line change
@@ -109,6 +109,10 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Java.Base", "src\Java.Base\
109109
EndProject
110110
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Java.Base-Tests", "tests\Java.Base-Tests\Java.Base-Tests.csproj", "{CB05E11B-B96F-4179-A4E9-5D6BDE29A8FC}"
111111
EndProject
112+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Java.Interop.Tools.Expressions", "src\Java.Interop.Tools.Expressions\Java.Interop.Tools.Expressions.csproj", "{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A}"
113+
EndProject
114+
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Java.Interop.Tools.Expressions-Tests", "tests\Java.Interop.Tools.Expressions-Tests\Java.Interop.Tools.Expressions-Tests.csproj", "{211BAA88-66B1-41B2-88B2-530DBD8DF702}"
115+
EndProject
112116
Global
113117
GlobalSection(SharedMSBuildProjectFiles) = preSolution
114118
src\Java.Interop.NamingCustomAttributes\Java.Interop.NamingCustomAttributes.projitems*{58b564a1-570d-4da2-b02d-25bddb1a9f4f}*SharedItemsImports = 5
@@ -308,6 +312,14 @@ Global
308312
{CB05E11B-B96F-4179-A4E9-5D6BDE29A8FC}.Debug|Any CPU.Build.0 = Debug|Any CPU
309313
{CB05E11B-B96F-4179-A4E9-5D6BDE29A8FC}.Release|Any CPU.ActiveCfg = Release|Any CPU
310314
{CB05E11B-B96F-4179-A4E9-5D6BDE29A8FC}.Release|Any CPU.Build.0 = Release|Any CPU
315+
{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
316+
{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A}.Debug|Any CPU.Build.0 = Debug|Any CPU
317+
{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A}.Release|Any CPU.ActiveCfg = Release|Any CPU
318+
{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A}.Release|Any CPU.Build.0 = Release|Any CPU
319+
{211BAA88-66B1-41B2-88B2-530DBD8DF702}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
320+
{211BAA88-66B1-41B2-88B2-530DBD8DF702}.Debug|Any CPU.Build.0 = Debug|Any CPU
321+
{211BAA88-66B1-41B2-88B2-530DBD8DF702}.Release|Any CPU.ActiveCfg = Release|Any CPU
322+
{211BAA88-66B1-41B2-88B2-530DBD8DF702}.Release|Any CPU.Build.0 = Release|Any CPU
311323
EndGlobalSection
312324
GlobalSection(SolutionProperties) = preSolution
313325
HideSolutionNode = FALSE
@@ -360,6 +372,8 @@ Global
360372
{11942DE9-AEC2-4B95-87AB-CA707C37643D} = {271C9F30-F679-4793-942B-0D9527CB3E2F}
361373
{30DCECA5-16FD-4FD0-883C-E5E83B11565D} = {0998E45F-8BCE-4791-A944-962CD54E2D80}
362374
{CB05E11B-B96F-4179-A4E9-5D6BDE29A8FC} = {271C9F30-F679-4793-942B-0D9527CB3E2F}
375+
{1A0262FE-3CDB-4AF2-AAD8-65C59524FE8A} = {0998E45F-8BCE-4791-A944-962CD54E2D80}
376+
{211BAA88-66B1-41B2-88B2-530DBD8DF702} = {271C9F30-F679-4793-942B-0D9527CB3E2F}
363377
EndGlobalSection
364378
GlobalSection(ExtensibilityGlobals) = postSolution
365379
SolutionGuid = {29204E0C-382A-49A0-A814-AD7FBF9774A5}

‎Makefile

+26-7
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ PREPARE_EXTERNAL_FILES = \
1818
DEPENDENCIES = \
1919
bin/Test$(CONFIGURATION)/libNativeTiming$(NATIVE_EXT)
2020

21+
NET_SUFFIX = -net7.0
22+
2123
TESTS = \
2224
bin/Test$(CONFIGURATION)/Java.Interop-Tests.dll \
2325
bin/Test$(CONFIGURATION)/Java.Interop.Dynamic-Tests.dll \
@@ -33,7 +35,7 @@ TESTS = \
3335
bin/Test$(CONFIGURATION)/Xamarin.SourceWriter-Tests.dll
3436

3537
NET_TESTS = \
36-
bin/Test$(CONFIGURATION)-net7.0/Java.Base-Tests.dll
38+
bin/Test$(CONFIGURATION)$(NET_SUFFIX)/Java.Base-Tests.dll
3739

3840
PTESTS = \
3941
bin/Test$(CONFIGURATION)/Java.Interop-PerformanceTests.dll
@@ -43,6 +45,10 @@ ATESTS = \
4345

4446
all: $(DEPENDENCIES) $(TESTS)
4547

48+
bin/ilverify:
49+
-mkdir bin
50+
dotnet tool install --tool-path bin dotnet-ilverify
51+
4652
run-all-tests:
4753
r=0; \
4854
$(MAKE) run-tests || r=1 ; \
@@ -127,7 +133,7 @@ run-tests: $(TESTS) bin/Test$(CONFIGURATION)/$(JAVA_INTEROP_LIB)
127133
$(foreach t,$(TESTS), $(call RUN_TEST,$(t),1)) \
128134
exit $$r;
129135

130-
run-net-tests: $(NET_TESTS) bin/Test$(CONFIGURATION)-net7.0/$(JAVA_INTEROP_LIB)
136+
run-net-tests: $(NET_TESTS) bin/Test$(CONFIGURATION)$(NET_SUFFIX)/$(JAVA_INTEROP_LIB)
131137
r=0; \
132138
$(foreach t,$(NET_TESTS), dotnet test $(t) || r=1) \
133139
exit $$r;
@@ -150,15 +156,28 @@ $(JRE_DLL_CONFIG): src/Java.Runtime.Environment/Java.Runtime.Environment.csproj
150156

151157
define run-jnimarshalmethod-gen
152158
MONO_TRACE_LISTENER=Console.Out \
153-
$(RUNTIME) bin/$(CONFIGURATION)/jnimarshalmethod-gen.exe -v --jvm "$(JI_JVM_PATH)" -L "$(JI_MONO_LIB_PATH)mono/4.5" -L "$(JI_MONO_LIB_PATH)mono/4.5/Facades" $(2) $(1)
159+
dotnet bin/$(CONFIGURATION)$(NET_SUFFIX)/jnimarshalmethod-gen.dll $(2) $(1)
154160
endef
155161

156-
run-test-jnimarshal: bin/Test$(CONFIGURATION)/Java.Interop.Export-Tests.dll bin/Test$(CONFIGURATION)/$(JAVA_INTEROP_LIB) $(JRE_DLL_CONFIG)
162+
# want: /usr/local/share/dotnet/shared/Microsoft.NETCore.App/7.0.0
163+
# have: Microsoft.NETCore.App 7.0.0 [/usr/local/share/dotnet/shared/Microsoft.NETCore.App]
164+
# use: shell pipeline!
165+
SYSTEM_NET_ASSEMBLIES_PATH := $(shell dotnet --list-runtimes | grep ^Microsoft.NETCore.App | tail -1 | sed -E 's,^Microsoft.NETCore.App ([^ ]+) \[([^]]+)\]$$,\2/\1,g' )
166+
167+
run-test-jnimarshal: bin/Test$(CONFIGURATION)$(NET_SUFFIX)/Java.Interop.Export-Tests.dll bin/Test$(CONFIGURATION)$(NET_SUFFIX)/$(JAVA_INTEROP_LIB) bin/ilverify
157168
mkdir -p test-jni-output
158-
$(call run-jnimarshalmethod-gen,"$<",-f -o test-jni-output --keeptemp)
159-
(test -f test-jni-output/$(notdir $<) && test -f test-jni-output/Java.Interop.Export-Tests-JniMarshalMethods.dll) || { echo "jnimarshalmethod-gen did not create the expected assemblies in the test-jni-output directory"; exit 1; }
169+
# Do we run w/o error?
170+
$(call run-jnimarshalmethod-gen,"$<", -v -v -o test-jni-output --keeptemp)
171+
(test -f test-jni-output/$(notdir $<) ) || { echo "jnimarshalmethod-gen did not create the expected assemblies in the test-jni-output directory"; exit 1; }
172+
# Is output valid?
173+
ikdasm test-jni-output/Java.Interop.Export-Tests.dll || { echo "output can not be processed by ikdasm"; exit 1; }
174+
bin/ilverify test-jni-output/Java.Interop.Export-Tests.dll \
175+
--tokens --system-module System.Private.CoreLib -r '$(dir $<)/*.dll' \
176+
-r '$(SYSTEM_NET_ASSEMBLIES_PATH)/*.dll' || { echo "ilverify found issues"; exit 1; }
177+
# replace "original" assembly
160178
$(call run-jnimarshalmethod-gen,"$<")
161-
$(call RUN_TEST,$<)
179+
# make sure tests still pass
180+
dotnet test $<
162181

163182
bin/Test$(CONFIGURATION)/generator.exe: bin/$(CONFIGURATION)/generator.exe
164183
cp $<* `dirname "$@"`

‎build-tools/automation/templates/core-tests.yaml

+19-1
Original file line numberDiff line numberDiff line change
@@ -105,13 +105,31 @@ steps:
105105

106106
- task: DotNetCoreCLI@2
107107
displayName: 'Tests: Java.Interop.Export'
108-
condition: eq('${{ parameters.runNativeTests }}', 'true')
108+
condition: or(eq('${{ parameters.runNativeDotnetTests }}', 'true'), eq('${{ parameters.runNativeTests }}', 'true'))
109109
inputs:
110110
command: test
111111
testRunTitle: Java.Interop.Export (${{ parameters.platformName }})
112112
arguments: bin/Test$(Build.Configuration)$(NetCoreTargetFrameworkPathSuffix)/Java.Interop.Export-Tests.dll
113113
continueOnError: true
114114

115+
- task: DotNetCoreCLI@2
116+
displayName: 'jnimarshalmethod-gen Java.Interop.Export-Tests.dll'
117+
condition: or(eq('${{ parameters.runNativeDotnetTests }}', 'true'), eq('${{ parameters.runNativeTests }}', 'true'))
118+
inputs:
119+
command: custom
120+
custom: bin/$(Build.Configuration)$(NetCoreTargetFrameworkPathSuffix)/jnimarshalmethod-gen.dll
121+
arguments: bin/Test$(Build.Configuration)$(NetCoreTargetFrameworkPathSuffix)/Java.Interop.Export-Tests.dll -v -v --keeptemp -o bin/Test$(Build.Configuration)$(NetCoreTargetFrameworkPathSuffix)
122+
continueOnError: true
123+
124+
- task: DotNetCoreCLI@2
125+
displayName: 'Tests: Java.Interop.Export w/ jnimarshalmethod-gen!'
126+
condition: or(eq('${{ parameters.runNativeDotnetTests }}', 'true'), eq('${{ parameters.runNativeTests }}', 'true'))
127+
inputs:
128+
command: test
129+
testRunTitle: Java.Interop.Export (jnimarshalmethod-gen + ${{ parameters.platformName }})
130+
arguments: bin/Test$(Build.Configuration)$(NetCoreTargetFrameworkPathSuffix)/Java.Interop.Export-Tests.dll
131+
continueOnError: true
132+
115133
- task: DotNetCoreCLI@2
116134
displayName: 'Tests: Java.Interop-Performance-net472'
117135
condition: eq('${{ parameters.runNativeTests }}', 'true')

‎src/Java.Base-ref.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -6408,7 +6408,7 @@ public partial class AccessibleObject : Java.Lang.Object, Java.Interop.IJavaPeer
64086408
{
64096409
protected AccessibleObject() { }
64106410
protected AccessibleObject(ref Java.Interop.JniObjectReference reference, Java.Interop.JniObjectReferenceOptions options) { }
6411-
public virtual bool Accessible { get { throw null; } set { } }
6411+
public virtual bool Accessible { [System.ObsoleteAttribute("deprecated")] get { throw null; } set { } }
64126412
[System.ComponentModel.EditorBrowsableAttribute(1)]
64136413
[System.Diagnostics.DebuggerBrowsableAttribute(0)]
64146414
public override Java.Interop.JniPeerMembers JniPeerMembers { get { throw null; } }

‎src/Java.Interop.Export/Java.Interop.Export.csproj

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
<Project Sdk="Microsoft.NET.Sdk">
33
<PropertyGroup>
44
<TargetFrameworks>netstandard2.0;$(DotNetTargetFramework)</TargetFrameworks>
5-
<LangVersion>8.0</LangVersion>
5+
<LangVersion>9.0</LangVersion>
66
<ProjectGuid>{B501D075-6183-4E1D-92C9-F7B5002475B1}</ProjectGuid>
77
<Nullable>enable</Nullable>
88
<SignAssembly>true</SignAssembly>
@@ -23,4 +23,4 @@
2323
<ItemGroup>
2424
<PackageReference Include="Microsoft.SourceLink.GitHub" PrivateAssets="All" />
2525
</ItemGroup>
26-
</Project>
26+
</Project>

‎src/Java.Interop.Export/Java.Interop/MarshalMemberBuilder.cs

+9-15
Original file line numberDiff line numberDiff line change
@@ -84,20 +84,6 @@ public string GetJniMethodSignature (JavaCallableAttribute export, MethodInfo me
8484
return export.Signature = GetJniMethodSignature (method);
8585
}
8686

87-
string GetTypeSignature (ParameterInfo p)
88-
{
89-
var info = Runtime.TypeManager.GetTypeSignature (p.ParameterType);
90-
if (info.IsValid)
91-
return info.QualifiedReference;
92-
93-
var marshaler = GetParameterMarshaler (p);
94-
info = Runtime.TypeManager.GetTypeSignature (marshaler.MarshalType);
95-
if (info.IsValid)
96-
return info.QualifiedReference;
97-
98-
throw new NotSupportedException ("Don't know how to determine JNI signature for parameter type: " + p.ParameterType.FullName + ".");
99-
}
100-
10187
Delegate CreateJniMethodMarshaler (MethodInfo method, JavaCallableAttribute? export, Type? type)
10288
{
10389
var e = CreateMarshalToManagedExpression (method, export, type);
@@ -242,6 +228,7 @@ public LambdaExpression CreateMarshalToManagedExpression (MethodInfo method, Jav
242228
: Expression.Lambda (marshalerType, body, bodyParams);
243229
}
244230

231+
// Keep in sync with ExpressionAssemblyBuilder.GetMarshalMethodDelegateType()
245232
static Type? GetMarshalerType (Type? returnType, List<Type> funcTypeParams, Type? declaringType)
246233
{
247234
// Too many parameters; does a `_JniMarshal_*` type exist in the type's declaring assembly?
@@ -277,6 +264,7 @@ public LambdaExpression CreateMarshalToManagedExpression (MethodInfo method, Jav
277264
static AssemblyBuilder? assemblyBuilder;
278265
static ModuleBuilder? moduleBuilder;
279266
static Type[]? DelegateCtorSignature;
267+
static Dictionary<string, Type>? marshalDelegateTypes;
280268

281269
static Type? CreateMarshalDelegateType (string name, Type? returnType, List<Type> funcTypeParams)
282270
{
@@ -290,6 +278,10 @@ public LambdaExpression CreateMarshalToManagedExpression (MethodInfo method, Jav
290278
typeof (object),
291279
typeof (IntPtr)
292280
};
281+
marshalDelegateTypes = new ();
282+
}
283+
if (marshalDelegateTypes!.TryGetValue (name, out var type)) {
284+
return type;
293285
}
294286
funcTypeParams.Insert (0, typeof (IntPtr));
295287
funcTypeParams.Insert (0, typeof (IntPtr));
@@ -307,7 +299,9 @@ public LambdaExpression CreateMarshalToManagedExpression (MethodInfo method, Jav
307299
.SetImplementationFlags (ImplAttributes);
308300
typeBuilder.DefineMethod ("Invoke", InvokeAttributes, returnType, funcTypeParams.ToArray ())
309301
.SetImplementationFlags (ImplAttributes);
310-
return typeBuilder.CreateTypeInfo ();
302+
var marshalDelType = typeBuilder.CreateTypeInfo ();
303+
marshalDelegateTypes.Add (name, marshalDelType);
304+
return marshalDelType;
311305
}
312306
}
313307
#endif // NET
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
<TargetFramework>$(DotNetTargetFramework)</TargetFramework>
5+
<ImplicitUsings>enable</ImplicitUsings>
6+
<Nullable>enable</Nullable>
7+
</PropertyGroup>
8+
9+
<Import Project="..\..\TargetFrameworkDependentValues.props" />
10+
11+
<PropertyGroup>
12+
<OutputPath>$(UtilityOutputFullPath)</OutputPath>
13+
</PropertyGroup>
14+
15+
<ItemGroup>
16+
<PackageReference Include="Microsoft.SourceLink.GitHub" PrivateAssets="All" />
17+
</ItemGroup>
18+
19+
<Import Project="..\..\build-tools\scripts\cecil.projitems" />
20+
21+
<ItemGroup>
22+
<ProjectReference Include="..\..\src\Java.Interop\Java.Interop.csproj" />
23+
<ProjectReference Include="..\..\src\Java.Interop.Tools.Cecil\Java.Interop.Tools.Cecil.csproj" />
24+
<ProjectReference Include="..\..\src\Java.Interop.Tools.Diagnostics\Java.Interop.Tools.Diagnostics.csproj" />
25+
</ItemGroup>
26+
27+
</Project>

‎src/Java.Interop.Tools.Expressions/Java.Interop.Tools.Expressions/CecilCompilerExpressionVisitor.cs

+827
Large diffs are not rendered by default.

‎src/Java.Interop.Tools.Expressions/Java.Interop.Tools.Expressions/ExpressionAssemblyBuilder.cs

+452
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
using System;
2+
3+
using Mono.Cecil;
4+
5+
namespace Java.Interop.Tools.Expressions;
6+
7+
public record ExpressionMethodRegistration (string JniName, string JniSignature, MethodDefinition MarshalMethodDefinition)
8+
{
9+
}

‎tests/Java.Interop.Export-Tests/Java.Interop/MarshalMemberBuilderTest.cs

+16-6
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@
1212

1313
namespace Java.InteropTests
1414
{
15-
#if !NET
1615
[TestFixture]
1716
class MarshalMemberBuilderTest : JavaVMFixture
1817
{
@@ -27,11 +26,17 @@ public void AddExportMethods ()
2726

2827
Assert.AreEqual ("action", methods [0].Name);
2928
Assert.AreEqual ("()V", methods [0].Signature);
30-
Assert.IsTrue (methods [0].Marshaler is Action<IntPtr, IntPtr>);
3129

32-
Assert.AreEqual ("staticAction", methods [1].Name);
33-
Assert.AreEqual ("()V", methods [1].Signature);
30+
Assert.AreEqual ("staticAction", methods [1].Name);
31+
Assert.AreEqual ("()V", methods [1].Signature);
32+
33+
#if NET
34+
Assert.AreEqual ("_JniMarshal_PP_V", methods [0].Marshaler.GetType ().FullName);
35+
Assert.AreEqual ("_JniMarshal_PP_V", methods [1].Marshaler.GetType ().FullName);
36+
#else
37+
Assert.IsTrue (methods [0].Marshaler is Action<IntPtr, IntPtr>);
3438
Assert.IsTrue (methods [1].Marshaler is Action<IntPtr, IntPtr>);
39+
#endif // NET
3540

3641
var m = t.GetStaticMethod ("testStaticMethods", "()V");
3742
JniEnvironment.StaticMethods.CallStaticVoidMethod (t.PeerReference, m);
@@ -201,6 +206,12 @@ static void CheckExpression (LambdaExpression expression, string memberName, Typ
201206
{
202207
Console.WriteLine ("## member: {0}", memberName);
203208
Console.WriteLine (expression.ToCSharpCode ());
209+
Assert.AreEqual (expectedBody, expression.ToCSharpCode ());
210+
#if NET
211+
// TODO: Use src/Java.Interop.Tools.Expressions to compile `expression`
212+
// and use the "IL decompiler" in tests/Java.Interop.Tools.Expressions-Tests
213+
// to verify the expected IL
214+
#else
204215
var da = AppDomain.CurrentDomain.DefineDynamicAssembly(
205216
new AssemblyName("dyn"), // call it whatever you want
206217
System.Reflection.Emit.AssemblyBuilderAccess.Save,
@@ -216,10 +227,10 @@ static void CheckExpression (LambdaExpression expression, string memberName, Typ
216227
expression.CompileToMethod (mb);
217228
dt.CreateType();
218229
Assert.AreEqual (expressionType, expression.Type);
219-
Assert.AreEqual (expectedBody, expression.ToCSharpCode ());
220230
#if !__ANDROID__
221231
da.Save (_name);
222232
#endif // !__ANDROID__
233+
#endif // !NET
223234
}
224235

225236
[Test]
@@ -556,5 +567,4 @@ public void CreateConstructActivationPeerExpression ()
556567
}}");
557568
}
558569
}
559-
#endif // !NET
560570
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
<Project Sdk="Microsoft.NET.Sdk">
2+
3+
<PropertyGroup>
4+
<TargetFramework>$(DotNetTargetFramework)</TargetFramework>
5+
<RootNamespace>Java.Interop.Tools.ExpressionsTests</RootNamespace>
6+
<ImplicitUsings>enable</ImplicitUsings>
7+
<Nullable>enable</Nullable>
8+
<IsPackable>false</IsPackable>
9+
</PropertyGroup>
10+
11+
<Import Project="..\..\TargetFrameworkDependentValues.props" />
12+
13+
<PropertyGroup>
14+
<OutputPath>$(TestOutputFullPath)</OutputPath>
15+
</PropertyGroup>
16+
17+
<ItemGroup>
18+
<PackageReference Include="Microsoft.NET.Test.Sdk" />
19+
<PackageReference Include="NUnit" />
20+
<PackageReference Include="NUnit3TestAdapter" />
21+
</ItemGroup>
22+
23+
<!--
24+
<Import Project="..\..\build-tools\scripts\cecil.projitems" />
25+
-->
26+
<ItemGroup>
27+
<PackageReference Include="Mono.Cecil" Version="0.11.4" />
28+
<PackageReference Include="Mono.Linq.Expressions" />
29+
</ItemGroup>
30+
31+
32+
<ItemGroup>
33+
<ProjectReference Include="..\..\src\Java.Interop.Tools.Expressions\Java.Interop.Tools.Expressions.csproj" />
34+
</ItemGroup>
35+
36+
</Project>

‎tests/Java.Interop.Tools.Expressions-Tests/Java.Interop.Tools.ExpressionsTests/ExpressionAssemblyBuilderTests.cs

+444
Large diffs are not rendered by default.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
global using NUnit.Framework;

‎tools/jnimarshalmethod-gen/App.cs

+252-96
Large diffs are not rendered by default.

‎tools/jnimarshalmethod-gen/Xamarin.Android.Tools.JniMarshalMethodGenerator.csproj

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<Project Sdk="Microsoft.NET.Sdk">
22

33
<PropertyGroup>
4-
<TargetFramework>net472</TargetFramework>
4+
<TargetFramework>$(DotNetTargetFramework)</TargetFramework>
55
<OutputType>Exe</OutputType>
66
<LangVersion>8.0</LangVersion>
77
<AssemblyName>jnimarshalmethod-gen</AssemblyName>
@@ -13,6 +13,7 @@
1313

1414
<PropertyGroup>
1515
<OutputPath>$(UtilityOutputFullPath)</OutputPath>
16+
<_DumpRegisterNativeMembers>True</_DumpRegisterNativeMembers>
1617
</PropertyGroup>
1718

1819
<PropertyGroup Condition=" '$(_DumpRegisterNativeMembers)' == 'True' ">
@@ -35,6 +36,7 @@
3536
<ProjectReference Include="..\..\src\Java.Interop\Java.Interop.csproj" />
3637
<ProjectReference Include="..\..\src\Java.Runtime.Environment\Java.Runtime.Environment.csproj" />
3738
<ProjectReference Include="..\..\src\Java.Interop.Tools.Cecil\Java.Interop.Tools.Cecil.csproj" />
39+
<ProjectReference Include="..\..\src\Java.Interop.Tools.Expressions\Java.Interop.Tools.Expressions.csproj" />
3840
</ItemGroup>
3941

4042
</Project>

0 commit comments

Comments
 (0)
Please sign in to comment.