Merge pull request #487 from mayerwin/patch-1
[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 bool IsDefined (Type attributeType, bool inherit)
132                 {
133                         throw new NotSupportedException ();
134                 }
135
136                 public override object [] GetCustomAttributes (bool inherit)
137                 {
138                         throw new NotSupportedException ();
139                 }
140
141                 public override object [] GetCustomAttributes (Type attributeType, bool inherit)
142                 {
143                         throw new NotSupportedException ();
144                 }
145
146                 public override string ToString ()
147                 {
148                          //IEnumerable`1 get_Item(TKey)
149                          StringBuilder sb = new StringBuilder (ReturnType.ToString ());
150                          sb.Append (" ");
151                          sb.Append (base_method.Name);
152                          sb.Append ("(");
153                          sb.Append (")");
154                          return sb.ToString ();
155                 }
156                 //
157                 // MethodBase members
158                 //
159
160                 public override MethodImplAttributes GetMethodImplementationFlags ()
161                 {
162                         return base_method.GetMethodImplementationFlags ();
163                 }
164
165                 public override ParameterInfo [] GetParameters ()
166                 {
167                         throw new NotSupportedException ();
168                 }
169
170                 public override int MetadataToken {
171                         get {
172                                 return base.MetadataToken;
173                         }
174                 }
175
176                 internal override int GetParameterCount ()
177                 {
178                         return base_method.GetParameterCount ();
179                 }
180
181                 public override Object Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture)
182                 {
183                         throw new NotSupportedException ();
184                 }
185
186                 public override RuntimeMethodHandle MethodHandle {
187                         get {
188                                 throw new NotSupportedException ();
189                         }
190                 }
191
192                 public override MethodAttributes Attributes {
193                         get {
194                                 return base_method.Attributes;
195                         }
196                 }
197
198                 public override CallingConventions CallingConvention {
199                         get {
200                                 return base_method.CallingConvention;
201                         }
202                 }
203
204                 public override MethodInfo MakeGenericMethod (params Type [] methodInstantiation)
205                 {
206                         if (!base_method.IsGenericMethodDefinition || (method_arguments != null))
207                                 throw new InvalidOperationException ("Method is not a generic method definition");
208
209                         if (methodInstantiation == null)
210                                 throw new ArgumentNullException ("methodInstantiation");
211
212                         if (base_method.GetGenericArguments ().Length != methodInstantiation.Length)
213                                 throw new ArgumentException ("Incorrect length", "methodInstantiation");
214
215                         foreach (Type type in methodInstantiation) {
216                                 if (type == null)
217                                         throw new ArgumentNullException ("methodInstantiation");
218                         }
219
220                         return new MethodOnTypeBuilderInst (this, methodInstantiation);
221                 }
222
223                 public override Type [] GetGenericArguments ()
224                 {
225                         if (!base_method.IsGenericMethodDefinition)
226                                 return null;
227                         Type[] source = method_arguments ?? base_method.GetGenericArguments ();
228                         Type[] result = new Type [source.Length];
229                         source.CopyTo (result, 0);
230                         return result;
231                 }
232
233                 public override MethodInfo GetGenericMethodDefinition ()
234                 {
235                         return generic_method_definition ?? base_method;
236                 }
237
238                 public override bool ContainsGenericParameters {
239                         get {
240                                 if (base_method.ContainsGenericParameters)
241                                         return true;
242                                 if (!base_method.IsGenericMethodDefinition)
243                                         throw new NotSupportedException ();
244                                 if (method_arguments == null)
245                                         return true;
246                                 foreach (Type t in method_arguments) {
247                                         if (t.ContainsGenericParameters)
248                                                 return true;
249                                 }
250                                 return false;
251                         }
252                 }
253
254                 public override bool IsGenericMethodDefinition {
255                         get {
256                                 return base_method.IsGenericMethodDefinition && method_arguments == null;
257                         }
258                 }
259
260                 public override bool IsGenericMethod {
261                         get {
262                                 return base_method.IsGenericMethodDefinition;
263                         }
264                 }
265
266                 //
267                 // MethodInfo members
268                 //
269
270                 public override MethodInfo GetBaseDefinition ()
271                 {
272                         throw new NotSupportedException ();
273                 }
274
275                 public override ICustomAttributeProvider ReturnTypeCustomAttributes {
276                         get {
277                                 throw new NotSupportedException ();
278                         }
279                 }
280         }
281 }
282
283 #endif