2 // expression.cs: Expression representation for the IL tree.
5 // Miguel de Icaza (miguel@ximian.com)
6 // Marek Safar (marek.safar@gmail.com)
8 // Copyright 2001, 2002, 2003 Ximian, Inc.
9 // Copyright 2003-2008 Novell, Inc.
13 namespace Mono
.CSharp
{
15 using System
.Collections
.Generic
;
16 using System
.Reflection
;
17 using System
.Reflection
.Emit
;
20 using SLE
= System
.Linq
.Expressions
;
23 // This is an user operator expression, automatically created during
26 public class UserOperatorCall
: Expression
{
27 public delegate Expression
ExpressionTreeExpression (ResolveContext ec
, MethodGroupExpr mg
);
29 protected readonly Arguments arguments
;
30 protected readonly MethodGroupExpr mg
;
31 readonly ExpressionTreeExpression expr_tree
;
33 public UserOperatorCall (MethodGroupExpr mg
, Arguments args
, ExpressionTreeExpression expr_tree
, Location loc
)
36 this.arguments
= args
;
37 this.expr_tree
= expr_tree
;
39 type
= mg
.BestCandidate
.ReturnType
;
40 eclass
= ExprClass
.Value
;
44 public override Expression
CreateExpressionTree (ResolveContext ec
)
46 if (expr_tree
!= null)
47 return expr_tree (ec
, mg
);
49 Arguments args
= Arguments
.CreateForExpressionTree (ec
, arguments
,
50 new NullLiteral (loc
),
51 mg
.CreateExpressionTree (ec
));
53 return CreateExpressionFactoryCall (ec
, "Call", args
);
56 protected override void CloneTo (CloneContext context
, Expression target
)
61 protected override Expression
DoResolve (ResolveContext ec
)
64 // We are born fully resolved
69 public override void Emit (EmitContext ec
)
71 mg
.EmitCall (ec
, arguments
);
74 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
76 var method
= mg
.BestCandidate
.GetMetaInfo () as MethodInfo
;
77 return SLE
.Expression
.Call (method
, Arguments
.MakeExpression (arguments
, ctx
));
80 public MethodGroupExpr Method
{
85 public class ParenthesizedExpression
: ShimExpression
87 public ParenthesizedExpression (Expression expr
)
93 protected override Expression
DoResolve (ResolveContext ec
)
95 return expr
.Resolve (ec
);
98 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
100 return expr
.DoResolveLValue (ec
, right_side
);
105 // Unary implements unary expressions.
107 public class Unary
: Expression
109 public enum Operator
: byte {
110 UnaryPlus
, UnaryNegation
, LogicalNot
, OnesComplement
,
114 static TypeSpec
[][] predefined_operators
;
116 public readonly Operator Oper
;
117 public Expression Expr
;
118 Expression enum_conversion
;
120 public Unary (Operator op
, Expression expr
, Location loc
)
128 // This routine will attempt to simplify the unary expression when the
129 // argument is a constant.
131 Constant
TryReduceConstant (ResolveContext ec
, Constant e
)
133 if (e
is EmptyConstantCast
)
134 return TryReduceConstant (ec
, ((EmptyConstantCast
) e
).child
);
136 if (e
is SideEffectConstant
) {
137 Constant r
= TryReduceConstant (ec
, ((SideEffectConstant
) e
).value);
138 return r
== null ? null : new SideEffectConstant (r
, e
, r
.Location
);
141 TypeSpec expr_type
= e
.Type
;
144 case Operator
.UnaryPlus
:
145 // Unary numeric promotions
146 if (expr_type
== TypeManager
.byte_type
)
147 return new IntConstant (((ByteConstant
)e
).Value
, e
.Location
);
148 if (expr_type
== TypeManager
.sbyte_type
)
149 return new IntConstant (((SByteConstant
)e
).Value
, e
.Location
);
150 if (expr_type
== TypeManager
.short_type
)
151 return new IntConstant (((ShortConstant
)e
).Value
, e
.Location
);
152 if (expr_type
== TypeManager
.ushort_type
)
153 return new IntConstant (((UShortConstant
)e
).Value
, e
.Location
);
154 if (expr_type
== TypeManager
.char_type
)
155 return new IntConstant (((CharConstant
)e
).Value
, e
.Location
);
157 // Predefined operators
158 if (expr_type
== TypeManager
.int32_type
|| expr_type
== TypeManager
.uint32_type
||
159 expr_type
== TypeManager
.int64_type
|| expr_type
== TypeManager
.uint64_type
||
160 expr_type
== TypeManager
.float_type
|| expr_type
== TypeManager
.double_type
||
161 expr_type
== TypeManager
.decimal_type
) {
167 case Operator
.UnaryNegation
:
168 // Unary numeric promotions
169 if (expr_type
== TypeManager
.byte_type
)
170 return new IntConstant (-((ByteConstant
)e
).Value
, e
.Location
);
171 if (expr_type
== TypeManager
.sbyte_type
)
172 return new IntConstant (-((SByteConstant
)e
).Value
, e
.Location
);
173 if (expr_type
== TypeManager
.short_type
)
174 return new IntConstant (-((ShortConstant
)e
).Value
, e
.Location
);
175 if (expr_type
== TypeManager
.ushort_type
)
176 return new IntConstant (-((UShortConstant
)e
).Value
, e
.Location
);
177 if (expr_type
== TypeManager
.char_type
)
178 return new IntConstant (-((CharConstant
)e
).Value
, e
.Location
);
180 // Predefined operators
181 if (expr_type
== TypeManager
.int32_type
) {
182 int value = ((IntConstant
)e
).Value
;
183 if (value == int.MinValue
) {
184 if (ec
.ConstantCheckState
) {
185 ConstantFold
.Error_CompileTimeOverflow (ec
, loc
);
190 return new IntConstant (-value, e
.Location
);
192 if (expr_type
== TypeManager
.int64_type
) {
193 long value = ((LongConstant
)e
).Value
;
194 if (value == long.MinValue
) {
195 if (ec
.ConstantCheckState
) {
196 ConstantFold
.Error_CompileTimeOverflow (ec
, loc
);
201 return new LongConstant (-value, e
.Location
);
204 if (expr_type
== TypeManager
.uint32_type
) {
205 UIntLiteral uil
= e
as UIntLiteral
;
207 if (uil
.Value
== int.MaxValue
+ (uint) 1)
208 return new IntLiteral (int.MinValue
, e
.Location
);
209 return new LongLiteral (-uil
.Value
, e
.Location
);
211 return new LongConstant (-((UIntConstant
)e
).Value
, e
.Location
);
214 if (expr_type
== TypeManager
.uint64_type
) {
215 ULongLiteral ull
= e
as ULongLiteral
;
216 if (ull
!= null && ull
.Value
== 9223372036854775808)
217 return new LongLiteral (long.MinValue
, e
.Location
);
221 if (expr_type
== TypeManager
.float_type
) {
222 FloatLiteral fl
= e
as FloatLiteral
;
223 // For better error reporting
225 return new FloatLiteral (-fl
.Value
, e
.Location
);
227 return new FloatConstant (-((FloatConstant
)e
).Value
, e
.Location
);
229 if (expr_type
== TypeManager
.double_type
) {
230 DoubleLiteral dl
= e
as DoubleLiteral
;
231 // For better error reporting
233 return new DoubleLiteral (-dl
.Value
, e
.Location
);
235 return new DoubleConstant (-((DoubleConstant
)e
).Value
, e
.Location
);
237 if (expr_type
== TypeManager
.decimal_type
)
238 return new DecimalConstant (-((DecimalConstant
)e
).Value
, e
.Location
);
242 case Operator
.LogicalNot
:
243 if (expr_type
!= TypeManager
.bool_type
)
246 bool b
= (bool)e
.GetValue ();
247 return new BoolConstant (!b
, e
.Location
);
249 case Operator
.OnesComplement
:
250 // Unary numeric promotions
251 if (expr_type
== TypeManager
.byte_type
)
252 return new IntConstant (~
((ByteConstant
)e
).Value
, e
.Location
);
253 if (expr_type
== TypeManager
.sbyte_type
)
254 return new IntConstant (~
((SByteConstant
)e
).Value
, e
.Location
);
255 if (expr_type
== TypeManager
.short_type
)
256 return new IntConstant (~
((ShortConstant
)e
).Value
, e
.Location
);
257 if (expr_type
== TypeManager
.ushort_type
)
258 return new IntConstant (~
((UShortConstant
)e
).Value
, e
.Location
);
259 if (expr_type
== TypeManager
.char_type
)
260 return new IntConstant (~
((CharConstant
)e
).Value
, e
.Location
);
262 // Predefined operators
263 if (expr_type
== TypeManager
.int32_type
)
264 return new IntConstant (~
((IntConstant
)e
).Value
, e
.Location
);
265 if (expr_type
== TypeManager
.uint32_type
)
266 return new UIntConstant (~
((UIntConstant
)e
).Value
, e
.Location
);
267 if (expr_type
== TypeManager
.int64_type
)
268 return new LongConstant (~
((LongConstant
)e
).Value
, e
.Location
);
269 if (expr_type
== TypeManager
.uint64_type
){
270 return new ULongConstant (~
((ULongConstant
)e
).Value
, e
.Location
);
272 if (e
is EnumConstant
) {
273 e
= TryReduceConstant (ec
, ((EnumConstant
)e
).Child
);
275 e
= new EnumConstant (e
, expr_type
);
280 throw new Exception ("Can not constant fold: " + Oper
.ToString());
283 protected Expression
ResolveOperator (ResolveContext ec
, Expression expr
)
285 eclass
= ExprClass
.Value
;
287 if (predefined_operators
== null)
288 CreatePredefinedOperatorsTable ();
290 TypeSpec expr_type
= expr
.Type
;
291 Expression best_expr
;
294 // Primitive types first
296 if (TypeManager
.IsPrimitiveType (expr_type
)) {
297 best_expr
= ResolvePrimitivePredefinedType (expr
);
298 if (best_expr
== null)
301 type
= best_expr
.Type
;
307 // E operator ~(E x);
309 if (Oper
== Operator
.OnesComplement
&& TypeManager
.IsEnumType (expr_type
))
310 return ResolveEnumOperator (ec
, expr
);
312 return ResolveUserType (ec
, expr
);
315 protected virtual Expression
ResolveEnumOperator (ResolveContext ec
, Expression expr
)
317 TypeSpec underlying_type
= EnumSpec
.GetUnderlyingType (expr
.Type
);
318 Expression best_expr
= ResolvePrimitivePredefinedType (EmptyCast
.Create (expr
, underlying_type
));
319 if (best_expr
== null)
323 enum_conversion
= Convert
.ExplicitNumericConversion (new EmptyExpression (best_expr
.Type
), underlying_type
);
325 return EmptyCast
.Create (this, type
);
328 public override Expression
CreateExpressionTree (ResolveContext ec
)
330 return CreateExpressionTree (ec
, null);
333 Expression
CreateExpressionTree (ResolveContext ec
, MethodGroupExpr user_op
)
337 case Operator
.AddressOf
:
338 Error_PointerInsideExpressionTree (ec
);
340 case Operator
.UnaryNegation
:
341 if (ec
.HasSet (ResolveContext
.Options
.CheckedScope
) && user_op
== null && !IsFloat (type
))
342 method_name
= "NegateChecked";
344 method_name
= "Negate";
346 case Operator
.OnesComplement
:
347 case Operator
.LogicalNot
:
350 case Operator
.UnaryPlus
:
351 method_name
= "UnaryPlus";
354 throw new InternalErrorException ("Unknown unary operator " + Oper
.ToString ());
357 Arguments args
= new Arguments (2);
358 args
.Add (new Argument (Expr
.CreateExpressionTree (ec
)));
360 args
.Add (new Argument (user_op
.CreateExpressionTree (ec
)));
361 return CreateExpressionFactoryCall (ec
, method_name
, args
);
364 static void CreatePredefinedOperatorsTable ()
366 predefined_operators
= new TypeSpec
[(int) Operator
.TOP
] [];
369 // 7.6.1 Unary plus operator
371 predefined_operators
[(int) Operator
.UnaryPlus
] = new TypeSpec
[] {
372 TypeManager
.int32_type
, TypeManager
.uint32_type
,
373 TypeManager
.int64_type
, TypeManager
.uint64_type
,
374 TypeManager
.float_type
, TypeManager
.double_type
,
375 TypeManager
.decimal_type
379 // 7.6.2 Unary minus operator
381 predefined_operators
[(int) Operator
.UnaryNegation
] = new TypeSpec
[] {
382 TypeManager
.int32_type
,
383 TypeManager
.int64_type
,
384 TypeManager
.float_type
, TypeManager
.double_type
,
385 TypeManager
.decimal_type
389 // 7.6.3 Logical negation operator
391 predefined_operators
[(int) Operator
.LogicalNot
] = new TypeSpec
[] {
392 TypeManager
.bool_type
396 // 7.6.4 Bitwise complement operator
398 predefined_operators
[(int) Operator
.OnesComplement
] = new TypeSpec
[] {
399 TypeManager
.int32_type
, TypeManager
.uint32_type
,
400 TypeManager
.int64_type
, TypeManager
.uint64_type
405 // Unary numeric promotions
407 static Expression
DoNumericPromotion (Operator op
, Expression expr
)
409 TypeSpec expr_type
= expr
.Type
;
410 if ((op
== Operator
.UnaryPlus
|| op
== Operator
.UnaryNegation
|| op
== Operator
.OnesComplement
) &&
411 expr_type
== TypeManager
.byte_type
|| expr_type
== TypeManager
.sbyte_type
||
412 expr_type
== TypeManager
.short_type
|| expr_type
== TypeManager
.ushort_type
||
413 expr_type
== TypeManager
.char_type
)
414 return Convert
.ImplicitNumericConversion (expr
, TypeManager
.int32_type
);
416 if (op
== Operator
.UnaryNegation
&& expr_type
== TypeManager
.uint32_type
)
417 return Convert
.ImplicitNumericConversion (expr
, TypeManager
.int64_type
);
422 protected override Expression
DoResolve (ResolveContext ec
)
424 if (Oper
== Operator
.AddressOf
) {
425 return ResolveAddressOf (ec
);
428 Expr
= Expr
.Resolve (ec
);
432 if (Expr
.Type
== InternalType
.Dynamic
) {
433 Arguments args
= new Arguments (1);
434 args
.Add (new Argument (Expr
));
435 return new DynamicUnaryConversion (GetOperatorExpressionTypeName (), args
, loc
).Resolve (ec
);
438 if (TypeManager
.IsNullableType (Expr
.Type
))
439 return new Nullable
.LiftedUnaryOperator (Oper
, Expr
, loc
).Resolve (ec
);
442 // Attempt to use a constant folding operation.
444 Constant cexpr
= Expr
as Constant
;
446 cexpr
= TryReduceConstant (ec
, cexpr
);
448 return cexpr
.Resolve (ec
);
451 Expression expr
= ResolveOperator (ec
, Expr
);
453 Error_OperatorCannotBeApplied (ec
, loc
, OperName (Oper
), Expr
.Type
);
456 // Reduce unary operator on predefined types
458 if (expr
== this && Oper
== Operator
.UnaryPlus
)
464 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right
)
469 public override void Emit (EmitContext ec
)
471 EmitOperator (ec
, type
);
474 protected void EmitOperator (EmitContext ec
, TypeSpec type
)
477 case Operator
.UnaryPlus
:
481 case Operator
.UnaryNegation
:
482 if (ec
.HasSet (EmitContext
.Options
.CheckedScope
) && !IsFloat (type
)) {
483 ec
.Emit (OpCodes
.Ldc_I4_0
);
484 if (type
== TypeManager
.int64_type
)
485 ec
.Emit (OpCodes
.Conv_U8
);
487 ec
.Emit (OpCodes
.Sub_Ovf
);
490 ec
.Emit (OpCodes
.Neg
);
495 case Operator
.LogicalNot
:
497 ec
.Emit (OpCodes
.Ldc_I4_0
);
498 ec
.Emit (OpCodes
.Ceq
);
501 case Operator
.OnesComplement
:
503 ec
.Emit (OpCodes
.Not
);
506 case Operator
.AddressOf
:
507 ((IMemoryLocation
)Expr
).AddressOf (ec
, AddressOp
.LoadStore
);
511 throw new Exception ("This should not happen: Operator = "
516 // Same trick as in Binary expression
518 if (enum_conversion
!= null)
519 enum_conversion
.Emit (ec
);
522 public override void EmitBranchable (EmitContext ec
, Label target
, bool on_true
)
524 if (Oper
== Operator
.LogicalNot
)
525 Expr
.EmitBranchable (ec
, target
, !on_true
);
527 base.EmitBranchable (ec
, target
, on_true
);
530 public override void EmitSideEffect (EmitContext ec
)
532 Expr
.EmitSideEffect (ec
);
535 public static void Error_OperatorCannotBeApplied (ResolveContext ec
, Location loc
, string oper
, TypeSpec t
)
537 ec
.Report
.Error (23, loc
, "The `{0}' operator cannot be applied to operand of type `{1}'",
538 oper
, TypeManager
.CSharpName (t
));
542 // Converts operator to System.Linq.Expressions.ExpressionType enum name
544 string GetOperatorExpressionTypeName ()
547 case Operator
.OnesComplement
:
548 return "OnesComplement";
549 case Operator
.LogicalNot
:
551 case Operator
.UnaryNegation
:
553 case Operator
.UnaryPlus
:
556 throw new NotImplementedException ("Unknown express type operator " + Oper
.ToString ());
560 static bool IsFloat (TypeSpec t
)
562 return t
== TypeManager
.float_type
|| t
== TypeManager
.double_type
;
566 // Returns a stringified representation of the Operator
568 public static string OperName (Operator oper
)
571 case Operator
.UnaryPlus
:
573 case Operator
.UnaryNegation
:
575 case Operator
.LogicalNot
:
577 case Operator
.OnesComplement
:
579 case Operator
.AddressOf
:
583 throw new NotImplementedException (oper
.ToString ());
586 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
588 var expr
= Expr
.MakeExpression (ctx
);
589 bool is_checked
= ctx
.HasSet (BuilderContext
.Options
.CheckedScope
);
592 case Operator
.UnaryNegation
:
593 return is_checked
? SLE
.Expression
.NegateChecked (expr
) : SLE
.Expression
.Negate (expr
);
594 case Operator
.LogicalNot
:
595 return SLE
.Expression
.Not (expr
);
597 case Operator
.OnesComplement
:
598 return SLE
.Expression
.OnesComplement (expr
);
601 throw new NotImplementedException (Oper
.ToString ());
605 public static void Reset ()
607 predefined_operators
= null;
610 Expression
ResolveAddressOf (ResolveContext ec
)
613 UnsafeError (ec
, loc
);
615 Expr
= Expr
.DoResolveLValue (ec
, EmptyExpression
.UnaryAddress
);
616 if (Expr
== null || Expr
.eclass
!= ExprClass
.Variable
) {
617 ec
.Report
.Error (211, loc
, "Cannot take the address of the given expression");
621 if (!TypeManager
.VerifyUnmanaged (ec
.Compiler
, Expr
.Type
, loc
)) {
625 IVariableReference vr
= Expr
as IVariableReference
;
628 VariableInfo vi
= vr
.VariableInfo
;
630 if (vi
.LocalInfo
!= null)
631 vi
.LocalInfo
.Used
= true;
634 // A variable is considered definitely assigned if you take its address.
639 is_fixed
= vr
.IsFixed
;
640 vr
.SetHasAddressTaken ();
643 AnonymousMethodExpression
.Error_AddressOfCapturedVar (ec
, vr
, loc
);
646 IFixedExpression fe
= Expr
as IFixedExpression
;
647 is_fixed
= fe
!= null && fe
.IsFixed
;
650 if (!is_fixed
&& !ec
.HasSet (ResolveContext
.Options
.FixedInitializerScope
)) {
651 ec
.Report
.Error (212, loc
, "You can only take the address of unfixed expression inside of a fixed statement initializer");
654 type
= PointerContainer
.MakeType (Expr
.Type
);
655 eclass
= ExprClass
.Value
;
659 Expression
ResolvePrimitivePredefinedType (Expression expr
)
661 expr
= DoNumericPromotion (Oper
, expr
);
662 TypeSpec expr_type
= expr
.Type
;
663 TypeSpec
[] predefined
= predefined_operators
[(int) Oper
];
664 foreach (TypeSpec t
in predefined
) {
672 // Perform user-operator overload resolution
674 protected virtual Expression
ResolveUserOperator (ResolveContext ec
, Expression expr
)
676 CSharp
.Operator
.OpType op_type
;
678 case Operator
.LogicalNot
:
679 op_type
= CSharp
.Operator
.OpType
.LogicalNot
; break;
680 case Operator
.OnesComplement
:
681 op_type
= CSharp
.Operator
.OpType
.OnesComplement
; break;
682 case Operator
.UnaryNegation
:
683 op_type
= CSharp
.Operator
.OpType
.UnaryNegation
; break;
684 case Operator
.UnaryPlus
:
685 op_type
= CSharp
.Operator
.OpType
.UnaryPlus
; break;
687 throw new InternalErrorException (Oper
.ToString ());
690 string op_name
= CSharp
.Operator
.GetMetadataName (op_type
);
691 MethodGroupExpr user_op
= MethodLookup (ec
.Compiler
, ec
.CurrentType
, expr
.Type
, MemberKind
.Operator
, op_name
, 0, expr
.Location
);
695 Arguments args
= new Arguments (1);
696 args
.Add (new Argument (expr
));
697 user_op
= user_op
.OverloadResolve (ec
, ref args
, false, expr
.Location
);
702 Expr
= args
[0].Expr
;
703 return new UserOperatorCall (user_op
, args
, CreateExpressionTree
, expr
.Location
);
707 // Unary user type overload resolution
709 Expression
ResolveUserType (ResolveContext ec
, Expression expr
)
711 Expression best_expr
= ResolveUserOperator (ec
, expr
);
712 if (best_expr
!= null)
715 TypeSpec
[] predefined
= predefined_operators
[(int) Oper
];
716 foreach (TypeSpec t
in predefined
) {
717 Expression oper_expr
= Convert
.UserDefinedConversion (ec
, expr
, t
, expr
.Location
, false, false);
718 if (oper_expr
== null)
722 // decimal type is predefined but has user-operators
724 if (oper_expr
.Type
== TypeManager
.decimal_type
)
725 oper_expr
= ResolveUserType (ec
, oper_expr
);
727 oper_expr
= ResolvePrimitivePredefinedType (oper_expr
);
729 if (oper_expr
== null)
732 if (best_expr
== null) {
733 best_expr
= oper_expr
;
737 int result
= MethodGroupExpr
.BetterTypeConversion (ec
, best_expr
.Type
, t
);
739 ec
.Report
.Error (35, loc
, "Operator `{0}' is ambiguous on an operand of type `{1}'",
740 OperName (Oper
), TypeManager
.CSharpName (expr
.Type
));
745 best_expr
= oper_expr
;
748 if (best_expr
== null)
752 // HACK: Decimal user-operator is included in standard operators
754 if (best_expr
.Type
== TypeManager
.decimal_type
)
758 type
= best_expr
.Type
;
762 protected override void CloneTo (CloneContext clonectx
, Expression t
)
764 Unary target
= (Unary
) t
;
766 target
.Expr
= Expr
.Clone (clonectx
);
771 // Unary operators are turned into Indirection expressions
772 // after semantic analysis (this is so we can take the address
773 // of an indirection).
775 public class Indirection
: Expression
, IMemoryLocation
, IAssignMethod
, IFixedExpression
{
777 LocalTemporary temporary
;
780 public Indirection (Expression expr
, Location l
)
786 public override Expression
CreateExpressionTree (ResolveContext ec
)
788 Error_PointerInsideExpressionTree (ec
);
792 protected override void CloneTo (CloneContext clonectx
, Expression t
)
794 Indirection target
= (Indirection
) t
;
795 target
.expr
= expr
.Clone (clonectx
);
798 public override void Emit (EmitContext ec
)
803 ec
.EmitLoadFromPtr (Type
);
806 public void Emit (EmitContext ec
, bool leave_copy
)
810 ec
.Emit (OpCodes
.Dup
);
811 temporary
= new LocalTemporary (expr
.Type
);
812 temporary
.Store (ec
);
816 public void EmitAssign (EmitContext ec
, Expression source
, bool leave_copy
, bool prepare_for_load
)
818 prepared
= prepare_for_load
;
822 if (prepare_for_load
)
823 ec
.Emit (OpCodes
.Dup
);
827 ec
.Emit (OpCodes
.Dup
);
828 temporary
= new LocalTemporary (expr
.Type
);
829 temporary
.Store (ec
);
832 ec
.EmitStoreFromPtr (type
);
834 if (temporary
!= null) {
836 temporary
.Release (ec
);
840 public void AddressOf (EmitContext ec
, AddressOp Mode
)
845 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
847 return DoResolve (ec
);
850 protected override Expression
DoResolve (ResolveContext ec
)
852 expr
= expr
.Resolve (ec
);
857 UnsafeError (ec
, loc
);
859 if (!expr
.Type
.IsPointer
) {
860 ec
.Report
.Error (193, loc
, "The * or -> operator must be applied to a pointer");
864 if (expr
.Type
== TypeManager
.void_ptr_type
) {
865 ec
.Report
.Error (242, loc
, "The operation in question is undefined on void pointers");
869 type
= TypeManager
.GetElementType (expr
.Type
);
870 eclass
= ExprClass
.Variable
;
874 public bool IsFixed
{
878 public override string ToString ()
880 return "*(" + expr
+ ")";
885 /// Unary Mutator expressions (pre and post ++ and --)
889 /// UnaryMutator implements ++ and -- expressions. It derives from
890 /// ExpressionStatement becuase the pre/post increment/decrement
891 /// operators can be used in a statement context.
893 /// FIXME: Idea, we could split this up in two classes, one simpler
894 /// for the common case, and one with the extra fields for more complex
895 /// classes (indexers require temporary access; overloaded require method)
898 public class UnaryMutator
: ExpressionStatement
900 class DynamicPostMutator
: Expression
, IAssignMethod
905 public DynamicPostMutator (Expression expr
)
908 this.type
= expr
.Type
;
909 this.loc
= expr
.Location
;
912 public override Expression
CreateExpressionTree (ResolveContext ec
)
914 throw new NotImplementedException ("ET");
917 protected override Expression
DoResolve (ResolveContext rc
)
919 eclass
= expr
.eclass
;
923 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
925 expr
.DoResolveLValue (ec
, right_side
);
926 return DoResolve (ec
);
929 public override void Emit (EmitContext ec
)
934 public void Emit (EmitContext ec
, bool leave_copy
)
936 throw new NotImplementedException ();
940 // Emits target assignment using unmodified source value
942 public void EmitAssign (EmitContext ec
, Expression source
, bool leave_copy
, bool prepare_for_load
)
945 // Allocate temporary variable to keep original value before it's modified
947 temp
= new LocalTemporary (type
);
951 ((IAssignMethod
) expr
).EmitAssign (ec
, source
, false, prepare_for_load
);
962 public enum Mode
: byte {
969 PreDecrement
= IsDecrement
,
970 PostIncrement
= IsPost
,
971 PostDecrement
= IsPost
| IsDecrement
975 bool is_expr
, recurse
;
979 // Holds the real operation
980 Expression operation
;
982 public UnaryMutator (Mode m
, Expression e
, Location loc
)
989 public override Expression
CreateExpressionTree (ResolveContext ec
)
991 return new SimpleAssign (this, this).CreateExpressionTree (ec
);
994 protected override Expression
DoResolve (ResolveContext ec
)
996 expr
= expr
.Resolve (ec
);
1001 if (expr
.Type
== InternalType
.Dynamic
) {
1003 // Handle postfix unary operators using local
1004 // temporary variable
1006 if ((mode
& Mode
.IsPost
) != 0)
1007 expr
= new DynamicPostMutator (expr
);
1009 Arguments args
= new Arguments (1);
1010 args
.Add (new Argument (expr
));
1011 return new SimpleAssign (expr
, new DynamicUnaryConversion (GetOperatorExpressionTypeName (), args
, loc
)).Resolve (ec
);
1014 if (TypeManager
.IsNullableType (expr
.Type
))
1015 return new Nullable
.LiftedUnaryMutator (mode
, expr
, loc
).Resolve (ec
);
1017 eclass
= ExprClass
.Value
;
1019 return ResolveOperator (ec
);
1022 void EmitCode (EmitContext ec
, bool is_expr
)
1025 this.is_expr
= is_expr
;
1026 ((IAssignMethod
) expr
).EmitAssign (ec
, this, is_expr
&& (mode
== Mode
.PreIncrement
|| mode
== Mode
.PreDecrement
), true);
1029 public override void Emit (EmitContext ec
)
1032 // We use recurse to allow ourselfs to be the source
1033 // of an assignment. This little hack prevents us from
1034 // having to allocate another expression
1037 ((IAssignMethod
) expr
).Emit (ec
, is_expr
&& (mode
== Mode
.PostIncrement
|| mode
== Mode
.PostDecrement
));
1039 operation
.Emit (ec
);
1045 EmitCode (ec
, true);
1048 public override void EmitStatement (EmitContext ec
)
1050 EmitCode (ec
, false);
1054 // Converts operator to System.Linq.Expressions.ExpressionType enum name
1056 string GetOperatorExpressionTypeName ()
1058 return IsDecrement
? "Decrement" : "Increment";
1062 get { return (mode & Mode.IsDecrement) != 0; }
1066 // Returns whether an object of type `t' can be incremented
1067 // or decremented with add/sub (ie, basically whether we can
1068 // use pre-post incr-decr operations on it, but it is not a
1069 // System.Decimal, which we require operator overloading to catch)
1071 static bool IsPredefinedOperator (TypeSpec t
)
1073 return (TypeManager
.IsPrimitiveType (t
) && t
!= TypeManager
.bool_type
) ||
1074 TypeManager
.IsEnumType (t
) ||
1075 t
.IsPointer
&& t
!= TypeManager
.void_ptr_type
;
1079 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
1081 var target
= ((RuntimeValueExpression
) expr
).MetaObject
.Expression
;
1082 var source
= SLE
.Expression
.Convert (operation
.MakeExpression (ctx
), target
.Type
);
1083 return SLE
.Expression
.Assign (target
, source
);
1087 protected override void CloneTo (CloneContext clonectx
, Expression t
)
1089 UnaryMutator target
= (UnaryMutator
) t
;
1091 target
.expr
= expr
.Clone (clonectx
);
1094 Expression
ResolveOperator (ResolveContext ec
)
1096 if (expr
is RuntimeValueExpression
) {
1099 // Use itself at the top of the stack
1100 operation
= new EmptyExpression (type
);
1104 // The operand of the prefix/postfix increment decrement operators
1105 // should be an expression that is classified as a variable,
1106 // a property access or an indexer access
1108 if (expr
.eclass
== ExprClass
.Variable
|| expr
.eclass
== ExprClass
.IndexerAccess
|| expr
.eclass
== ExprClass
.PropertyAccess
) {
1109 expr
= expr
.ResolveLValue (ec
, expr
);
1111 ec
.Report
.Error (1059, loc
, "The operand of an increment or decrement operator must be a variable, property or indexer");
1115 // 1. Check predefined types
1117 if (IsPredefinedOperator (type
)) {
1118 // TODO: Move to IntConstant once I get rid of int32_type
1119 var one
= new IntConstant (1, loc
);
1121 // TODO: Cache this based on type when using EmptyExpression in
1123 Binary
.Operator op
= IsDecrement
? Binary
.Operator
.Subtraction
: Binary
.Operator
.Addition
;
1124 operation
= new Binary (op
, operation
, one
, loc
);
1125 operation
= operation
.Resolve (ec
);
1126 if (operation
!= null && operation
.Type
!= type
)
1127 operation
= Convert
.ExplicitNumericConversion (operation
, type
);
1133 // Step 2: Perform Operator Overload location
1138 op_name
= Operator
.GetMetadataName (Operator
.OpType
.Decrement
);
1140 op_name
= Operator
.GetMetadataName (Operator
.OpType
.Increment
);
1142 var mg
= MethodLookup (ec
.Compiler
, ec
.CurrentType
, type
, MemberKind
.Operator
, op_name
, 0, loc
);
1145 Arguments args
= new Arguments (1);
1146 args
.Add (new Argument (expr
));
1147 mg
= mg
.OverloadResolve (ec
, ref args
, false, loc
);
1151 args
[0].Expr
= operation
;
1152 operation
= new UserOperatorCall (mg
, args
, null, loc
);
1153 operation
= Convert
.ImplicitConversionRequired (ec
, operation
, type
, loc
);
1157 string name
= IsDecrement
?
1158 Operator
.GetName (Operator
.OpType
.Decrement
) :
1159 Operator
.GetName (Operator
.OpType
.Increment
);
1161 Unary
.Error_OperatorCannotBeApplied (ec
, loc
, name
, type
);
1167 /// Base class for the `Is' and `As' classes.
1171 /// FIXME: Split this in two, and we get to save the `Operator' Oper
1174 public abstract class Probe
: Expression
{
1175 public Expression ProbeType
;
1176 protected Expression expr
;
1177 protected TypeExpr probe_type_expr
;
1179 public Probe (Expression expr
, Expression probe_type
, Location l
)
1181 ProbeType
= probe_type
;
1186 public Expression Expr
{
1192 protected override Expression
DoResolve (ResolveContext ec
)
1194 probe_type_expr
= ProbeType
.ResolveAsTypeTerminal (ec
, false);
1195 if (probe_type_expr
== null)
1198 expr
= expr
.Resolve (ec
);
1202 if (probe_type_expr
.Type
.IsStatic
) {
1203 ec
.Report
.Error (-244, loc
, "The `{0}' operator cannot be applied to an operand of a static type",
1207 if (expr
.Type
.IsPointer
|| probe_type_expr
.Type
.IsPointer
) {
1208 ec
.Report
.Error (244, loc
, "The `{0}' operator cannot be applied to an operand of pointer type",
1213 if (expr
.Type
== InternalType
.AnonymousMethod
) {
1214 ec
.Report
.Error (837, loc
, "The `{0}' operator cannot be applied to a lambda expression or anonymous method",
1222 protected abstract string OperatorName { get; }
1224 protected override void CloneTo (CloneContext clonectx
, Expression t
)
1226 Probe target
= (Probe
) t
;
1228 target
.expr
= expr
.Clone (clonectx
);
1229 target
.ProbeType
= ProbeType
.Clone (clonectx
);
1235 /// Implementation of the `is' operator.
1237 public class Is
: Probe
{
1238 Nullable
.Unwrap expr_unwrap
;
1240 public Is (Expression expr
, Expression probe_type
, Location l
)
1241 : base (expr
, probe_type
, l
)
1245 public override Expression
CreateExpressionTree (ResolveContext ec
)
1247 Arguments args
= Arguments
.CreateForExpressionTree (ec
, null,
1248 expr
.CreateExpressionTree (ec
),
1249 new TypeOf (probe_type_expr
, loc
));
1251 return CreateExpressionFactoryCall (ec
, "TypeIs", args
);
1254 public override void Emit (EmitContext ec
)
1256 if (expr_unwrap
!= null) {
1257 expr_unwrap
.EmitCheck (ec
);
1262 ec
.Emit (OpCodes
.Isinst
, probe_type_expr
.Type
);
1263 ec
.Emit (OpCodes
.Ldnull
);
1264 ec
.Emit (OpCodes
.Cgt_Un
);
1267 public override void EmitBranchable (EmitContext ec
, Label target
, bool on_true
)
1269 if (expr_unwrap
!= null) {
1270 expr_unwrap
.EmitCheck (ec
);
1273 ec
.Emit (OpCodes
.Isinst
, probe_type_expr
.Type
);
1275 ec
.Emit (on_true
? OpCodes
.Brtrue
: OpCodes
.Brfalse
, target
);
1278 Expression
CreateConstantResult (ResolveContext ec
, bool result
)
1281 ec
.Report
.Warning (183, 1, loc
, "The given expression is always of the provided (`{0}') type",
1282 TypeManager
.CSharpName (probe_type_expr
.Type
));
1284 ec
.Report
.Warning (184, 1, loc
, "The given expression is never of the provided (`{0}') type",
1285 TypeManager
.CSharpName (probe_type_expr
.Type
));
1287 return ReducedExpression
.Create (new BoolConstant (result
, loc
).Resolve (ec
), this);
1290 protected override Expression
DoResolve (ResolveContext ec
)
1292 if (base.DoResolve (ec
) == null)
1295 TypeSpec d
= expr
.Type
;
1296 bool d_is_nullable
= false;
1299 // If E is a method group or the null literal, or if the type of E is a reference
1300 // type or a nullable type and the value of E is null, the result is false
1302 if (expr
.IsNull
|| expr
.eclass
== ExprClass
.MethodGroup
)
1303 return CreateConstantResult (ec
, false);
1305 if (TypeManager
.IsNullableType (d
)) {
1306 var ut
= Nullable
.NullableInfo
.GetUnderlyingType (d
);
1307 if (!ut
.IsGenericParameter
) {
1309 d_is_nullable
= true;
1313 type
= TypeManager
.bool_type
;
1314 eclass
= ExprClass
.Value
;
1315 TypeSpec t
= probe_type_expr
.Type
;
1316 bool t_is_nullable
= false;
1317 if (TypeManager
.IsNullableType (t
)) {
1318 var ut
= Nullable
.NullableInfo
.GetUnderlyingType (t
);
1319 if (!ut
.IsGenericParameter
) {
1321 t_is_nullable
= true;
1325 if (TypeManager
.IsStruct (t
)) {
1328 // D and T are the same value types but D can be null
1330 if (d_is_nullable
&& !t_is_nullable
) {
1331 expr_unwrap
= Nullable
.Unwrap
.Create (expr
, false);
1336 // The result is true if D and T are the same value types
1338 return CreateConstantResult (ec
, true);
1341 var tp
= d
as TypeParameterSpec
;
1343 return ResolveGenericParameter (ec
, t
, tp
);
1346 // An unboxing conversion exists
1348 if (Convert
.ExplicitReferenceConversionExists (d
, t
))
1351 if (TypeManager
.IsGenericParameter (t
))
1352 return ResolveGenericParameter (ec
, d
, (TypeParameterSpec
) t
);
1354 if (TypeManager
.IsStruct (d
)) {
1356 if (Convert
.ImplicitBoxingConversionExists (d
, t
, out temp
))
1357 return CreateConstantResult (ec
, true);
1359 if (TypeManager
.IsGenericParameter (d
))
1360 return ResolveGenericParameter (ec
, t
, (TypeParameterSpec
) d
);
1362 if (TypeManager
.ContainsGenericParameters (d
))
1365 if (Convert
.ImplicitReferenceConversionExists (expr
, t
) ||
1366 Convert
.ExplicitReferenceConversionExists (d
, t
)) {
1372 return CreateConstantResult (ec
, false);
1375 Expression
ResolveGenericParameter (ResolveContext ec
, TypeSpec d
, TypeParameterSpec t
)
1377 if (t
.IsReferenceType
) {
1378 if (TypeManager
.IsStruct (d
))
1379 return CreateConstantResult (ec
, false);
1382 if (TypeManager
.IsGenericParameter (expr
.Type
)) {
1383 if (t
.IsValueType
&& expr
.Type
== t
)
1384 return CreateConstantResult (ec
, true);
1386 expr
= new BoxedCast (expr
, d
);
1392 protected override string OperatorName
{
1393 get { return "is"; }
1398 /// Implementation of the `as' operator.
1400 public class As
: Probe
{
1402 Expression resolved_type
;
1404 public As (Expression expr
, Expression probe_type
, Location l
)
1405 : base (expr
, probe_type
, l
)
1409 public override Expression
CreateExpressionTree (ResolveContext ec
)
1411 Arguments args
= Arguments
.CreateForExpressionTree (ec
, null,
1412 expr
.CreateExpressionTree (ec
),
1413 new TypeOf (probe_type_expr
, loc
));
1415 return CreateExpressionFactoryCall (ec
, "TypeAs", args
);
1418 public override void Emit (EmitContext ec
)
1423 ec
.Emit (OpCodes
.Isinst
, type
);
1425 if (TypeManager
.IsGenericParameter (type
) || TypeManager
.IsNullableType (type
))
1426 ec
.Emit (OpCodes
.Unbox_Any
, type
);
1429 protected override Expression
DoResolve (ResolveContext ec
)
1431 if (resolved_type
== null) {
1432 resolved_type
= base.DoResolve (ec
);
1434 if (resolved_type
== null)
1438 type
= probe_type_expr
.Type
;
1439 eclass
= ExprClass
.Value
;
1440 TypeSpec etype
= expr
.Type
;
1442 if (!TypeManager
.IsReferenceType (type
) && !TypeManager
.IsNullableType (type
)) {
1443 if (TypeManager
.IsGenericParameter (type
)) {
1444 ec
.Report
.Error (413, loc
,
1445 "The `as' operator cannot be used with a non-reference type parameter `{0}'. Consider adding `class' or a reference type constraint",
1446 probe_type_expr
.GetSignatureForError ());
1448 ec
.Report
.Error (77, loc
,
1449 "The `as' operator cannot be used with a non-nullable value type `{0}'",
1450 TypeManager
.CSharpName (type
));
1455 if (expr
.IsNull
&& TypeManager
.IsNullableType (type
)) {
1456 return Nullable
.LiftedNull
.CreateFromExpression (ec
, this);
1459 Expression e
= Convert
.ImplicitConversion (ec
, expr
, type
, loc
);
1466 if (Convert
.ExplicitReferenceConversionExists (etype
, type
)){
1467 if (TypeManager
.IsGenericParameter (etype
))
1468 expr
= new BoxedCast (expr
, etype
);
1474 if (TypeManager
.ContainsGenericParameters (etype
) ||
1475 TypeManager
.ContainsGenericParameters (type
)) {
1476 expr
= new BoxedCast (expr
, etype
);
1481 ec
.Report
.Error (39, loc
, "Cannot convert type `{0}' to `{1}' via a built-in conversion",
1482 TypeManager
.CSharpName (etype
), TypeManager
.CSharpName (type
));
1487 protected override string OperatorName
{
1488 get { return "as"; }
1493 /// This represents a typecast in the source language.
1495 /// FIXME: Cast expressions have an unusual set of parsing
1496 /// rules, we need to figure those out.
1498 public class Cast
: ShimExpression
{
1499 Expression target_type
;
1501 public Cast (Expression cast_type
, Expression expr
)
1502 : this (cast_type
, expr
, cast_type
.Location
)
1506 public Cast (Expression cast_type
, Expression expr
, Location loc
)
1509 this.target_type
= cast_type
;
1513 public Expression TargetType
{
1514 get { return target_type; }
1517 protected override Expression
DoResolve (ResolveContext ec
)
1519 expr
= expr
.Resolve (ec
);
1523 TypeExpr target
= target_type
.ResolveAsTypeTerminal (ec
, false);
1529 if (type
.IsStatic
) {
1530 ec
.Report
.Error (716, loc
, "Cannot convert to static type `{0}'", TypeManager
.CSharpName (type
));
1534 eclass
= ExprClass
.Value
;
1536 Constant c
= expr
as Constant
;
1538 c
= c
.TryReduce (ec
, type
, loc
);
1543 if (type
.IsPointer
&& !ec
.IsUnsafe
) {
1544 UnsafeError (ec
, loc
);
1545 } else if (expr
.Type
== InternalType
.Dynamic
) {
1546 Arguments arg
= new Arguments (1);
1547 arg
.Add (new Argument (expr
));
1548 return new DynamicConversion (type
, CSharpBinderFlags
.ConvertExplicit
, arg
, loc
).Resolve (ec
);
1551 expr
= Convert
.ExplicitConversion (ec
, expr
, type
, loc
);
1555 protected override void CloneTo (CloneContext clonectx
, Expression t
)
1557 Cast target
= (Cast
) t
;
1559 target
.target_type
= target_type
.Clone (clonectx
);
1560 target
.expr
= expr
.Clone (clonectx
);
1564 public class ImplicitCast
: ShimExpression
1568 public ImplicitCast (Expression expr
, TypeSpec target
, bool arrayAccess
)
1571 this.loc
= expr
.Location
;
1573 this.arrayAccess
= arrayAccess
;
1576 protected override Expression
DoResolve (ResolveContext ec
)
1578 expr
= expr
.Resolve (ec
);
1583 expr
= ConvertExpressionToArrayIndex (ec
, expr
);
1585 expr
= Convert
.ImplicitConversionRequired (ec
, expr
, type
, loc
);
1592 // C# 2.0 Default value expression
1594 public class DefaultValueExpression
: Expression
1598 public DefaultValueExpression (Expression expr
, Location loc
)
1604 public override Expression
CreateExpressionTree (ResolveContext ec
)
1606 Arguments args
= new Arguments (2);
1607 args
.Add (new Argument (this));
1608 args
.Add (new Argument (new TypeOf (new TypeExpression (type
, loc
), loc
)));
1609 return CreateExpressionFactoryCall (ec
, "Constant", args
);
1612 protected override Expression
DoResolve (ResolveContext ec
)
1614 TypeExpr texpr
= expr
.ResolveAsTypeTerminal (ec
, false);
1620 if (type
.IsStatic
) {
1621 ec
.Report
.Error (-244, loc
, "The `default value' operator cannot be applied to an operand of a static type");
1625 return new NullLiteral (Location
).ConvertImplicitly (ec
, type
);
1627 if (TypeManager
.IsReferenceType (type
))
1628 return new NullConstant (type
, loc
);
1630 Constant c
= New
.Constantify (type
);
1632 return c
.Resolve (ec
);
1634 eclass
= ExprClass
.Variable
;
1638 public override void Emit (EmitContext ec
)
1640 LocalTemporary temp_storage
= new LocalTemporary(type
);
1642 temp_storage
.AddressOf(ec
, AddressOp
.LoadStore
);
1643 ec
.Emit(OpCodes
.Initobj
, type
);
1644 temp_storage
.Emit(ec
);
1647 protected override void CloneTo (CloneContext clonectx
, Expression t
)
1649 DefaultValueExpression target
= (DefaultValueExpression
) t
;
1651 target
.expr
= expr
.Clone (clonectx
);
1656 /// Binary operators
1658 public class Binary
: Expression
, IDynamicBinder
1660 protected class PredefinedOperator
{
1661 protected readonly TypeSpec left
;
1662 protected readonly TypeSpec right
;
1663 public readonly Operator OperatorsMask
;
1664 public TypeSpec ReturnType
;
1666 public PredefinedOperator (TypeSpec ltype
, TypeSpec rtype
, Operator op_mask
)
1667 : this (ltype
, rtype
, op_mask
, ltype
)
1671 public PredefinedOperator (TypeSpec type
, Operator op_mask
, TypeSpec return_type
)
1672 : this (type
, type
, op_mask
, return_type
)
1676 public PredefinedOperator (TypeSpec type
, Operator op_mask
)
1677 : this (type
, type
, op_mask
, type
)
1681 public PredefinedOperator (TypeSpec ltype
, TypeSpec rtype
, Operator op_mask
, TypeSpec return_type
)
1683 if ((op_mask
& Operator
.ValuesOnlyMask
) != 0)
1684 throw new InternalErrorException ("Only masked values can be used");
1688 this.OperatorsMask
= op_mask
;
1689 this.ReturnType
= return_type
;
1692 public virtual Expression
ConvertResult (ResolveContext ec
, Binary b
)
1694 b
.type
= ReturnType
;
1696 b
.left
= Convert
.ImplicitConversion (ec
, b
.left
, left
, b
.left
.Location
);
1697 b
.right
= Convert
.ImplicitConversion (ec
, b
.right
, right
, b
.right
.Location
);
1700 // A user operators does not support multiple user conversions, but decimal type
1701 // is considered to be predefined type therefore we apply predefined operators rules
1702 // and then look for decimal user-operator implementation
1704 if (left
== TypeManager
.decimal_type
)
1705 return b
.ResolveUserOperator (ec
, b
.left
.Type
, b
.right
.Type
);
1707 var c
= b
.right
as Constant
;
1709 if (c
.IsDefaultValue
&& (b
.oper
== Operator
.Addition
|| b
.oper
== Operator
.BitwiseOr
|| b
.oper
== Operator
.Subtraction
))
1710 return ReducedExpression
.Create (b
.left
, b
).Resolve (ec
);
1711 if ((b
.oper
== Operator
.Multiply
|| b
.oper
== Operator
.Division
) && c
.IsOneInteger
)
1712 return ReducedExpression
.Create (b
.left
, b
).Resolve (ec
);
1716 c
= b
.left
as Constant
;
1718 if (c
.IsDefaultValue
&& (b
.oper
== Operator
.Addition
|| b
.oper
== Operator
.BitwiseOr
))
1719 return ReducedExpression
.Create (b
.right
, b
).Resolve (ec
);
1720 if (b
.oper
== Operator
.Multiply
&& c
.IsOneInteger
)
1721 return ReducedExpression
.Create (b
.right
, b
).Resolve (ec
);
1728 public bool IsPrimitiveApplicable (TypeSpec ltype
, TypeSpec rtype
)
1731 // We are dealing with primitive types only
1733 return left
== ltype
&& ltype
== rtype
;
1736 public virtual bool IsApplicable (ResolveContext ec
, Expression lexpr
, Expression rexpr
)
1738 if (TypeManager
.IsEqual (left
, lexpr
.Type
) &&
1739 TypeManager
.IsEqual (right
, rexpr
.Type
))
1742 return Convert
.ImplicitConversionExists (ec
, lexpr
, left
) &&
1743 Convert
.ImplicitConversionExists (ec
, rexpr
, right
);
1746 public PredefinedOperator
ResolveBetterOperator (ResolveContext ec
, PredefinedOperator best_operator
)
1749 if (left
!= null && best_operator
.left
!= null) {
1750 result
= MethodGroupExpr
.BetterTypeConversion (ec
, best_operator
.left
, left
);
1754 // When second arguments are same as the first one, the result is same
1756 if (right
!= null && (left
!= right
|| best_operator
.left
!= best_operator
.right
)) {
1757 result
|= MethodGroupExpr
.BetterTypeConversion (ec
, best_operator
.right
, right
);
1760 if (result
== 0 || result
> 2)
1763 return result
== 1 ? best_operator
: this;
1767 class PredefinedStringOperator
: PredefinedOperator
{
1768 public PredefinedStringOperator (TypeSpec type
, Operator op_mask
)
1769 : base (type
, op_mask
, type
)
1771 ReturnType
= TypeManager
.string_type
;
1774 public PredefinedStringOperator (TypeSpec ltype
, TypeSpec rtype
, Operator op_mask
)
1775 : base (ltype
, rtype
, op_mask
)
1777 ReturnType
= TypeManager
.string_type
;
1780 public override Expression
ConvertResult (ResolveContext ec
, Binary b
)
1783 // Use original expression for nullable arguments
1785 Nullable
.Unwrap unwrap
= b
.left
as Nullable
.Unwrap
;
1787 b
.left
= unwrap
.Original
;
1789 unwrap
= b
.right
as Nullable
.Unwrap
;
1791 b
.right
= unwrap
.Original
;
1793 b
.left
= Convert
.ImplicitConversion (ec
, b
.left
, left
, b
.left
.Location
);
1794 b
.right
= Convert
.ImplicitConversion (ec
, b
.right
, right
, b
.right
.Location
);
1797 // Start a new concat expression using converted expression
1799 return StringConcat
.Create (ec
, b
.left
, b
.right
, b
.loc
);
1803 class PredefinedShiftOperator
: PredefinedOperator
{
1804 public PredefinedShiftOperator (TypeSpec ltype
, Operator op_mask
) :
1805 base (ltype
, TypeManager
.int32_type
, op_mask
)
1809 public override Expression
ConvertResult (ResolveContext ec
, Binary b
)
1811 b
.left
= Convert
.ImplicitConversion (ec
, b
.left
, left
, b
.left
.Location
);
1813 Expression expr_tree_expr
= Convert
.ImplicitConversion (ec
, b
.right
, TypeManager
.int32_type
, b
.right
.Location
);
1815 int right_mask
= left
== TypeManager
.int32_type
|| left
== TypeManager
.uint32_type
? 0x1f : 0x3f;
1818 // b = b.left >> b.right & (0x1f|0x3f)
1820 b
.right
= new Binary (Operator
.BitwiseAnd
,
1821 b
.right
, new IntConstant (right_mask
, b
.right
.Location
), b
.loc
).Resolve (ec
);
1824 // Expression tree representation does not use & mask
1826 b
.right
= ReducedExpression
.Create (b
.right
, expr_tree_expr
).Resolve (ec
);
1827 b
.type
= ReturnType
;
1830 // Optimize shift by 0
1832 var c
= b
.right
as Constant
;
1833 if (c
!= null && c
.IsDefaultValue
)
1834 return ReducedExpression
.Create (b
.left
, b
).Resolve (ec
);
1840 class PredefinedPointerOperator
: PredefinedOperator
{
1841 public PredefinedPointerOperator (TypeSpec ltype
, TypeSpec rtype
, Operator op_mask
)
1842 : base (ltype
, rtype
, op_mask
)
1846 public PredefinedPointerOperator (TypeSpec ltype
, TypeSpec rtype
, Operator op_mask
, TypeSpec retType
)
1847 : base (ltype
, rtype
, op_mask
, retType
)
1851 public PredefinedPointerOperator (TypeSpec type
, Operator op_mask
, TypeSpec return_type
)
1852 : base (type
, op_mask
, return_type
)
1856 public override bool IsApplicable (ResolveContext ec
, Expression lexpr
, Expression rexpr
)
1859 if (!lexpr
.Type
.IsPointer
)
1862 if (!Convert
.ImplicitConversionExists (ec
, lexpr
, left
))
1866 if (right
== null) {
1867 if (!rexpr
.Type
.IsPointer
)
1870 if (!Convert
.ImplicitConversionExists (ec
, rexpr
, right
))
1877 public override Expression
ConvertResult (ResolveContext ec
, Binary b
)
1880 b
.left
= EmptyCast
.Create (b
.left
, left
);
1881 } else if (right
!= null) {
1882 b
.right
= EmptyCast
.Create (b
.right
, right
);
1885 TypeSpec r_type
= ReturnType
;
1886 Expression left_arg
, right_arg
;
1887 if (r_type
== null) {
1890 right_arg
= b
.right
;
1891 r_type
= b
.left
.Type
;
1895 r_type
= b
.right
.Type
;
1899 right_arg
= b
.right
;
1902 return new PointerArithmetic (b
.oper
, left_arg
, right_arg
, r_type
, b
.loc
).Resolve (ec
);
1907 public enum Operator
{
1908 Multiply
= 0 | ArithmeticMask
,
1909 Division
= 1 | ArithmeticMask
,
1910 Modulus
= 2 | ArithmeticMask
,
1911 Addition
= 3 | ArithmeticMask
| AdditionMask
,
1912 Subtraction
= 4 | ArithmeticMask
| SubtractionMask
,
1914 LeftShift
= 5 | ShiftMask
,
1915 RightShift
= 6 | ShiftMask
,
1917 LessThan
= 7 | ComparisonMask
| RelationalMask
,
1918 GreaterThan
= 8 | ComparisonMask
| RelationalMask
,
1919 LessThanOrEqual
= 9 | ComparisonMask
| RelationalMask
,
1920 GreaterThanOrEqual
= 10 | ComparisonMask
| RelationalMask
,
1921 Equality
= 11 | ComparisonMask
| EqualityMask
,
1922 Inequality
= 12 | ComparisonMask
| EqualityMask
,
1924 BitwiseAnd
= 13 | BitwiseMask
,
1925 ExclusiveOr
= 14 | BitwiseMask
,
1926 BitwiseOr
= 15 | BitwiseMask
,
1928 LogicalAnd
= 16 | LogicalMask
,
1929 LogicalOr
= 17 | LogicalMask
,
1934 ValuesOnlyMask
= ArithmeticMask
- 1,
1935 ArithmeticMask
= 1 << 5,
1937 ComparisonMask
= 1 << 7,
1938 EqualityMask
= 1 << 8,
1939 BitwiseMask
= 1 << 9,
1940 LogicalMask
= 1 << 10,
1941 AdditionMask
= 1 << 11,
1942 SubtractionMask
= 1 << 12,
1943 RelationalMask
= 1 << 13
1946 readonly Operator oper
;
1947 protected Expression left
, right
;
1948 readonly bool is_compound
;
1949 Expression enum_conversion
;
1951 static PredefinedOperator
[] standard_operators
;
1952 static PredefinedOperator
[] pointer_operators
;
1954 public Binary (Operator oper
, Expression left
, Expression right
, bool isCompound
, Location loc
)
1955 : this (oper
, left
, right
, loc
)
1957 this.is_compound
= isCompound
;
1960 public Binary (Operator oper
, Expression left
, Expression right
, Location loc
)
1968 public Operator Oper
{
1975 /// Returns a stringified representation of the Operator
1977 string OperName (Operator oper
)
1981 case Operator
.Multiply
:
1984 case Operator
.Division
:
1987 case Operator
.Modulus
:
1990 case Operator
.Addition
:
1993 case Operator
.Subtraction
:
1996 case Operator
.LeftShift
:
1999 case Operator
.RightShift
:
2002 case Operator
.LessThan
:
2005 case Operator
.GreaterThan
:
2008 case Operator
.LessThanOrEqual
:
2011 case Operator
.GreaterThanOrEqual
:
2014 case Operator
.Equality
:
2017 case Operator
.Inequality
:
2020 case Operator
.BitwiseAnd
:
2023 case Operator
.BitwiseOr
:
2026 case Operator
.ExclusiveOr
:
2029 case Operator
.LogicalOr
:
2032 case Operator
.LogicalAnd
:
2036 s
= oper
.ToString ();
2046 public static void Error_OperatorCannotBeApplied (ResolveContext ec
, Expression left
, Expression right
, Operator oper
, Location loc
)
2048 new Binary (oper
, left
, right
, loc
).Error_OperatorCannotBeApplied (ec
, left
, right
);
2051 public static void Error_OperatorCannotBeApplied (ResolveContext ec
, Expression left
, Expression right
, string oper
, Location loc
)
2054 l
= TypeManager
.CSharpName (left
.Type
);
2055 r
= TypeManager
.CSharpName (right
.Type
);
2057 ec
.Report
.Error (19, loc
, "Operator `{0}' cannot be applied to operands of type `{1}' and `{2}'",
2061 protected void Error_OperatorCannotBeApplied (ResolveContext ec
, Expression left
, Expression right
)
2063 Error_OperatorCannotBeApplied (ec
, left
, right
, OperName (oper
), loc
);
2067 // Converts operator to System.Linq.Expressions.ExpressionType enum name
2069 string GetOperatorExpressionTypeName ()
2072 case Operator
.Addition
:
2073 return is_compound
? "AddAssign" : "Add";
2074 case Operator
.BitwiseAnd
:
2075 return is_compound
? "AndAssign" : "And";
2076 case Operator
.BitwiseOr
:
2077 return is_compound
? "OrAssign" : "Or";
2078 case Operator
.Division
:
2079 return is_compound
? "DivideAssign" : "Divide";
2080 case Operator
.ExclusiveOr
:
2081 return is_compound
? "ExclusiveOrAssign" : "ExclusiveOr";
2082 case Operator
.Equality
:
2084 case Operator
.GreaterThan
:
2085 return "GreaterThan";
2086 case Operator
.GreaterThanOrEqual
:
2087 return "GreaterThanOrEqual";
2088 case Operator
.Inequality
:
2090 case Operator
.LeftShift
:
2091 return is_compound
? "LeftShiftAssign" : "LeftShift";
2092 case Operator
.LessThan
:
2094 case Operator
.LessThanOrEqual
:
2095 return "LessThanOrEqual";
2096 case Operator
.LogicalAnd
:
2098 case Operator
.LogicalOr
:
2100 case Operator
.Modulus
:
2101 return is_compound
? "ModuloAssign" : "Modulo";
2102 case Operator
.Multiply
:
2103 return is_compound
? "MultiplyAssign" : "Multiply";
2104 case Operator
.RightShift
:
2105 return is_compound
? "RightShiftAssign" : "RightShift";
2106 case Operator
.Subtraction
:
2107 return is_compound
? "SubtractAssign" : "Subtract";
2109 throw new NotImplementedException ("Unknown expression type operator " + oper
.ToString ());
2113 static string GetOperatorMetadataName (Operator op
)
2115 CSharp
.Operator
.OpType op_type
;
2117 case Operator
.Addition
:
2118 op_type
= CSharp
.Operator
.OpType
.Addition
; break;
2119 case Operator
.BitwiseAnd
:
2120 op_type
= CSharp
.Operator
.OpType
.BitwiseAnd
; break;
2121 case Operator
.BitwiseOr
:
2122 op_type
= CSharp
.Operator
.OpType
.BitwiseOr
; break;
2123 case Operator
.Division
:
2124 op_type
= CSharp
.Operator
.OpType
.Division
; break;
2125 case Operator
.Equality
:
2126 op_type
= CSharp
.Operator
.OpType
.Equality
; break;
2127 case Operator
.ExclusiveOr
:
2128 op_type
= CSharp
.Operator
.OpType
.ExclusiveOr
; break;
2129 case Operator
.GreaterThan
:
2130 op_type
= CSharp
.Operator
.OpType
.GreaterThan
; break;
2131 case Operator
.GreaterThanOrEqual
:
2132 op_type
= CSharp
.Operator
.OpType
.GreaterThanOrEqual
; break;
2133 case Operator
.Inequality
:
2134 op_type
= CSharp
.Operator
.OpType
.Inequality
; break;
2135 case Operator
.LeftShift
:
2136 op_type
= CSharp
.Operator
.OpType
.LeftShift
; break;
2137 case Operator
.LessThan
:
2138 op_type
= CSharp
.Operator
.OpType
.LessThan
; break;
2139 case Operator
.LessThanOrEqual
:
2140 op_type
= CSharp
.Operator
.OpType
.LessThanOrEqual
; break;
2141 case Operator
.Modulus
:
2142 op_type
= CSharp
.Operator
.OpType
.Modulus
; break;
2143 case Operator
.Multiply
:
2144 op_type
= CSharp
.Operator
.OpType
.Multiply
; break;
2145 case Operator
.RightShift
:
2146 op_type
= CSharp
.Operator
.OpType
.RightShift
; break;
2147 case Operator
.Subtraction
:
2148 op_type
= CSharp
.Operator
.OpType
.Subtraction
; break;
2150 throw new InternalErrorException (op
.ToString ());
2153 return CSharp
.Operator
.GetMetadataName (op_type
);
2156 public static void EmitOperatorOpcode (EmitContext ec
, Operator oper
, TypeSpec l
)
2161 case Operator
.Multiply
:
2162 if (ec
.HasSet (EmitContext
.Options
.CheckedScope
)) {
2163 if (l
== TypeManager
.int32_type
|| l
== TypeManager
.int64_type
)
2164 opcode
= OpCodes
.Mul_Ovf
;
2165 else if (!IsFloat (l
))
2166 opcode
= OpCodes
.Mul_Ovf_Un
;
2168 opcode
= OpCodes
.Mul
;
2170 opcode
= OpCodes
.Mul
;
2174 case Operator
.Division
:
2176 opcode
= OpCodes
.Div_Un
;
2178 opcode
= OpCodes
.Div
;
2181 case Operator
.Modulus
:
2183 opcode
= OpCodes
.Rem_Un
;
2185 opcode
= OpCodes
.Rem
;
2188 case Operator
.Addition
:
2189 if (ec
.HasSet (EmitContext
.Options
.CheckedScope
)) {
2190 if (l
== TypeManager
.int32_type
|| l
== TypeManager
.int64_type
)
2191 opcode
= OpCodes
.Add_Ovf
;
2192 else if (!IsFloat (l
))
2193 opcode
= OpCodes
.Add_Ovf_Un
;
2195 opcode
= OpCodes
.Add
;
2197 opcode
= OpCodes
.Add
;
2200 case Operator
.Subtraction
:
2201 if (ec
.HasSet (EmitContext
.Options
.CheckedScope
)) {
2202 if (l
== TypeManager
.int32_type
|| l
== TypeManager
.int64_type
)
2203 opcode
= OpCodes
.Sub_Ovf
;
2204 else if (!IsFloat (l
))
2205 opcode
= OpCodes
.Sub_Ovf_Un
;
2207 opcode
= OpCodes
.Sub
;
2209 opcode
= OpCodes
.Sub
;
2212 case Operator
.RightShift
:
2214 opcode
= OpCodes
.Shr_Un
;
2216 opcode
= OpCodes
.Shr
;
2219 case Operator
.LeftShift
:
2220 opcode
= OpCodes
.Shl
;
2223 case Operator
.Equality
:
2224 opcode
= OpCodes
.Ceq
;
2227 case Operator
.Inequality
:
2228 ec
.Emit (OpCodes
.Ceq
);
2229 ec
.Emit (OpCodes
.Ldc_I4_0
);
2231 opcode
= OpCodes
.Ceq
;
2234 case Operator
.LessThan
:
2236 opcode
= OpCodes
.Clt_Un
;
2238 opcode
= OpCodes
.Clt
;
2241 case Operator
.GreaterThan
:
2243 opcode
= OpCodes
.Cgt_Un
;
2245 opcode
= OpCodes
.Cgt
;
2248 case Operator
.LessThanOrEqual
:
2249 if (IsUnsigned (l
) || IsFloat (l
))
2250 ec
.Emit (OpCodes
.Cgt_Un
);
2252 ec
.Emit (OpCodes
.Cgt
);
2253 ec
.Emit (OpCodes
.Ldc_I4_0
);
2255 opcode
= OpCodes
.Ceq
;
2258 case Operator
.GreaterThanOrEqual
:
2259 if (IsUnsigned (l
) || IsFloat (l
))
2260 ec
.Emit (OpCodes
.Clt_Un
);
2262 ec
.Emit (OpCodes
.Clt
);
2264 ec
.Emit (OpCodes
.Ldc_I4_0
);
2266 opcode
= OpCodes
.Ceq
;
2269 case Operator
.BitwiseOr
:
2270 opcode
= OpCodes
.Or
;
2273 case Operator
.BitwiseAnd
:
2274 opcode
= OpCodes
.And
;
2277 case Operator
.ExclusiveOr
:
2278 opcode
= OpCodes
.Xor
;
2282 throw new InternalErrorException (oper
.ToString ());
2288 static bool IsUnsigned (TypeSpec t
)
2293 return (t
== TypeManager
.uint32_type
|| t
== TypeManager
.uint64_type
||
2294 t
== TypeManager
.ushort_type
|| t
== TypeManager
.byte_type
);
2297 static bool IsFloat (TypeSpec t
)
2299 return t
== TypeManager
.float_type
|| t
== TypeManager
.double_type
;
2302 public static void Reset ()
2304 pointer_operators
= standard_operators
= null;
2307 Expression
ResolveOperator (ResolveContext ec
)
2309 TypeSpec l
= left
.Type
;
2310 TypeSpec r
= right
.Type
;
2312 bool primitives_only
= false;
2314 if (standard_operators
== null)
2315 CreateStandardOperatorsTable ();
2318 // Handles predefined primitive types
2320 if (TypeManager
.IsPrimitiveType (l
) && TypeManager
.IsPrimitiveType (r
)) {
2321 if ((oper
& Operator
.ShiftMask
) == 0) {
2322 if (l
!= TypeManager
.bool_type
&& !DoBinaryOperatorPromotion (ec
))
2325 primitives_only
= true;
2329 if (l
.IsPointer
|| r
.IsPointer
)
2330 return ResolveOperatorPointer (ec
, l
, r
);
2333 bool lenum
= TypeManager
.IsEnumType (l
);
2334 bool renum
= TypeManager
.IsEnumType (r
);
2335 if (lenum
|| renum
) {
2336 expr
= ResolveOperatorEnum (ec
, lenum
, renum
, l
, r
);
2338 // TODO: Can this be ambiguous
2344 if ((oper
== Operator
.Addition
|| oper
== Operator
.Subtraction
|| (oper
& Operator
.EqualityMask
) != 0) &&
2345 (TypeManager
.IsDelegateType (l
) || TypeManager
.IsDelegateType (r
))) {
2347 expr
= ResolveOperatorDelegate (ec
, l
, r
);
2349 // TODO: Can this be ambiguous
2355 expr
= ResolveUserOperator (ec
, l
, r
);
2359 // Predefined reference types equality
2360 if ((oper
& Operator
.EqualityMask
) != 0) {
2361 expr
= ResolveOperatorEqualityRerefence (ec
, l
, r
);
2367 return ResolveOperatorPredefined (ec
, standard_operators
, primitives_only
, null);
2370 // at least one of 'left' or 'right' is an enumeration constant (EnumConstant or SideEffectConstant or ...)
2371 // if 'left' is not an enumeration constant, create one from the type of 'right'
2372 Constant
EnumLiftUp (ResolveContext ec
, Constant left
, Constant right
, Location loc
)
2375 case Operator
.BitwiseOr
:
2376 case Operator
.BitwiseAnd
:
2377 case Operator
.ExclusiveOr
:
2378 case Operator
.Equality
:
2379 case Operator
.Inequality
:
2380 case Operator
.LessThan
:
2381 case Operator
.LessThanOrEqual
:
2382 case Operator
.GreaterThan
:
2383 case Operator
.GreaterThanOrEqual
:
2384 if (TypeManager
.IsEnumType (left
.Type
))
2387 if (left
.IsZeroInteger
)
2388 return left
.TryReduce (ec
, right
.Type
, loc
);
2392 case Operator
.Addition
:
2393 case Operator
.Subtraction
:
2396 case Operator
.Multiply
:
2397 case Operator
.Division
:
2398 case Operator
.Modulus
:
2399 case Operator
.LeftShift
:
2400 case Operator
.RightShift
:
2401 if (TypeManager
.IsEnumType (right
.Type
) || TypeManager
.IsEnumType (left
.Type
))
2405 Error_OperatorCannotBeApplied (ec
, this.left
, this.right
);
2410 // The `|' operator used on types which were extended is dangerous
2412 void CheckBitwiseOrOnSignExtended (ResolveContext ec
)
2414 OpcodeCast lcast
= left
as OpcodeCast
;
2415 if (lcast
!= null) {
2416 if (IsUnsigned (lcast
.UnderlyingType
))
2420 OpcodeCast rcast
= right
as OpcodeCast
;
2421 if (rcast
!= null) {
2422 if (IsUnsigned (rcast
.UnderlyingType
))
2426 if (lcast
== null && rcast
== null)
2429 // FIXME: consider constants
2431 ec
.Report
.Warning (675, 3, loc
,
2432 "The operator `|' used on the sign-extended type `{0}'. Consider casting to a smaller unsigned type first",
2433 TypeManager
.CSharpName (lcast
!= null ? lcast
.UnderlyingType
: rcast
.UnderlyingType
));
2436 static void CreatePointerOperatorsTable ()
2438 var temp
= new List
<PredefinedPointerOperator
> ();
2441 // Pointer arithmetic:
2443 // T* operator + (T* x, int y); T* operator - (T* x, int y);
2444 // T* operator + (T* x, uint y); T* operator - (T* x, uint y);
2445 // T* operator + (T* x, long y); T* operator - (T* x, long y);
2446 // T* operator + (T* x, ulong y); T* operator - (T* x, ulong y);
2448 temp
.Add (new PredefinedPointerOperator (null, TypeManager
.int32_type
, Operator
.AdditionMask
| Operator
.SubtractionMask
));
2449 temp
.Add (new PredefinedPointerOperator (null, TypeManager
.uint32_type
, Operator
.AdditionMask
| Operator
.SubtractionMask
));
2450 temp
.Add (new PredefinedPointerOperator (null, TypeManager
.int64_type
, Operator
.AdditionMask
| Operator
.SubtractionMask
));
2451 temp
.Add (new PredefinedPointerOperator (null, TypeManager
.uint64_type
, Operator
.AdditionMask
| Operator
.SubtractionMask
));
2454 // T* operator + (int y, T* x);
2455 // T* operator + (uint y, T *x);
2456 // T* operator + (long y, T *x);
2457 // T* operator + (ulong y, T *x);
2459 temp
.Add (new PredefinedPointerOperator (TypeManager
.int32_type
, null, Operator
.AdditionMask
, null));
2460 temp
.Add (new PredefinedPointerOperator (TypeManager
.uint32_type
, null, Operator
.AdditionMask
, null));
2461 temp
.Add (new PredefinedPointerOperator (TypeManager
.int64_type
, null, Operator
.AdditionMask
, null));
2462 temp
.Add (new PredefinedPointerOperator (TypeManager
.uint64_type
, null, Operator
.AdditionMask
, null));
2465 // long operator - (T* x, T *y)
2467 temp
.Add (new PredefinedPointerOperator (null, Operator
.SubtractionMask
, TypeManager
.int64_type
));
2469 pointer_operators
= temp
.ToArray ();
2472 static void CreateStandardOperatorsTable ()
2474 var temp
= new List
<PredefinedOperator
> ();
2475 TypeSpec bool_type
= TypeManager
.bool_type
;
2477 temp
.Add (new PredefinedOperator (TypeManager
.int32_type
, Operator
.ArithmeticMask
| Operator
.BitwiseMask
));
2478 temp
.Add (new PredefinedOperator (TypeManager
.uint32_type
, Operator
.ArithmeticMask
| Operator
.BitwiseMask
));
2479 temp
.Add (new PredefinedOperator (TypeManager
.int64_type
, Operator
.ArithmeticMask
| Operator
.BitwiseMask
));
2480 temp
.Add (new PredefinedOperator (TypeManager
.uint64_type
, Operator
.ArithmeticMask
| Operator
.BitwiseMask
));
2481 temp
.Add (new PredefinedOperator (TypeManager
.float_type
, Operator
.ArithmeticMask
));
2482 temp
.Add (new PredefinedOperator (TypeManager
.double_type
, Operator
.ArithmeticMask
));
2483 temp
.Add (new PredefinedOperator (TypeManager
.decimal_type
, Operator
.ArithmeticMask
));
2485 temp
.Add (new PredefinedOperator (TypeManager
.int32_type
, Operator
.ComparisonMask
, bool_type
));
2486 temp
.Add (new PredefinedOperator (TypeManager
.uint32_type
, Operator
.ComparisonMask
, bool_type
));
2487 temp
.Add (new PredefinedOperator (TypeManager
.int64_type
, Operator
.ComparisonMask
, bool_type
));
2488 temp
.Add (new PredefinedOperator (TypeManager
.uint64_type
, Operator
.ComparisonMask
, bool_type
));
2489 temp
.Add (new PredefinedOperator (TypeManager
.float_type
, Operator
.ComparisonMask
, bool_type
));
2490 temp
.Add (new PredefinedOperator (TypeManager
.double_type
, Operator
.ComparisonMask
, bool_type
));
2491 temp
.Add (new PredefinedOperator (TypeManager
.decimal_type
, Operator
.ComparisonMask
, bool_type
));
2493 temp
.Add (new PredefinedOperator (TypeManager
.string_type
, Operator
.EqualityMask
, bool_type
));
2495 temp
.Add (new PredefinedStringOperator (TypeManager
.string_type
, Operator
.AdditionMask
));
2496 temp
.Add (new PredefinedStringOperator (TypeManager
.string_type
, TypeManager
.object_type
, Operator
.AdditionMask
));
2497 temp
.Add (new PredefinedStringOperator (TypeManager
.object_type
, TypeManager
.string_type
, Operator
.AdditionMask
));
2499 temp
.Add (new PredefinedOperator (bool_type
,
2500 Operator
.BitwiseMask
| Operator
.LogicalMask
| Operator
.EqualityMask
, bool_type
));
2502 temp
.Add (new PredefinedShiftOperator (TypeManager
.int32_type
, Operator
.ShiftMask
));
2503 temp
.Add (new PredefinedShiftOperator (TypeManager
.uint32_type
, Operator
.ShiftMask
));
2504 temp
.Add (new PredefinedShiftOperator (TypeManager
.int64_type
, Operator
.ShiftMask
));
2505 temp
.Add (new PredefinedShiftOperator (TypeManager
.uint64_type
, Operator
.ShiftMask
));
2507 standard_operators
= temp
.ToArray ();
2511 // Rules used during binary numeric promotion
2513 static bool DoNumericPromotion (ResolveContext rc
, ref Expression prim_expr
, ref Expression second_expr
, TypeSpec type
)
2518 Constant c
= prim_expr
as Constant
;
2520 temp
= c
.ConvertImplicitly (rc
, type
);
2527 if (type
== TypeManager
.uint32_type
) {
2528 etype
= prim_expr
.Type
;
2529 if (etype
== TypeManager
.int32_type
|| etype
== TypeManager
.short_type
|| etype
== TypeManager
.sbyte_type
) {
2530 type
= TypeManager
.int64_type
;
2532 if (type
!= second_expr
.Type
) {
2533 c
= second_expr
as Constant
;
2535 temp
= c
.ConvertImplicitly (rc
, type
);
2537 temp
= Convert
.ImplicitNumericConversion (second_expr
, type
);
2543 } else if (type
== TypeManager
.uint64_type
) {
2545 // A compile-time error occurs if the other operand is of type sbyte, short, int, or long
2547 if (type
== TypeManager
.int32_type
|| type
== TypeManager
.int64_type
||
2548 type
== TypeManager
.short_type
|| type
== TypeManager
.sbyte_type
)
2552 temp
= Convert
.ImplicitNumericConversion (prim_expr
, type
);
2561 // 7.2.6.2 Binary numeric promotions
2563 public bool DoBinaryOperatorPromotion (ResolveContext ec
)
2565 TypeSpec ltype
= left
.Type
;
2566 TypeSpec rtype
= right
.Type
;
2569 foreach (TypeSpec t
in ConstantFold
.BinaryPromotionsTypes
) {
2571 return t
== rtype
|| DoNumericPromotion (ec
, ref right
, ref left
, t
);
2574 return t
== ltype
|| DoNumericPromotion (ec
, ref left
, ref right
, t
);
2577 TypeSpec int32
= TypeManager
.int32_type
;
2578 if (ltype
!= int32
) {
2579 Constant c
= left
as Constant
;
2581 temp
= c
.ConvertImplicitly (ec
, int32
);
2583 temp
= Convert
.ImplicitNumericConversion (left
, int32
);
2590 if (rtype
!= int32
) {
2591 Constant c
= right
as Constant
;
2593 temp
= c
.ConvertImplicitly (ec
, int32
);
2595 temp
= Convert
.ImplicitNumericConversion (right
, int32
);
2605 protected override Expression
DoResolve (ResolveContext ec
)
2610 if ((oper
== Operator
.Subtraction
) && (left
is ParenthesizedExpression
)) {
2611 left
= ((ParenthesizedExpression
) left
).Expr
;
2612 left
= left
.Resolve (ec
, ResolveFlags
.VariableOrValue
| ResolveFlags
.Type
);
2616 if (left
.eclass
== ExprClass
.Type
) {
2617 ec
.Report
.Error (75, loc
, "To cast a negative value, you must enclose the value in parentheses");
2621 left
= left
.Resolve (ec
);
2626 Constant lc
= left
as Constant
;
2628 if (lc
!= null && lc
.Type
== TypeManager
.bool_type
&&
2629 ((oper
== Operator
.LogicalAnd
&& lc
.IsDefaultValue
) ||
2630 (oper
== Operator
.LogicalOr
&& !lc
.IsDefaultValue
))) {
2632 // FIXME: resolve right expression as unreachable
2633 // right.Resolve (ec);
2635 ec
.Report
.Warning (429, 4, loc
, "Unreachable expression code detected");
2639 right
= right
.Resolve (ec
);
2643 eclass
= ExprClass
.Value
;
2644 Constant rc
= right
as Constant
;
2646 // The conversion rules are ignored in enum context but why
2647 if (!ec
.HasSet (ResolveContext
.Options
.EnumScope
) && lc
!= null && rc
!= null && (TypeManager
.IsEnumType (left
.Type
) || TypeManager
.IsEnumType (right
.Type
))) {
2648 lc
= EnumLiftUp (ec
, lc
, rc
, loc
);
2650 rc
= EnumLiftUp (ec
, rc
, lc
, loc
);
2653 if (rc
!= null && lc
!= null) {
2654 int prev_e
= ec
.Report
.Errors
;
2655 Expression e
= ConstantFold
.BinaryFold (ec
, oper
, lc
, rc
, loc
);
2659 if (e
!= null || ec
.Report
.Errors
!= prev_e
)
2663 // Comparison warnings
2664 if ((oper
& Operator
.ComparisonMask
) != 0) {
2665 if (left
.Equals (right
)) {
2666 ec
.Report
.Warning (1718, 3, loc
, "A comparison made to same variable. Did you mean to compare something else?");
2668 CheckUselessComparison (ec
, lc
, right
.Type
);
2669 CheckUselessComparison (ec
, rc
, left
.Type
);
2672 if (left
.Type
== InternalType
.Dynamic
|| right
.Type
== InternalType
.Dynamic
) {
2673 Arguments args
= new Arguments (2);
2674 args
.Add (new Argument (left
));
2675 args
.Add (new Argument (right
));
2676 return new DynamicExpressionStatement (this, args
, loc
).Resolve (ec
);
2679 if (RootContext
.Version
>= LanguageVersion
.ISO_2
&&
2680 ((TypeManager
.IsNullableType (left
.Type
) && (right
is NullLiteral
|| TypeManager
.IsNullableType (right
.Type
) || TypeManager
.IsValueType (right
.Type
))) ||
2681 (TypeManager
.IsValueType (left
.Type
) && right
is NullLiteral
) ||
2682 (TypeManager
.IsNullableType (right
.Type
) && (left
is NullLiteral
|| TypeManager
.IsNullableType (left
.Type
) || TypeManager
.IsValueType (left
.Type
))) ||
2683 (TypeManager
.IsValueType (right
.Type
) && left
is NullLiteral
)))
2684 return new Nullable
.LiftedBinaryOperator (oper
, left
, right
, loc
).Resolve (ec
);
2686 return DoResolveCore (ec
, left
, right
);
2689 protected Expression
DoResolveCore (ResolveContext ec
, Expression left_orig
, Expression right_orig
)
2691 Expression expr
= ResolveOperator (ec
);
2693 Error_OperatorCannotBeApplied (ec
, left_orig
, right_orig
);
2695 if (left
== null || right
== null)
2696 throw new InternalErrorException ("Invalid conversion");
2698 if (oper
== Operator
.BitwiseOr
)
2699 CheckBitwiseOrOnSignExtended (ec
);
2704 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
2706 var le
= left
.MakeExpression (ctx
);
2707 var re
= right
.MakeExpression (ctx
);
2708 bool is_checked
= ctx
.HasSet (BuilderContext
.Options
.CheckedScope
);
2711 case Operator
.Addition
:
2712 return is_checked
? SLE
.Expression
.AddChecked (le
, re
) : SLE
.Expression
.Add (le
, re
);
2713 case Operator
.BitwiseAnd
:
2714 return SLE
.Expression
.And (le
, re
);
2715 case Operator
.BitwiseOr
:
2716 return SLE
.Expression
.Or (le
, re
);
2717 case Operator
.Division
:
2718 return SLE
.Expression
.Divide (le
, re
);
2719 case Operator
.Equality
:
2720 return SLE
.Expression
.Equal (le
, re
);
2721 case Operator
.ExclusiveOr
:
2722 return SLE
.Expression
.ExclusiveOr (le
, re
);
2723 case Operator
.GreaterThan
:
2724 return SLE
.Expression
.GreaterThan (le
, re
);
2725 case Operator
.GreaterThanOrEqual
:
2726 return SLE
.Expression
.GreaterThanOrEqual (le
, re
);
2727 case Operator
.Inequality
:
2728 return SLE
.Expression
.NotEqual (le
, re
);
2729 case Operator
.LeftShift
:
2730 return SLE
.Expression
.LeftShift (le
, re
);
2731 case Operator
.LessThan
:
2732 return SLE
.Expression
.LessThan (le
, re
);
2733 case Operator
.LessThanOrEqual
:
2734 return SLE
.Expression
.LessThanOrEqual (le
, re
);
2735 case Operator
.LogicalAnd
:
2736 return SLE
.Expression
.AndAlso (le
, re
);
2737 case Operator
.LogicalOr
:
2738 return SLE
.Expression
.OrElse (le
, re
);
2739 case Operator
.Modulus
:
2740 return SLE
.Expression
.Modulo (le
, re
);
2741 case Operator
.Multiply
:
2742 return is_checked
? SLE
.Expression
.MultiplyChecked (le
, re
) : SLE
.Expression
.Multiply (le
, re
);
2743 case Operator
.RightShift
:
2744 return SLE
.Expression
.RightShift (le
, re
);
2745 case Operator
.Subtraction
:
2746 return is_checked
? SLE
.Expression
.SubtractChecked (le
, re
) : SLE
.Expression
.Subtract (le
, re
);
2748 throw new NotImplementedException (oper
.ToString ());
2753 // D operator + (D x, D y)
2754 // D operator - (D x, D y)
2755 // bool operator == (D x, D y)
2756 // bool operator != (D x, D y)
2758 Expression
ResolveOperatorDelegate (ResolveContext ec
, TypeSpec l
, TypeSpec r
)
2760 bool is_equality
= (oper
& Operator
.EqualityMask
) != 0;
2761 if (!TypeManager
.IsEqual (l
, r
) && !TypeSpecComparer
.Variant
.IsEqual (r
, l
)) {
2763 if (right
.eclass
== ExprClass
.MethodGroup
|| (r
== InternalType
.AnonymousMethod
&& !is_equality
)) {
2764 tmp
= Convert
.ImplicitConversionRequired (ec
, right
, l
, loc
);
2769 } else if (left
.eclass
== ExprClass
.MethodGroup
|| (l
== InternalType
.AnonymousMethod
&& !is_equality
)) {
2770 tmp
= Convert
.ImplicitConversionRequired (ec
, left
, r
, loc
);
2781 // Resolve delegate equality as a user operator
2784 return ResolveUserOperator (ec
, l
, r
);
2787 Arguments args
= new Arguments (2);
2788 args
.Add (new Argument (left
));
2789 args
.Add (new Argument (right
));
2791 if (oper
== Operator
.Addition
) {
2792 if (TypeManager
.delegate_combine_delegate_delegate
== null) {
2793 TypeManager
.delegate_combine_delegate_delegate
= TypeManager
.GetPredefinedMethod (
2794 TypeManager
.delegate_type
, "Combine", loc
, TypeManager
.delegate_type
, TypeManager
.delegate_type
);
2797 method
= TypeManager
.delegate_combine_delegate_delegate
;
2799 if (TypeManager
.delegate_remove_delegate_delegate
== null) {
2800 TypeManager
.delegate_remove_delegate_delegate
= TypeManager
.GetPredefinedMethod (
2801 TypeManager
.delegate_type
, "Remove", loc
, TypeManager
.delegate_type
, TypeManager
.delegate_type
);
2804 method
= TypeManager
.delegate_remove_delegate_delegate
;
2808 return new EmptyExpression (TypeManager
.decimal_type
);
2810 MethodGroupExpr mg
= new MethodGroupExpr (method
, TypeManager
.delegate_type
, loc
);
2811 mg
= mg
.OverloadResolve (ec
, ref args
, false, loc
);
2813 return new ClassCast (new UserOperatorCall (mg
, args
, CreateExpressionTree
, loc
), l
);
2817 // Enumeration operators
2819 Expression
ResolveOperatorEnum (ResolveContext ec
, bool lenum
, bool renum
, TypeSpec ltype
, TypeSpec rtype
)
2822 // bool operator == (E x, E y);
2823 // bool operator != (E x, E y);
2824 // bool operator < (E x, E y);
2825 // bool operator > (E x, E y);
2826 // bool operator <= (E x, E y);
2827 // bool operator >= (E x, E y);
2829 // E operator & (E x, E y);
2830 // E operator | (E x, E y);
2831 // E operator ^ (E x, E y);
2833 // U operator - (E e, E f)
2834 // E operator - (E e, U x)
2836 // E operator + (U x, E e)
2837 // E operator + (E e, U x)
2839 if (!((oper
& (Operator
.ComparisonMask
| Operator
.BitwiseMask
)) != 0 ||
2840 (oper
== Operator
.Subtraction
&& lenum
) ||
2841 (oper
== Operator
.Addition
&& (lenum
!= renum
|| type
!= null)))) // type != null for lifted null
2844 Expression ltemp
= left
;
2845 Expression rtemp
= right
;
2846 TypeSpec underlying_type
;
2849 if ((oper
& (Operator
.ComparisonMask
| Operator
.BitwiseMask
)) != 0) {
2851 expr
= Convert
.ImplicitConversion (ec
, left
, rtype
, loc
);
2857 expr
= Convert
.ImplicitConversion (ec
, right
, ltype
, loc
);
2865 if (TypeManager
.IsEqual (ltype
, rtype
)) {
2866 underlying_type
= EnumSpec
.GetUnderlyingType (ltype
);
2868 if (left
is Constant
)
2869 left
= ((Constant
) left
).ConvertExplicitly (false, underlying_type
).Resolve (ec
);
2871 left
= EmptyCast
.Create (left
, underlying_type
);
2873 if (right
is Constant
)
2874 right
= ((Constant
) right
).ConvertExplicitly (false, underlying_type
).Resolve (ec
);
2876 right
= EmptyCast
.Create (right
, underlying_type
);
2878 underlying_type
= EnumSpec
.GetUnderlyingType (ltype
);
2880 if (oper
!= Operator
.Subtraction
&& oper
!= Operator
.Addition
) {
2881 Constant c
= right
as Constant
;
2882 if (c
== null || !c
.IsDefaultValue
)
2885 if (!Convert
.ImplicitStandardConversionExists (right
, underlying_type
))
2888 right
= Convert
.ImplicitConversionStandard (ec
, right
, underlying_type
, right
.Location
);
2891 if (left
is Constant
)
2892 left
= ((Constant
) left
).ConvertExplicitly (false, underlying_type
).Resolve (ec
);
2894 left
= EmptyCast
.Create (left
, underlying_type
);
2897 underlying_type
= EnumSpec
.GetUnderlyingType (rtype
);
2899 if (oper
!= Operator
.Addition
) {
2900 Constant c
= left
as Constant
;
2901 if (c
== null || !c
.IsDefaultValue
)
2904 if (!Convert
.ImplicitStandardConversionExists (left
, underlying_type
))
2907 left
= Convert
.ImplicitConversionStandard (ec
, left
, underlying_type
, left
.Location
);
2910 if (right
is Constant
)
2911 right
= ((Constant
) right
).ConvertExplicitly (false, underlying_type
).Resolve (ec
);
2913 right
= EmptyCast
.Create (right
, underlying_type
);
2920 // C# specification uses explicit cast syntax which means binary promotion
2921 // should happen, however it seems that csc does not do that
2923 if (!DoBinaryOperatorPromotion (ec
)) {
2929 TypeSpec res_type
= null;
2930 if ((oper
& Operator
.BitwiseMask
) != 0 || oper
== Operator
.Subtraction
|| oper
== Operator
.Addition
) {
2931 TypeSpec promoted_type
= lenum
? left
.Type
: right
.Type
;
2932 enum_conversion
= Convert
.ExplicitNumericConversion (
2933 new EmptyExpression (promoted_type
), underlying_type
);
2935 if (oper
== Operator
.Subtraction
&& renum
&& lenum
)
2936 res_type
= underlying_type
;
2937 else if (oper
== Operator
.Addition
&& renum
)
2943 expr
= ResolveOperatorPredefined (ec
, standard_operators
, true, res_type
);
2944 if (!is_compound
|| expr
== null)
2952 // If the return type of the selected operator is implicitly convertible to the type of x
2954 if (Convert
.ImplicitConversionExists (ec
, expr
, ltype
))
2958 // Otherwise, if the selected operator is a predefined operator, if the return type of the
2959 // selected operator is explicitly convertible to the type of x, and if y is implicitly
2960 // convertible to the type of x or the operator is a shift operator, then the operation
2961 // is evaluated as x = (T)(x op y), where T is the type of x
2963 expr
= Convert
.ExplicitConversion (ec
, expr
, ltype
, loc
);
2967 if (Convert
.ImplicitConversionExists (ec
, ltemp
, ltype
))
2974 // 7.9.6 Reference type equality operators
2976 Binary
ResolveOperatorEqualityRerefence (ResolveContext ec
, TypeSpec l
, TypeSpec r
)
2979 // operator != (object a, object b)
2980 // operator == (object a, object b)
2983 // TODO: this method is almost equivalent to Convert.ImplicitReferenceConversion
2985 if (left
.eclass
== ExprClass
.MethodGroup
|| right
.eclass
== ExprClass
.MethodGroup
)
2988 type
= TypeManager
.bool_type
;
2990 var lgen
= l
as TypeParameterSpec
;
2993 if (l
is InternalType
)
2998 // Only allow to compare same reference type parameter
3000 if (TypeManager
.IsReferenceType (l
)) {
3001 left
= new BoxedCast (left
, TypeManager
.object_type
);
3002 right
= new BoxedCast (right
, TypeManager
.object_type
);
3009 if (TypeManager
.IsValueType (l
))
3015 var rgen
= r
as TypeParameterSpec
;
3018 // a, Both operands are reference-type values or the value null
3019 // b, One operand is a value of type T where T is a type-parameter and
3020 // the other operand is the value null. Furthermore T does not have the
3021 // value type constrain
3023 if (left
is NullLiteral
|| right
is NullLiteral
) {
3025 if (lgen
.HasSpecialStruct
)
3028 left
= new BoxedCast (left
, TypeManager
.object_type
);
3033 if (rgen
.HasSpecialStruct
)
3036 right
= new BoxedCast (right
, TypeManager
.object_type
);
3042 // An interface is converted to the object before the
3043 // standard conversion is applied. It's not clear from the
3044 // standard but it looks like it works like that.
3047 if (!TypeManager
.IsReferenceType (l
))
3050 l
= TypeManager
.object_type
;
3051 left
= new BoxedCast (left
, l
);
3052 } else if (l
.IsInterface
) {
3053 l
= TypeManager
.object_type
;
3054 } else if (TypeManager
.IsStruct (l
)) {
3059 if (!TypeManager
.IsReferenceType (r
))
3062 r
= TypeManager
.object_type
;
3063 right
= new BoxedCast (right
, r
);
3064 } else if (r
.IsInterface
) {
3065 r
= TypeManager
.object_type
;
3066 } else if (TypeManager
.IsStruct (r
)) {
3071 const string ref_comparison
= "Possible unintended reference comparison. " +
3072 "Consider casting the {0} side of the expression to `string' to compare the values";
3075 // A standard implicit conversion exists from the type of either
3076 // operand to the type of the other operand
3078 if (Convert
.ImplicitReferenceConversionExists (left
, r
)) {
3079 if (l
== TypeManager
.string_type
)
3080 ec
.Report
.Warning (253, 2, loc
, ref_comparison
, "right");
3085 if (Convert
.ImplicitReferenceConversionExists (right
, l
)) {
3086 if (r
== TypeManager
.string_type
)
3087 ec
.Report
.Warning (252, 2, loc
, ref_comparison
, "left");
3096 Expression
ResolveOperatorPointer (ResolveContext ec
, TypeSpec l
, TypeSpec r
)
3099 // bool operator == (void* x, void* y);
3100 // bool operator != (void* x, void* y);
3101 // bool operator < (void* x, void* y);
3102 // bool operator > (void* x, void* y);
3103 // bool operator <= (void* x, void* y);
3104 // bool operator >= (void* x, void* y);
3106 if ((oper
& Operator
.ComparisonMask
) != 0) {
3109 temp
= Convert
.ImplicitConversion (ec
, left
, r
, left
.Location
);
3116 temp
= Convert
.ImplicitConversion (ec
, right
, l
, right
.Location
);
3122 type
= TypeManager
.bool_type
;
3126 if (pointer_operators
== null)
3127 CreatePointerOperatorsTable ();
3129 return ResolveOperatorPredefined (ec
, pointer_operators
, false, null);
3133 // Build-in operators method overloading
3135 protected virtual Expression
ResolveOperatorPredefined (ResolveContext ec
, PredefinedOperator
[] operators
, bool primitives_only
, TypeSpec enum_type
)
3137 PredefinedOperator best_operator
= null;
3138 TypeSpec l
= left
.Type
;
3139 TypeSpec r
= right
.Type
;
3140 Operator oper_mask
= oper
& ~Operator
.ValuesOnlyMask
;
3142 foreach (PredefinedOperator po
in operators
) {
3143 if ((po
.OperatorsMask
& oper_mask
) == 0)
3146 if (primitives_only
) {
3147 if (!po
.IsPrimitiveApplicable (l
, r
))
3150 if (!po
.IsApplicable (ec
, left
, right
))
3154 if (best_operator
== null) {
3156 if (primitives_only
)
3162 best_operator
= po
.ResolveBetterOperator (ec
, best_operator
);
3164 if (best_operator
== null) {
3165 ec
.Report
.Error (34, loc
, "Operator `{0}' is ambiguous on operands of type `{1}' and `{2}'",
3166 OperName (oper
), TypeManager
.CSharpName (l
), TypeManager
.CSharpName (r
));
3173 if (best_operator
== null)
3176 Expression expr
= best_operator
.ConvertResult (ec
, this);
3179 // Optimize &/&& constant expressions with 0 value
3181 if (oper
== Operator
.BitwiseAnd
|| oper
== Operator
.LogicalAnd
) {
3182 Constant rc
= right
as Constant
;
3183 Constant lc
= left
as Constant
;
3184 if ((lc
!= null && lc
.IsDefaultValue
) || (rc
!= null && rc
.IsDefaultValue
)) {
3186 // The result is a constant with side-effect
3188 Constant side_effect
= rc
== null ?
3189 new SideEffectConstant (lc
, right
, loc
) :
3190 new SideEffectConstant (rc
, left
, loc
);
3192 return ReducedExpression
.Create (side_effect
.Resolve (ec
), expr
);
3196 if (enum_type
== null)
3200 // HACK: required by enum_conversion
3202 expr
.Type
= enum_type
;
3203 return EmptyCast
.Create (expr
, enum_type
);
3207 // Performs user-operator overloading
3209 protected virtual Expression
ResolveUserOperator (ResolveContext ec
, TypeSpec l
, TypeSpec r
)
3212 if (oper
== Operator
.LogicalAnd
)
3213 user_oper
= Operator
.BitwiseAnd
;
3214 else if (oper
== Operator
.LogicalOr
)
3215 user_oper
= Operator
.BitwiseOr
;
3219 string op
= GetOperatorMetadataName (user_oper
);
3221 MethodGroupExpr left_operators
= MethodLookup (ec
.Compiler
, ec
.CurrentType
, l
, MemberKind
.Operator
, op
, 0, loc
);
3222 MethodGroupExpr right_operators
= null;
3224 if (!TypeManager
.IsEqual (r
, l
)) {
3225 right_operators
= MethodLookup (ec
.Compiler
, ec
.CurrentType
, r
, MemberKind
.Operator
, op
, 0, loc
);
3226 if (right_operators
== null && left_operators
== null)
3228 } else if (left_operators
== null) {
3232 Arguments args
= new Arguments (2);
3233 Argument larg
= new Argument (left
);
3235 Argument rarg
= new Argument (right
);
3238 MethodGroupExpr union
;
3241 // User-defined operator implementations always take precedence
3242 // over predefined operator implementations
3244 if (left_operators
!= null && right_operators
!= null) {
3245 if (IsPredefinedUserOperator (l
, user_oper
)) {
3246 union
= right_operators
.OverloadResolve (ec
, ref args
, true, loc
);
3248 union
= left_operators
;
3249 } else if (IsPredefinedUserOperator (r
, user_oper
)) {
3250 union
= left_operators
.OverloadResolve (ec
, ref args
, true, loc
);
3252 union
= right_operators
;
3254 union
= MethodGroupExpr
.MakeUnionSet (left_operators
, right_operators
, loc
);
3256 } else if (left_operators
!= null) {
3257 union
= left_operators
;
3259 union
= right_operators
;
3262 union
= union
.OverloadResolve (ec
, ref args
, true, loc
);
3266 Expression oper_expr
;
3268 // TODO: CreateExpressionTree is allocated every time
3269 if (user_oper
!= oper
) {
3270 oper_expr
= new ConditionalLogicalOperator (union
, args
, CreateExpressionTree
,
3271 oper
== Operator
.LogicalAnd
, loc
).Resolve (ec
);
3273 oper_expr
= new UserOperatorCall (union
, args
, CreateExpressionTree
, loc
);
3276 // This is used to check if a test 'x == null' can be optimized to a reference equals,
3277 // and not invoke user operator
3279 if ((oper
& Operator
.EqualityMask
) != 0) {
3280 if ((left
is NullLiteral
&& IsBuildInEqualityOperator (r
)) ||
3281 (right
is NullLiteral
&& IsBuildInEqualityOperator (l
))) {
3282 type
= TypeManager
.bool_type
;
3283 if (left
is NullLiteral
|| right
is NullLiteral
)
3284 oper_expr
= ReducedExpression
.Create (this, oper_expr
);
3285 } else if (l
!= r
) {
3286 var mi
= union
.BestCandidate
;
3289 // Two System.Delegate(s) are never equal
3291 if (mi
.DeclaringType
== TypeManager
.multicast_delegate_type
)
3302 public override TypeExpr
ResolveAsTypeTerminal (IMemberContext ec
, bool silent
)
3307 private void CheckUselessComparison (ResolveContext ec
, Constant c
, TypeSpec type
)
3309 if (c
== null || !IsTypeIntegral (type
)
3310 || c
is StringConstant
3311 || c
is BoolConstant
3312 || c
is FloatConstant
3313 || c
is DoubleConstant
3314 || c
is DecimalConstant
3320 if (c
is ULongConstant
) {
3321 ulong uvalue
= ((ULongConstant
) c
).Value
;
3322 if (uvalue
> long.MaxValue
) {
3323 if (type
== TypeManager
.byte_type
||
3324 type
== TypeManager
.sbyte_type
||
3325 type
== TypeManager
.short_type
||
3326 type
== TypeManager
.ushort_type
||
3327 type
== TypeManager
.int32_type
||
3328 type
== TypeManager
.uint32_type
||
3329 type
== TypeManager
.int64_type
||
3330 type
== TypeManager
.char_type
)
3331 WarnUselessComparison (ec
, type
);
3334 value = (long) uvalue
;
3336 else if (c
is ByteConstant
)
3337 value = ((ByteConstant
) c
).Value
;
3338 else if (c
is SByteConstant
)
3339 value = ((SByteConstant
) c
).Value
;
3340 else if (c
is ShortConstant
)
3341 value = ((ShortConstant
) c
).Value
;
3342 else if (c
is UShortConstant
)
3343 value = ((UShortConstant
) c
).Value
;
3344 else if (c
is IntConstant
)
3345 value = ((IntConstant
) c
).Value
;
3346 else if (c
is UIntConstant
)
3347 value = ((UIntConstant
) c
).Value
;
3348 else if (c
is LongConstant
)
3349 value = ((LongConstant
) c
).Value
;
3350 else if (c
is CharConstant
)
3351 value = ((CharConstant
)c
).Value
;
3356 if (IsValueOutOfRange (value, type
))
3357 WarnUselessComparison (ec
, type
);
3360 static bool IsValueOutOfRange (long value, TypeSpec type
)
3362 if (IsTypeUnsigned (type
) && value < 0)
3364 return type
== TypeManager
.sbyte_type
&& (value >= 0x80 || value < -0x80) ||
3365 type
== TypeManager
.byte_type
&& value >= 0x100 ||
3366 type
== TypeManager
.short_type
&& (value >= 0x8000 || value < -0x8000) ||
3367 type
== TypeManager
.ushort_type
&& value >= 0x10000 ||
3368 type
== TypeManager
.int32_type
&& (value >= 0x80000000 || value < -0x80000000) ||
3369 type
== TypeManager
.uint32_type
&& value >= 0x100000000;
3372 static bool IsBuildInEqualityOperator (TypeSpec t
)
3374 return t
== TypeManager
.object_type
|| t
== TypeManager
.string_type
||
3375 t
== TypeManager
.delegate_type
|| TypeManager
.IsDelegateType (t
);
3378 static bool IsPredefinedUserOperator (TypeSpec t
, Operator op
)
3381 // Some predefined types have user operators
3383 return (op
& Operator
.EqualityMask
) != 0 && (t
== TypeManager
.string_type
|| t
== TypeManager
.decimal_type
);
3386 private static bool IsTypeIntegral (TypeSpec type
)
3388 return type
== TypeManager
.uint64_type
||
3389 type
== TypeManager
.int64_type
||
3390 type
== TypeManager
.uint32_type
||
3391 type
== TypeManager
.int32_type
||
3392 type
== TypeManager
.ushort_type
||
3393 type
== TypeManager
.short_type
||
3394 type
== TypeManager
.sbyte_type
||
3395 type
== TypeManager
.byte_type
||
3396 type
== TypeManager
.char_type
;
3399 private static bool IsTypeUnsigned (TypeSpec type
)
3401 return type
== TypeManager
.uint64_type
||
3402 type
== TypeManager
.uint32_type
||
3403 type
== TypeManager
.ushort_type
||
3404 type
== TypeManager
.byte_type
||
3405 type
== TypeManager
.char_type
;
3408 private void WarnUselessComparison (ResolveContext ec
, TypeSpec type
)
3410 ec
.Report
.Warning (652, 2, loc
, "A comparison between a constant and a variable is useless. The constant is out of the range of the variable type `{0}'",
3411 TypeManager
.CSharpName (type
));
3415 /// EmitBranchable is called from Statement.EmitBoolExpression in the
3416 /// context of a conditional bool expression. This function will return
3417 /// false if it is was possible to use EmitBranchable, or true if it was.
3419 /// The expression's code is generated, and we will generate a branch to `target'
3420 /// if the resulting expression value is equal to isTrue
3422 public override void EmitBranchable (EmitContext ec
, Label target
, bool on_true
)
3425 // This is more complicated than it looks, but its just to avoid
3426 // duplicated tests: basically, we allow ==, !=, >, <, >= and <=
3427 // but on top of that we want for == and != to use a special path
3428 // if we are comparing against null
3430 if ((oper
== Operator
.Equality
|| oper
== Operator
.Inequality
) && (left
is Constant
|| right
is Constant
)) {
3431 bool my_on_true
= oper
== Operator
.Inequality
? on_true
: !on_true
;
3434 // put the constant on the rhs, for simplicity
3436 if (left
is Constant
) {
3437 Expression swap
= right
;
3442 if (((Constant
) right
).IsZeroInteger
) {
3443 left
.EmitBranchable (ec
, target
, my_on_true
);
3446 if (right
.Type
== TypeManager
.bool_type
) {
3447 // right is a boolean, and it's not 'false' => it is 'true'
3448 left
.EmitBranchable (ec
, target
, !my_on_true
);
3452 } else if (oper
== Operator
.LogicalAnd
) {
3455 Label tests_end
= ec
.DefineLabel ();
3457 left
.EmitBranchable (ec
, tests_end
, false);
3458 right
.EmitBranchable (ec
, target
, true);
3459 ec
.MarkLabel (tests_end
);
3462 // This optimizes code like this
3463 // if (true && i > 4)
3465 if (!(left
is Constant
))
3466 left
.EmitBranchable (ec
, target
, false);
3468 if (!(right
is Constant
))
3469 right
.EmitBranchable (ec
, target
, false);
3474 } else if (oper
== Operator
.LogicalOr
){
3476 left
.EmitBranchable (ec
, target
, true);
3477 right
.EmitBranchable (ec
, target
, true);
3480 Label tests_end
= ec
.DefineLabel ();
3481 left
.EmitBranchable (ec
, tests_end
, true);
3482 right
.EmitBranchable (ec
, target
, false);
3483 ec
.MarkLabel (tests_end
);
3488 } else if (!(oper
== Operator
.LessThan
|| oper
== Operator
.GreaterThan
||
3489 oper
== Operator
.LessThanOrEqual
|| oper
== Operator
.GreaterThanOrEqual
||
3490 oper
== Operator
.Equality
|| oper
== Operator
.Inequality
)) {
3491 base.EmitBranchable (ec
, target
, on_true
);
3498 TypeSpec t
= left
.Type
;
3499 bool is_float
= IsFloat (t
);
3500 bool is_unsigned
= is_float
|| IsUnsigned (t
);
3503 case Operator
.Equality
:
3505 ec
.Emit (OpCodes
.Beq
, target
);
3507 ec
.Emit (OpCodes
.Bne_Un
, target
);
3510 case Operator
.Inequality
:
3512 ec
.Emit (OpCodes
.Bne_Un
, target
);
3514 ec
.Emit (OpCodes
.Beq
, target
);
3517 case Operator
.LessThan
:
3519 if (is_unsigned
&& !is_float
)
3520 ec
.Emit (OpCodes
.Blt_Un
, target
);
3522 ec
.Emit (OpCodes
.Blt
, target
);
3525 ec
.Emit (OpCodes
.Bge_Un
, target
);
3527 ec
.Emit (OpCodes
.Bge
, target
);
3530 case Operator
.GreaterThan
:
3532 if (is_unsigned
&& !is_float
)
3533 ec
.Emit (OpCodes
.Bgt_Un
, target
);
3535 ec
.Emit (OpCodes
.Bgt
, target
);
3538 ec
.Emit (OpCodes
.Ble_Un
, target
);
3540 ec
.Emit (OpCodes
.Ble
, target
);
3543 case Operator
.LessThanOrEqual
:
3545 if (is_unsigned
&& !is_float
)
3546 ec
.Emit (OpCodes
.Ble_Un
, target
);
3548 ec
.Emit (OpCodes
.Ble
, target
);
3551 ec
.Emit (OpCodes
.Bgt_Un
, target
);
3553 ec
.Emit (OpCodes
.Bgt
, target
);
3557 case Operator
.GreaterThanOrEqual
:
3559 if (is_unsigned
&& !is_float
)
3560 ec
.Emit (OpCodes
.Bge_Un
, target
);
3562 ec
.Emit (OpCodes
.Bge
, target
);
3565 ec
.Emit (OpCodes
.Blt_Un
, target
);
3567 ec
.Emit (OpCodes
.Blt
, target
);
3570 throw new InternalErrorException (oper
.ToString ());
3574 public override void Emit (EmitContext ec
)
3576 EmitOperator (ec
, left
.Type
);
3579 protected virtual void EmitOperator (EmitContext ec
, TypeSpec l
)
3582 // Handle short-circuit operators differently
3585 if ((oper
& Operator
.LogicalMask
) != 0) {
3586 Label load_result
= ec
.DefineLabel ();
3587 Label end
= ec
.DefineLabel ();
3589 bool is_or
= oper
== Operator
.LogicalOr
;
3590 left
.EmitBranchable (ec
, load_result
, is_or
);
3592 ec
.Emit (OpCodes
.Br_S
, end
);
3594 ec
.MarkLabel (load_result
);
3595 ec
.Emit (is_or
? OpCodes
.Ldc_I4_1
: OpCodes
.Ldc_I4_0
);
3601 // Optimize zero-based operations which cannot be optimized at expression level
3603 if (oper
== Operator
.Subtraction
) {
3604 var lc
= left
as IntegralConstant
;
3605 if (lc
!= null && lc
.IsDefaultValue
) {
3607 ec
.Emit (OpCodes
.Neg
);
3614 EmitOperatorOpcode (ec
, oper
, l
);
3617 // Nullable enum could require underlying type cast and we cannot simply wrap binary
3618 // expression because that would wrap lifted binary operation
3620 if (enum_conversion
!= null)
3621 enum_conversion
.Emit (ec
);
3624 public override void EmitSideEffect (EmitContext ec
)
3626 if ((oper
& Operator
.LogicalMask
) != 0 ||
3627 (ec
.HasSet (EmitContext
.Options
.CheckedScope
) && (oper
== Operator
.Multiply
|| oper
== Operator
.Addition
|| oper
== Operator
.Subtraction
))) {
3628 base.EmitSideEffect (ec
);
3630 left
.EmitSideEffect (ec
);
3631 right
.EmitSideEffect (ec
);
3635 protected override void CloneTo (CloneContext clonectx
, Expression t
)
3637 Binary target
= (Binary
) t
;
3639 target
.left
= left
.Clone (clonectx
);
3640 target
.right
= right
.Clone (clonectx
);
3643 public Expression
CreateCallSiteBinder (ResolveContext ec
, Arguments args
)
3645 Arguments binder_args
= new Arguments (4);
3647 MemberAccess sle
= new MemberAccess (new MemberAccess (
3648 new QualifiedAliasMember (QualifiedAliasMember
.GlobalAlias
, "System", loc
), "Linq", loc
), "Expressions", loc
);
3650 CSharpBinderFlags flags
= 0;
3651 if (ec
.HasSet (ResolveContext
.Options
.CheckedScope
))
3652 flags
= CSharpBinderFlags
.CheckedContext
;
3654 if ((oper
& Operator
.LogicalMask
) != 0)
3655 flags
|= CSharpBinderFlags
.BinaryOperationLogical
;
3657 binder_args
.Add (new Argument (new EnumConstant (new IntLiteral ((int) flags
, loc
), TypeManager
.binder_flags
)));
3658 binder_args
.Add (new Argument (new MemberAccess (new MemberAccess (sle
, "ExpressionType", loc
), GetOperatorExpressionTypeName (), loc
)));
3659 binder_args
.Add (new Argument (new TypeOf (new TypeExpression (ec
.CurrentType
, loc
), loc
)));
3660 binder_args
.Add (new Argument (new ImplicitlyTypedArrayCreation ("[]", args
.CreateDynamicBinderArguments (ec
), loc
)));
3662 return new Invocation (DynamicExpressionStatement
.GetBinder ("BinaryOperation", loc
), binder_args
);
3665 public override Expression
CreateExpressionTree (ResolveContext ec
)
3667 return CreateExpressionTree (ec
, null);
3670 Expression
CreateExpressionTree (ResolveContext ec
, MethodGroupExpr method
)
3673 bool lift_arg
= false;
3676 case Operator
.Addition
:
3677 if (method
== null && ec
.HasSet (ResolveContext
.Options
.CheckedScope
) && !IsFloat (type
))
3678 method_name
= "AddChecked";
3680 method_name
= "Add";
3682 case Operator
.BitwiseAnd
:
3683 method_name
= "And";
3685 case Operator
.BitwiseOr
:
3688 case Operator
.Division
:
3689 method_name
= "Divide";
3691 case Operator
.Equality
:
3692 method_name
= "Equal";
3695 case Operator
.ExclusiveOr
:
3696 method_name
= "ExclusiveOr";
3698 case Operator
.GreaterThan
:
3699 method_name
= "GreaterThan";
3702 case Operator
.GreaterThanOrEqual
:
3703 method_name
= "GreaterThanOrEqual";
3706 case Operator
.Inequality
:
3707 method_name
= "NotEqual";
3710 case Operator
.LeftShift
:
3711 method_name
= "LeftShift";
3713 case Operator
.LessThan
:
3714 method_name
= "LessThan";
3717 case Operator
.LessThanOrEqual
:
3718 method_name
= "LessThanOrEqual";
3721 case Operator
.LogicalAnd
:
3722 method_name
= "AndAlso";
3724 case Operator
.LogicalOr
:
3725 method_name
= "OrElse";
3727 case Operator
.Modulus
:
3728 method_name
= "Modulo";
3730 case Operator
.Multiply
:
3731 if (method
== null && ec
.HasSet (ResolveContext
.Options
.CheckedScope
) && !IsFloat (type
))
3732 method_name
= "MultiplyChecked";
3734 method_name
= "Multiply";
3736 case Operator
.RightShift
:
3737 method_name
= "RightShift";
3739 case Operator
.Subtraction
:
3740 if (method
== null && ec
.HasSet (ResolveContext
.Options
.CheckedScope
) && !IsFloat (type
))
3741 method_name
= "SubtractChecked";
3743 method_name
= "Subtract";
3747 throw new InternalErrorException ("Unknown expression tree binary operator " + oper
);
3750 Arguments args
= new Arguments (2);
3751 args
.Add (new Argument (left
.CreateExpressionTree (ec
)));
3752 args
.Add (new Argument (right
.CreateExpressionTree (ec
)));
3753 if (method
!= null) {
3755 args
.Add (new Argument (new BoolConstant (false, loc
)));
3757 args
.Add (new Argument (method
.CreateExpressionTree (ec
)));
3760 return CreateExpressionFactoryCall (ec
, method_name
, args
);
3765 // Represents the operation a + b [+ c [+ d [+ ...]]], where a is a string
3766 // b, c, d... may be strings or objects.
3768 public class StringConcat
: Expression
{
3769 Arguments arguments
;
3771 public StringConcat (Expression left
, Expression right
, Location loc
)
3774 type
= TypeManager
.string_type
;
3775 eclass
= ExprClass
.Value
;
3777 arguments
= new Arguments (2);
3780 public static StringConcat
Create (ResolveContext rc
, Expression left
, Expression right
, Location loc
)
3782 if (left
.eclass
== ExprClass
.Unresolved
|| right
.eclass
== ExprClass
.Unresolved
)
3783 throw new ArgumentException ();
3785 var s
= new StringConcat (left
, right
, loc
);
3786 s
.Append (rc
, left
);
3787 s
.Append (rc
, right
);
3791 public override Expression
CreateExpressionTree (ResolveContext ec
)
3793 Argument arg
= arguments
[0];
3794 return CreateExpressionAddCall (ec
, arg
, arg
.CreateExpressionTree (ec
), 1);
3798 // Creates nested calls tree from an array of arguments used for IL emit
3800 Expression
CreateExpressionAddCall (ResolveContext ec
, Argument left
, Expression left_etree
, int pos
)
3802 Arguments concat_args
= new Arguments (2);
3803 Arguments add_args
= new Arguments (3);
3805 concat_args
.Add (left
);
3806 add_args
.Add (new Argument (left_etree
));
3808 concat_args
.Add (arguments
[pos
]);
3809 add_args
.Add (new Argument (arguments
[pos
].CreateExpressionTree (ec
)));
3811 MethodGroupExpr method
= CreateConcatMemberExpression ().Resolve (ec
) as MethodGroupExpr
;
3815 method
= method
.OverloadResolve (ec
, ref concat_args
, false, loc
);
3819 add_args
.Add (new Argument (method
.CreateExpressionTree (ec
)));
3821 Expression expr
= CreateExpressionFactoryCall (ec
, "Add", add_args
);
3822 if (++pos
== arguments
.Count
)
3825 left
= new Argument (new EmptyExpression (method
.BestCandidate
.ReturnType
));
3826 return CreateExpressionAddCall (ec
, left
, expr
, pos
);
3829 protected override Expression
DoResolve (ResolveContext ec
)
3834 void Append (ResolveContext rc
, Expression operand
)
3839 StringConstant sc
= operand
as StringConstant
;
3841 if (arguments
.Count
!= 0) {
3842 Argument last_argument
= arguments
[arguments
.Count
- 1];
3843 StringConstant last_expr_constant
= last_argument
.Expr
as StringConstant
;
3844 if (last_expr_constant
!= null) {
3845 last_argument
.Expr
= new StringConstant (
3846 last_expr_constant
.Value
+ sc
.Value
, sc
.Location
).Resolve (rc
);
3852 // Multiple (3+) concatenation are resolved as multiple StringConcat instances
3854 StringConcat concat_oper
= operand
as StringConcat
;
3855 if (concat_oper
!= null) {
3856 arguments
.AddRange (concat_oper
.arguments
);
3861 arguments
.Add (new Argument (operand
));
3864 Expression
CreateConcatMemberExpression ()
3866 return new MemberAccess (new MemberAccess (new QualifiedAliasMember ("global", "System", loc
), "String", loc
), "Concat", loc
);
3869 public override void Emit (EmitContext ec
)
3871 Expression concat
= new Invocation (CreateConcatMemberExpression (), arguments
, true);
3872 concat
= concat
.Resolve (new ResolveContext (ec
.MemberContext
));
3877 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
3879 if (arguments
.Count
!= 2)
3880 throw new NotImplementedException ("arguments.Count != 2");
3882 var concat
= typeof (string).GetMethod ("Concat", new[] { typeof (object), typeof (object) }
);
3883 return SLE
.Expression
.Add (arguments
[0].Expr
.MakeExpression (ctx
), arguments
[1].Expr
.MakeExpression (ctx
), concat
);
3888 // User-defined conditional logical operator
3890 public class ConditionalLogicalOperator
: UserOperatorCall
{
3891 readonly bool is_and
;
3894 public ConditionalLogicalOperator (MethodGroupExpr oper_method
, Arguments arguments
,
3895 ExpressionTreeExpression expr_tree
, bool is_and
, Location loc
)
3896 : base (oper_method
, arguments
, expr_tree
, loc
)
3898 this.is_and
= is_and
;
3899 eclass
= ExprClass
.Unresolved
;
3902 protected override Expression
DoResolve (ResolveContext ec
)
3904 var method
= mg
.BestCandidate
;
3905 type
= method
.ReturnType
;
3906 AParametersCollection pd
= method
.Parameters
;
3907 if (!TypeManager
.IsEqual (type
, type
) || !TypeManager
.IsEqual (type
, pd
.Types
[0]) || !TypeManager
.IsEqual (type
, pd
.Types
[1])) {
3908 ec
.Report
.Error (217, loc
,
3909 "A user-defined operator `{0}' must have parameters and return values of the same type in order to be applicable as a short circuit operator",
3910 TypeManager
.CSharpSignature (method
));
3914 Expression left_dup
= new EmptyExpression (type
);
3915 Expression op_true
= GetOperatorTrue (ec
, left_dup
, loc
);
3916 Expression op_false
= GetOperatorFalse (ec
, left_dup
, loc
);
3917 if (op_true
== null || op_false
== null) {
3918 ec
.Report
.Error (218, loc
,
3919 "The type `{0}' must have operator `true' and operator `false' defined when `{1}' is used as a short circuit operator",
3920 TypeManager
.CSharpName (type
), TypeManager
.CSharpSignature (method
));
3924 oper
= is_and
? op_false
: op_true
;
3925 eclass
= ExprClass
.Value
;
3929 public override void Emit (EmitContext ec
)
3931 Label end_target
= ec
.DefineLabel ();
3934 // Emit and duplicate left argument
3936 arguments
[0].Expr
.Emit (ec
);
3937 ec
.Emit (OpCodes
.Dup
);
3938 arguments
.RemoveAt (0);
3940 oper
.EmitBranchable (ec
, end_target
, true);
3942 ec
.MarkLabel (end_target
);
3946 public class PointerArithmetic
: Expression
{
3947 Expression left
, right
;
3951 // We assume that `l' is always a pointer
3953 public PointerArithmetic (Binary
.Operator op
, Expression l
, Expression r
, TypeSpec t
, Location loc
)
3962 public override Expression
CreateExpressionTree (ResolveContext ec
)
3964 Error_PointerInsideExpressionTree (ec
);
3968 protected override Expression
DoResolve (ResolveContext ec
)
3970 eclass
= ExprClass
.Variable
;
3972 if (left
.Type
== TypeManager
.void_ptr_type
) {
3973 ec
.Report
.Error (242, loc
, "The operation in question is undefined on void pointers");
3980 public override void Emit (EmitContext ec
)
3982 TypeSpec op_type
= left
.Type
;
3984 // It must be either array or fixed buffer
3986 if (TypeManager
.HasElementType (op_type
)) {
3987 element
= TypeManager
.GetElementType (op_type
);
3989 FieldExpr fe
= left
as FieldExpr
;
3991 element
= ((FixedFieldSpec
) (fe
.Spec
)).ElementType
;
3996 int size
= GetTypeSize (element
);
3997 TypeSpec rtype
= right
.Type
;
3999 if ((op
& Binary
.Operator
.SubtractionMask
) != 0 && rtype
.IsPointer
){
4001 // handle (pointer - pointer)
4005 ec
.Emit (OpCodes
.Sub
);
4009 ec
.Emit (OpCodes
.Sizeof
, element
);
4012 ec
.Emit (OpCodes
.Div
);
4014 ec
.Emit (OpCodes
.Conv_I8
);
4017 // handle + and - on (pointer op int)
4019 Constant left_const
= left
as Constant
;
4020 if (left_const
!= null) {
4022 // Optimize ((T*)null) pointer operations
4024 if (left_const
.IsDefaultValue
) {
4025 left
= EmptyExpression
.Null
;
4033 var right_const
= right
as Constant
;
4034 if (right_const
!= null) {
4036 // Optimize 0-based arithmetic
4038 if (right_const
.IsDefaultValue
)
4042 right
= new IntConstant (size
, right
.Location
);
4044 right
= new SizeOf (new TypeExpression (element
, right
.Location
), right
.Location
);
4046 // TODO: Should be the checks resolve context sensitive?
4047 ResolveContext rc
= new ResolveContext (ec
.MemberContext
, ResolveContext
.Options
.UnsafeScope
);
4048 right
= new Binary (Binary
.Operator
.Multiply
, right
, right_const
, loc
).Resolve (rc
);
4054 if (rtype
== TypeManager
.sbyte_type
|| rtype
== TypeManager
.byte_type
||
4055 rtype
== TypeManager
.short_type
|| rtype
== TypeManager
.ushort_type
) {
4056 ec
.Emit (OpCodes
.Conv_I
);
4057 } else if (rtype
== TypeManager
.uint32_type
) {
4058 ec
.Emit (OpCodes
.Conv_U
);
4061 if (right_const
== null && size
!= 1){
4063 ec
.Emit (OpCodes
.Sizeof
, element
);
4066 if (rtype
== TypeManager
.int64_type
|| rtype
== TypeManager
.uint64_type
)
4067 ec
.Emit (OpCodes
.Conv_I8
);
4069 Binary
.EmitOperatorOpcode (ec
, Binary
.Operator
.Multiply
, rtype
);
4072 if (left_const
== null) {
4073 if (rtype
== TypeManager
.int64_type
)
4074 ec
.Emit (OpCodes
.Conv_I
);
4075 else if (rtype
== TypeManager
.uint64_type
)
4076 ec
.Emit (OpCodes
.Conv_U
);
4078 Binary
.EmitOperatorOpcode (ec
, op
, op_type
);
4085 // A boolean-expression is an expression that yields a result
4088 public class BooleanExpression
: ShimExpression
4090 public BooleanExpression (Expression expr
)
4093 this.loc
= expr
.Location
;
4096 public override Expression
CreateExpressionTree (ResolveContext ec
)
4098 // TODO: We should emit IsTrue (v4) instead of direct user operator
4099 // call but that would break csc compatibility
4100 return base.CreateExpressionTree (ec
);
4103 protected override Expression
DoResolve (ResolveContext ec
)
4105 // A boolean-expression is required to be of a type
4106 // that can be implicitly converted to bool or of
4107 // a type that implements operator true
4109 expr
= expr
.Resolve (ec
);
4113 Assign ass
= expr
as Assign
;
4114 if (ass
!= null && ass
.Source
is Constant
) {
4115 ec
.Report
.Warning (665, 3, loc
,
4116 "Assignment in conditional expression is always constant. Did you mean to use `==' instead ?");
4119 if (expr
.Type
== TypeManager
.bool_type
)
4122 if (expr
.Type
== InternalType
.Dynamic
) {
4123 Arguments args
= new Arguments (1);
4124 args
.Add (new Argument (expr
));
4125 return new DynamicUnaryConversion ("IsTrue", args
, loc
).Resolve (ec
);
4128 type
= TypeManager
.bool_type
;
4129 Expression converted
= Convert
.ImplicitConversion (ec
, expr
, type
, loc
);
4130 if (converted
!= null)
4134 // If no implicit conversion to bool exists, try using `operator true'
4136 converted
= GetOperatorTrue (ec
, expr
, loc
);
4137 if (converted
== null) {
4138 expr
.Error_ValueCannotBeConverted (ec
, loc
, type
, false);
4147 /// Implements the ternary conditional operator (?:)
4149 public class Conditional
: Expression
{
4150 Expression expr
, true_expr
, false_expr
;
4152 public Conditional (BooleanExpression expr
, Expression true_expr
, Expression false_expr
)
4155 this.true_expr
= true_expr
;
4156 this.false_expr
= false_expr
;
4157 this.loc
= expr
.Location
;
4160 public Expression Expr
{
4166 public Expression TrueExpr
{
4172 public Expression FalseExpr
{
4178 public override Expression
CreateExpressionTree (ResolveContext ec
)
4180 Arguments args
= new Arguments (3);
4181 args
.Add (new Argument (expr
.CreateExpressionTree (ec
)));
4182 args
.Add (new Argument (true_expr
.CreateExpressionTree (ec
)));
4183 args
.Add (new Argument (false_expr
.CreateExpressionTree (ec
)));
4184 return CreateExpressionFactoryCall (ec
, "Condition", args
);
4187 protected override Expression
DoResolve (ResolveContext ec
)
4189 expr
= expr
.Resolve (ec
);
4190 true_expr
= true_expr
.Resolve (ec
);
4191 false_expr
= false_expr
.Resolve (ec
);
4193 if (true_expr
== null || false_expr
== null || expr
== null)
4196 eclass
= ExprClass
.Value
;
4197 TypeSpec true_type
= true_expr
.Type
;
4198 TypeSpec false_type
= false_expr
.Type
;
4202 // First, if an implicit conversion exists from true_expr
4203 // to false_expr, then the result type is of type false_expr.Type
4205 if (!TypeManager
.IsEqual (true_type
, false_type
)) {
4206 Expression conv
= Convert
.ImplicitConversion (ec
, true_expr
, false_type
, loc
);
4209 // Check if both can convert implicitly to each other's type
4211 if (Convert
.ImplicitConversion (ec
, false_expr
, true_type
, loc
) != null) {
4212 ec
.Report
.Error (172, true_expr
.Location
,
4213 "Type of conditional expression cannot be determined as `{0}' and `{1}' convert implicitly to each other",
4214 TypeManager
.CSharpName (true_type
), TypeManager
.CSharpName (false_type
));
4219 } else if ((conv
= Convert
.ImplicitConversion (ec
, false_expr
, true_type
, loc
)) != null) {
4222 ec
.Report
.Error (173, true_expr
.Location
,
4223 "Type of conditional expression cannot be determined because there is no implicit conversion between `{0}' and `{1}'",
4224 TypeManager
.CSharpName (true_type
), TypeManager
.CSharpName (false_type
));
4229 // Dead code optimalization
4230 Constant c
= expr
as Constant
;
4232 bool is_false
= c
.IsDefaultValue
;
4233 ec
.Report
.Warning (429, 4, is_false
? true_expr
.Location
: false_expr
.Location
, "Unreachable expression code detected");
4234 return ReducedExpression
.Create (is_false
? false_expr
: true_expr
, this).Resolve (ec
);
4240 public override TypeExpr
ResolveAsTypeTerminal (IMemberContext ec
, bool silent
)
4245 public override void Emit (EmitContext ec
)
4247 Label false_target
= ec
.DefineLabel ();
4248 Label end_target
= ec
.DefineLabel ();
4250 expr
.EmitBranchable (ec
, false_target
, false);
4251 true_expr
.Emit (ec
);
4253 if (type
.IsInterface
) {
4254 LocalBuilder temp
= ec
.GetTemporaryLocal (type
);
4255 ec
.Emit (OpCodes
.Stloc
, temp
);
4256 ec
.Emit (OpCodes
.Ldloc
, temp
);
4257 ec
.FreeTemporaryLocal (temp
, type
);
4260 ec
.Emit (OpCodes
.Br
, end_target
);
4261 ec
.MarkLabel (false_target
);
4262 false_expr
.Emit (ec
);
4263 ec
.MarkLabel (end_target
);
4266 protected override void CloneTo (CloneContext clonectx
, Expression t
)
4268 Conditional target
= (Conditional
) t
;
4270 target
.expr
= expr
.Clone (clonectx
);
4271 target
.true_expr
= true_expr
.Clone (clonectx
);
4272 target
.false_expr
= false_expr
.Clone (clonectx
);
4276 public abstract class VariableReference
: Expression
, IAssignMethod
, IMemoryLocation
, IVariableReference
{
4277 LocalTemporary temp
;
4280 public abstract HoistedVariable
GetHoistedVariable (AnonymousExpression ae
);
4281 public abstract bool IsFixed { get; }
4282 public abstract bool IsRef { get; }
4283 public abstract string Name { get; }
4284 public abstract void SetHasAddressTaken ();
4287 // Variable IL data, it has to be protected to encapsulate hoisted variables
4289 protected abstract ILocalVariable Variable { get; }
4292 // Variable flow-analysis data
4294 public abstract VariableInfo VariableInfo { get; }
4297 public virtual void AddressOf (EmitContext ec
, AddressOp mode
)
4299 HoistedVariable hv
= GetHoistedVariable (ec
);
4301 hv
.AddressOf (ec
, mode
);
4305 Variable
.EmitAddressOf (ec
);
4308 public HoistedVariable
GetHoistedVariable (ResolveContext rc
)
4310 return GetHoistedVariable (rc
.CurrentAnonymousMethod
);
4313 public HoistedVariable
GetHoistedVariable (EmitContext ec
)
4315 return GetHoistedVariable (ec
.CurrentAnonymousMethod
);
4318 public override string GetSignatureForError ()
4323 public override void Emit (EmitContext ec
)
4328 public override void EmitSideEffect (EmitContext ec
)
4334 // This method is used by parameters that are references, that are
4335 // being passed as references: we only want to pass the pointer (that
4336 // is already stored in the parameter, not the address of the pointer,
4337 // and not the value of the variable).
4339 public void EmitLoad (EmitContext ec
)
4344 public void Emit (EmitContext ec
, bool leave_copy
)
4346 Report
.Debug (64, "VARIABLE EMIT", this, Variable
, type
, IsRef
, loc
);
4348 HoistedVariable hv
= GetHoistedVariable (ec
);
4350 hv
.Emit (ec
, leave_copy
);
4358 // If we are a reference, we loaded on the stack a pointer
4359 // Now lets load the real value
4361 ec
.EmitLoadFromPtr (type
);
4365 ec
.Emit (OpCodes
.Dup
);
4368 temp
= new LocalTemporary (Type
);
4374 public void EmitAssign (EmitContext ec
, Expression source
, bool leave_copy
,
4375 bool prepare_for_load
)
4377 HoistedVariable hv
= GetHoistedVariable (ec
);
4379 hv
.EmitAssign (ec
, source
, leave_copy
, prepare_for_load
);
4383 New n_source
= source
as New
;
4384 if (n_source
!= null) {
4385 if (!n_source
.Emit (ec
, this)) {
4398 ec
.Emit (OpCodes
.Dup
);
4400 temp
= new LocalTemporary (Type
);
4406 ec
.EmitStoreFromPtr (type
);
4408 Variable
.EmitAssign (ec
);
4416 public bool IsHoisted
{
4417 get { return GetHoistedVariable ((AnonymousExpression) null) != null; }
4424 public class LocalVariableReference
: VariableReference
{
4425 readonly string name
;
4427 public LocalInfo local_info
;
4430 public LocalVariableReference (Block block
, string name
, Location l
)
4438 // Setting `is_readonly' to false will allow you to create a writable
4439 // reference to a read-only variable. This is used by foreach and using.
4441 public LocalVariableReference (Block block
, string name
, Location l
,
4442 LocalInfo local_info
, bool is_readonly
)
4443 : this (block
, name
, l
)
4445 this.local_info
= local_info
;
4446 this.is_readonly
= is_readonly
;
4449 public override VariableInfo VariableInfo
{
4450 get { return local_info.VariableInfo; }
4453 public override HoistedVariable
GetHoistedVariable (AnonymousExpression ae
)
4455 return local_info
.HoistedVariant
;
4459 // A local variable is always fixed
4461 public override bool IsFixed
{
4462 get { return true; }
4465 public override bool IsRef
{
4466 get { return false; }
4469 public bool IsReadOnly
{
4470 get { return is_readonly; }
4473 public override string Name
{
4474 get { return name; }
4477 public bool VerifyAssigned (ResolveContext ec
)
4479 VariableInfo variable_info
= local_info
.VariableInfo
;
4480 return variable_info
== null || variable_info
.IsAssigned (ec
, loc
);
4483 void ResolveLocalInfo ()
4485 if (local_info
== null) {
4486 local_info
= Block
.GetLocalInfo (Name
);
4487 type
= local_info
.VariableType
;
4488 is_readonly
= local_info
.ReadOnly
;
4492 public override void SetHasAddressTaken ()
4494 local_info
.AddressTaken
= true;
4497 public override Expression
CreateExpressionTree (ResolveContext ec
)
4499 HoistedVariable hv
= GetHoistedVariable (ec
);
4501 return hv
.CreateExpressionTree ();
4503 Arguments arg
= new Arguments (1);
4504 arg
.Add (new Argument (this));
4505 return CreateExpressionFactoryCall (ec
, "Constant", arg
);
4508 Expression
DoResolveBase (ResolveContext ec
)
4510 Expression e
= Block
.GetConstantExpression (Name
);
4512 return e
.Resolve (ec
);
4514 VerifyAssigned (ec
);
4517 // If we are referencing a variable from the external block
4518 // flag it for capturing
4520 if (ec
.MustCaptureVariable (local_info
)) {
4521 if (local_info
.AddressTaken
)
4522 AnonymousMethodExpression
.Error_AddressOfCapturedVar (ec
, this, loc
);
4524 if (ec
.IsVariableCapturingRequired
) {
4525 AnonymousMethodStorey storey
= local_info
.Block
.Explicit
.CreateAnonymousMethodStorey (ec
);
4526 storey
.CaptureLocalVariable (ec
, local_info
);
4530 eclass
= ExprClass
.Variable
;
4531 type
= local_info
.VariableType
;
4535 protected override Expression
DoResolve (ResolveContext ec
)
4537 ResolveLocalInfo ();
4538 local_info
.Used
= true;
4540 if (type
== null && local_info
.Type
is VarExpr
) {
4541 local_info
.VariableType
= TypeManager
.object_type
;
4542 Error_VariableIsUsedBeforeItIsDeclared (ec
.Report
, Name
);
4546 return DoResolveBase (ec
);
4549 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
4551 ResolveLocalInfo ();
4554 if (right_side
== EmptyExpression
.OutAccess
.Instance
)
4555 local_info
.Used
= true;
4557 // Infer implicitly typed local variable
4559 VarExpr ve
= local_info
.Type
as VarExpr
;
4561 if (!ve
.InferType (ec
, right_side
))
4563 type
= local_info
.VariableType
= ve
.Type
;
4570 if (right_side
== EmptyExpression
.OutAccess
.Instance
) {
4571 code
= 1657; msg
= "Cannot pass `{0}' as a ref or out argument because it is a `{1}'";
4572 } else if (right_side
== EmptyExpression
.LValueMemberAccess
) {
4573 code
= 1654; msg
= "Cannot assign to members of `{0}' because it is a `{1}'";
4574 } else if (right_side
== EmptyExpression
.LValueMemberOutAccess
) {
4575 code
= 1655; msg
= "Cannot pass members of `{0}' as ref or out arguments because it is a `{1}'";
4576 } else if (right_side
== EmptyExpression
.UnaryAddress
) {
4577 code
= 459; msg
= "Cannot take the address of {1} `{0}'";
4579 code
= 1656; msg
= "Cannot assign to `{0}' because it is a `{1}'";
4581 ec
.Report
.Error (code
, loc
, msg
, Name
, local_info
.GetReadOnlyContext ());
4582 } else if (VariableInfo
!= null) {
4583 VariableInfo
.SetAssigned (ec
);
4586 return DoResolveBase (ec
);
4589 public override int GetHashCode ()
4591 return Name
.GetHashCode ();
4594 public override bool Equals (object obj
)
4596 LocalVariableReference lvr
= obj
as LocalVariableReference
;
4600 return Name
== lvr
.Name
&& Block
== lvr
.Block
;
4603 protected override ILocalVariable Variable
{
4604 get { return local_info; }
4607 public override string ToString ()
4609 return String
.Format ("{0} ({1}:{2})", GetType (), Name
, loc
);
4612 protected override void CloneTo (CloneContext clonectx
, Expression t
)
4614 LocalVariableReference target
= (LocalVariableReference
) t
;
4616 target
.Block
= clonectx
.LookupBlock (Block
);
4617 if (local_info
!= null)
4618 target
.local_info
= clonectx
.LookupVariable (local_info
);
4623 /// This represents a reference to a parameter in the intermediate
4626 public class ParameterReference
: VariableReference
{
4627 readonly ToplevelParameterInfo pi
;
4629 public ParameterReference (ToplevelParameterInfo pi
, Location loc
)
4635 public override bool IsRef
{
4636 get { return (pi.Parameter.ModFlags & Parameter.Modifier.ISBYREF) != 0; }
4639 bool HasOutModifier
{
4640 get { return pi.Parameter.ModFlags == Parameter.Modifier.OUT; }
4643 public override HoistedVariable
GetHoistedVariable (AnonymousExpression ae
)
4645 return pi
.Parameter
.HoistedVariant
;
4649 // A ref or out parameter is classified as a moveable variable, even
4650 // if the argument given for the parameter is a fixed variable
4652 public override bool IsFixed
{
4653 get { return !IsRef; }
4656 public override string Name
{
4657 get { return Parameter.Name; }
4660 public Parameter Parameter
{
4661 get { return pi.Parameter; }
4664 public override VariableInfo VariableInfo
{
4665 get { return pi.VariableInfo; }
4668 protected override ILocalVariable Variable
{
4669 get { return Parameter; }
4672 public bool IsAssigned (ResolveContext ec
, Location loc
)
4674 // HACK: Variables are not captured in probing mode
4675 if (ec
.IsInProbingMode
)
4678 if (!ec
.DoFlowAnalysis
|| !HasOutModifier
|| ec
.CurrentBranching
.IsAssigned (VariableInfo
))
4681 ec
.Report
.Error (269, loc
, "Use of unassigned out parameter `{0}'", Name
);
4685 public override void SetHasAddressTaken ()
4687 Parameter
.HasAddressTaken
= true;
4690 void SetAssigned (ResolveContext ec
)
4692 if (HasOutModifier
&& ec
.DoFlowAnalysis
)
4693 ec
.CurrentBranching
.SetAssigned (VariableInfo
);
4696 bool DoResolveBase (ResolveContext ec
)
4698 type
= pi
.ParameterType
;
4699 eclass
= ExprClass
.Variable
;
4701 AnonymousExpression am
= ec
.CurrentAnonymousMethod
;
4705 Block b
= ec
.CurrentBlock
;
4708 IParameterData
[] p
= b
.Toplevel
.Parameters
.FixedParameters
;
4709 for (int i
= 0; i
< p
.Length
; ++i
) {
4710 if (p
[i
] != Parameter
)
4714 // Don't capture local parameters
4716 if (b
== ec
.CurrentBlock
.Toplevel
&& !am
.IsIterator
)
4720 ec
.Report
.Error (1628, loc
,
4721 "Parameter `{0}' cannot be used inside `{1}' when using `ref' or `out' modifier",
4722 Name
, am
.ContainerType
);
4725 if (pi
.Parameter
.HasAddressTaken
)
4726 AnonymousMethodExpression
.Error_AddressOfCapturedVar (ec
, this, loc
);
4728 if (ec
.IsVariableCapturingRequired
&& !b
.Toplevel
.IsExpressionTree
) {
4729 AnonymousMethodStorey storey
= pi
.Block
.CreateAnonymousMethodStorey (ec
);
4730 storey
.CaptureParameter (ec
, this);
4742 public override int GetHashCode ()
4744 return Name
.GetHashCode ();
4747 public override bool Equals (object obj
)
4749 ParameterReference pr
= obj
as ParameterReference
;
4753 return Name
== pr
.Name
;
4756 public override void AddressOf (EmitContext ec
, AddressOp mode
)
4759 // ParameterReferences might already be a reference
4766 base.AddressOf (ec
, mode
);
4769 protected override void CloneTo (CloneContext clonectx
, Expression target
)
4774 public override Expression
CreateExpressionTree (ResolveContext ec
)
4776 HoistedVariable hv
= GetHoistedVariable (ec
);
4778 return hv
.CreateExpressionTree ();
4780 return Parameter
.ExpressionTreeVariableReference ();
4784 // Notice that for ref/out parameters, the type exposed is not the
4785 // same type exposed externally.
4788 // externally we expose "int&"
4789 // here we expose "int".
4791 // We record this in "is_ref". This means that the type system can treat
4792 // the type as it is expected, but when we generate the code, we generate
4793 // the alternate kind of code.
4795 protected override Expression
DoResolve (ResolveContext ec
)
4797 if (!DoResolveBase (ec
))
4800 // HACK: Variables are not captured in probing mode
4801 if (ec
.IsInProbingMode
)
4804 if (HasOutModifier
&& ec
.DoFlowAnalysis
&&
4805 (!ec
.OmitStructFlowAnalysis
|| !VariableInfo
.TypeInfo
.IsStruct
) && !IsAssigned (ec
, loc
))
4811 override public Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
4813 if (!DoResolveBase (ec
))
4816 // HACK: parameters are not captured when probing is on
4817 if (!ec
.IsInProbingMode
)
4823 static public void EmitLdArg (EmitContext ec
, int x
)
4826 case 0: ec
.Emit (OpCodes
.Ldarg_0
); break;
4827 case 1: ec
.Emit (OpCodes
.Ldarg_1
); break;
4828 case 2: ec
.Emit (OpCodes
.Ldarg_2
); break;
4829 case 3: ec
.Emit (OpCodes
.Ldarg_3
); break;
4831 if (x
> byte.MaxValue
)
4832 ec
.Emit (OpCodes
.Ldarg
, x
);
4834 ec
.Emit (OpCodes
.Ldarg_S
, (byte) x
);
4841 /// Invocation of methods or delegates.
4843 public class Invocation
: ExpressionStatement
4845 protected Arguments arguments
;
4846 protected Expression expr
;
4847 protected MethodGroupExpr mg
;
4848 bool arguments_resolved
;
4851 // arguments is an ArrayList, but we do not want to typecast,
4852 // as it might be null.
4854 public Invocation (Expression expr
, Arguments arguments
)
4856 SimpleName sn
= expr
as SimpleName
;
4858 this.expr
= sn
.GetMethodGroup ();
4862 this.arguments
= arguments
;
4864 loc
= expr
.Location
;
4867 public Invocation (Expression expr
, Arguments arguments
, bool arguments_resolved
)
4868 : this (expr
, arguments
)
4870 this.arguments_resolved
= arguments_resolved
;
4873 public override Expression
CreateExpressionTree (ResolveContext ec
)
4875 Expression instance
= mg
.IsInstance
?
4876 mg
.InstanceExpression
.CreateExpressionTree (ec
) :
4877 new NullLiteral (loc
);
4879 var args
= Arguments
.CreateForExpressionTree (ec
, arguments
,
4881 mg
.CreateExpressionTree (ec
));
4884 MemberExpr
.Error_BaseAccessInExpressionTree (ec
, loc
);
4886 return CreateExpressionFactoryCall (ec
, "Call", args
);
4889 protected override Expression
DoResolve (ResolveContext ec
)
4891 Expression member_expr
= expr
.Resolve (ec
, ResolveFlags
.VariableOrValue
| ResolveFlags
.MethodGroup
);
4892 if (member_expr
== null)
4896 // Next, evaluate all the expressions in the argument list
4898 bool dynamic_arg
= false;
4899 if (arguments
!= null && !arguments_resolved
)
4900 arguments
.Resolve (ec
, out dynamic_arg
);
4902 TypeSpec expr_type
= member_expr
.Type
;
4903 mg
= member_expr
as MethodGroupExpr
;
4905 bool dynamic_member
= expr_type
== InternalType
.Dynamic
;
4907 if (!dynamic_member
) {
4908 Expression invoke
= null;
4911 if (expr_type
!= null && TypeManager
.IsDelegateType (expr_type
)) {
4912 invoke
= new DelegateInvocation (member_expr
, arguments
, loc
);
4913 invoke
= invoke
.Resolve (ec
);
4914 if (invoke
== null || !dynamic_arg
)
4917 MemberExpr me
= member_expr
as MemberExpr
;
4919 member_expr
.Error_UnexpectedKind (ec
, ResolveFlags
.MethodGroup
, loc
);
4923 mg
= ec
.LookupExtensionMethod (me
.Type
, me
.Name
, -1, loc
);
4925 ec
.Report
.Error (1955, loc
, "The member `{0}' cannot be used as method or delegate",
4926 member_expr
.GetSignatureForError ());
4930 ((ExtensionMethodGroupExpr
) mg
).ExtensionExpression
= me
.InstanceExpression
;
4934 if (invoke
== null) {
4935 mg
= DoResolveOverload (ec
);
4941 if (dynamic_arg
|| dynamic_member
)
4942 return DoResolveDynamic (ec
, member_expr
);
4944 var method
= mg
.BestCandidate
;
4945 if (method
!= null) {
4946 type
= method
.ReturnType
;
4950 // Only base will allow this invocation to happen.
4952 if (mg
.IsBase
&& method
.IsAbstract
){
4953 Error_CannotCallAbstractBase (ec
, TypeManager
.CSharpSignature (method
));
4957 if (arguments
== null && method
.DeclaringType
== TypeManager
.object_type
&& method
.Name
== Destructor
.MetadataName
) {
4959 ec
.Report
.Error (250, loc
, "Do not directly call your base class Finalize method. It is called automatically from your destructor");
4961 ec
.Report
.Error (245, loc
, "Destructors and object.Finalize cannot be called directly. Consider calling IDisposable.Dispose if available");
4965 IsSpecialMethodInvocation (ec
, method
, loc
);
4967 if (mg
.InstanceExpression
!= null)
4968 mg
.InstanceExpression
.CheckMarshalByRefAccess (ec
);
4970 eclass
= ExprClass
.Value
;
4974 Expression
DoResolveDynamic (ResolveContext ec
, Expression memberExpr
)
4977 DynamicMemberBinder dmb
= memberExpr
as DynamicMemberBinder
;
4979 args
= dmb
.Arguments
;
4980 if (arguments
!= null)
4981 args
.AddRange (arguments
);
4982 } else if (mg
== null) {
4983 if (arguments
== null)
4984 args
= new Arguments (1);
4988 args
.Insert (0, new Argument (memberExpr
));
4992 ec
.Report
.Error (1971, loc
,
4993 "The base call to method `{0}' cannot be dynamically dispatched. Consider casting the dynamic arguments or eliminating the base access",
5000 if (mg
.IsStatic
!= mg
.IsInstance
) {
5002 args
= new Arguments (1);
5005 args
.Insert (0, new Argument (new TypeOf (new TypeExpression (mg
.DeclaringType
, loc
), loc
).Resolve (ec
), Argument
.AType
.DynamicTypeName
));
5007 MemberAccess ma
= expr
as MemberAccess
;
5009 args
.Insert (0, new Argument (ma
.Left
.Resolve (ec
)));
5011 args
.Insert (0, new Argument (new This (loc
).Resolve (ec
)));
5016 return new DynamicInvocation (expr
as ATypeNameExpression
, args
, loc
).Resolve (ec
);
5019 protected virtual MethodGroupExpr
DoResolveOverload (ResolveContext ec
)
5021 return mg
.OverloadResolve (ec
, ref arguments
, false, loc
);
5024 public static bool IsSpecialMethodInvocation (ResolveContext ec
, MethodSpec method
, Location loc
)
5026 if (!method
.IsReservedMethod
)
5029 if (ec
.HasSet (ResolveContext
.Options
.InvokeSpecialName
))
5032 ec
.Report
.SymbolRelatedToPreviousError (method
);
5033 ec
.Report
.Error (571, loc
, "`{0}': cannot explicitly call operator or accessor",
5034 method
.GetSignatureForError ());
5039 static Type
[] GetVarargsTypes (MethodSpec mb
, Arguments arguments
)
5041 AParametersCollection pd
= mb
.Parameters
;
5043 Argument a
= arguments
[pd
.Count
- 1];
5044 Arglist list
= (Arglist
) a
.Expr
;
5046 return list
.ArgumentTypes
;
5050 /// is_base tells whether we want to force the use of the `call'
5051 /// opcode instead of using callvirt. Call is required to call
5052 /// a specific method, while callvirt will always use the most
5053 /// recent method in the vtable.
5055 /// is_static tells whether this is an invocation on a static method
5057 /// instance_expr is an expression that represents the instance
5058 /// it must be non-null if is_static is false.
5060 /// method is the method to invoke.
5062 /// Arguments is the list of arguments to pass to the method or constructor.
5064 public static void EmitCall (EmitContext ec
, bool is_base
,
5065 Expression instance_expr
,
5066 MethodSpec method
, Arguments Arguments
, Location loc
)
5068 EmitCall (ec
, is_base
, instance_expr
, method
, Arguments
, loc
, false, false);
5071 // `dup_args' leaves an extra copy of the arguments on the stack
5072 // `omit_args' does not leave any arguments at all.
5073 // So, basically, you could make one call with `dup_args' set to true,
5074 // and then another with `omit_args' set to true, and the two calls
5075 // would have the same set of arguments. However, each argument would
5076 // only have been evaluated once.
5077 public static void EmitCall (EmitContext ec
, bool is_base
,
5078 Expression instance_expr
,
5079 MethodSpec method
, Arguments Arguments
, Location loc
,
5080 bool dup_args
, bool omit_args
)
5082 LocalTemporary this_arg
= null;
5084 TypeSpec decl_type
= method
.DeclaringType
;
5086 // Speed up the check by not doing it on not allowed targets
5087 if (method
.ReturnType
== TypeManager
.void_type
&& method
.IsConditionallyExcluded (loc
))
5091 TypeSpec iexpr_type
;
5093 if (method
.IsStatic
) {
5095 call_op
= OpCodes
.Call
;
5097 iexpr_type
= instance_expr
.Type
;
5099 if (is_base
|| decl_type
.IsStruct
|| decl_type
.IsEnum
|| (instance_expr
is This
&& !method
.IsVirtual
)) {
5100 call_op
= OpCodes
.Call
;
5102 call_op
= OpCodes
.Callvirt
;
5106 // If this is ourselves, push "this"
5109 TypeSpec t
= iexpr_type
;
5112 // Push the instance expression
5114 if ((iexpr_type
.IsStruct
&& (call_op
== OpCodes
.Callvirt
|| (call_op
== OpCodes
.Call
&& decl_type
== iexpr_type
))) ||
5115 iexpr_type
.IsGenericParameter
|| TypeManager
.IsNullableType (decl_type
)) {
5117 // If the expression implements IMemoryLocation, then
5118 // we can optimize and use AddressOf on the
5121 // If not we have to use some temporary storage for
5123 var iml
= instance_expr
as IMemoryLocation
;
5125 iml
.AddressOf (ec
, AddressOp
.LoadStore
);
5127 LocalTemporary temp
= new LocalTemporary (iexpr_type
);
5128 instance_expr
.Emit (ec
);
5130 temp
.AddressOf (ec
, AddressOp
.Load
);
5133 // avoid the overhead of doing this all the time.
5135 t
= ReferenceContainer
.MakeType (iexpr_type
);
5136 } else if (iexpr_type
.IsEnum
|| iexpr_type
.IsStruct
) {
5137 instance_expr
.Emit (ec
);
5138 ec
.Emit (OpCodes
.Box
, iexpr_type
);
5139 t
= iexpr_type
= TypeManager
.object_type
;
5141 instance_expr
.Emit (ec
);
5145 ec
.Emit (OpCodes
.Dup
);
5146 if (Arguments
!= null && Arguments
.Count
!= 0) {
5147 this_arg
= new LocalTemporary (t
);
5148 this_arg
.Store (ec
);
5154 if (!omit_args
&& Arguments
!= null)
5155 Arguments
.Emit (ec
, dup_args
, this_arg
);
5157 if (call_op
== OpCodes
.Callvirt
&& (iexpr_type
.IsGenericParameter
|| iexpr_type
.IsStruct
)) {
5158 ec
.Emit (OpCodes
.Constrained
, iexpr_type
);
5161 if (method
.Parameters
.HasArglist
) {
5162 Type
[] varargs_types
= GetVarargsTypes (method
, Arguments
);
5163 ec
.Emit (call_op
, method
, varargs_types
);
5170 // and DoFoo is not virtual, you can omit the callvirt,
5171 // because you don't need the null checking behavior.
5173 ec
.Emit (call_op
, method
);
5176 public override void Emit (EmitContext ec
)
5178 mg
.EmitCall (ec
, arguments
);
5181 public override void EmitStatement (EmitContext ec
)
5186 // Pop the return value if there is one
5188 if (type
!= TypeManager
.void_type
)
5189 ec
.Emit (OpCodes
.Pop
);
5192 protected override void CloneTo (CloneContext clonectx
, Expression t
)
5194 Invocation target
= (Invocation
) t
;
5196 if (arguments
!= null)
5197 target
.arguments
= arguments
.Clone (clonectx
);
5199 target
.expr
= expr
.Clone (clonectx
);
5202 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
5204 return MakeExpression (ctx
, mg
.InstanceExpression
, (MethodSpec
) mg
, arguments
);
5207 public static SLE
.Expression
MakeExpression (BuilderContext ctx
, Expression instance
, MethodSpec mi
, Arguments args
)
5209 var instance_expr
= instance
== null ? null : instance
.MakeExpression (ctx
);
5210 return SLE
.Expression
.Call (instance_expr
, (MethodInfo
) mi
.GetMetaInfo (), Arguments
.MakeExpression (args
, ctx
));
5215 /// Implements the new expression
5217 public class New
: ExpressionStatement
, IMemoryLocation
{
5218 protected Arguments Arguments
;
5221 // During bootstrap, it contains the RequestedType,
5222 // but if `type' is not null, it *might* contain a NewDelegate
5223 // (because of field multi-initialization)
5225 protected Expression RequestedType
;
5227 protected MethodGroupExpr method
;
5229 public New (Expression requested_type
, Arguments arguments
, Location l
)
5231 RequestedType
= requested_type
;
5232 Arguments
= arguments
;
5237 /// Converts complex core type syntax like 'new int ()' to simple constant
5239 public static Constant
Constantify (TypeSpec t
)
5241 if (t
== TypeManager
.int32_type
)
5242 return new IntConstant (0, Location
.Null
);
5243 if (t
== TypeManager
.uint32_type
)
5244 return new UIntConstant (0, Location
.Null
);
5245 if (t
== TypeManager
.int64_type
)
5246 return new LongConstant (0, Location
.Null
);
5247 if (t
== TypeManager
.uint64_type
)
5248 return new ULongConstant (0, Location
.Null
);
5249 if (t
== TypeManager
.float_type
)
5250 return new FloatConstant (0, Location
.Null
);
5251 if (t
== TypeManager
.double_type
)
5252 return new DoubleConstant (0, Location
.Null
);
5253 if (t
== TypeManager
.short_type
)
5254 return new ShortConstant (0, Location
.Null
);
5255 if (t
== TypeManager
.ushort_type
)
5256 return new UShortConstant (0, Location
.Null
);
5257 if (t
== TypeManager
.sbyte_type
)
5258 return new SByteConstant (0, Location
.Null
);
5259 if (t
== TypeManager
.byte_type
)
5260 return new ByteConstant (0, Location
.Null
);
5261 if (t
== TypeManager
.char_type
)
5262 return new CharConstant ('\0', Location
.Null
);
5263 if (t
== TypeManager
.bool_type
)
5264 return new BoolConstant (false, Location
.Null
);
5265 if (t
== TypeManager
.decimal_type
)
5266 return new DecimalConstant (0, Location
.Null
);
5267 if (TypeManager
.IsEnumType (t
))
5268 return new EnumConstant (Constantify (EnumSpec
.GetUnderlyingType (t
)), t
);
5269 if (TypeManager
.IsNullableType (t
))
5270 return Nullable
.LiftedNull
.Create (t
, Location
.Null
);
5276 // Checks whether the type is an interface that has the
5277 // [ComImport, CoClass] attributes and must be treated
5280 public Expression
CheckComImport (ResolveContext ec
)
5282 if (!type
.IsInterface
)
5286 // Turn the call into:
5287 // (the-interface-stated) (new class-referenced-in-coclassattribute ())
5289 var real_class
= type
.MemberDefinition
.GetAttributeCoClass ();
5290 if (real_class
== null)
5293 New proxy
= new New (new TypeExpression (real_class
, loc
), Arguments
, loc
);
5294 Cast cast
= new Cast (new TypeExpression (type
, loc
), proxy
, loc
);
5295 return cast
.Resolve (ec
);
5298 public override Expression
CreateExpressionTree (ResolveContext ec
)
5301 if (method
== null) {
5302 args
= new Arguments (1);
5303 args
.Add (new Argument (new TypeOf (new TypeExpression (type
, loc
), loc
)));
5305 args
= Arguments
.CreateForExpressionTree (ec
,
5307 method
.CreateExpressionTree (ec
));
5310 return CreateExpressionFactoryCall (ec
, "New", args
);
5313 protected override Expression
DoResolve (ResolveContext ec
)
5316 // The New DoResolve might be called twice when initializing field
5317 // expressions (see EmitFieldInitializers, the call to
5318 // GetInitializerExpression will perform a resolve on the expression,
5319 // and later the assign will trigger another resolution
5321 // This leads to bugs (#37014)
5324 if (RequestedType
is NewDelegate
)
5325 return RequestedType
;
5329 TypeExpr texpr
= RequestedType
.ResolveAsTypeTerminal (ec
, false);
5335 if (type
.IsPointer
) {
5336 ec
.Report
.Error (1919, loc
, "Unsafe type `{0}' cannot be used in an object creation expression",
5337 TypeManager
.CSharpName (type
));
5341 if (Arguments
== null) {
5342 Constant c
= Constantify (type
);
5344 return ReducedExpression
.Create (c
.Resolve (ec
), this);
5347 if (TypeManager
.IsDelegateType (type
)) {
5348 return (new NewDelegate (type
, Arguments
, loc
)).Resolve (ec
);
5351 var tparam
= type
as TypeParameterSpec
;
5352 if (tparam
!= null) {
5353 if (!tparam
.HasSpecialConstructor
&& !tparam
.HasSpecialStruct
) {
5354 ec
.Report
.Error (304, loc
,
5355 "Cannot create an instance of the variable type `{0}' because it does not have the new() constraint",
5356 TypeManager
.CSharpName (type
));
5359 if ((Arguments
!= null) && (Arguments
.Count
!= 0)) {
5360 ec
.Report
.Error (417, loc
,
5361 "`{0}': cannot provide arguments when creating an instance of a variable type",
5362 TypeManager
.CSharpName (type
));
5365 if (TypeManager
.activator_create_instance
== null) {
5366 TypeSpec activator_type
= TypeManager
.CoreLookupType (ec
.Compiler
, "System", "Activator", MemberKind
.Class
, true);
5367 if (activator_type
!= null) {
5368 TypeManager
.activator_create_instance
= TypeManager
.GetPredefinedMethod (
5369 activator_type
, MemberFilter
.Method ("CreateInstance", 1, ParametersCompiled
.EmptyReadOnlyParameters
, null), loc
);
5373 eclass
= ExprClass
.Value
;
5377 if (type
.IsStatic
) {
5378 ec
.Report
.SymbolRelatedToPreviousError (type
);
5379 ec
.Report
.Error (712, loc
, "Cannot create an instance of the static class `{0}'", TypeManager
.CSharpName (type
));
5383 if (type
.IsInterface
|| type
.IsAbstract
){
5384 if (!TypeManager
.IsGenericType (type
)) {
5385 RequestedType
= CheckComImport (ec
);
5386 if (RequestedType
!= null)
5387 return RequestedType
;
5390 ec
.Report
.SymbolRelatedToPreviousError (type
);
5391 ec
.Report
.Error (144, loc
, "Cannot create an instance of the abstract class or interface `{0}'", TypeManager
.CSharpName (type
));
5395 bool is_struct
= TypeManager
.IsStruct (type
);
5396 eclass
= ExprClass
.Value
;
5399 // SRE returns a match for .ctor () on structs (the object constructor),
5400 // so we have to manually ignore it.
5402 if (is_struct
&& Arguments
== null)
5405 // For member-lookup, treat 'new Foo (bar)' as call to 'foo.ctor (bar)', where 'foo' is of type 'Foo'.
5406 Expression ml
= MemberLookupFinal (ec
, type
, type
, ConstructorInfo
.ConstructorName
, 0,
5407 MemberKind
.Constructor
, BindingRestriction
.AccessibleOnly
| BindingRestriction
.DeclaredOnly
, loc
);
5410 if (Arguments
!= null) {
5411 Arguments
.Resolve (ec
, out dynamic);
5419 method
= ml
as MethodGroupExpr
;
5420 if (method
== null) {
5421 ml
.Error_UnexpectedKind (ec
, ResolveFlags
.MethodGroup
, loc
);
5425 method
= method
.OverloadResolve (ec
, ref Arguments
, false, loc
);
5430 Arguments
.Insert (0, new Argument (new TypeOf (texpr
, loc
).Resolve (ec
), Argument
.AType
.DynamicTypeName
));
5431 return new DynamicConstructorBinder (type
, Arguments
, loc
).Resolve (ec
);
5437 bool DoEmitTypeParameter (EmitContext ec
)
5439 var ctor_factory
= TypeManager
.activator_create_instance
.MakeGenericMethod (type
);
5440 var tparam
= (TypeParameterSpec
) type
;
5442 if (tparam
.IsReferenceType
) {
5443 ec
.Emit (OpCodes
.Call
, ctor_factory
);
5447 // Allow DoEmit() to be called multiple times.
5448 // We need to create a new LocalTemporary each time since
5449 // you can't share LocalBuilders among ILGeneators.
5450 LocalTemporary temp
= new LocalTemporary (type
);
5452 Label label_activator
= ec
.DefineLabel ();
5453 Label label_end
= ec
.DefineLabel ();
5455 temp
.AddressOf (ec
, AddressOp
.Store
);
5456 ec
.Emit (OpCodes
.Initobj
, type
);
5459 ec
.Emit (OpCodes
.Box
, type
);
5460 ec
.Emit (OpCodes
.Brfalse
, label_activator
);
5462 temp
.AddressOf (ec
, AddressOp
.Store
);
5463 ec
.Emit (OpCodes
.Initobj
, type
);
5465 ec
.Emit (OpCodes
.Br_S
, label_end
);
5467 ec
.MarkLabel (label_activator
);
5469 ec
.Emit (OpCodes
.Call
, ctor_factory
);
5470 ec
.MarkLabel (label_end
);
5475 // This Emit can be invoked in two contexts:
5476 // * As a mechanism that will leave a value on the stack (new object)
5477 // * As one that wont (init struct)
5479 // If we are dealing with a ValueType, we have a few
5480 // situations to deal with:
5482 // * The target is a ValueType, and we have been provided
5483 // the instance (this is easy, we are being assigned).
5485 // * The target of New is being passed as an argument,
5486 // to a boxing operation or a function that takes a
5489 // In this case, we need to create a temporary variable
5490 // that is the argument of New.
5492 // Returns whether a value is left on the stack
5494 // *** Implementation note ***
5496 // To benefit from this optimization, each assignable expression
5497 // has to manually cast to New and call this Emit.
5499 // TODO: It's worth to implement it for arrays and fields
5501 public virtual bool Emit (EmitContext ec
, IMemoryLocation target
)
5503 bool is_value_type
= TypeManager
.IsValueType (type
);
5504 VariableReference vr
= target
as VariableReference
;
5506 if (target
!= null && is_value_type
&& (vr
!= null || method
== null)) {
5507 target
.AddressOf (ec
, AddressOp
.Store
);
5508 } else if (vr
!= null && vr
.IsRef
) {
5512 if (Arguments
!= null)
5513 Arguments
.Emit (ec
);
5515 if (is_value_type
) {
5516 if (method
== null) {
5517 ec
.Emit (OpCodes
.Initobj
, type
);
5522 ec
.Emit (OpCodes
.Call
, method
.BestCandidate
);
5527 if (type
is TypeParameterSpec
)
5528 return DoEmitTypeParameter (ec
);
5530 ec
.Emit (OpCodes
.Newobj
, method
.BestCandidate
);
5534 public override void Emit (EmitContext ec
)
5536 LocalTemporary v
= null;
5537 if (method
== null && TypeManager
.IsValueType (type
)) {
5538 // TODO: Use temporary variable from pool
5539 v
= new LocalTemporary (type
);
5546 public override void EmitStatement (EmitContext ec
)
5548 LocalTemporary v
= null;
5549 if (method
== null && TypeManager
.IsValueType (type
)) {
5550 // TODO: Use temporary variable from pool
5551 v
= new LocalTemporary (type
);
5555 ec
.Emit (OpCodes
.Pop
);
5558 public virtual bool HasInitializer
{
5564 public void AddressOf (EmitContext ec
, AddressOp mode
)
5566 EmitAddressOf (ec
, mode
);
5569 protected virtual IMemoryLocation
EmitAddressOf (EmitContext ec
, AddressOp mode
)
5571 LocalTemporary value_target
= new LocalTemporary (type
);
5573 if (type
is TypeParameterSpec
) {
5574 DoEmitTypeParameter (ec
);
5575 value_target
.Store (ec
);
5576 value_target
.AddressOf (ec
, mode
);
5577 return value_target
;
5580 if (!TypeManager
.IsStruct (type
)){
5582 // We throw an exception. So far, I believe we only need to support
5584 // foreach (int j in new StructType ())
5587 throw new Exception ("AddressOf should not be used for classes");
5590 value_target
.AddressOf (ec
, AddressOp
.Store
);
5592 if (method
== null) {
5593 ec
.Emit (OpCodes
.Initobj
, type
);
5595 if (Arguments
!= null)
5596 Arguments
.Emit (ec
);
5598 ec
.Emit (OpCodes
.Call
, method
.BestCandidate
);
5601 value_target
.AddressOf (ec
, mode
);
5602 return value_target
;
5605 protected override void CloneTo (CloneContext clonectx
, Expression t
)
5607 New target
= (New
) t
;
5609 target
.RequestedType
= RequestedType
.Clone (clonectx
);
5610 if (Arguments
!= null){
5611 target
.Arguments
= Arguments
.Clone (clonectx
);
5615 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
5617 return SLE
.Expression
.New ((ConstructorInfo
) method
.BestCandidate
.GetMetaInfo (), Arguments
.MakeExpression (Arguments
, ctx
));
5621 public class ArrayInitializer
: ShimExpression
5623 List
<Expression
> elements
;
5625 public ArrayInitializer (List
<Expression
> init
, Location loc
)
5631 public ArrayInitializer (int count
, Location loc
)
5634 elements
= new List
<Expression
> (count
);
5637 public ArrayInitializer (Location loc
)
5642 public void Add (Expression expr
)
5644 elements
.Add (expr
);
5647 protected override void CloneTo (CloneContext clonectx
, Expression t
)
5649 var target
= (ArrayInitializer
) t
;
5651 target
.elements
= new List
<Expression
> (elements
.Count
);
5652 foreach (var element
in elements
)
5653 target
.elements
.Add (element
.Clone (clonectx
));
5655 base.CloneTo (clonectx
, t
);
5659 get { return elements.Count; }
5662 protected override Expression
DoResolve (ResolveContext rc
)
5664 throw new NotImplementedException ();
5667 public Expression
this [int index
] {
5668 get { return elements [index]; }
5673 /// 14.5.10.2: Represents an array creation expression.
5677 /// There are two possible scenarios here: one is an array creation
5678 /// expression that specifies the dimensions and optionally the
5679 /// initialization data and the other which does not need dimensions
5680 /// specified but where initialization data is mandatory.
5682 public class ArrayCreation
: Expression
5684 FullNamedExpression requested_base_type
;
5685 ArrayInitializer initializers
;
5688 // The list of Argument types.
5689 // This is used to construct the `newarray' or constructor signature
5691 protected List
<Expression
> arguments
;
5693 protected TypeSpec array_element_type
;
5694 int num_arguments
= 0;
5695 protected int dimensions
;
5696 protected readonly string rank
;
5697 Expression first_emit
;
5698 LocalTemporary first_emit_temp
;
5700 protected List
<Expression
> array_data
;
5702 Dictionary
<int, int> bounds
;
5704 // The number of constants in array initializers
5705 int const_initializers_count
;
5706 bool only_constant_initializers
;
5708 public ArrayCreation (FullNamedExpression requested_base_type
, List
<Expression
> exprs
, string rank
, ArrayInitializer initializers
, Location l
)
5710 this.requested_base_type
= requested_base_type
;
5711 this.initializers
= initializers
;
5715 arguments
= new List
<Expression
> (exprs
);
5716 num_arguments
= arguments
.Count
;
5719 public ArrayCreation (FullNamedExpression requested_base_type
, string rank
, ArrayInitializer initializers
, Location l
)
5721 this.requested_base_type
= requested_base_type
;
5722 this.initializers
= initializers
;
5727 protected override void Error_NegativeArrayIndex (ResolveContext ec
, Location loc
)
5729 ec
.Report
.Error (248, loc
, "Cannot create an array with a negative size");
5732 bool CheckIndices (ResolveContext ec
, ArrayInitializer probe
, int idx
, bool specified_dims
, int child_bounds
)
5734 if (initializers
!= null && bounds
== null) {
5736 // We use this to store all the date values in the order in which we
5737 // will need to store them in the byte blob later
5739 array_data
= new List
<Expression
> ();
5740 bounds
= new Dictionary
<int, int> ();
5743 if (specified_dims
) {
5744 Expression a
= arguments
[idx
];
5749 a
= ConvertExpressionToArrayIndex (ec
, a
);
5755 if (initializers
!= null) {
5756 Constant c
= a
as Constant
;
5757 if (c
== null && a
is ArrayIndexCast
)
5758 c
= ((ArrayIndexCast
) a
).Child
as Constant
;
5761 ec
.Report
.Error (150, a
.Location
, "A constant value is expected");
5767 value = System
.Convert
.ToInt32 (c
.GetValue ());
5769 ec
.Report
.Error (150, a
.Location
, "A constant value is expected");
5773 // TODO: probe.Count does not fit ulong in
5774 if (value != probe
.Count
) {
5775 ec
.Report
.Error (847, loc
, "An array initializer of length `{0}' was expected", value.ToString ());
5779 bounds
[idx
] = value;
5783 if (initializers
== null)
5786 only_constant_initializers
= true;
5787 for (int i
= 0; i
< probe
.Count
; ++i
) {
5789 if (o
is ArrayInitializer
) {
5790 var sub_probe
= o
as ArrayInitializer
;
5791 if (idx
+ 1 >= dimensions
){
5792 ec
.Report
.Error (623, loc
, "Array initializers can only be used in a variable or field initializer. Try using a new expression instead");
5796 bool ret
= CheckIndices (ec
, sub_probe
, idx
+ 1, specified_dims
, child_bounds
- 1);
5799 } else if (child_bounds
> 1) {
5800 ec
.Report
.Error (846, o
.Location
, "A nested array initializer was expected");
5802 Expression element
= ResolveArrayElement (ec
, o
);
5803 if (element
== null)
5806 // Initializers with the default values can be ignored
5807 Constant c
= element
as Constant
;
5809 if (!c
.IsDefaultInitializer (array_element_type
)) {
5810 ++const_initializers_count
;
5813 only_constant_initializers
= false;
5816 array_data
.Add (element
);
5823 public override Expression
CreateExpressionTree (ResolveContext ec
)
5827 if (array_data
== null) {
5828 args
= new Arguments (arguments
.Count
+ 1);
5829 args
.Add (new Argument (new TypeOf (new TypeExpression (array_element_type
, loc
), loc
)));
5830 foreach (Expression a
in arguments
)
5831 args
.Add (new Argument (a
.CreateExpressionTree (ec
)));
5833 return CreateExpressionFactoryCall (ec
, "NewArrayBounds", args
);
5836 if (dimensions
> 1) {
5837 ec
.Report
.Error (838, loc
, "An expression tree cannot contain a multidimensional array initializer");
5841 args
= new Arguments (array_data
== null ? 1 : array_data
.Count
+ 1);
5842 args
.Add (new Argument (new TypeOf (new TypeExpression (array_element_type
, loc
), loc
)));
5843 if (array_data
!= null) {
5844 for (int i
= 0; i
< array_data
.Count
; ++i
) {
5845 Expression e
= array_data
[i
];
5846 args
.Add (new Argument (e
.CreateExpressionTree (ec
)));
5850 return CreateExpressionFactoryCall (ec
, "NewArrayInit", args
);
5853 public void UpdateIndices ()
5856 for (var probe
= initializers
; probe
!= null;) {
5857 if (probe
.Count
> 0 && probe
[0] is ArrayInitializer
) {
5858 Expression e
= new IntConstant (probe
.Count
, Location
.Null
);
5861 bounds
[i
++] = probe
.Count
;
5863 probe
= (ArrayInitializer
) probe
[0];
5866 Expression e
= new IntConstant (probe
.Count
, Location
.Null
);
5869 bounds
[i
++] = probe
.Count
;
5875 protected virtual Expression
ResolveArrayElement (ResolveContext ec
, Expression element
)
5877 element
= element
.Resolve (ec
);
5878 if (element
== null)
5881 if (element
is CompoundAssign
.TargetExpression
) {
5882 if (first_emit
!= null)
5883 throw new InternalErrorException ("Can only handle one mutator at a time");
5884 first_emit
= element
;
5885 element
= first_emit_temp
= new LocalTemporary (element
.Type
);
5888 return Convert
.ImplicitConversionRequired (
5889 ec
, element
, array_element_type
, loc
);
5892 protected bool ResolveInitializers (ResolveContext ec
)
5894 if (arguments
!= null) {
5896 for (int i
= 0; i
< arguments
.Count
; ++i
) {
5897 res
&= CheckIndices (ec
, initializers
, i
, true, dimensions
);
5898 if (initializers
!= null)
5905 arguments
= new List
<Expression
> ();
5907 if (!CheckIndices (ec
, initializers
, 0, false, dimensions
))
5916 // Resolved the type of the array
5918 bool ResolveArrayType (ResolveContext ec
)
5920 if (requested_base_type
is VarExpr
) {
5921 ec
.Report
.Error (820, loc
, "An implicitly typed local variable declarator cannot use an array initializer");
5925 StringBuilder array_qualifier
= new StringBuilder ();
5928 // `In the first form allocates an array instace of the type that results
5929 // from deleting each of the individual expression from the expression list'
5931 if (num_arguments
> 0) {
5932 array_qualifier
.Append ("[");
5933 for (int i
= num_arguments
-1; i
> 0; i
--)
5934 array_qualifier
.Append (",");
5935 array_qualifier
.Append ("]");
5938 array_qualifier
.Append (rank
);
5943 TypeExpr array_type_expr
;
5944 array_type_expr
= new ComposedCast (requested_base_type
, array_qualifier
.ToString (), loc
);
5945 array_type_expr
= array_type_expr
.ResolveAsTypeTerminal (ec
, false);
5946 if (array_type_expr
== null)
5949 type
= array_type_expr
.Type
;
5950 var ac
= type
as ArrayContainer
;
5952 ec
.Report
.Error (622, loc
, "Can only use array initializer expressions to assign to array types. Try using a new expression instead");
5956 array_element_type
= ac
.Element
;
5957 dimensions
= ac
.Rank
;
5962 protected override Expression
DoResolve (ResolveContext ec
)
5967 if (!ResolveArrayType (ec
))
5971 // validate the initializers and fill in any missing bits
5973 if (!ResolveInitializers (ec
))
5976 eclass
= ExprClass
.Value
;
5980 byte [] MakeByteBlob ()
5985 int count
= array_data
.Count
;
5987 TypeSpec element_type
= array_element_type
;
5988 if (TypeManager
.IsEnumType (element_type
))
5989 element_type
= EnumSpec
.GetUnderlyingType (element_type
);
5991 factor
= GetTypeSize (element_type
);
5993 throw new Exception ("unrecognized type in MakeByteBlob: " + element_type
);
5995 data
= new byte [(count
* factor
+ 3) & ~
3];
5998 for (int i
= 0; i
< count
; ++i
) {
5999 object v
= array_data
[i
];
6001 if (v
is EnumConstant
)
6002 v
= ((EnumConstant
) v
).Child
;
6004 if (v
is Constant
&& !(v
is StringConstant
))
6005 v
= ((Constant
) v
).GetValue ();
6011 if (element_type
== TypeManager
.int64_type
){
6012 if (!(v
is Expression
)){
6013 long val
= (long) v
;
6015 for (int j
= 0; j
< factor
; ++j
) {
6016 data
[idx
+ j
] = (byte) (val
& 0xFF);
6020 } else if (element_type
== TypeManager
.uint64_type
){
6021 if (!(v
is Expression
)){
6022 ulong val
= (ulong) v
;
6024 for (int j
= 0; j
< factor
; ++j
) {
6025 data
[idx
+ j
] = (byte) (val
& 0xFF);
6029 } else if (element_type
== TypeManager
.float_type
) {
6030 if (!(v
is Expression
)){
6031 element
= BitConverter
.GetBytes ((float) v
);
6033 for (int j
= 0; j
< factor
; ++j
)
6034 data
[idx
+ j
] = element
[j
];
6035 if (!BitConverter
.IsLittleEndian
)
6036 System
.Array
.Reverse (data
, idx
, 4);
6038 } else if (element_type
== TypeManager
.double_type
) {
6039 if (!(v
is Expression
)){
6040 element
= BitConverter
.GetBytes ((double) v
);
6042 for (int j
= 0; j
< factor
; ++j
)
6043 data
[idx
+ j
] = element
[j
];
6045 // FIXME: Handle the ARM float format.
6046 if (!BitConverter
.IsLittleEndian
)
6047 System
.Array
.Reverse (data
, idx
, 8);
6049 } else if (element_type
== TypeManager
.char_type
){
6050 if (!(v
is Expression
)){
6051 int val
= (int) ((char) v
);
6053 data
[idx
] = (byte) (val
& 0xff);
6054 data
[idx
+1] = (byte) (val
>> 8);
6056 } else if (element_type
== TypeManager
.short_type
){
6057 if (!(v
is Expression
)){
6058 int val
= (int) ((short) v
);
6060 data
[idx
] = (byte) (val
& 0xff);
6061 data
[idx
+1] = (byte) (val
>> 8);
6063 } else if (element_type
== TypeManager
.ushort_type
){
6064 if (!(v
is Expression
)){
6065 int val
= (int) ((ushort) v
);
6067 data
[idx
] = (byte) (val
& 0xff);
6068 data
[idx
+1] = (byte) (val
>> 8);
6070 } else if (element_type
== TypeManager
.int32_type
) {
6071 if (!(v
is Expression
)){
6074 data
[idx
] = (byte) (val
& 0xff);
6075 data
[idx
+1] = (byte) ((val
>> 8) & 0xff);
6076 data
[idx
+2] = (byte) ((val
>> 16) & 0xff);
6077 data
[idx
+3] = (byte) (val
>> 24);
6079 } else if (element_type
== TypeManager
.uint32_type
) {
6080 if (!(v
is Expression
)){
6081 uint val
= (uint) v
;
6083 data
[idx
] = (byte) (val
& 0xff);
6084 data
[idx
+1] = (byte) ((val
>> 8) & 0xff);
6085 data
[idx
+2] = (byte) ((val
>> 16) & 0xff);
6086 data
[idx
+3] = (byte) (val
>> 24);
6088 } else if (element_type
== TypeManager
.sbyte_type
) {
6089 if (!(v
is Expression
)){
6090 sbyte val
= (sbyte) v
;
6091 data
[idx
] = (byte) val
;
6093 } else if (element_type
== TypeManager
.byte_type
) {
6094 if (!(v
is Expression
)){
6095 byte val
= (byte) v
;
6096 data
[idx
] = (byte) val
;
6098 } else if (element_type
== TypeManager
.bool_type
) {
6099 if (!(v
is Expression
)){
6100 bool val
= (bool) v
;
6101 data
[idx
] = (byte) (val
? 1 : 0);
6103 } else if (element_type
== TypeManager
.decimal_type
){
6104 if (!(v
is Expression
)){
6105 int [] bits
= Decimal
.GetBits ((decimal) v
);
6108 // FIXME: For some reason, this doesn't work on the MS runtime.
6109 int [] nbits
= new int [4];
6110 nbits
[0] = bits
[3];
6111 nbits
[1] = bits
[2];
6112 nbits
[2] = bits
[0];
6113 nbits
[3] = bits
[1];
6115 for (int j
= 0; j
< 4; j
++){
6116 data
[p
++] = (byte) (nbits
[j
] & 0xff);
6117 data
[p
++] = (byte) ((nbits
[j
] >> 8) & 0xff);
6118 data
[p
++] = (byte) ((nbits
[j
] >> 16) & 0xff);
6119 data
[p
++] = (byte) (nbits
[j
] >> 24);
6123 throw new Exception ("Unrecognized type in MakeByteBlob: " + element_type
);
6133 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
6135 var initializers
= new SLE
.Expression
[array_data
.Count
];
6136 for (var i
= 0; i
< initializers
.Length
; i
++) {
6137 if (array_data
[i
] == null)
6138 initializers
[i
] = SLE
.Expression
.Default (array_element_type
.GetMetaInfo ());
6140 initializers
[i
] = array_data
[i
].MakeExpression (ctx
);
6143 return SLE
.Expression
.NewArrayInit (array_element_type
.GetMetaInfo (), initializers
);
6147 // Emits the initializers for the array
6149 void EmitStaticInitializers (EmitContext ec
)
6151 // FIXME: This should go to Resolve !
6152 if (TypeManager
.void_initializearray_array_fieldhandle
== null) {
6153 TypeManager
.void_initializearray_array_fieldhandle
= TypeManager
.GetPredefinedMethod (
6154 TypeManager
.runtime_helpers_type
, "InitializeArray", loc
,
6155 TypeManager
.array_type
, TypeManager
.runtime_field_handle_type
);
6156 if (TypeManager
.void_initializearray_array_fieldhandle
== null)
6161 // First, the static data
6165 byte [] data
= MakeByteBlob ();
6167 fb
= RootContext
.MakeStaticData (data
);
6169 ec
.Emit (OpCodes
.Dup
);
6170 ec
.Emit (OpCodes
.Ldtoken
, fb
);
6171 ec
.Emit (OpCodes
.Call
, TypeManager
.void_initializearray_array_fieldhandle
);
6175 // Emits pieces of the array that can not be computed at compile
6176 // time (variables and string locations).
6178 // This always expect the top value on the stack to be the array
6180 void EmitDynamicInitializers (EmitContext ec
, bool emitConstants
)
6182 int dims
= bounds
.Count
;
6183 var current_pos
= new int [dims
];
6185 for (int i
= 0; i
< array_data
.Count
; i
++){
6187 Expression e
= array_data
[i
];
6188 var c
= e
as Constant
;
6190 // Constant can be initialized via StaticInitializer
6191 if (c
== null || (c
!= null && emitConstants
&& !c
.IsDefaultInitializer (array_element_type
))) {
6192 TypeSpec etype
= e
.Type
;
6194 ec
.Emit (OpCodes
.Dup
);
6196 for (int idx
= 0; idx
< dims
; idx
++)
6197 ec
.EmitInt (current_pos
[idx
]);
6200 // If we are dealing with a struct, get the
6201 // address of it, so we can store it.
6203 if ((dims
== 1) && TypeManager
.IsStruct (etype
) &&
6204 (!TypeManager
.IsBuiltinOrEnum (etype
) ||
6205 etype
== TypeManager
.decimal_type
)) {
6207 ec
.Emit (OpCodes
.Ldelema
, etype
);
6212 ec
.EmitArrayStore ((ArrayContainer
) type
);
6218 for (int j
= dims
- 1; j
>= 0; j
--){
6220 if (current_pos
[j
] < bounds
[j
])
6222 current_pos
[j
] = 0;
6227 public override void Emit (EmitContext ec
)
6229 if (first_emit
!= null) {
6230 first_emit
.Emit (ec
);
6231 first_emit_temp
.Store (ec
);
6234 foreach (Expression e
in arguments
)
6237 ec
.EmitArrayNew ((ArrayContainer
) type
);
6239 if (initializers
== null)
6242 // Emit static initializer for arrays which have contain more than 2 items and
6243 // the static initializer will initialize at least 25% of array values.
6244 // NOTE: const_initializers_count does not contain default constant values.
6245 if (const_initializers_count
> 2 && const_initializers_count
* 4 > (array_data
.Count
) &&
6246 (TypeManager
.IsPrimitiveType (array_element_type
) || TypeManager
.IsEnumType (array_element_type
))) {
6247 EmitStaticInitializers (ec
);
6249 if (!only_constant_initializers
)
6250 EmitDynamicInitializers (ec
, false);
6252 EmitDynamicInitializers (ec
, true);
6255 if (first_emit_temp
!= null)
6256 first_emit_temp
.Release (ec
);
6259 public override void EncodeAttributeValue (IMemberContext rc
, AttributeEncoder enc
, TypeSpec targetType
)
6261 // no multi dimensional or jagged arrays
6262 if (arguments
.Count
!= 1 || array_element_type
.IsArray
) {
6263 base.EncodeAttributeValue (rc
, enc
, targetType
);
6267 // No array covariance, except for array -> object
6268 if (type
!= targetType
) {
6269 if (targetType
!= TypeManager
.object_type
) {
6270 base.EncodeAttributeValue (rc
, enc
, targetType
);
6277 // Single dimensional array of 0 size
6278 if (array_data
== null) {
6279 IntConstant ic
= arguments
[0] as IntConstant
;
6280 if (ic
== null || !ic
.IsDefaultValue
) {
6281 base.EncodeAttributeValue (rc
, enc
, targetType
);
6283 enc
.Stream
.Write (0);
6289 enc
.Stream
.Write ((int) array_data
.Count
);
6290 foreach (var element
in array_data
) {
6291 element
.EncodeAttributeValue (rc
, enc
, array_element_type
);
6295 protected override void CloneTo (CloneContext clonectx
, Expression t
)
6297 ArrayCreation target
= (ArrayCreation
) t
;
6299 if (requested_base_type
!= null)
6300 target
.requested_base_type
= (FullNamedExpression
)requested_base_type
.Clone (clonectx
);
6302 if (arguments
!= null){
6303 target
.arguments
= new List
<Expression
> (arguments
.Count
);
6304 foreach (Expression e
in arguments
)
6305 target
.arguments
.Add (e
.Clone (clonectx
));
6308 if (initializers
!= null)
6309 target
.initializers
= (ArrayInitializer
) initializers
.Clone (clonectx
);
6314 // Represents an implicitly typed array epxression
6316 class ImplicitlyTypedArrayCreation
: ArrayCreation
6318 public ImplicitlyTypedArrayCreation (string rank
, ArrayInitializer initializers
, Location loc
)
6319 : base (null, rank
, initializers
, loc
)
6321 if (rank
.Length
> 2) {
6322 while (rank
[++dimensions
] == ',');
6328 protected override Expression
DoResolve (ResolveContext ec
)
6333 if (!ResolveInitializers (ec
))
6336 if (array_element_type
== null || array_element_type
== TypeManager
.null_type
||
6337 array_element_type
== TypeManager
.void_type
|| array_element_type
== InternalType
.AnonymousMethod
||
6338 array_element_type
== InternalType
.MethodGroup
||
6339 arguments
.Count
!= dimensions
) {
6340 Error_NoBestType (ec
);
6345 // At this point we found common base type for all initializer elements
6346 // but we have to be sure that all static initializer elements are of
6349 UnifyInitializerElement (ec
);
6351 type
= TypeManager
.GetConstructedType (array_element_type
, rank
);
6352 eclass
= ExprClass
.Value
;
6356 void Error_NoBestType (ResolveContext ec
)
6358 ec
.Report
.Error (826, loc
,
6359 "The type of an implicitly typed array cannot be inferred from the initializer. Try specifying array type explicitly");
6363 // Converts static initializer only
6365 void UnifyInitializerElement (ResolveContext ec
)
6367 for (int i
= 0; i
< array_data
.Count
; ++i
) {
6368 Expression e
= (Expression
)array_data
[i
];
6370 array_data
[i
] = Convert
.ImplicitConversion (ec
, e
, array_element_type
, Location
.Null
);
6374 protected override Expression
ResolveArrayElement (ResolveContext ec
, Expression element
)
6376 element
= element
.Resolve (ec
);
6377 if (element
== null)
6380 if (array_element_type
== null) {
6381 if (element
.Type
!= TypeManager
.null_type
)
6382 array_element_type
= element
.Type
;
6387 if (Convert
.ImplicitConversionExists (ec
, element
, array_element_type
)) {
6391 if (Convert
.ImplicitConversionExists (ec
, new TypeExpression (array_element_type
, loc
), element
.Type
)) {
6392 array_element_type
= element
.Type
;
6396 Error_NoBestType (ec
);
6401 public sealed class CompilerGeneratedThis
: This
6403 public static This Instance
= new CompilerGeneratedThis ();
6405 private CompilerGeneratedThis ()
6406 : base (Location
.Null
)
6410 public CompilerGeneratedThis (TypeSpec type
, Location loc
)
6416 protected override Expression
DoResolve (ResolveContext ec
)
6418 eclass
= ExprClass
.Variable
;
6420 type
= ec
.CurrentType
;
6425 public override HoistedVariable
GetHoistedVariable (AnonymousExpression ae
)
6432 /// Represents the `this' construct
6435 public class This
: VariableReference
6437 sealed class ThisVariable
: ILocalVariable
6439 public static readonly ILocalVariable Instance
= new ThisVariable ();
6441 public void Emit (EmitContext ec
)
6443 ec
.Emit (OpCodes
.Ldarg_0
);
6446 public void EmitAssign (EmitContext ec
)
6448 throw new InvalidOperationException ();
6451 public void EmitAddressOf (EmitContext ec
)
6453 ec
.Emit (OpCodes
.Ldarg_0
);
6457 VariableInfo variable_info
;
6459 public This (Location loc
)
6464 public override VariableInfo VariableInfo
{
6465 get { return variable_info; }
6468 public override bool IsFixed
{
6469 get { return false; }
6472 public override HoistedVariable
GetHoistedVariable (AnonymousExpression ae
)
6477 AnonymousMethodStorey storey
= ae
.Storey
;
6478 while (storey
!= null) {
6479 AnonymousMethodStorey temp
= storey
.Parent
as AnonymousMethodStorey
;
6481 return storey
.HoistedThis
;
6489 public override bool IsRef
{
6490 get { return type.IsStruct; }
6493 protected override ILocalVariable Variable
{
6494 get { return ThisVariable.Instance; }
6497 public static bool IsThisAvailable (ResolveContext ec
, bool ignoreAnonymous
)
6499 if (ec
.IsStatic
|| ec
.HasAny (ResolveContext
.Options
.FieldInitializerScope
| ResolveContext
.Options
.BaseInitializer
| ResolveContext
.Options
.ConstantScope
))
6502 if (ignoreAnonymous
|| ec
.CurrentAnonymousMethod
== null)
6505 if (TypeManager
.IsStruct (ec
.CurrentType
) && ec
.CurrentIterator
== null)
6511 public bool ResolveBase (ResolveContext ec
)
6513 eclass
= ExprClass
.Variable
;
6514 type
= ec
.CurrentType
;
6516 if (!IsThisAvailable (ec
, false)) {
6517 if (ec
.IsStatic
&& !ec
.HasSet (ResolveContext
.Options
.ConstantScope
)) {
6518 ec
.Report
.Error (26, loc
, "Keyword `this' is not valid in a static property, static method, or static field initializer");
6519 } else if (ec
.CurrentAnonymousMethod
!= null) {
6520 ec
.Report
.Error (1673, loc
,
6521 "Anonymous methods inside structs cannot access instance members of `this'. " +
6522 "Consider copying `this' to a local variable outside the anonymous method and using the local instead");
6524 ec
.Report
.Error (27, loc
, "Keyword `this' is not available in the current context");
6528 var block
= ec
.CurrentBlock
;
6529 if (block
!= null) {
6530 if (block
.Toplevel
.ThisVariable
!= null)
6531 variable_info
= block
.Toplevel
.ThisVariable
.VariableInfo
;
6533 AnonymousExpression am
= ec
.CurrentAnonymousMethod
;
6534 if (am
!= null && ec
.IsVariableCapturingRequired
) {
6535 am
.SetHasThisAccess ();
6543 // Called from Invocation to check if the invocation is correct
6545 public override void CheckMarshalByRefAccess (ResolveContext ec
)
6547 if ((variable_info
!= null) && !(TypeManager
.IsStruct (type
) && ec
.OmitStructFlowAnalysis
) &&
6548 !variable_info
.IsAssigned (ec
)) {
6549 ec
.Report
.Error (188, loc
,
6550 "The `this' object cannot be used before all of its fields are assigned to");
6551 variable_info
.SetAssigned (ec
);
6555 public override Expression
CreateExpressionTree (ResolveContext ec
)
6557 Arguments args
= new Arguments (1);
6558 args
.Add (new Argument (this));
6560 // Use typeless constant for ldarg.0 to save some
6561 // space and avoid problems with anonymous stories
6562 return CreateExpressionFactoryCall (ec
, "Constant", args
);
6565 protected override Expression
DoResolve (ResolveContext ec
)
6571 override public Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
6573 if (!ResolveBase (ec
))
6576 if (variable_info
!= null)
6577 variable_info
.SetAssigned (ec
);
6579 if (ec
.CurrentType
.IsClass
){
6580 if (right_side
== EmptyExpression
.UnaryAddress
)
6581 ec
.Report
.Error (459, loc
, "Cannot take the address of `this' because it is read-only");
6582 else if (right_side
== EmptyExpression
.OutAccess
.Instance
)
6583 ec
.Report
.Error (1605, loc
, "Cannot pass `this' as a ref or out argument because it is read-only");
6585 ec
.Report
.Error (1604, loc
, "Cannot assign to `this' because it is read-only");
6591 public override int GetHashCode()
6593 throw new NotImplementedException ();
6596 public override string Name
{
6597 get { return "this"; }
6600 public override bool Equals (object obj
)
6602 This t
= obj
as This
;
6609 protected override void CloneTo (CloneContext clonectx
, Expression t
)
6614 public override void SetHasAddressTaken ()
6621 /// Represents the `__arglist' construct
6623 public class ArglistAccess
: Expression
6625 public ArglistAccess (Location loc
)
6630 public override Expression
CreateExpressionTree (ResolveContext ec
)
6632 throw new NotSupportedException ("ET");
6635 protected override Expression
DoResolve (ResolveContext ec
)
6637 eclass
= ExprClass
.Variable
;
6638 type
= TypeManager
.runtime_argument_handle_type
;
6640 if (ec
.HasSet (ResolveContext
.Options
.FieldInitializerScope
) || !ec
.CurrentBlock
.Toplevel
.Parameters
.HasArglist
) {
6641 ec
.Report
.Error (190, loc
,
6642 "The __arglist construct is valid only within a variable argument method");
6648 public override void Emit (EmitContext ec
)
6650 ec
.Emit (OpCodes
.Arglist
);
6653 protected override void CloneTo (CloneContext clonectx
, Expression target
)
6660 /// Represents the `__arglist (....)' construct
6662 public class Arglist
: Expression
6664 Arguments Arguments
;
6666 public Arglist (Location loc
)
6671 public Arglist (Arguments args
, Location l
)
6677 public Type
[] ArgumentTypes
{
6679 if (Arguments
== null)
6680 return System
.Type
.EmptyTypes
;
6682 var retval
= new Type
[Arguments
.Count
];
6683 for (int i
= 0; i
< retval
.Length
; i
++)
6684 retval
[i
] = Arguments
[i
].Expr
.Type
.GetMetaInfo ();
6690 public override Expression
CreateExpressionTree (ResolveContext ec
)
6692 ec
.Report
.Error (1952, loc
, "An expression tree cannot contain a method with variable arguments");
6696 protected override Expression
DoResolve (ResolveContext ec
)
6698 eclass
= ExprClass
.Variable
;
6699 type
= InternalType
.Arglist
;
6700 if (Arguments
!= null) {
6701 bool dynamic; // Can be ignored as there is always only 1 overload
6702 Arguments
.Resolve (ec
, out dynamic);
6708 public override void Emit (EmitContext ec
)
6710 if (Arguments
!= null)
6711 Arguments
.Emit (ec
);
6714 protected override void CloneTo (CloneContext clonectx
, Expression t
)
6716 Arglist target
= (Arglist
) t
;
6718 if (Arguments
!= null)
6719 target
.Arguments
= Arguments
.Clone (clonectx
);
6724 /// Implements the typeof operator
6726 public class TypeOf
: Expression
{
6727 Expression QueriedType
;
6728 protected TypeSpec typearg
;
6730 public TypeOf (Expression queried_type
, Location l
)
6732 QueriedType
= queried_type
;
6736 public override Expression
CreateExpressionTree (ResolveContext ec
)
6738 Arguments args
= new Arguments (2);
6739 args
.Add (new Argument (this));
6740 args
.Add (new Argument (new TypeOf (new TypeExpression (type
, loc
), loc
)));
6741 return CreateExpressionFactoryCall (ec
, "Constant", args
);
6744 protected override Expression
DoResolve (ResolveContext ec
)
6746 TypeExpr texpr
= QueriedType
.ResolveAsTypeTerminal (ec
, false);
6750 typearg
= texpr
.Type
;
6753 // Get generic type definition for unbounded type arguments
6755 var tne
= QueriedType
as ATypeNameExpression
;
6756 if (tne
!= null && typearg
.IsGeneric
&& !tne
.HasTypeArguments
)
6757 typearg
= typearg
.GetDefinition ();
6759 if (typearg
== TypeManager
.void_type
) {
6760 ec
.Report
.Error (673, loc
, "System.Void cannot be used from C#. Use typeof (void) to get the void type object");
6761 } else if (typearg
.IsPointer
&& !ec
.IsUnsafe
){
6762 UnsafeError (ec
, loc
);
6763 } else if (texpr
is DynamicTypeExpr
) {
6764 ec
.Report
.Error (1962, QueriedType
.Location
,
6765 "The typeof operator cannot be used on the dynamic type");
6768 type
= TypeManager
.type_type
;
6770 return DoResolveBase ();
6773 protected Expression
DoResolveBase ()
6775 if (TypeManager
.system_type_get_type_from_handle
== null) {
6776 TypeManager
.system_type_get_type_from_handle
= TypeManager
.GetPredefinedMethod (
6777 TypeManager
.type_type
, "GetTypeFromHandle", loc
, TypeManager
.runtime_handle_type
);
6780 // Even though what is returned is a type object, it's treated as a value by the compiler.
6781 // In particular, 'typeof (Foo).X' is something totally different from 'Foo.X'.
6782 eclass
= ExprClass
.Value
;
6786 public override void EncodeAttributeValue (IMemberContext rc
, AttributeEncoder enc
, TypeSpec targetType
)
6788 // Target type is not System.Type therefore must be object
6789 // and we need to use different encoding sequence
6790 if (targetType
!= type
)
6794 var gi = typearg as InflatedTypeSpec;
6796 // TODO: This has to be recursive, handle arrays, etc.
6797 // I could probably do it after CustomAttribute encoder rewrite
6798 foreach (var ta in gi.TypeArguments) {
6799 if (ta.IsGenericParameter) {
6800 ec.Report.SymbolRelatedToPreviousError (typearg);
6801 ec.Report.Error (416, loc, "`{0}': an attribute argument cannot use type parameters",
6802 TypeManager.CSharpName (typearg));
6810 if (!enc
.EncodeTypeName (typearg
)) {
6811 rc
.Compiler
.Report
.SymbolRelatedToPreviousError (typearg
);
6812 rc
.Compiler
.Report
.Error (416, loc
, "`{0}': an attribute argument cannot use type parameters",
6813 TypeManager
.CSharpName (typearg
));
6817 public override void Emit (EmitContext ec
)
6819 ec
.Emit (OpCodes
.Ldtoken
, typearg
);
6820 ec
.Emit (OpCodes
.Call
, TypeManager
.system_type_get_type_from_handle
);
6823 public TypeSpec TypeArgument
{
6829 protected override void CloneTo (CloneContext clonectx
, Expression t
)
6831 TypeOf target
= (TypeOf
) t
;
6832 if (QueriedType
!= null)
6833 target
.QueriedType
= QueriedType
.Clone (clonectx
);
6838 /// Implements the `typeof (void)' operator
6840 public class TypeOfVoid
: TypeOf
{
6841 public TypeOfVoid (Location l
) : base (null, l
)
6846 protected override Expression
DoResolve (ResolveContext ec
)
6848 type
= TypeManager
.type_type
;
6849 typearg
= TypeManager
.void_type
;
6851 return DoResolveBase ();
6855 class TypeOfMethod
: TypeOfMember
<MethodSpec
>
6857 public TypeOfMethod (MethodSpec method
, Location loc
)
6858 : base (method
, loc
)
6862 protected override Expression
DoResolve (ResolveContext ec
)
6864 if (member
.IsConstructor
) {
6865 type
= TypeManager
.ctorinfo_type
;
6867 type
= TypeManager
.ctorinfo_type
= TypeManager
.CoreLookupType (ec
.Compiler
, "System.Reflection", "ConstructorInfo", MemberKind
.Class
, true);
6869 type
= TypeManager
.methodinfo_type
;
6871 type
= TypeManager
.methodinfo_type
= TypeManager
.CoreLookupType (ec
.Compiler
, "System.Reflection", "MethodInfo", MemberKind
.Class
, true);
6874 return base.DoResolve (ec
);
6877 public override void Emit (EmitContext ec
)
6879 ec
.Emit (OpCodes
.Ldtoken
, member
);
6882 ec
.Emit (OpCodes
.Castclass
, type
);
6885 protected override string GetMethodName
{
6886 get { return "GetMethodFromHandle"; }
6889 protected override string RuntimeHandleName
{
6890 get { return "RuntimeMethodHandle"; }
6893 protected override MethodSpec TypeFromHandle
{
6895 return TypeManager
.methodbase_get_type_from_handle
;
6898 TypeManager
.methodbase_get_type_from_handle
= value;
6902 protected override MethodSpec TypeFromHandleGeneric
{
6904 return TypeManager
.methodbase_get_type_from_handle_generic
;
6907 TypeManager
.methodbase_get_type_from_handle_generic
= value;
6911 protected override string TypeName
{
6912 get { return "MethodBase"; }
6916 abstract class TypeOfMember
<T
> : Expression where T
: MemberSpec
6918 protected readonly T member
;
6920 protected TypeOfMember (T member
, Location loc
)
6922 this.member
= member
;
6926 public override Expression
CreateExpressionTree (ResolveContext ec
)
6928 Arguments args
= new Arguments (2);
6929 args
.Add (new Argument (this));
6930 args
.Add (new Argument (new TypeOf (new TypeExpression (type
, loc
), loc
)));
6931 return CreateExpressionFactoryCall (ec
, "Constant", args
);
6934 protected override Expression
DoResolve (ResolveContext ec
)
6936 bool is_generic
= member
.DeclaringType
.IsGenericOrParentIsGeneric
;
6937 var mi
= is_generic
? TypeFromHandleGeneric
: TypeFromHandle
;
6940 TypeSpec t
= TypeManager
.CoreLookupType (ec
.Compiler
, "System.Reflection", TypeName
, MemberKind
.Class
, true);
6941 TypeSpec handle_type
= TypeManager
.CoreLookupType (ec
.Compiler
, "System", RuntimeHandleName
, MemberKind
.Struct
, true);
6943 if (t
== null || handle_type
== null)
6946 mi
= TypeManager
.GetPredefinedMethod (t
, GetMethodName
, loc
,
6948 new TypeSpec
[] { handle_type, TypeManager.runtime_handle_type }
:
6949 new TypeSpec
[] { handle_type }
);
6952 TypeFromHandleGeneric
= mi
;
6954 TypeFromHandle
= mi
;
6957 eclass
= ExprClass
.Value
;
6961 public override void Emit (EmitContext ec
)
6963 bool is_generic
= member
.DeclaringType
.IsGenericOrParentIsGeneric
;
6966 mi
= TypeFromHandleGeneric
;
6967 ec
.Emit (OpCodes
.Ldtoken
, member
.DeclaringType
);
6969 mi
= TypeFromHandle
;
6972 ec
.Emit (OpCodes
.Call
, mi
);
6975 protected abstract string GetMethodName { get; }
6976 protected abstract string RuntimeHandleName { get; }
6977 protected abstract MethodSpec TypeFromHandle { get; set; }
6978 protected abstract MethodSpec TypeFromHandleGeneric { get; set; }
6979 protected abstract string TypeName { get; }
6982 class TypeOfField
: TypeOfMember
<FieldSpec
>
6984 public TypeOfField (FieldSpec field
, Location loc
)
6989 protected override Expression
DoResolve (ResolveContext ec
)
6991 if (TypeManager
.fieldinfo_type
== null)
6992 TypeManager
.fieldinfo_type
= TypeManager
.CoreLookupType (ec
.Compiler
, "System.Reflection", TypeName
, MemberKind
.Class
, true);
6994 type
= TypeManager
.fieldinfo_type
;
6995 return base.DoResolve (ec
);
6998 public override void Emit (EmitContext ec
)
7000 ec
.Emit (OpCodes
.Ldtoken
, member
);
7004 protected override string GetMethodName
{
7005 get { return "GetFieldFromHandle"; }
7008 protected override string RuntimeHandleName
{
7009 get { return "RuntimeFieldHandle"; }
7012 protected override MethodSpec TypeFromHandle
{
7014 return TypeManager
.fieldinfo_get_field_from_handle
;
7017 TypeManager
.fieldinfo_get_field_from_handle
= value;
7021 protected override MethodSpec TypeFromHandleGeneric
{
7023 return TypeManager
.fieldinfo_get_field_from_handle_generic
;
7026 TypeManager
.fieldinfo_get_field_from_handle_generic
= value;
7030 protected override string TypeName
{
7031 get { return "FieldInfo"; }
7036 /// Implements the sizeof expression
7038 public class SizeOf
: Expression
{
7039 readonly Expression QueriedType
;
7040 TypeSpec type_queried
;
7042 public SizeOf (Expression queried_type
, Location l
)
7044 this.QueriedType
= queried_type
;
7048 public override Expression
CreateExpressionTree (ResolveContext ec
)
7050 Error_PointerInsideExpressionTree (ec
);
7054 protected override Expression
DoResolve (ResolveContext ec
)
7056 TypeExpr texpr
= QueriedType
.ResolveAsTypeTerminal (ec
, false);
7060 type_queried
= texpr
.Type
;
7061 if (TypeManager
.IsEnumType (type_queried
))
7062 type_queried
= EnumSpec
.GetUnderlyingType (type_queried
);
7064 int size_of
= GetTypeSize (type_queried
);
7066 return new IntConstant (size_of
, loc
).Resolve (ec
);
7069 if (!TypeManager
.VerifyUnmanaged (ec
.Compiler
, type_queried
, loc
)){
7074 ec
.Report
.Error (233, loc
,
7075 "`{0}' does not have a predefined size, therefore sizeof can only be used in an unsafe context (consider using System.Runtime.InteropServices.Marshal.SizeOf)",
7076 TypeManager
.CSharpName (type_queried
));
7079 type
= TypeManager
.int32_type
;
7080 eclass
= ExprClass
.Value
;
7084 public override void Emit (EmitContext ec
)
7086 ec
.Emit (OpCodes
.Sizeof
, type_queried
);
7089 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7095 /// Implements the qualified-alias-member (::) expression.
7097 public class QualifiedAliasMember
: MemberAccess
7099 readonly string alias;
7100 public static readonly string GlobalAlias
= "global";
7102 public QualifiedAliasMember (string alias, string identifier
, Location l
)
7103 : base (null, identifier
, l
)
7108 public QualifiedAliasMember (string alias, string identifier
, TypeArguments targs
, Location l
)
7109 : base (null, identifier
, targs
, l
)
7114 public QualifiedAliasMember (string alias, string identifier
, int arity
, Location l
)
7115 : base (null, identifier
, arity
, l
)
7120 public override FullNamedExpression
ResolveAsTypeStep (IMemberContext ec
, bool silent
)
7122 if (alias == GlobalAlias
) {
7123 expr
= GlobalRootNamespace
.Instance
;
7124 return base.ResolveAsTypeStep (ec
, silent
);
7127 int errors
= ec
.Compiler
.Report
.Errors
;
7128 expr
= ec
.LookupNamespaceAlias (alias);
7130 if (errors
== ec
.Compiler
.Report
.Errors
)
7131 ec
.Compiler
.Report
.Error (432, loc
, "Alias `{0}' not found", alias);
7135 FullNamedExpression fne
= base.ResolveAsTypeStep (ec
, silent
);
7139 if (expr
.eclass
== ExprClass
.Type
) {
7141 ec
.Compiler
.Report
.Error (431, loc
,
7142 "Alias `{0}' cannot be used with '::' since it denotes a type. Consider replacing '::' with '.'", alias);
7150 protected override Expression
DoResolve (ResolveContext ec
)
7152 return ResolveAsTypeStep (ec
, false);
7155 protected override void Error_IdentifierNotFound (IMemberContext rc
, TypeSpec expr_type
, string identifier
)
7157 rc
.Compiler
.Report
.Error (687, loc
,
7158 "A namespace alias qualifier `{0}' did not resolve to a namespace or a type",
7159 GetSignatureForError ());
7162 public override string GetSignatureForError ()
7165 if (targs
!= null) {
7166 name
= Name
+ "<" + targs
.GetSignatureForError () + ">";
7169 return alias + "::" + name
;
7172 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7179 /// Implements the member access expression
7181 public class MemberAccess
: ATypeNameExpression
{
7182 protected Expression expr
;
7184 public MemberAccess (Expression expr
, string id
)
7185 : base (id
, expr
.Location
)
7190 public MemberAccess (Expression expr
, string identifier
, Location loc
)
7191 : base (identifier
, loc
)
7196 public MemberAccess (Expression expr
, string identifier
, TypeArguments args
, Location loc
)
7197 : base (identifier
, args
, loc
)
7202 public MemberAccess (Expression expr
, string identifier
, int arity
, Location loc
)
7203 : base (identifier
, arity
, loc
)
7208 Expression
DoResolve (ResolveContext ec
, Expression right_side
)
7211 throw new Exception ();
7214 // Resolve the expression with flow analysis turned off, we'll do the definite
7215 // assignment checks later. This is because we don't know yet what the expression
7216 // will resolve to - it may resolve to a FieldExpr and in this case we must do the
7217 // definite assignment check on the actual field and not on the whole struct.
7220 SimpleName original
= expr
as SimpleName
;
7221 Expression expr_resolved
;
7222 const ResolveFlags flags
= ResolveFlags
.VariableOrValue
| ResolveFlags
.Type
;
7224 using (ec
.Set (ResolveContext
.Options
.OmitStructFlowAnalysis
)) {
7225 if (original
!= null) {
7226 expr_resolved
= original
.DoResolve (ec
, true);
7227 if (expr_resolved
!= null) {
7228 // Ugly, simulate skipped Resolve
7229 if (expr_resolved
is ConstantExpr
) {
7230 expr_resolved
= expr_resolved
.Resolve (ec
);
7231 } else if (expr_resolved
is FieldExpr
|| expr_resolved
is PropertyExpr
) {
7233 } else if ((flags
& expr_resolved
.ExprClassToResolveFlags
) == 0) {
7234 expr_resolved
.Error_UnexpectedKind (ec
, flags
, expr
.Location
);
7235 expr_resolved
= null;
7239 expr_resolved
= expr
.Resolve (ec
, flags
);
7243 if (expr_resolved
== null)
7246 Namespace ns
= expr_resolved
as Namespace
;
7248 FullNamedExpression retval
= ns
.Lookup (ec
.Compiler
, Name
, Arity
, loc
);
7251 ns
.Error_NamespaceDoesNotExist (loc
, Name
, Arity
, ec
);
7252 else if (HasTypeArguments
)
7253 retval
= new GenericTypeExpr (retval
.Type
, targs
, loc
).ResolveAsTypeStep (ec
, false);
7258 TypeSpec expr_type
= expr_resolved
.Type
;
7259 if (expr_type
== InternalType
.Dynamic
) {
7260 Arguments args
= new Arguments (1);
7261 args
.Add (new Argument (expr_resolved
.Resolve (ec
)));
7262 expr
= new DynamicMemberBinder (Name
, args
, loc
);
7263 if (right_side
!= null)
7264 return expr
.DoResolveLValue (ec
, right_side
);
7266 return expr
.Resolve (ec
);
7269 const MemberKind dot_kinds
= MemberKind
.Class
| MemberKind
.Struct
| MemberKind
.Delegate
| MemberKind
.Enum
| MemberKind
.Interface
| MemberKind
.TypeParameter
;
7270 if ((expr_type
.Kind
& dot_kinds
) == 0 || expr_type
== TypeManager
.void_type
) {
7271 Unary
.Error_OperatorCannotBeApplied (ec
, loc
, ".", expr_type
);
7275 var arity
= HasTypeArguments
? targs
.Count
: -1;
7277 var member_lookup
= MemberLookup (ec
.Compiler
,
7278 ec
.CurrentType
, expr_type
, expr_type
, Name
, arity
, BindingRestriction
.NoOverrides
, loc
);
7280 if (member_lookup
== null) {
7281 expr
= expr_resolved
.Resolve (ec
);
7283 ExprClass expr_eclass
= expr
.eclass
;
7286 // Extension methods are not allowed on all expression types
7288 if (expr_eclass
== ExprClass
.Value
|| expr_eclass
== ExprClass
.Variable
||
7289 expr_eclass
== ExprClass
.IndexerAccess
|| expr_eclass
== ExprClass
.PropertyAccess
||
7290 expr_eclass
== ExprClass
.EventAccess
) {
7291 ExtensionMethodGroupExpr ex_method_lookup
= ec
.LookupExtensionMethod (expr_type
, Name
, arity
, loc
);
7292 if (ex_method_lookup
!= null) {
7293 ex_method_lookup
.ExtensionExpression
= expr
;
7295 if (HasTypeArguments
) {
7296 if (!targs
.Resolve (ec
))
7299 ex_method_lookup
.SetTypeArguments (ec
, targs
);
7302 return ex_method_lookup
.Resolve (ec
);
7306 member_lookup
= Error_MemberLookupFailed (ec
,
7307 ec
.CurrentType
, expr_type
, expr_type
, Name
, arity
, null,
7308 MemberKind
.All
, BindingRestriction
.AccessibleOnly
);
7309 if (member_lookup
== null)
7314 TypeExpr texpr
= member_lookup
as TypeExpr
;
7315 if (texpr
!= null) {
7316 if (!(expr_resolved
is TypeExpr
)) {
7317 me
= expr_resolved
as MemberExpr
;
7318 if (me
== null || me
.ProbeIdenticalTypeName (ec
, expr_resolved
, original
) == expr_resolved
) {
7319 ec
.Report
.Error (572, loc
, "`{0}': cannot reference a type through an expression; try `{1}' instead",
7320 Name
, member_lookup
.GetSignatureForError ());
7325 if (!texpr
.CheckAccessLevel (ec
.MemberContext
)) {
7326 ec
.Report
.SymbolRelatedToPreviousError (member_lookup
.Type
);
7327 ErrorIsInaccesible (loc
, TypeManager
.CSharpName (member_lookup
.Type
), ec
.Report
);
7331 if (HasTypeArguments
) {
7332 var ct
= new GenericTypeExpr (member_lookup
.Type
, targs
, loc
);
7333 return ct
.ResolveAsTypeStep (ec
, false);
7336 return member_lookup
;
7339 me
= (MemberExpr
) member_lookup
;
7341 if (original
!= null && me
.IsStatic
)
7342 expr_resolved
= me
.ProbeIdenticalTypeName (ec
, expr_resolved
, original
);
7344 me
= me
.ResolveMemberAccess (ec
, expr_resolved
, original
);
7346 if (HasTypeArguments
) {
7347 if (!targs
.Resolve (ec
))
7350 me
.SetTypeArguments (ec
, targs
);
7353 if (original
!= null && (!TypeManager
.IsValueType (expr_type
) || me
is PropertyExpr
)) {
7354 if (me
.IsInstance
) {
7355 LocalVariableReference
var = expr_resolved
as LocalVariableReference
;
7356 if (var != null && !var.VerifyAssigned (ec
))
7361 // The following DoResolve/DoResolveLValue will do the definite assignment
7364 if (right_side
!= null)
7365 return me
.DoResolveLValue (ec
, right_side
);
7367 return me
.Resolve (ec
);
7370 protected override Expression
DoResolve (ResolveContext ec
)
7372 return DoResolve (ec
, null);
7375 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
7377 return DoResolve (ec
, right_side
);
7380 public override FullNamedExpression
ResolveAsTypeStep (IMemberContext ec
, bool silent
)
7382 return ResolveNamespaceOrType (ec
, silent
);
7385 public FullNamedExpression
ResolveNamespaceOrType (IMemberContext rc
, bool silent
)
7387 FullNamedExpression expr_resolved
= expr
.ResolveAsTypeStep (rc
, silent
);
7389 if (expr_resolved
== null)
7392 Namespace ns
= expr_resolved
as Namespace
;
7394 FullNamedExpression retval
= ns
.Lookup (rc
.Compiler
, Name
, Arity
, loc
);
7396 if (retval
== null) {
7398 ns
.Error_NamespaceDoesNotExist (loc
, Name
, Arity
, rc
);
7399 } else if (HasTypeArguments
) {
7400 retval
= new GenericTypeExpr (retval
.Type
, targs
, loc
).ResolveAsTypeStep (rc
, silent
);
7406 TypeExpr tnew_expr
= expr_resolved
.ResolveAsTypeTerminal (rc
, false);
7407 if (tnew_expr
== null)
7410 TypeSpec expr_type
= tnew_expr
.Type
;
7411 if (TypeManager
.IsGenericParameter (expr_type
)) {
7412 rc
.Compiler
.Report
.Error (704, loc
, "A nested type cannot be specified through a type parameter `{0}'",
7413 tnew_expr
.GetSignatureForError ());
7417 var nested
= MemberCache
.FindNestedType (expr_type
, Name
, Arity
);
7418 if (nested
== null) {
7422 Error_IdentifierNotFound (rc
, expr_type
, Name
);
7427 if (!IsMemberAccessible (rc
.CurrentType
?? InternalType
.FakeInternalType
, nested
, out extra_check
)) {
7428 ErrorIsInaccesible (loc
, nested
.GetSignatureForError (), rc
.Compiler
.Report
);
7432 if (HasTypeArguments
) {
7433 texpr
= new GenericTypeExpr (nested
, targs
, loc
);
7435 texpr
= new TypeExpression (nested
, loc
);
7438 return texpr
.ResolveAsTypeStep (rc
, false);
7441 protected virtual void Error_IdentifierNotFound (IMemberContext rc
, TypeSpec expr_type
, string identifier
)
7443 var nested
= MemberCache
.FindNestedType (expr_type
, Name
, -System
.Math
.Max (1, Arity
));
7445 if (nested
!= null) {
7446 Error_TypeArgumentsCannotBeUsed (rc
.Compiler
.Report
, expr
.Location
, nested
, Arity
);
7450 var member_lookup
= MemberLookup (rc
.Compiler
,
7451 rc
.CurrentType
, expr_type
, expr_type
, identifier
, -1,
7452 MemberKind
.All
, BindingRestriction
.None
, loc
);
7454 if (member_lookup
== null) {
7455 rc
.Compiler
.Report
.Error (426, loc
, "The nested type `{0}' does not exist in the type `{1}'",
7456 Name
, expr_type
.GetSignatureForError ());
7458 // TODO: Report.SymbolRelatedToPreviousError
7459 member_lookup
.Error_UnexpectedKind (rc
.Compiler
.Report
, null, "type", loc
);
7463 protected override void Error_TypeDoesNotContainDefinition (ResolveContext ec
, TypeSpec type
, string name
)
7465 if (RootContext
.Version
> LanguageVersion
.ISO_2
&& !ec
.Compiler
.IsRuntimeBinder
&&
7466 ((expr
.eclass
& (ExprClass
.Value
| ExprClass
.Variable
)) != 0)) {
7467 ec
.Report
.Error (1061, loc
, "Type `{0}' does not contain a definition for `{1}' and no " +
7468 "extension method `{1}' of type `{0}' could be found " +
7469 "(are you missing a using directive or an assembly reference?)",
7470 TypeManager
.CSharpName (type
), name
);
7474 base.Error_TypeDoesNotContainDefinition (ec
, type
, name
);
7477 public override string GetSignatureForError ()
7479 return expr
.GetSignatureForError () + "." + base.GetSignatureForError ();
7482 public Expression Left
{
7488 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7490 MemberAccess target
= (MemberAccess
) t
;
7492 target
.expr
= expr
.Clone (clonectx
);
7497 /// Implements checked expressions
7499 public class CheckedExpr
: Expression
{
7501 public Expression Expr
;
7503 public CheckedExpr (Expression e
, Location l
)
7509 public override Expression
CreateExpressionTree (ResolveContext ec
)
7511 using (ec
.With (ResolveContext
.Options
.AllCheckStateFlags
, true))
7512 return Expr
.CreateExpressionTree (ec
);
7515 protected override Expression
DoResolve (ResolveContext ec
)
7517 using (ec
.With (ResolveContext
.Options
.AllCheckStateFlags
, true))
7518 Expr
= Expr
.Resolve (ec
);
7523 if (Expr
is Constant
|| Expr
is MethodGroupExpr
|| Expr
is AnonymousMethodExpression
|| Expr
is DefaultValueExpression
)
7526 eclass
= Expr
.eclass
;
7531 public override void Emit (EmitContext ec
)
7533 using (ec
.With (EmitContext
.Options
.AllCheckStateFlags
, true))
7537 public override void EmitBranchable (EmitContext ec
, Label target
, bool on_true
)
7539 using (ec
.With (EmitContext
.Options
.AllCheckStateFlags
, true))
7540 Expr
.EmitBranchable (ec
, target
, on_true
);
7543 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
7545 using (ctx
.With (BuilderContext
.Options
.AllCheckStateFlags
, true)) {
7546 return Expr
.MakeExpression (ctx
);
7550 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7552 CheckedExpr target
= (CheckedExpr
) t
;
7554 target
.Expr
= Expr
.Clone (clonectx
);
7559 /// Implements the unchecked expression
7561 public class UnCheckedExpr
: Expression
{
7563 public Expression Expr
;
7565 public UnCheckedExpr (Expression e
, Location l
)
7571 public override Expression
CreateExpressionTree (ResolveContext ec
)
7573 using (ec
.With (ResolveContext
.Options
.AllCheckStateFlags
, false))
7574 return Expr
.CreateExpressionTree (ec
);
7577 protected override Expression
DoResolve (ResolveContext ec
)
7579 using (ec
.With (ResolveContext
.Options
.AllCheckStateFlags
, false))
7580 Expr
= Expr
.Resolve (ec
);
7585 if (Expr
is Constant
|| Expr
is MethodGroupExpr
|| Expr
is AnonymousMethodExpression
|| Expr
is DefaultValueExpression
)
7588 eclass
= Expr
.eclass
;
7593 public override void Emit (EmitContext ec
)
7595 using (ec
.With (EmitContext
.Options
.AllCheckStateFlags
, false))
7599 public override void EmitBranchable (EmitContext ec
, Label target
, bool on_true
)
7601 using (ec
.With (EmitContext
.Options
.AllCheckStateFlags
, false))
7602 Expr
.EmitBranchable (ec
, target
, on_true
);
7605 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7607 UnCheckedExpr target
= (UnCheckedExpr
) t
;
7609 target
.Expr
= Expr
.Clone (clonectx
);
7614 /// An Element Access expression.
7616 /// During semantic analysis these are transformed into
7617 /// IndexerAccess, ArrayAccess or a PointerArithmetic.
7619 public class ElementAccess
: Expression
{
7620 public Arguments Arguments
;
7621 public Expression Expr
;
7623 public ElementAccess (Expression e
, Arguments args
)
7627 this.Arguments
= args
;
7630 public override Expression
CreateExpressionTree (ResolveContext ec
)
7632 Arguments args
= Arguments
.CreateForExpressionTree (ec
, Arguments
,
7633 Expr
.CreateExpressionTree (ec
));
7635 return CreateExpressionFactoryCall (ec
, "ArrayIndex", args
);
7638 Expression
MakePointerAccess (ResolveContext ec
, TypeSpec t
)
7640 if (Arguments
.Count
!= 1){
7641 ec
.Report
.Error (196, loc
, "A pointer must be indexed by only one value");
7645 if (Arguments
[0] is NamedArgument
)
7646 Error_NamedArgument ((NamedArgument
) Arguments
[0], ec
.Report
);
7648 Expression p
= new PointerArithmetic (Binary
.Operator
.Addition
, Expr
, Arguments
[0].Expr
.Resolve (ec
), t
, loc
);
7649 return new Indirection (p
, loc
).Resolve (ec
);
7652 protected override Expression
DoResolve (ResolveContext ec
)
7654 Expr
= Expr
.Resolve (ec
);
7659 // We perform some simple tests, and then to "split" the emit and store
7660 // code we create an instance of a different class, and return that.
7662 // I am experimenting with this pattern.
7664 TypeSpec t
= Expr
.Type
;
7666 if (t
== TypeManager
.array_type
){
7667 ec
.Report
.Error (21, loc
, "Cannot apply indexing with [] to an expression of type `System.Array'");
7672 return (new ArrayAccess (this, loc
)).Resolve (ec
);
7674 return MakePointerAccess (ec
, t
);
7676 FieldExpr fe
= Expr
as FieldExpr
;
7678 var ff
= fe
.Spec
as FixedFieldSpec
;
7680 return MakePointerAccess (ec
, ff
.ElementType
);
7683 return (new IndexerAccess (this, loc
)).Resolve (ec
);
7686 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
7688 Expr
= Expr
.Resolve (ec
);
7694 return (new ArrayAccess (this, loc
)).DoResolveLValue (ec
, right_side
);
7697 return MakePointerAccess (ec
, type
);
7699 if (Expr
.eclass
!= ExprClass
.Variable
&& TypeManager
.IsStruct (type
))
7700 Error_CannotModifyIntermediateExpressionValue (ec
);
7702 return (new IndexerAccess (this, loc
)).DoResolveLValue (ec
, right_side
);
7705 public override void Emit (EmitContext ec
)
7707 throw new Exception ("Should never be reached");
7710 public static void Error_NamedArgument (NamedArgument na
, Report Report
)
7712 Report
.Error (1742, na
.Location
, "An element access expression cannot use named argument");
7715 public override string GetSignatureForError ()
7717 return Expr
.GetSignatureForError ();
7720 protected override void CloneTo (CloneContext clonectx
, Expression t
)
7722 ElementAccess target
= (ElementAccess
) t
;
7724 target
.Expr
= Expr
.Clone (clonectx
);
7725 if (Arguments
!= null)
7726 target
.Arguments
= Arguments
.Clone (clonectx
);
7731 /// Implements array access
7733 public class ArrayAccess
: Expression
, IDynamicAssign
, IMemoryLocation
{
7735 // Points to our "data" repository
7739 LocalTemporary temp
;
7743 public ArrayAccess (ElementAccess ea_data
, Location l
)
7749 public override Expression
CreateExpressionTree (ResolveContext ec
)
7751 return ea
.CreateExpressionTree (ec
);
7754 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
7756 return DoResolve (ec
);
7759 protected override Expression
DoResolve (ResolveContext ec
)
7761 // dynamic is used per argument in ConvertExpressionToArrayIndex case
7763 ea
.Arguments
.Resolve (ec
, out dynamic);
7765 TypeSpec t
= ea
.Expr
.Type
;
7766 int rank
= ea
.Arguments
.Count
;
7767 if (t
.GetMetaInfo ().GetArrayRank () != rank
) {
7768 ec
.Report
.Error (22, ea
.Location
, "Wrong number of indexes `{0}' inside [], expected `{1}'",
7769 ea
.Arguments
.Count
.ToString (), t
.GetMetaInfo ().GetArrayRank ().ToString ());
7773 type
= TypeManager
.GetElementType (t
);
7774 if (type
.IsPointer
&& !ec
.IsUnsafe
) {
7775 UnsafeError (ec
, ea
.Location
);
7778 foreach (Argument a
in ea
.Arguments
) {
7779 if (a
is NamedArgument
)
7780 ElementAccess
.Error_NamedArgument ((NamedArgument
) a
, ec
.Report
);
7782 a
.Expr
= ConvertExpressionToArrayIndex (ec
, a
.Expr
);
7785 eclass
= ExprClass
.Variable
;
7790 protected override void Error_NegativeArrayIndex (ResolveContext ec
, Location loc
)
7792 ec
.Report
.Warning (251, 2, loc
, "Indexing an array with a negative index (array indices always start at zero)");
7796 // Load the array arguments into the stack.
7798 void LoadArrayAndArguments (EmitContext ec
)
7802 for (int i
= 0; i
< ea
.Arguments
.Count
; ++i
) {
7803 ea
.Arguments
[i
].Emit (ec
);
7807 public void Emit (EmitContext ec
, bool leave_copy
)
7809 var ac
= ea
.Expr
.Type
as ArrayContainer
;
7812 ec
.EmitLoadFromPtr (type
);
7814 LoadArrayAndArguments (ec
);
7815 ec
.EmitArrayLoad (ac
);
7819 ec
.Emit (OpCodes
.Dup
);
7820 temp
= new LocalTemporary (this.type
);
7825 public override void Emit (EmitContext ec
)
7830 public void EmitAssign (EmitContext ec
, Expression source
, bool leave_copy
, bool prepare_for_load
)
7832 var ac
= (ArrayContainer
) ea
.Expr
.Type
;
7833 TypeSpec t
= source
.Type
;
7834 prepared
= prepare_for_load
;
7837 AddressOf (ec
, AddressOp
.LoadStore
);
7838 ec
.Emit (OpCodes
.Dup
);
7840 LoadArrayAndArguments (ec
);
7843 // If we are dealing with a struct, get the
7844 // address of it, so we can store it.
7846 // The stobj opcode used by value types will need
7847 // an address on the stack, not really an array/array
7850 if (ac
.Rank
== 1 && TypeManager
.IsStruct (t
) &&
7851 (!TypeManager
.IsBuiltinOrEnum (t
) ||
7852 t
== TypeManager
.decimal_type
)) {
7854 ec
.Emit (OpCodes
.Ldelema
, t
);
7860 ec
.Emit (OpCodes
.Dup
);
7861 temp
= new LocalTemporary (this.type
);
7866 ec
.EmitStoreFromPtr (t
);
7868 ec
.EmitArrayStore (ac
);
7877 public void EmitNew (EmitContext ec
, New source
, bool leave_copy
)
7879 if (!source
.Emit (ec
, this)) {
7881 throw new NotImplementedException ();
7886 throw new NotImplementedException ();
7889 public void AddressOf (EmitContext ec
, AddressOp mode
)
7891 var ac
= (ArrayContainer
) ea
.Expr
.Type
;
7893 LoadArrayAndArguments (ec
);
7894 ec
.EmitArrayAddress (ac
);
7898 public SLE
.Expression
MakeAssignExpression (BuilderContext ctx
)
7900 return SLE
.Expression
.ArrayAccess (
7901 ea
.Expr
.MakeExpression (ctx
),
7902 Arguments
.MakeExpression (ea
.Arguments
, ctx
));
7906 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
7908 return SLE
.Expression
.ArrayIndex (
7909 ea
.Expr
.MakeExpression (ctx
),
7910 Arguments
.MakeExpression (ea
.Arguments
, ctx
));
7915 /// Expressions that represent an indexer call.
7917 public class IndexerAccess
: Expression
, IDynamicAssign
7919 class IndexerMethodGroupExpr
: MethodGroupExpr
7921 IEnumerable
<IndexerSpec
> candidates
;
7923 public IndexerMethodGroupExpr (IEnumerable
<IndexerSpec
> indexers
, Location loc
)
7924 : base (FilterAccessors (indexers
).ToList (), null, loc
)
7926 candidates
= indexers
;
7929 public IndexerSpec
BestIndexer ()
7931 return candidates
.Where (l
=> l
.Get
== BestCandidate
|| l
.Set
== BestCandidate
).First ();
7934 static IEnumerable
<MemberSpec
> FilterAccessors (IEnumerable
<IndexerSpec
> indexers
)
7936 foreach (IndexerSpec i
in indexers
) {
7944 protected override IList
<MemberSpec
> GetBaseTypeMethods (ResolveContext rc
, TypeSpec type
)
7946 candidates
= GetIndexersForType (type
, false);
7947 if (candidates
== null)
7950 return FilterAccessors (candidates
).ToList ();
7953 public override string Name
{
7959 protected override int GetApplicableParametersCount (MethodSpec method
, AParametersCollection parameters
)
7962 // Here is the trick, decrease number of arguments by 1 when only
7963 // available property method is setter. This makes overload resolution
7964 // work correctly for indexers.
7967 if (method
.Name
[0] == 'g')
7968 return parameters
.Count
;
7970 return parameters
.Count
- 1;
7975 // Points to our "data" repository
7978 bool is_base_indexer
;
7980 LocalTemporary temp
;
7981 LocalTemporary prepared_value
;
7982 Expression set_expr
;
7984 protected TypeSpec indexer_type
;
7985 protected TypeSpec current_type
;
7986 protected Expression instance_expr
;
7987 protected Arguments arguments
;
7989 public IndexerAccess (ElementAccess ea
, Location loc
)
7990 : this (ea
.Expr
, false, loc
)
7992 this.arguments
= ea
.Arguments
;
7995 protected IndexerAccess (Expression instance_expr
, bool is_base_indexer
,
7998 this.instance_expr
= instance_expr
;
7999 this.is_base_indexer
= is_base_indexer
;
8003 static string GetAccessorName (bool isSet
)
8005 return isSet
? "set" : "get";
8008 public override Expression
CreateExpressionTree (ResolveContext ec
)
8010 Arguments args
= Arguments
.CreateForExpressionTree (ec
, arguments
,
8011 instance_expr
.CreateExpressionTree (ec
),
8012 new TypeOfMethod (spec
.Get
, loc
));
8014 return CreateExpressionFactoryCall (ec
, "Call", args
);
8017 static IEnumerable
<IndexerSpec
> GetIndexersForType (TypeSpec lookup_type
, bool baseAccess
)
8019 BindingRestriction restrictions
= BindingRestriction
.AccessibleOnly
;
8021 restrictions
|= BindingRestriction
.NoOverrides
;
8023 return MemberCache
.FindIndexers (lookup_type
, restrictions
);
8026 protected virtual void CommonResolve (ResolveContext ec
)
8028 indexer_type
= instance_expr
.Type
;
8029 current_type
= ec
.CurrentType
;
8032 protected override Expression
DoResolve (ResolveContext ec
)
8034 return ResolveAccessor (ec
, null);
8037 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
8039 if (right_side
== EmptyExpression
.OutAccess
.Instance
) {
8040 right_side
.DoResolveLValue (ec
, this);
8044 // if the indexer returns a value type, and we try to set a field in it
8045 if (right_side
== EmptyExpression
.LValueMemberAccess
|| right_side
== EmptyExpression
.LValueMemberOutAccess
) {
8046 Error_CannotModifyIntermediateExpressionValue (ec
);
8049 return ResolveAccessor (ec
, right_side
);
8052 Expression
ResolveAccessor (ResolveContext ec
, Expression right_side
)
8058 arguments
.Resolve (ec
, out dynamic);
8060 if (indexer_type
== InternalType
.Dynamic
) {
8063 var ilist
= GetIndexersForType (indexer_type
, this is BaseIndexerAccess
);
8064 if (ilist
== null) {
8065 ec
.Report
.Error (21, loc
, "Cannot apply indexing with [] to an expression of type `{0}'",
8066 TypeManager
.CSharpName (indexer_type
));
8070 var mg
= new IndexerMethodGroupExpr (ilist
, loc
);
8071 mg
.InstanceExpression
= instance_expr
;
8072 mg
= mg
.OverloadResolve (ec
, ref arguments
, false, loc
) as IndexerMethodGroupExpr
;
8077 spec
= mg
.BestIndexer ();
8081 Arguments args
= new Arguments (arguments
.Count
+ 1);
8082 if (is_base_indexer
) {
8083 ec
.Report
.Error (1972, loc
, "The indexer base access cannot be dynamically dispatched. Consider casting the dynamic arguments or eliminating the base access");
8085 args
.Add (new Argument (instance_expr
));
8087 args
.AddRange (arguments
);
8089 var expr
= new DynamicIndexBinder (args
, loc
);
8090 if (right_side
!= null)
8091 return expr
.ResolveLValue (ec
, right_side
);
8093 return expr
.Resolve (ec
);
8096 type
= spec
.MemberType
;
8097 if (type
.IsPointer
&& !ec
.IsUnsafe
)
8098 UnsafeError (ec
, loc
);
8100 MethodSpec accessor
;
8101 if (right_side
== null) {
8102 accessor
= spec
.Get
;
8104 accessor
= spec
.Set
;
8105 if (!spec
.HasSet
&& spec
.HasGet
) {
8106 ec
.Report
.SymbolRelatedToPreviousError (spec
);
8107 ec
.Report
.Error (200, loc
, "The read only property or indexer `{0}' cannot be assigned to",
8108 spec
.GetSignatureForError ());
8112 set_expr
= Convert
.ImplicitConversion (ec
, right_side
, type
, loc
);
8115 if (accessor
== null || accessor
.Kind
== MemberKind
.FakeMethod
) {
8116 ec
.Report
.SymbolRelatedToPreviousError (spec
);
8117 ec
.Report
.Error (154, loc
, "The property or indexer `{0}' cannot be used in this context because it lacks a `{1}' accessor",
8118 spec
.GetSignatureForError (), GetAccessorName (right_side
!= null));
8123 // Only base will allow this invocation to happen.
8125 if (spec
.IsAbstract
&& this is BaseIndexerAccess
) {
8126 Error_CannotCallAbstractBase (ec
, spec
.GetSignatureForError ());
8129 bool must_do_cs1540_check
;
8130 if (!IsMemberAccessible (ec
.CurrentType
, accessor
, out must_do_cs1540_check
)) {
8131 if (spec
.HasDifferentAccessibility
) {
8132 ec
.Report
.SymbolRelatedToPreviousError (accessor
);
8133 ec
.Report
.Error (271, loc
, "The property or indexer `{0}' cannot be used in this context because a `{1}' accessor is inaccessible",
8134 TypeManager
.GetFullNameSignature (spec
), GetAccessorName (right_side
!= null));
8136 ec
.Report
.SymbolRelatedToPreviousError (spec
);
8137 ErrorIsInaccesible (loc
, TypeManager
.GetFullNameSignature (spec
), ec
.Report
);
8141 instance_expr
.CheckMarshalByRefAccess (ec
);
8143 if (must_do_cs1540_check
&& (instance_expr
!= EmptyExpression
.Null
) &&
8144 !TypeManager
.IsInstantiationOfSameGenericType (instance_expr
.Type
, ec
.CurrentType
) &&
8145 !TypeManager
.IsNestedChildOf (ec
.CurrentType
, instance_expr
.Type
) &&
8146 !TypeManager
.IsSubclassOf (instance_expr
.Type
, ec
.CurrentType
)) {
8147 ec
.Report
.SymbolRelatedToPreviousError (accessor
);
8148 Error_CannotAccessProtected (ec
, loc
, spec
, instance_expr
.Type
, ec
.CurrentType
);
8152 eclass
= ExprClass
.IndexerAccess
;
8156 public override void Emit (EmitContext ec
)
8161 public void Emit (EmitContext ec
, bool leave_copy
)
8164 prepared_value
.Emit (ec
);
8166 Invocation
.EmitCall (ec
, is_base_indexer
, instance_expr
, spec
.Get
,
8167 arguments
, loc
, false, false);
8171 ec
.Emit (OpCodes
.Dup
);
8172 temp
= new LocalTemporary (Type
);
8178 // source is ignored, because we already have a copy of it from the
8179 // LValue resolution and we have already constructed a pre-cached
8180 // version of the arguments (ea.set_arguments);
8182 public void EmitAssign (EmitContext ec
, Expression source
, bool leave_copy
, bool prepare_for_load
)
8184 prepared
= prepare_for_load
;
8185 Expression
value = set_expr
;
8188 Invocation
.EmitCall (ec
, is_base_indexer
, instance_expr
, spec
.Get
,
8189 arguments
, loc
, true, false);
8191 prepared_value
= new LocalTemporary (type
);
8192 prepared_value
.Store (ec
);
8194 prepared_value
.Release (ec
);
8197 ec
.Emit (OpCodes
.Dup
);
8198 temp
= new LocalTemporary (Type
);
8201 } else if (leave_copy
) {
8202 temp
= new LocalTemporary (Type
);
8209 arguments
.Add (new Argument (value));
8211 Invocation
.EmitCall (ec
, is_base_indexer
, instance_expr
, spec
.Set
, arguments
, loc
, false, prepared
);
8219 public override string GetSignatureForError ()
8221 return spec
.GetSignatureForError ();
8225 public SLE
.Expression
MakeAssignExpression (BuilderContext ctx
)
8227 var value = new[] { set_expr.MakeExpression (ctx) }
;
8228 var args
= Arguments
.MakeExpression (arguments
, ctx
).Concat (value);
8230 return SLE
.Expression
.Block (
8231 SLE
.Expression
.Call (instance_expr
.MakeExpression (ctx
), (MethodInfo
) spec
.Set
.GetMetaInfo (), args
),
8236 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
8238 var args
= Arguments
.MakeExpression (arguments
, ctx
);
8239 return SLE
.Expression
.Call (instance_expr
.MakeExpression (ctx
), (MethodInfo
) spec
.Get
.GetMetaInfo (), args
);
8242 protected override void CloneTo (CloneContext clonectx
, Expression t
)
8244 IndexerAccess target
= (IndexerAccess
) t
;
8246 if (arguments
!= null)
8247 target
.arguments
= arguments
.Clone (clonectx
);
8249 if (instance_expr
!= null)
8250 target
.instance_expr
= instance_expr
.Clone (clonectx
);
8255 /// The base operator for method names
8257 public class BaseAccess
: Expression
{
8258 public readonly string Identifier
;
8261 public BaseAccess (string member
, Location l
)
8263 this.Identifier
= member
;
8267 public BaseAccess (string member
, TypeArguments args
, Location l
)
8273 public override Expression
CreateExpressionTree (ResolveContext ec
)
8275 throw new NotSupportedException ("ET");
8278 protected override Expression
DoResolve (ResolveContext ec
)
8280 Expression c
= CommonResolve (ec
);
8286 // MethodGroups use this opportunity to flag an error on lacking ()
8288 if (!(c
is MethodGroupExpr
))
8289 return c
.Resolve (ec
);
8293 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
8295 Expression c
= CommonResolve (ec
);
8301 // MethodGroups use this opportunity to flag an error on lacking ()
8303 if (! (c
is MethodGroupExpr
))
8304 return c
.DoResolveLValue (ec
, right_side
);
8309 Expression
CommonResolve (ResolveContext ec
)
8311 Expression member_lookup
;
8312 TypeSpec current_type
= ec
.CurrentType
;
8313 TypeSpec base_type
= current_type
.BaseType
;
8315 if (!This
.IsThisAvailable (ec
, false)) {
8317 ec
.Report
.Error (1511, loc
, "Keyword `base' is not available in a static method");
8319 ec
.Report
.Error (1512, loc
, "Keyword `base' is not available in the current context");
8324 var arity
= args
== null ? -1 : args
.Count
;
8325 member_lookup
= MemberLookup (ec
.Compiler
, ec
.CurrentType
, null, base_type
, Identifier
, arity
,
8326 MemberKind
.All
, BindingRestriction
.AccessibleOnly
, loc
);
8327 if (member_lookup
== null) {
8328 Error_MemberLookupFailed (ec
, ec
.CurrentType
, base_type
, base_type
, Identifier
, arity
,
8329 null, MemberKind
.All
, BindingRestriction
.AccessibleOnly
);
8336 left
= new TypeExpression (base_type
, loc
);
8338 left
= ec
.GetThis (loc
);
8340 MemberExpr me
= member_lookup
as MemberExpr
;
8342 if (member_lookup
is TypeExpression
){
8343 ec
.Report
.Error (582, loc
, "{0}: Can not reference a type through an expression, try `{1}' instead",
8344 Identifier
, member_lookup
.GetSignatureForError ());
8346 ec
.Report
.Error (582, loc
, "{0}: Can not reference a {1} through an expression",
8347 Identifier
, member_lookup
.ExprClassName
);
8353 me
= me
.ResolveMemberAccess (ec
, left
, null);
8358 me
.SetTypeArguments (ec
, args
);
8364 public override void Emit (EmitContext ec
)
8366 throw new Exception ("Should never be called");
8369 protected override void CloneTo (CloneContext clonectx
, Expression t
)
8371 BaseAccess target
= (BaseAccess
) t
;
8374 target
.args
= args
.Clone ();
8379 /// The base indexer operator
8381 public class BaseIndexerAccess
: IndexerAccess
{
8382 public BaseIndexerAccess (Arguments args
, Location loc
)
8383 : base (null, true, loc
)
8385 this.arguments
= args
;
8388 protected override void CommonResolve (ResolveContext ec
)
8390 instance_expr
= ec
.GetThis (loc
);
8392 current_type
= ec
.CurrentType
.BaseType
;
8393 indexer_type
= current_type
;
8396 public override Expression
CreateExpressionTree (ResolveContext ec
)
8398 MemberExpr
.Error_BaseAccessInExpressionTree (ec
, loc
);
8399 return base.CreateExpressionTree (ec
);
8404 /// This class exists solely to pass the Type around and to be a dummy
8405 /// that can be passed to the conversion functions (this is used by
8406 /// foreach implementation to typecast the object return value from
8407 /// get_Current into the proper type. All code has been generated and
8408 /// we only care about the side effect conversions to be performed
8410 /// This is also now used as a placeholder where a no-action expression
8411 /// is needed (the `New' class).
8413 public class EmptyExpression
: Expression
{
8414 public static readonly Expression Null
= new EmptyExpression ();
8416 public class OutAccess
: EmptyExpression
8418 public static readonly OutAccess Instance
= new OutAccess ();
8420 public override Expression
DoResolveLValue (ResolveContext rc
, Expression right_side
)
8422 rc
.Report
.Error (206, right_side
.Location
,
8423 "A property, indexer or dynamic member access may not be passed as `ref' or `out' parameter");
8429 public static readonly EmptyExpression LValueMemberAccess
= new EmptyExpression ();
8430 public static readonly EmptyExpression LValueMemberOutAccess
= new EmptyExpression ();
8431 public static readonly EmptyExpression UnaryAddress
= new EmptyExpression ();
8433 static EmptyExpression temp
= new EmptyExpression ();
8434 public static EmptyExpression
Grab ()
8436 EmptyExpression retval
= temp
== null ? new EmptyExpression () : temp
;
8441 public static void Release (EmptyExpression e
)
8448 // FIXME: Don't set to object
8449 type
= TypeManager
.object_type
;
8450 eclass
= ExprClass
.Value
;
8451 loc
= Location
.Null
;
8454 public EmptyExpression (TypeSpec t
)
8457 eclass
= ExprClass
.Value
;
8458 loc
= Location
.Null
;
8461 public override Expression
CreateExpressionTree (ResolveContext ec
)
8463 throw new NotSupportedException ("ET");
8466 protected override Expression
DoResolve (ResolveContext ec
)
8471 public override void Emit (EmitContext ec
)
8473 // nothing, as we only exist to not do anything.
8476 public override void EmitSideEffect (EmitContext ec
)
8481 // This is just because we might want to reuse this bad boy
8482 // instead of creating gazillions of EmptyExpressions.
8483 // (CanImplicitConversion uses it)
8485 public void SetType (TypeSpec t
)
8492 // Empty statement expression
8494 public sealed class EmptyExpressionStatement
: ExpressionStatement
8496 public static readonly EmptyExpressionStatement Instance
= new EmptyExpressionStatement ();
8498 private EmptyExpressionStatement ()
8500 loc
= Location
.Null
;
8503 public override Expression
CreateExpressionTree (ResolveContext ec
)
8508 public override void EmitStatement (EmitContext ec
)
8513 protected override Expression
DoResolve (ResolveContext ec
)
8515 eclass
= ExprClass
.Value
;
8516 type
= TypeManager
.object_type
;
8520 public override void Emit (EmitContext ec
)
8526 public class UserCast
: Expression
{
8530 public UserCast (MethodSpec method
, Expression source
, Location l
)
8532 this.method
= method
;
8533 this.source
= source
;
8534 type
= method
.ReturnType
;
8538 public Expression Source
{
8544 public override Expression
CreateExpressionTree (ResolveContext ec
)
8546 Arguments args
= new Arguments (3);
8547 args
.Add (new Argument (source
.CreateExpressionTree (ec
)));
8548 args
.Add (new Argument (new TypeOf (new TypeExpression (type
, loc
), loc
)));
8549 args
.Add (new Argument (new TypeOfMethod (method
, loc
)));
8550 return CreateExpressionFactoryCall (ec
, "Convert", args
);
8553 protected override Expression
DoResolve (ResolveContext ec
)
8555 ObsoleteAttribute oa
= method
.GetAttributeObsolete ();
8557 AttributeTester
.Report_ObsoleteMessage (oa
, GetSignatureForError (), loc
, ec
.Report
);
8559 eclass
= ExprClass
.Value
;
8563 public override void Emit (EmitContext ec
)
8566 ec
.Emit (OpCodes
.Call
, method
);
8569 public override string GetSignatureForError ()
8571 return TypeManager
.CSharpSignature (method
);
8574 public override SLE
.Expression
MakeExpression (BuilderContext ctx
)
8576 return SLE
.Expression
.Convert (source
.MakeExpression (ctx
), type
.GetMetaInfo (), (MethodInfo
) method
.GetMetaInfo ());
8581 // This class is used to "construct" the type during a typecast
8582 // operation. Since the Type.GetType class in .NET can parse
8583 // the type specification, we just use this to construct the type
8584 // one bit at a time.
8586 public class ComposedCast
: TypeExpr
{
8587 FullNamedExpression left
;
8590 public ComposedCast (FullNamedExpression left
, string dim
)
8591 : this (left
, dim
, left
.Location
)
8595 public ComposedCast (FullNamedExpression left
, string dim
, Location l
)
8602 protected override TypeExpr
DoResolveAsTypeStep (IMemberContext ec
)
8604 TypeExpr lexpr
= left
.ResolveAsTypeTerminal (ec
, false);
8608 TypeSpec ltype
= lexpr
.Type
;
8609 if ((dim
.Length
> 0) && (dim
[0] == '?')) {
8610 TypeExpr nullable
= new Nullable
.NullableType (lexpr
, loc
);
8612 nullable
= new ComposedCast (nullable
, dim
.Substring (1), loc
);
8613 return nullable
.ResolveAsTypeTerminal (ec
, false);
8616 if (dim
== "*" && !TypeManager
.VerifyUnmanaged (ec
.Compiler
, ltype
, loc
))
8619 if (dim
.Length
!= 0 && dim
[0] == '[') {
8620 if (TypeManager
.IsSpecialType (ltype
)) {
8621 ec
.Compiler
.Report
.Error (611, loc
, "Array elements cannot be of type `{0}'", TypeManager
.CSharpName (ltype
));
8625 if (ltype
.IsStatic
) {
8626 ec
.Compiler
.Report
.SymbolRelatedToPreviousError (ltype
);
8627 ec
.Compiler
.Report
.Error (719, loc
, "Array elements cannot be of static type `{0}'",
8628 TypeManager
.CSharpName (ltype
));
8633 type
= TypeManager
.GetConstructedType (ltype
, dim
);
8638 throw new InternalErrorException ("Couldn't create computed type " + ltype
+ dim
);
8640 if (type
.IsPointer
&& !ec
.IsUnsafe
){
8641 UnsafeError (ec
.Compiler
.Report
, loc
);
8644 eclass
= ExprClass
.Type
;
8648 public override string GetSignatureForError ()
8650 return left
.GetSignatureForError () + dim
;
8654 public class FixedBufferPtr
: Expression
{
8657 public FixedBufferPtr (Expression array
, TypeSpec array_type
, Location l
)
8662 type
= PointerContainer
.MakeType (array_type
);
8663 eclass
= ExprClass
.Value
;
8666 public override Expression
CreateExpressionTree (ResolveContext ec
)
8668 Error_PointerInsideExpressionTree (ec
);
8672 public override void Emit(EmitContext ec
)
8677 protected override Expression
DoResolve (ResolveContext ec
)
8680 // We are born fully resolved
8688 // This class is used to represent the address of an array, used
8689 // only by the Fixed statement, this generates "&a [0]" construct
8690 // for fixed (char *pa = a)
8692 public class ArrayPtr
: FixedBufferPtr
{
8693 TypeSpec array_type
;
8695 public ArrayPtr (Expression array
, TypeSpec array_type
, Location l
):
8696 base (array
, array_type
, l
)
8698 this.array_type
= array_type
;
8701 public override void Emit (EmitContext ec
)
8706 ec
.Emit (OpCodes
.Ldelema
, array_type
);
8711 // Encapsulates a conversion rules required for array indexes
8713 public class ArrayIndexCast
: TypeCast
8715 public ArrayIndexCast (Expression expr
)
8716 : base (expr
, TypeManager
.int32_type
)
8718 if (expr
.Type
== TypeManager
.int32_type
)
8719 throw new ArgumentException ("unnecessary array index conversion");
8722 public override Expression
CreateExpressionTree (ResolveContext ec
)
8724 using (ec
.Set (ResolveContext
.Options
.CheckedScope
)) {
8725 return base.CreateExpressionTree (ec
);
8729 public override void Emit (EmitContext ec
)
8733 var expr_type
= child
.Type
;
8735 if (expr_type
== TypeManager
.uint32_type
)
8736 ec
.Emit (OpCodes
.Conv_U
);
8737 else if (expr_type
== TypeManager
.int64_type
)
8738 ec
.Emit (OpCodes
.Conv_Ovf_I
);
8739 else if (expr_type
== TypeManager
.uint64_type
)
8740 ec
.Emit (OpCodes
.Conv_Ovf_I_Un
);
8742 throw new InternalErrorException ("Cannot emit cast to unknown array element type", type
);
8747 // Implements the `stackalloc' keyword
8749 public class StackAlloc
: Expression
{
8754 public StackAlloc (Expression type
, Expression count
, Location l
)
8761 public override Expression
CreateExpressionTree (ResolveContext ec
)
8763 throw new NotSupportedException ("ET");
8766 protected override Expression
DoResolve (ResolveContext ec
)
8768 count
= count
.Resolve (ec
);
8772 if (count
.Type
!= TypeManager
.uint32_type
){
8773 count
= Convert
.ImplicitConversionRequired (ec
, count
, TypeManager
.int32_type
, loc
);
8778 Constant c
= count
as Constant
;
8779 if (c
!= null && c
.IsNegative
) {
8780 ec
.Report
.Error (247, loc
, "Cannot use a negative size with stackalloc");
8783 if (ec
.HasAny (ResolveContext
.Options
.CatchScope
| ResolveContext
.Options
.FinallyScope
)) {
8784 ec
.Report
.Error (255, loc
, "Cannot use stackalloc in finally or catch");
8787 TypeExpr texpr
= t
.ResolveAsTypeTerminal (ec
, false);
8793 if (!TypeManager
.VerifyUnmanaged (ec
.Compiler
, otype
, loc
))
8796 type
= PointerContainer
.MakeType (otype
);
8797 eclass
= ExprClass
.Value
;
8802 public override void Emit (EmitContext ec
)
8804 int size
= GetTypeSize (otype
);
8809 ec
.Emit (OpCodes
.Sizeof
, otype
);
8813 ec
.Emit (OpCodes
.Mul_Ovf_Un
);
8814 ec
.Emit (OpCodes
.Localloc
);
8817 protected override void CloneTo (CloneContext clonectx
, Expression t
)
8819 StackAlloc target
= (StackAlloc
) t
;
8820 target
.count
= count
.Clone (clonectx
);
8821 target
.t
= t
.Clone (clonectx
);
8826 // An object initializer expression
8828 public class ElementInitializer
: Assign
8830 public readonly string Name
;
8832 public ElementInitializer (string name
, Expression initializer
, Location loc
)
8833 : base (null, initializer
, loc
)
8838 protected override void CloneTo (CloneContext clonectx
, Expression t
)
8840 ElementInitializer target
= (ElementInitializer
) t
;
8841 target
.source
= source
.Clone (clonectx
);
8844 public override Expression
CreateExpressionTree (ResolveContext ec
)
8846 Arguments args
= new Arguments (2);
8847 FieldExpr fe
= target
as FieldExpr
;
8849 args
.Add (new Argument (fe
.CreateTypeOfExpression ()));
8851 args
.Add (new Argument (((PropertyExpr
)target
).CreateSetterTypeOfExpression ()));
8853 args
.Add (new Argument (source
.CreateExpressionTree (ec
)));
8854 return CreateExpressionFactoryCall (ec
,
8855 source
is CollectionOrObjectInitializers
? "ListBind" : "Bind",
8859 protected override Expression
DoResolve (ResolveContext ec
)
8862 return EmptyExpressionStatement
.Instance
;
8864 MemberExpr me
= MemberLookupFinal (ec
, ec
.CurrentInitializerVariable
.Type
, ec
.CurrentInitializerVariable
.Type
,
8865 Name
, 0, MemberKind
.Field
| MemberKind
.Property
, BindingRestriction
.AccessibleOnly
| BindingRestriction
.InstanceOnly
, loc
) as MemberExpr
;
8871 me
.InstanceExpression
= ec
.CurrentInitializerVariable
;
8873 if (source
is CollectionOrObjectInitializers
) {
8874 Expression previous
= ec
.CurrentInitializerVariable
;
8875 ec
.CurrentInitializerVariable
= target
;
8876 source
= source
.Resolve (ec
);
8877 ec
.CurrentInitializerVariable
= previous
;
8881 eclass
= source
.eclass
;
8886 Expression expr
= base.DoResolve (ec
);
8891 // Ignore field initializers with default value
8893 Constant c
= source
as Constant
;
8894 if (c
!= null && c
.IsDefaultInitializer (type
) && target
.eclass
== ExprClass
.Variable
)
8895 return EmptyExpressionStatement
.Instance
.Resolve (ec
);
8900 protected override MemberExpr
Error_MemberLookupFailed (ResolveContext ec
, TypeSpec type
, IList
<MemberSpec
> members
)
8902 var member
= members
.First ();
8903 if (member
.Kind
!= MemberKind
.Property
&& member
.Kind
!= MemberKind
.Field
)
8904 ec
.Report
.Error (1913, loc
, "Member `{0}' cannot be initialized. An object " +
8905 "initializer may only be used for fields, or properties", TypeManager
.GetFullNameSignature (member
));
8907 ec
.Report
.Error (1914, loc
, " Static field or property `{0}' cannot be assigned in an object initializer",
8908 TypeManager
.GetFullNameSignature (member
));
8913 public override void EmitStatement (EmitContext ec
)
8915 if (source
is CollectionOrObjectInitializers
)
8918 base.EmitStatement (ec
);
8923 // A collection initializer expression
8925 class CollectionElementInitializer
: Invocation
8927 public class ElementInitializerArgument
: Argument
8929 public ElementInitializerArgument (Expression e
)
8935 sealed class AddMemberAccess
: MemberAccess
8937 public AddMemberAccess (Expression expr
, Location loc
)
8938 : base (expr
, "Add", loc
)
8942 protected override void Error_TypeDoesNotContainDefinition (ResolveContext ec
, TypeSpec type
, string name
)
8944 if (TypeManager
.HasElementType (type
))
8947 base.Error_TypeDoesNotContainDefinition (ec
, type
, name
);
8951 public CollectionElementInitializer (Expression argument
)
8952 : base (null, new Arguments (1))
8954 base.arguments
.Add (new ElementInitializerArgument (argument
));
8955 this.loc
= argument
.Location
;
8958 public CollectionElementInitializer (List
<Expression
> arguments
, Location loc
)
8959 : base (null, new Arguments (arguments
.Count
))
8961 foreach (Expression e
in arguments
)
8962 base.arguments
.Add (new ElementInitializerArgument (e
));
8967 public override Expression
CreateExpressionTree (ResolveContext ec
)
8969 Arguments args
= new Arguments (2);
8970 args
.Add (new Argument (mg
.CreateExpressionTree (ec
)));
8972 var expr_initializers
= new ArrayInitializer (arguments
.Count
, loc
);
8973 foreach (Argument a
in arguments
)
8974 expr_initializers
.Add (a
.CreateExpressionTree (ec
));
8976 args
.Add (new Argument (new ArrayCreation (
8977 CreateExpressionTypeExpression (ec
, loc
), "[]", expr_initializers
, loc
)));
8978 return CreateExpressionFactoryCall (ec
, "ElementInit", args
);
8981 protected override void CloneTo (CloneContext clonectx
, Expression t
)
8983 CollectionElementInitializer target
= (CollectionElementInitializer
) t
;
8984 if (arguments
!= null)
8985 target
.arguments
= arguments
.Clone (clonectx
);
8988 protected override Expression
DoResolve (ResolveContext ec
)
8990 base.expr
= new AddMemberAccess (ec
.CurrentInitializerVariable
, loc
);
8992 return base.DoResolve (ec
);
8997 // A block of object or collection initializers
8999 public class CollectionOrObjectInitializers
: ExpressionStatement
9001 IList
<Expression
> initializers
;
9002 bool is_collection_initialization
;
9004 public static readonly CollectionOrObjectInitializers Empty
=
9005 new CollectionOrObjectInitializers (Array
.AsReadOnly (new Expression
[0]), Location
.Null
);
9007 public CollectionOrObjectInitializers (IList
<Expression
> initializers
, Location loc
)
9009 this.initializers
= initializers
;
9013 public bool IsEmpty
{
9015 return initializers
.Count
== 0;
9019 public bool IsCollectionInitializer
{
9021 return is_collection_initialization
;
9025 protected override void CloneTo (CloneContext clonectx
, Expression target
)
9027 CollectionOrObjectInitializers t
= (CollectionOrObjectInitializers
) target
;
9029 t
.initializers
= new List
<Expression
> (initializers
.Count
);
9030 foreach (var e
in initializers
)
9031 t
.initializers
.Add (e
.Clone (clonectx
));
9034 public override Expression
CreateExpressionTree (ResolveContext ec
)
9036 var expr_initializers
= new ArrayInitializer (initializers
.Count
, loc
);
9037 foreach (Expression e
in initializers
) {
9038 Expression expr
= e
.CreateExpressionTree (ec
);
9040 expr_initializers
.Add (expr
);
9043 return new ImplicitlyTypedArrayCreation ("[]", expr_initializers
, loc
);
9046 protected override Expression
DoResolve (ResolveContext ec
)
9048 List
<string> element_names
= null;
9049 for (int i
= 0; i
< initializers
.Count
; ++i
) {
9050 Expression initializer
= (Expression
) initializers
[i
];
9051 ElementInitializer element_initializer
= initializer
as ElementInitializer
;
9054 if (element_initializer
!= null) {
9055 element_names
= new List
<string> (initializers
.Count
);
9056 element_names
.Add (element_initializer
.Name
);
9057 } else if (initializer
is CompletingExpression
){
9058 initializer
.Resolve (ec
);
9059 throw new InternalErrorException ("This line should never be reached");
9061 if (!ec
.CurrentInitializerVariable
.Type
.ImplementsInterface (TypeManager
.ienumerable_type
)) {
9062 ec
.Report
.Error (1922, loc
, "A field or property `{0}' cannot be initialized with a collection " +
9063 "object initializer because type `{1}' does not implement `{2}' interface",
9064 ec
.CurrentInitializerVariable
.GetSignatureForError (),
9065 TypeManager
.CSharpName (ec
.CurrentInitializerVariable
.Type
),
9066 TypeManager
.CSharpName (TypeManager
.ienumerable_type
));
9069 is_collection_initialization
= true;
9072 if (is_collection_initialization
!= (element_initializer
== null)) {
9073 ec
.Report
.Error (747, initializer
.Location
, "Inconsistent `{0}' member declaration",
9074 is_collection_initialization
? "collection initializer" : "object initializer");
9078 if (!is_collection_initialization
) {
9079 if (element_names
.Contains (element_initializer
.Name
)) {
9080 ec
.Report
.Error (1912, element_initializer
.Location
,
9081 "An object initializer includes more than one member `{0}' initialization",
9082 element_initializer
.Name
);
9084 element_names
.Add (element_initializer
.Name
);
9089 Expression e
= initializer
.Resolve (ec
);
9090 if (e
== EmptyExpressionStatement
.Instance
)
9091 initializers
.RemoveAt (i
--);
9093 initializers
[i
] = e
;
9096 type
= ec
.CurrentInitializerVariable
.Type
;
9097 if (is_collection_initialization
) {
9098 if (TypeManager
.HasElementType (type
)) {
9099 ec
.Report
.Error (1925, loc
, "Cannot initialize object of type `{0}' with a collection initializer",
9100 TypeManager
.CSharpName (type
));
9104 eclass
= ExprClass
.Variable
;
9108 public override void Emit (EmitContext ec
)
9113 public override void EmitStatement (EmitContext ec
)
9115 foreach (ExpressionStatement e
in initializers
)
9116 e
.EmitStatement (ec
);
9121 // New expression with element/object initializers
9123 public class NewInitialize
: New
9126 // This class serves as a proxy for variable initializer target instances.
9127 // A real variable is assigned later when we resolve left side of an
9130 sealed class InitializerTargetExpression
: Expression
, IMemoryLocation
9132 NewInitialize new_instance
;
9134 public InitializerTargetExpression (NewInitialize newInstance
)
9136 this.type
= newInstance
.type
;
9137 this.loc
= newInstance
.loc
;
9138 this.eclass
= newInstance
.eclass
;
9139 this.new_instance
= newInstance
;
9142 public override Expression
CreateExpressionTree (ResolveContext ec
)
9144 // Should not be reached
9145 throw new NotSupportedException ("ET");
9148 protected override Expression
DoResolve (ResolveContext ec
)
9153 public override Expression
DoResolveLValue (ResolveContext ec
, Expression right_side
)
9158 public override void Emit (EmitContext ec
)
9160 Expression e
= (Expression
) new_instance
.instance
;
9164 #region IMemoryLocation Members
9166 public void AddressOf (EmitContext ec
, AddressOp mode
)
9168 new_instance
.instance
.AddressOf (ec
, mode
);
9174 CollectionOrObjectInitializers initializers
;
9175 IMemoryLocation instance
;
9177 public NewInitialize (Expression requested_type
, Arguments arguments
, CollectionOrObjectInitializers initializers
, Location l
)
9178 : base (requested_type
, arguments
, l
)
9180 this.initializers
= initializers
;
9183 protected override IMemoryLocation
EmitAddressOf (EmitContext ec
, AddressOp Mode
)
9185 instance
= base.EmitAddressOf (ec
, Mode
);
9187 if (!initializers
.IsEmpty
)
9188 initializers
.Emit (ec
);
9193 protected override void CloneTo (CloneContext clonectx
, Expression t
)
9195 base.CloneTo (clonectx
, t
);
9197 NewInitialize target
= (NewInitialize
) t
;
9198 target
.initializers
= (CollectionOrObjectInitializers
) initializers
.Clone (clonectx
);
9201 public override Expression
CreateExpressionTree (ResolveContext ec
)
9203 Arguments args
= new Arguments (2);
9204 args
.Add (new Argument (base.CreateExpressionTree (ec
)));
9205 if (!initializers
.IsEmpty
)
9206 args
.Add (new Argument (initializers
.CreateExpressionTree (ec
)));
9208 return CreateExpressionFactoryCall (ec
,
9209 initializers
.IsCollectionInitializer
? "ListInit" : "MemberInit",
9213 protected override Expression
DoResolve (ResolveContext ec
)
9215 Expression e
= base.DoResolve (ec
);
9219 Expression previous
= ec
.CurrentInitializerVariable
;
9220 ec
.CurrentInitializerVariable
= new InitializerTargetExpression (this);
9221 initializers
.Resolve (ec
);
9222 ec
.CurrentInitializerVariable
= previous
;
9226 public override bool Emit (EmitContext ec
, IMemoryLocation target
)
9228 bool left_on_stack
= base.Emit (ec
, target
);
9230 if (initializers
.IsEmpty
)
9231 return left_on_stack
;
9233 LocalTemporary temp
= target
as LocalTemporary
;
9235 if (!left_on_stack
) {
9236 VariableReference vr
= target
as VariableReference
;
9238 // FIXME: This still does not work correctly for pre-set variables
9239 if (vr
!= null && vr
.IsRef
)
9240 target
.AddressOf (ec
, AddressOp
.Load
);
9242 ((Expression
) target
).Emit (ec
);
9243 left_on_stack
= true;
9246 temp
= new LocalTemporary (type
);
9253 initializers
.Emit (ec
);
9255 if (left_on_stack
) {
9260 return left_on_stack
;
9263 public override bool HasInitializer
{
9265 return !initializers
.IsEmpty
;
9270 public class NewAnonymousType
: New
9272 static readonly AnonymousTypeParameter
[] EmptyParameters
= new AnonymousTypeParameter
[0];
9274 List
<AnonymousTypeParameter
> parameters
;
9275 readonly TypeContainer parent
;
9276 AnonymousTypeClass anonymous_type
;
9278 public NewAnonymousType (List
<AnonymousTypeParameter
> parameters
, TypeContainer parent
, Location loc
)
9279 : base (null, null, loc
)
9281 this.parameters
= parameters
;
9282 this.parent
= parent
;
9285 protected override void CloneTo (CloneContext clonectx
, Expression target
)
9287 if (parameters
== null)
9290 NewAnonymousType t
= (NewAnonymousType
) target
;
9291 t
.parameters
= new List
<AnonymousTypeParameter
> (parameters
.Count
);
9292 foreach (AnonymousTypeParameter atp
in parameters
)
9293 t
.parameters
.Add ((AnonymousTypeParameter
) atp
.Clone (clonectx
));
9296 AnonymousTypeClass
CreateAnonymousType (ResolveContext ec
, IList
<AnonymousTypeParameter
> parameters
)
9298 AnonymousTypeClass type
= parent
.Module
.Compiled
.GetAnonymousType (parameters
);
9302 type
= AnonymousTypeClass
.Create (ec
.Compiler
, parent
, parameters
, loc
);
9308 type
.ResolveTypeParameters ();
9311 if (ec
.Report
.Errors
== 0)
9314 parent
.Module
.Compiled
.AddAnonymousType (type
);
9318 public override Expression
CreateExpressionTree (ResolveContext ec
)
9320 if (parameters
== null)
9321 return base.CreateExpressionTree (ec
);
9323 var init
= new ArrayInitializer (parameters
.Count
, loc
);
9324 foreach (Property p
in anonymous_type
.Properties
)
9325 init
.Add (new TypeOfMethod (MemberCache
.GetMember (type
, p
.Get
.Spec
), loc
));
9327 var ctor_args
= new ArrayInitializer (Arguments
.Count
, loc
);
9328 foreach (Argument a
in Arguments
)
9329 ctor_args
.Add (a
.CreateExpressionTree (ec
));
9331 Arguments args
= new Arguments (3);
9332 args
.Add (new Argument (method
.CreateExpressionTree (ec
)));
9333 args
.Add (new Argument (new ArrayCreation (TypeManager
.expression_type_expr
, "[]", ctor_args
, loc
)));
9334 args
.Add (new Argument (new ImplicitlyTypedArrayCreation ("[]", init
, loc
)));
9336 return CreateExpressionFactoryCall (ec
, "New", args
);
9339 protected override Expression
DoResolve (ResolveContext ec
)
9341 if (ec
.HasSet (ResolveContext
.Options
.ConstantScope
)) {
9342 ec
.Report
.Error (836, loc
, "Anonymous types cannot be used in this expression");
9346 if (parameters
== null) {
9347 anonymous_type
= CreateAnonymousType (ec
, EmptyParameters
);
9348 RequestedType
= new TypeExpression (anonymous_type
.Definition
, loc
);
9349 return base.DoResolve (ec
);
9353 Arguments
= new Arguments (parameters
.Count
);
9354 TypeExpression
[] t_args
= new TypeExpression
[parameters
.Count
];
9355 for (int i
= 0; i
< parameters
.Count
; ++i
) {
9356 Expression e
= ((AnonymousTypeParameter
) parameters
[i
]).Resolve (ec
);
9362 Arguments
.Add (new Argument (e
));
9363 t_args
[i
] = new TypeExpression (e
.Type
, e
.Location
);
9369 anonymous_type
= CreateAnonymousType (ec
, parameters
);
9370 if (anonymous_type
== null)
9373 RequestedType
= new GenericTypeExpr (anonymous_type
.Definition
, new TypeArguments (t_args
), loc
);
9374 return base.DoResolve (ec
);
9378 public class AnonymousTypeParameter
: ShimExpression
9380 public readonly string Name
;
9382 public AnonymousTypeParameter (Expression initializer
, string name
, Location loc
)
9383 : base (initializer
)
9389 public AnonymousTypeParameter (Parameter parameter
)
9390 : base (new SimpleName (parameter
.Name
, parameter
.Location
))
9392 this.Name
= parameter
.Name
;
9393 this.loc
= parameter
.Location
;
9396 public override bool Equals (object o
)
9398 AnonymousTypeParameter other
= o
as AnonymousTypeParameter
;
9399 return other
!= null && Name
== other
.Name
;
9402 public override int GetHashCode ()
9404 return Name
.GetHashCode ();
9407 protected override Expression
DoResolve (ResolveContext ec
)
9409 Expression e
= expr
.Resolve (ec
);
9413 if (e
.eclass
== ExprClass
.MethodGroup
) {
9414 Error_InvalidInitializer (ec
, e
.ExprClassName
);
9419 if (type
== TypeManager
.void_type
|| type
== TypeManager
.null_type
||
9420 type
== InternalType
.AnonymousMethod
|| type
.IsPointer
) {
9421 Error_InvalidInitializer (ec
, e
.GetSignatureForError ());
9428 protected virtual void Error_InvalidInitializer (ResolveContext ec
, string initializer
)
9430 ec
.Report
.Error (828, loc
, "An anonymous type property `{0}' cannot be initialized with `{1}'",