-
Notifications
You must be signed in to change notification settings - Fork 569
Expand file tree
/
Copy pathGenerateAdditionalProviderSources.cs
More file actions
187 lines (153 loc) · 6.62 KB
/
GenerateAdditionalProviderSources.cs
File metadata and controls
187 lines (153 loc) · 6.62 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
#nullable enable
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using Java.Interop.Tools.JavaCallableWrappers;
using Microsoft.Android.Build.Tasks;
using Microsoft.Build.Framework;
namespace Xamarin.Android.Tasks;
public class GenerateAdditionalProviderSources : AndroidTask
{
public override string TaskPrefix => "GPS";
[Required]
public string [] AdditionalProviderSources { get; set; } = [];
[Required]
public string AndroidRuntime { get; set; } = "";
public string CodeGenerationTarget { get; set; } = "";
[Required]
public string IntermediateOutputDirectory { get; set; } = "";
public string? OutputDirectory { get; set; }
[Required]
public string TargetName { get; set; } = "";
public ITaskItem[]? Environments { get; set; }
// We need to pass these two to the environment builder, otherwise not used
// by this task. See also GenerateNativeApplicationSources.cs
public string? HttpClientHandlerType { get; set; }
public bool EnableSGenConcurrent { get; set; }
/// <summary>
/// When true, <c>ApplicationRegistration.java</c> is generated by the inner build's
/// <see cref="RewriteMarshalMethods"/> task (which has the classifier to filter types).
/// This task skips generating it to avoid overwriting the inner build's output.
/// </summary>
public bool EnableMarshalMethods { get; set; }
AndroidRuntime androidRuntime;
JavaPeerStyle codeGenerationTarget;
public override bool RunTask ()
{
androidRuntime = MonoAndroidHelper.ParseAndroidRuntime (AndroidRuntime);
codeGenerationTarget = MonoAndroidHelper.ParseCodeGenerationTarget (CodeGenerationTarget);
// Retrieve the stored NativeCodeGenStateObject
var nativeCodeGenStates = BuildEngine4.GetRegisteredTaskObjectAssemblyLocal<NativeCodeGenStateCollection> (
MonoAndroidHelper.GetProjectBuildSpecificTaskObjectKey (GenerateJavaStubs.NativeCodeGenStateObjectRegisterTaskKey, WorkingDirectory, IntermediateOutputDirectory),
RegisteredTaskObjectLifetime.Build
);
// We only need the first architecture, since this task is architecture-agnostic
var templateCodeGenState = nativeCodeGenStates.States.First ().Value;
Generate (templateCodeGenState);
return !Log.HasLoggedErrors;
}
void Generate (NativeCodeGenStateObject codeGenState)
{
// Create additional runtime provider java sources.
bool isMonoVM = androidRuntime switch {
Xamarin.Android.Tasks.AndroidRuntime.MonoVM => true,
Xamarin.Android.Tasks.AndroidRuntime.CoreCLR => true,
_ => false,
};
string providerTemplateFile = isMonoVM ?
"MonoRuntimeProvider.Bundled.java" :
"NativeAotRuntimeProvider.java";
string providerTemplate = GetResource (providerTemplateFile);
foreach (var provider in AdditionalProviderSources) {
var contents = providerTemplate.Replace (isMonoVM ? "MonoRuntimeProvider" : "NativeAotRuntimeProvider", provider);
var real_provider = isMonoVM ?
Path.Combine (OutputDirectory, "src", "mono", provider + ".java") :
Path.Combine (OutputDirectory, "src", "net", "dot", "jni", "nativeaot", provider + ".java");
Files.CopyIfStringChanged (contents, real_provider);
}
// For NativeAOT, generate JavaInteropRuntime.java and NativeAotEnvironmentVars.java
if (androidRuntime == Xamarin.Android.Tasks.AndroidRuntime.NativeAOT) {
GenerateJavaSource (
"JavaInteropRuntime.java",
new Dictionary<string, string> (StringComparer.Ordinal) {
{ "@MAIN_ASSEMBLY_NAME@", TargetName },
}
);
// We care only about environment variables here
var envBuilder = new EnvironmentBuilder (Log);
envBuilder.Read (Environments);
GenerateNativeApplicationConfigSources.AddDefaultEnvironmentVariables (envBuilder, HttpClientHandlerType, EnableSGenConcurrent);
var envVarNames = new StringBuilder ();
var envVarValues = new StringBuilder ();
foreach (var kvp in envBuilder.EnvironmentVariables) {
// All the strings already have double-quotes properly quoted, EnvironmentBuilder took care of that
AppendEnvVarEntry (envVarNames, kvp.Key);
AppendEnvVarEntry (envVarValues, kvp.Value);
}
var envVars = new Dictionary<string, string> (StringComparer.Ordinal) {
{ "@ENVIRONMENT_VAR_NAMES@", envVarNames.ToString () },
{ "@ENVIRONMENT_VAR_VALUES@", envVarValues.ToString () },
};
GenerateJavaSource (
"NativeAotEnvironmentVars.java",
envVars
);
}
// Create additional application java sources.
// When marshal methods are enabled, ApplicationRegistration.java is generated
// by RewriteMarshalMethods in the inner build, which has the classifier to
// filter out types with no dynamically registered methods.
if (!EnableMarshalMethods) {
StringWriter regCallsWriter = new StringWriter ();
regCallsWriter.WriteLine ("// Application and Instrumentation ACWs must be registered first.");
foreach ((string jniName, string assemblyQualifiedName) in codeGenState.ApplicationsAndInstrumentationsToRegister) {
regCallsWriter.WriteLine (
codeGenerationTarget == JavaPeerStyle.XAJavaInterop1 ?
"\t\tmono.android.Runtime.register (\"{0}\", {1}.class, {1}.__md_methods);" :
"\t\tnet.dot.jni.ManagedPeer.registerNativeMembers ({1}.class, {1}.__md_methods);",
assemblyQualifiedName,
jniName
);
}
regCallsWriter.Close ();
var real_app_dir = Path.Combine (OutputDirectory, "src", "net", "dot", "android");
string applicationTemplateFile = "ApplicationRegistration.java";
SaveResource (
applicationTemplateFile,
applicationTemplateFile,
real_app_dir,
template => template.Replace ("// REGISTER_APPLICATION_AND_INSTRUMENTATION_CLASSES_HERE", regCallsWriter.ToString ())
);
}
void AppendEnvVarEntry (StringBuilder sb, string value)
{
sb.Append ("\t\t\"");
sb.Append (value);
sb.Append ("\",\n");
}
void GenerateJavaSource (string fileName, Dictionary<string, string> replacements)
{
var template = new StringBuilder (GetResource (fileName));
foreach (var kvp in replacements) {
template.Replace (kvp.Key, kvp.Value);
}
var path = Path.Combine (OutputDirectory, "src", "net", "dot", "jni", "nativeaot", fileName);
Log.LogDebugMessage ($"Writing: {path}");
Files.CopyIfStringChanged (template.ToString (), path);
}
}
string GetResource (string resource)
{
using (var stream = GetType ().Assembly.GetManifestResourceStream (resource))
using (var reader = new StreamReader (stream))
return reader.ReadToEnd ();
}
void SaveResource (string resource, string filename, string destDir, Func<string, string> applyTemplate)
{
string template = GetResource (resource);
template = applyTemplate (template);
Files.CopyIfStringChanged (template, Path.Combine (destDir, filename));
}
}