|
| 1 | +using System; |
| 2 | +using System.Text; |
| 3 | +using System.Xml.Linq; |
| 4 | +using Android.Runtime; |
| 5 | +using Java.Interop.Tools.Diagnostics; |
| 6 | +using Java.Interop.Tools.JavaCallableWrappers.CallableWrapperMembers; |
| 7 | +using Java.Interop.Tools.TypeNameMappings; |
| 8 | +using Mono.Cecil; |
| 9 | + |
| 10 | +namespace Java.Interop.Tools.JavaCallableWrappers.Adapters; |
| 11 | + |
| 12 | +class CecilImporter |
| 13 | +{ |
| 14 | + public static CallableWrapperField CreateField (MethodDefinition method, string fieldName, IMetadataResolver resolver) |
| 15 | + { |
| 16 | + var visibility = JavaCallableWrapperGenerator.GetJavaAccess (method.Attributes & MethodAttributes.MemberAccessMask); |
| 17 | + var type_name = JavaNativeTypeManager.ReturnTypeFromSignature (JavaNativeTypeManager.GetJniSignature (method, resolver))?.Type |
| 18 | + ?? throw new ArgumentException ($"Could not get JNI signature for method `{method.Name}`", nameof (method)); |
| 19 | + var annotations = JavaCallableWrapperGenerator.GetAnnotationsString ("\t", method.CustomAttributes, resolver); |
| 20 | + |
| 21 | + return new CallableWrapperField ( |
| 22 | + fieldName: fieldName, |
| 23 | + typeName: type_name, |
| 24 | + visibility: visibility, |
| 25 | + isStatic: method.IsStatic, |
| 26 | + initializerName: method.Name, |
| 27 | + annotations: annotations); |
| 28 | + } |
| 29 | + |
| 30 | + // Temporary conversion function |
| 31 | + public static CallableWrapperField CreateField (JavaCallableWrapperGenerator.JavaFieldInfo field) |
| 32 | + { |
| 33 | + return new CallableWrapperField ( |
| 34 | + fieldName: field.FieldName, |
| 35 | + typeName: field.TypeName, |
| 36 | + visibility: field.GetJavaAccess (), |
| 37 | + isStatic: field.IsStatic, |
| 38 | + initializerName: field.InitializerName, |
| 39 | + annotations: field.Annotations); |
| 40 | + } |
| 41 | + |
| 42 | + public static CallableWrapperMethod CreateMethod (MethodDefinition method, RegisterAttribute register, IMetadataResolver cache, bool shouldBeDynamicallyRegistered = true) |
| 43 | + => CreateMethod (method, register, null, null, cache, shouldBeDynamicallyRegistered); |
| 44 | + |
| 45 | + public static CallableWrapperMethod CreateMethod (MethodDefinition methodDefinition, RegisterAttribute register, string? managedParameters, string? outerType, IMetadataResolver cache, bool shouldBeDynamicallyRegistered = true) |
| 46 | + { |
| 47 | + var method = CreateMethod (register.Name, register.Signature, register.Connector, managedParameters, outerType, null); |
| 48 | + |
| 49 | + method.Annotations = JavaCallableWrapperGenerator.GetAnnotationsString ("\t", methodDefinition.CustomAttributes, cache); |
| 50 | + method.IsDynamicallyRegistered = shouldBeDynamicallyRegistered; |
| 51 | + |
| 52 | + return method; |
| 53 | + } |
| 54 | + |
| 55 | + public static CallableWrapperMethod CreateMethod (MethodDefinition methodDefinition, ExportAttribute export, string? managedParameters, IMetadataResolver resolver) |
| 56 | + { |
| 57 | + var method = CreateMethod (methodDefinition.Name, JavaNativeTypeManager.GetJniSignature (methodDefinition, resolver), "__export__", null, null, export.SuperArgumentsString); |
| 58 | + |
| 59 | + method.IsExport = true; |
| 60 | + method.IsStatic = methodDefinition.IsStatic; |
| 61 | + method.JavaAccess = JavaCallableWrapperGenerator.GetJavaAccess (methodDefinition.Attributes & MethodAttributes.MemberAccessMask); |
| 62 | + method.ThrownTypeNames = export.ThrownNames; |
| 63 | + method.JavaNameOverride = export.Name; |
| 64 | + method.ManagedParameters = managedParameters; |
| 65 | + method.Annotations = JavaCallableWrapperGenerator.GetAnnotationsString ("\t", methodDefinition.CustomAttributes, resolver); |
| 66 | + |
| 67 | + return method; |
| 68 | + } |
| 69 | + |
| 70 | + public static CallableWrapperMethod CreateMethod (MethodDefinition methodDefinition, IMetadataResolver resolver) |
| 71 | + { |
| 72 | + var method = CreateMethod (methodDefinition.Name, JavaNativeTypeManager.GetJniSignature (methodDefinition, resolver), "__export__", null, null, null); |
| 73 | + |
| 74 | + if (methodDefinition.HasParameters) |
| 75 | + Diagnostic.Error (4205, JavaCallableWrapperGenerator.LookupSource (methodDefinition), Localization.Resources.JavaCallableWrappers_XA4205); |
| 76 | + if (methodDefinition.ReturnType.MetadataType == MetadataType.Void) |
| 77 | + Diagnostic.Error (4208, JavaCallableWrapperGenerator.LookupSource (methodDefinition), Localization.Resources.JavaCallableWrappers_XA4208); |
| 78 | + |
| 79 | + method.IsExport = true; |
| 80 | + method.IsStatic = method.IsStatic; |
| 81 | + method.JavaAccess = JavaCallableWrapperGenerator.GetJavaAccess (methodDefinition.Attributes & MethodAttributes.MemberAccessMask); |
| 82 | + |
| 83 | + // Annotations are processed within CallableWrapperField, not the initializer method. So we don't generate them here. |
| 84 | + |
| 85 | + return method; |
| 86 | + } |
| 87 | + |
| 88 | + public static CallableWrapperMethod CreateMethod (string name, string? signature, string? connector, string? managedParameters, string? outerType, string? superCall) |
| 89 | + { |
| 90 | + signature = signature ?? throw new ArgumentNullException ("`connector` cannot be null.", nameof (connector)); |
| 91 | + var method_name = "n_" + name + ":" + signature + ":" + connector; |
| 92 | + |
| 93 | + var method = new CallableWrapperMethod (name, method_name, signature) { |
| 94 | + ManagedParameters = managedParameters |
| 95 | + }; |
| 96 | + |
| 97 | + var jnisig = signature; |
| 98 | + var closer = jnisig.IndexOf (')'); |
| 99 | + var ret = jnisig.Substring (closer + 1); |
| 100 | + method.Retval = JavaNativeTypeManager.Parse (ret)?.Type; |
| 101 | + |
| 102 | + var jniparms = jnisig.Substring (1, closer - 1); |
| 103 | + |
| 104 | + if (string.IsNullOrEmpty (jniparms) && string.IsNullOrEmpty (superCall)) |
| 105 | + return method; |
| 106 | + |
| 107 | + var parms = new StringBuilder (); |
| 108 | + var scall = new StringBuilder (); |
| 109 | + var acall = new StringBuilder (); |
| 110 | + var first = true; |
| 111 | + var i = 0; |
| 112 | + |
| 113 | + foreach (var jti in JavaNativeTypeManager.FromSignature (jniparms)) { |
| 114 | + if (outerType != null) { |
| 115 | + acall.Append (outerType).Append (".this"); |
| 116 | + outerType = null; |
| 117 | + continue; |
| 118 | + } |
| 119 | + |
| 120 | + var parmType = jti.Type; |
| 121 | + |
| 122 | + if (!first) { |
| 123 | + parms.Append (", "); |
| 124 | + scall.Append (", "); |
| 125 | + acall.Append (", "); |
| 126 | + } |
| 127 | + |
| 128 | + first = false; |
| 129 | + parms.Append (parmType).Append (" p").Append (i); |
| 130 | + scall.Append ("p").Append (i); |
| 131 | + acall.Append ("p").Append (i); |
| 132 | + ++i; |
| 133 | + } |
| 134 | + |
| 135 | + method.Params = parms.ToString (); |
| 136 | + method.SuperCall = superCall ?? scall.ToString (); |
| 137 | + method.ActivateCall = acall.ToString (); |
| 138 | + |
| 139 | + return method; |
| 140 | + } |
| 141 | + |
| 142 | + // Temporary conversion function |
| 143 | + public static CallableWrapperMethod CreateMethod (JavaCallableWrapperGenerator.Signature signature) |
| 144 | + { |
| 145 | + return new CallableWrapperMethod (signature.Name, signature.Method, signature.JniSignature) { |
| 146 | + ManagedParameters = signature.ManagedParameters, |
| 147 | + JavaNameOverride = signature.JavaNameOverride, |
| 148 | + Params = signature.Params, |
| 149 | + Retval = signature.Retval, |
| 150 | + ThrowsDeclaration = signature.ThrowsDeclaration, |
| 151 | + JavaAccess = signature.JavaAccess, |
| 152 | + IsExport = signature.IsExport, |
| 153 | + IsStatic = signature.IsStatic, |
| 154 | + IsDynamicallyRegistered = signature.IsDynamicallyRegistered, |
| 155 | + ThrownTypeNames = signature.ThrownTypeNames, |
| 156 | + Annotations = signature.Annotations, |
| 157 | + SuperCall = signature.SuperCall, |
| 158 | + ActivateCall = signature.ActivateCall, |
| 159 | + }; |
| 160 | + } |
| 161 | + |
| 162 | + // Temporary conversion function |
| 163 | + public static CallableWrapperConstructor CreateConstructor (JavaCallableWrapperGenerator.Signature signature) |
| 164 | + { |
| 165 | + return new CallableWrapperConstructor (signature.Name, signature.Method, signature.JniSignature) { |
| 166 | + ManagedParameters = signature.ManagedParameters, |
| 167 | + JavaNameOverride = signature.JavaNameOverride, |
| 168 | + Params = signature.Params, |
| 169 | + Retval = signature.Retval, |
| 170 | + ThrowsDeclaration = signature.ThrowsDeclaration, |
| 171 | + JavaAccess = signature.JavaAccess, |
| 172 | + IsExport = signature.IsExport, |
| 173 | + IsStatic = signature.IsStatic, |
| 174 | + IsDynamicallyRegistered = signature.IsDynamicallyRegistered, |
| 175 | + ThrownTypeNames = signature.ThrownTypeNames, |
| 176 | + Annotations = signature.Annotations, |
| 177 | + SuperCall = signature.SuperCall, |
| 178 | + ActivateCall = signature.ActivateCall |
| 179 | + }; |
| 180 | + } |
| 181 | + |
| 182 | + public static CallableWrapperApplicationConstructor? CreateApplicationConstructor (string name, TypeDefinition type, IMetadataResolver resolver) |
| 183 | + { |
| 184 | + if (!JavaNativeTypeManager.IsApplication (type, resolver)) |
| 185 | + return null; |
| 186 | + |
| 187 | + return new CallableWrapperApplicationConstructor (name); |
| 188 | + } |
| 189 | + |
| 190 | + // Temporary conversion function |
| 191 | + public static CallableWrapperType CreateType (JavaCallableWrapperGenerator generator) |
| 192 | + { |
| 193 | + var type = new CallableWrapperType (generator.name, generator.package) { |
| 194 | + Type = generator.type, |
| 195 | + Cache = generator.cache, |
| 196 | + IsAbstract = generator.type.IsAbstract, |
| 197 | + ApplicationJavaClass = generator.ApplicationJavaClass, |
| 198 | + Generator = generator, |
| 199 | + HasDynamicallyRegisteredMethods = generator.HasDynamicallyRegisteredMethods, |
| 200 | + GenerateOnCreateOverrides = generator.GenerateOnCreateOverrides, |
| 201 | + MonoRuntimeInitialization = generator.MonoRuntimeInitialization, |
| 202 | + }; |
| 203 | + |
| 204 | + if (generator.children is not null) |
| 205 | + foreach (var nested in generator.children) |
| 206 | + type.NestedTypes.Add (CreateType (nested)); |
| 207 | + |
| 208 | + return type; |
| 209 | + } |
| 210 | +} |
0 commit comments