Merge pull request #1631 from alexanderkyte/stringbuilder-referencesource
[mono.git] / mcs / class / corlib / System.Reflection.Emit / MethodOnTypeBuilderInst.cs
1 //
2 // System.Reflection.Emit/MethodOnTypeBuilderInst.cs
3 //
4 // Author:
5 //   Zoltan Varga (vargaz@gmail.com)
6 //
7 //
8 // Copyright (C) 2008 Novell, Inc (http://www.novell.com)
9 //
10 // Permission is hereby granted, free of charge, to any person obtaining
11 // a copy of this software and associated documentation files (the
12 // "Software"), to deal in the Software without restriction, including
13 // without limitation the rights to use, copy, modify, merge, publish,
14 // distribute, sublicense, and/or sell copies of the Software, and to
15 // permit persons to whom the Software is furnished to do so, subject to
16 // the following conditions:
17 // 
18 // The above copyright notice and this permission notice shall be
19 // included in all copies or substantial portions of the Software.
20 // 
21 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
22 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
23 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
24 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
25 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
26 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
27 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
28 //
29
30 #if !FULL_AOT_RUNTIME
31 using System;
32 using System.Globalization;
33 using System.Reflection;
34 using System.Text;
35 using System.Runtime.InteropServices;
36
37
38 namespace System.Reflection.Emit
39 {
40         /*
41          * This class represents a method of an instantiation of a generic type builder.
42          */
43         [StructLayout (LayoutKind.Sequential)]
44         internal class MethodOnTypeBuilderInst : MethodInfo
45         {
46                 #region Keep in sync with object-internals.h
47                 Type instantiation;
48                 MethodInfo base_method; /*This is the base method definition, it must be non-inflated and belong to a non-inflated type.*/
49                 Type[] method_arguments;
50                 #endregion
51                 MethodInfo generic_method_definition;
52
53                 public MethodOnTypeBuilderInst (MonoGenericClass instantiation, MethodInfo base_method)
54                 {
55                         this.instantiation = instantiation;
56                         this.base_method = base_method;
57                 }
58
59                 internal MethodOnTypeBuilderInst (MethodOnTypeBuilderInst gmd, Type[] typeArguments)
60                 {
61                         this.instantiation = gmd.instantiation;
62                         this.base_method = gmd.base_method;
63                         this.method_arguments = new Type [typeArguments.Length];
64                         typeArguments.CopyTo (this.method_arguments, 0);
65                         this.generic_method_definition = gmd;
66                 }
67
68                 internal MethodOnTypeBuilderInst (MethodInfo method, Type[] typeArguments)
69                 {
70                         this.instantiation = method.DeclaringType;
71                         this.base_method = ExtractBaseMethod (method);
72                         this.method_arguments = new Type [typeArguments.Length];
73                         typeArguments.CopyTo (this.method_arguments, 0);
74                         if (base_method != method)
75                                 this.generic_method_definition = method;
76                 }
77
78                 static MethodInfo ExtractBaseMethod (MethodInfo info)
79                 {
80                         if (info is MethodBuilder)
81                                 return info;
82                         if (info is MethodOnTypeBuilderInst)
83                                 return ((MethodOnTypeBuilderInst)info).base_method;
84
85                         if (info.IsGenericMethod)
86                                 info = info.GetGenericMethodDefinition ();
87
88                         Type t = info.DeclaringType;
89                         if (!t.IsGenericType || t.IsGenericTypeDefinition)
90                                 return info;
91
92                         return (MethodInfo)t.Module.ResolveMethod (info.MetadataToken);
93                 }
94
95                 internal Type[] GetTypeArgs ()
96                 {
97                         if (!instantiation.IsGenericType || instantiation.IsGenericParameter)
98                                 return null;                            
99
100                         return instantiation.GetGenericArguments ();
101                 }
102
103                 //
104                 // MemberInfo members
105                 //
106                 
107                 public override Type DeclaringType {
108                         get {
109                                 return instantiation;
110                         }
111                 }
112
113                 public override string Name {
114                         get {
115                                 return base_method.Name;
116                         }
117                 }
118
119                 public override Type ReflectedType {
120                         get {
121                                 return instantiation;
122                         }
123                 }
124
125                 public override Type ReturnType {
126                         get { 
127                                 return base_method.ReturnType;
128                         }
129                 }
130
131                 public override Module Module {
132                         get {
133                                 return base_method.Module;
134                         }
135                 }
136
137                 public override bool IsDefined (Type attributeType, bool inherit)
138                 {
139                         throw new NotSupportedException ();
140                 }
141
142                 public override object [] GetCustomAttributes (bool inherit)
143                 {
144                         throw new NotSupportedException ();
145                 }
146
147                 public override object [] GetCustomAttributes (Type attributeType, bool inherit)
148                 {
149                         throw new NotSupportedException ();
150                 }
151
152                 public override string ToString ()
153                 {
154                          //IEnumerable`1 get_Item(TKey)
155                          StringBuilder sb = new StringBuilder (ReturnType.ToString ());
156                          sb.Append (" ");
157                          sb.Append (base_method.Name);
158                          sb.Append ("(");
159                          sb.Append (")");
160                          return sb.ToString ();
161                 }
162                 //
163                 // MethodBase members
164                 //
165
166                 public override MethodImplAttributes GetMethodImplementationFlags ()
167                 {
168                         return base_method.GetMethodImplementationFlags ();
169                 }
170
171                 public override ParameterInfo [] GetParameters ()
172                 {
173                         return GetParametersInternal ();
174                 }
175
176                 internal override ParameterInfo [] GetParametersInternal ()
177                 {
178                         throw new NotSupportedException ();
179                 }
180
181                 public override int MetadataToken {
182                         get {
183                                 return base.MetadataToken;
184                         }
185                 }
186
187                 internal override int GetParametersCount ()
188                 {
189                         return base_method.GetParametersCount ();
190                 }
191
192                 public override Object Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture)
193                 {
194                         throw new NotSupportedException ();
195                 }
196
197                 public override RuntimeMethodHandle MethodHandle {
198                         get {
199                                 throw new NotSupportedException ();
200                         }
201                 }
202
203                 public override MethodAttributes Attributes {
204                         get {
205                                 return base_method.Attributes;
206                         }
207                 }
208
209                 public override CallingConventions CallingConvention {
210                         get {
211                                 return base_method.CallingConvention;
212                         }
213                 }
214
215                 public override MethodInfo MakeGenericMethod (params Type [] methodInstantiation)
216                 {
217                         if (!base_method.IsGenericMethodDefinition || (method_arguments != null))
218                                 throw new InvalidOperationException ("Method is not a generic method definition");
219
220                         if (methodInstantiation == null)
221                                 throw new ArgumentNullException ("methodInstantiation");
222
223                         if (base_method.GetGenericArguments ().Length != methodInstantiation.Length)
224                                 throw new ArgumentException ("Incorrect length", "methodInstantiation");
225
226                         foreach (Type type in methodInstantiation) {
227                                 if (type == null)
228                                         throw new ArgumentNullException ("methodInstantiation");
229                         }
230
231                         return new MethodOnTypeBuilderInst (this, methodInstantiation);
232                 }
233
234                 public override Type [] GetGenericArguments ()
235                 {
236                         if (!base_method.IsGenericMethodDefinition)
237                                 return null;
238                         Type[] source = method_arguments ?? base_method.GetGenericArguments ();
239                         Type[] result = new Type [source.Length];
240                         source.CopyTo (result, 0);
241                         return result;
242                 }
243
244                 public override MethodInfo GetGenericMethodDefinition ()
245                 {
246                         return generic_method_definition ?? base_method;
247                 }
248
249                 public override bool ContainsGenericParameters {
250                         get {
251                                 if (base_method.ContainsGenericParameters)
252                                         return true;
253                                 if (!base_method.IsGenericMethodDefinition)
254                                         throw new NotSupportedException ();
255                                 if (method_arguments == null)
256                                         return true;
257                                 foreach (Type t in method_arguments) {
258                                         if (t.ContainsGenericParameters)
259                                                 return true;
260                                 }
261                                 return false;
262                         }
263                 }
264
265                 public override bool IsGenericMethodDefinition {
266                         get {
267                                 return base_method.IsGenericMethodDefinition && method_arguments == null;
268                         }
269                 }
270
271                 public override bool IsGenericMethod {
272                         get {
273                                 return base_method.IsGenericMethodDefinition;
274                         }
275                 }
276
277                 //
278                 // MethodInfo members
279                 //
280
281                 public override MethodInfo GetBaseDefinition ()
282                 {
283                         throw new NotSupportedException ();
284                 }
285
286                 public override ICustomAttributeProvider ReturnTypeCustomAttributes {
287                         get {
288                                 throw new NotSupportedException ();
289                         }
290                 }
291         }
292 }
293
294 #endif