2 // async.cs: Asynchronous functions
5 // Marek Safar (marek.safar@gmail.com)
7 // Dual licensed under the terms of the MIT X11 or GNU GPL
9 // Copyright 2011 Novell, Inc.
10 // Copyright 2011 Xamarin Inc.
14 using System.Collections.Generic;
16 using System.Collections;
19 using IKVM.Reflection.Emit;
21 using System.Reflection.Emit;
26 public class Await : ExpressionStatement
31 public Await (Expression expr, Location loc)
37 public Expression Expr {
43 protected override void CloneTo (CloneContext clonectx, Expression target)
45 var t = (Await) target;
47 t.expr = expr.Clone (clonectx);
50 public override Expression CreateExpressionTree (ResolveContext ec)
52 throw new NotImplementedException ("ET");
55 public override bool ContainsEmitWithAwait ()
60 protected override Expression DoResolve (ResolveContext rc)
62 if (rc.HasSet (ResolveContext.Options.LockScope)) {
63 rc.Report.Error (1996, loc,
64 "The `await' operator cannot be used in the body of a lock statement");
68 rc.Report.Error (4004, loc,
69 "The `await' operator cannot be used in an unsafe context");
72 var bc = (BlockContext) rc;
74 stmt = new AwaitStatement (expr, loc);
75 if (!stmt.Resolve (bc))
78 type = stmt.ResultType;
79 eclass = ExprClass.Variable;
83 public override void Emit (EmitContext ec)
85 stmt.EmitPrologue (ec);
87 using (ec.With (BuilderContext.Options.OmitDebugInfo, true)) {
92 public override Expression EmitToField (EmitContext ec)
94 stmt.EmitPrologue (ec);
95 return stmt.GetResultExpression (ec);
98 public void EmitAssign (EmitContext ec, FieldExpr field)
100 stmt.EmitPrologue (ec);
101 field.InstanceExpression.Emit (ec);
105 public override void EmitStatement (EmitContext ec)
107 stmt.EmitStatement (ec);
110 public override object Accept (StructuralVisitor visitor)
112 return visitor.Visit (this);
116 class AwaitStatement : YieldStatement<AsyncInitializer>
118 sealed class AwaitableMemberAccess : MemberAccess
120 public AwaitableMemberAccess (Expression expr)
121 : base (expr, "GetAwaiter")
125 protected override void Error_TypeDoesNotContainDefinition (ResolveContext rc, TypeSpec type, string name)
127 Error_OperatorCannotBeApplied (rc, type);
130 protected override void Error_OperatorCannotBeApplied (ResolveContext rc, TypeSpec type)
132 var invocation = LeftExpression as Invocation;
133 if (invocation != null && invocation.MethodGroup != null && (invocation.MethodGroup.BestCandidate.Modifiers & Modifiers.ASYNC) != 0) {
134 rc.Report.Error (4008, loc, "Cannot await void method `{0}'. Consider changing method return type to `Task'",
135 invocation.GetSignatureForError ());
137 rc.Report.Error (4001, loc, "Cannot await `{0}' expression", type.GetSignatureForError ());
142 sealed class GetResultInvocation : Invocation
144 public GetResultInvocation (MethodGroupExpr mge, Arguments arguments)
145 : base (null, arguments)
148 type = mg.BestCandidateReturnType;
151 public override Expression EmitToField (EmitContext ec)
158 PropertySpec is_completed;
159 MethodSpec get_result;
161 TypeSpec result_type;
163 public AwaitStatement (Expression expr, Location loc)
172 return is_completed == null;
176 public TypeSpec ResultType {
184 protected override void DoEmit (EmitContext ec)
186 GetResultExpression (ec).Emit (ec);
189 public Expression GetResultExpression (EmitContext ec)
191 var fe_awaiter = new FieldExpr (awaiter, loc);
192 fe_awaiter.InstanceExpression = new CompilerGeneratedThis (ec.CurrentType, loc);
195 // result = awaiter.GetResult ();
198 var rc = new ResolveContext (ec.MemberContext);
199 return new Invocation (new MemberAccess (fe_awaiter, "GetResult"), new Arguments (0)).Resolve (rc);
201 var mg_result = MethodGroupExpr.CreatePredefined (get_result, fe_awaiter.Type, loc);
202 mg_result.InstanceExpression = fe_awaiter;
204 return new GetResultInvocation (mg_result, new Arguments (0));
208 public void EmitPrologue (EmitContext ec)
210 var fe_awaiter = new FieldExpr (awaiter, loc);
211 fe_awaiter.InstanceExpression = new CompilerGeneratedThis (ec.CurrentType, loc);
214 // awaiter = expr.GetAwaiter ();
216 fe_awaiter.EmitAssign (ec, expr, false, false);
218 Label skip_continuation = ec.DefineLabel ();
220 Expression completed_expr;
222 var rc = new ResolveContext (ec.MemberContext);
224 Arguments dargs = new Arguments (1);
225 dargs.Add (new Argument (fe_awaiter));
226 completed_expr = new DynamicMemberBinder ("IsCompleted", dargs, loc).Resolve (rc);
228 dargs = new Arguments (1);
229 dargs.Add (new Argument (completed_expr));
230 completed_expr = new DynamicConversion (ec.Module.Compiler.BuiltinTypes.Bool, 0, dargs, loc).Resolve (rc);
232 var pe = PropertyExpr.CreatePredefined (is_completed, loc);
233 pe.InstanceExpression = fe_awaiter;
237 completed_expr.EmitBranchable (ec, skip_continuation, true);
242 // The stack has to be empty before calling await continuation. We handle this
243 // by lifting values which would be left on stack into class fields. The process
244 // is quite complicated and quite hard to test because any expression can possibly
245 // leave a value on the stack.
247 // Following assert fails when some of expression called before is missing EmitToField
248 // or parent expression fails to find await in children expressions
250 ec.AssertEmptyStack ();
252 var storey = (AsyncTaskStorey) machine_initializer.Storey;
254 storey.EmitAwaitOnCompletedDynamic (ec, fe_awaiter);
256 storey.EmitAwaitOnCompleted (ec, fe_awaiter);
260 machine_initializer.EmitLeave (ec, unwind_protect);
262 ec.MarkLabel (resume_point);
263 ec.MarkLabel (skip_continuation);
266 public void EmitStatement (EmitContext ec)
271 if (ResultType.Kind != MemberKind.Void) {
272 var storey = (AsyncTaskStorey) machine_initializer.Storey;
274 if (storey.HoistedReturn != null)
275 storey.HoistedReturn.EmitAssign (ec);
277 ec.Emit (OpCodes.Pop);
281 void Error_WrongAwaiterPattern (ResolveContext rc, TypeSpec awaiter)
283 rc.Report.Error (4011, loc, "The awaiter type `{0}' must have suitable IsCompleted and GetResult members",
284 awaiter.GetSignatureForError ());
287 public override bool Resolve (BlockContext bc)
289 if (bc.CurrentBlock is Linq.QueryBlock) {
290 bc.Report.Error (1995, loc,
291 "The `await' operator may only be used in a query expression within the first collection expression of the initial `from' clause or within the collection expression of a `join' clause");
295 if (!base.Resolve (bc))
298 Arguments args = new Arguments (0);
303 // The await expression is of dynamic type
305 if (type.BuiltinType == BuiltinTypeSpec.Type.Dynamic) {
308 awaiter = ((AsyncTaskStorey) machine_initializer.Storey).AddAwaiter (type, loc);
310 expr = new Invocation (new MemberAccess (expr, "GetAwaiter"), args).Resolve (bc);
315 // Check whether the expression is awaitable
317 Expression ama = new AwaitableMemberAccess (expr).Resolve (bc);
321 var errors_printer = new SessionReportPrinter ();
322 var old = bc.Report.SetPrinter (errors_printer);
323 ama = new Invocation (ama, args).Resolve (bc);
324 bc.Report.SetPrinter (old);
326 if (errors_printer.ErrorsCount > 0 || !MemberAccess.IsValidDotExpression (ama.Type)) {
327 bc.Report.Error (1986, expr.Location,
328 "The `await' operand type `{0}' must have suitable GetAwaiter method",
329 expr.Type.GetSignatureForError ());
334 var awaiter_type = ama.Type;
335 awaiter = ((AsyncTaskStorey) machine_initializer.Storey).AddAwaiter (awaiter_type, loc);
340 // Predefined: bool IsCompleted { get; }
342 is_completed = MemberCache.FindMember (awaiter_type, MemberFilter.Property ("IsCompleted", bc.Module.Compiler.BuiltinTypes.Bool),
343 BindingRestriction.InstanceOnly) as PropertySpec;
345 if (is_completed == null || !is_completed.HasGet) {
346 Error_WrongAwaiterPattern (bc, awaiter_type);
351 // Predefined: GetResult ()
353 // The method return type is also result type of await expression
355 get_result = MemberCache.FindMember (awaiter_type, MemberFilter.Method ("GetResult", 0,
356 ParametersCompiled.EmptyReadOnlyParameters, null),
357 BindingRestriction.InstanceOnly) as MethodSpec;
359 if (get_result == null) {
360 Error_WrongAwaiterPattern (bc, awaiter_type);
365 // Predefined: INotifyCompletion.OnCompleted (System.Action)
367 var nc = bc.Module.PredefinedTypes.INotifyCompletion;
368 if (nc.Define () && !awaiter_type.ImplementsInterface (nc.TypeSpec, false)) {
369 bc.Report.Error (4027, loc, "The awaiter type `{0}' must implement interface `{1}'",
370 awaiter_type.GetSignatureForError (), nc.GetSignatureForError ());
374 result_type = get_result.ReturnType;
380 public class AsyncInitializer : StateMachineInitializer
382 TypeInferenceContext return_inference;
384 public AsyncInitializer (ParametersBlock block, TypeDefinition host, TypeSpec returnType)
385 : base (block, host, returnType)
391 public override string ContainerType {
393 return "async state machine block";
397 public override bool IsIterator {
403 public Block OriginalBlock {
409 public TypeInferenceContext ReturnTypeInference {
411 return return_inference;
417 public static void Create (IMemberContext context, ParametersBlock block, ParametersCompiled parameters, TypeDefinition host, TypeSpec returnType, Location loc)
419 for (int i = 0; i < parameters.Count; i++) {
420 Parameter p = parameters[i];
421 Parameter.Modifier mod = p.ModFlags;
422 if ((mod & Parameter.Modifier.RefOutMask) != 0) {
423 host.Compiler.Report.Error (1988, p.Location,
424 "Async methods cannot have ref or out parameters");
428 if (p is ArglistParameter) {
429 host.Compiler.Report.Error (4006, p.Location,
430 "__arglist is not allowed in parameter list of async methods");
434 if (parameters.Types[i].IsPointer) {
435 host.Compiler.Report.Error (4005, p.Location,
436 "Async methods cannot have unsafe parameters");
441 if (!block.HasAwait) {
442 host.Compiler.Report.Warning (1998, 1, loc,
443 "Async block lacks `await' operator and will run synchronously");
446 block.WrapIntoAsyncTask (context, host, returnType);
449 protected override BlockContext CreateBlockContext (ResolveContext rc)
451 var ctx = base.CreateBlockContext (rc);
452 var lambda = rc.CurrentAnonymousMethod as LambdaMethod;
454 return_inference = lambda.ReturnTypeInference;
456 ctx.StartFlowBranching (this, rc.CurrentBranching);
460 public override Expression CreateExpressionTree (ResolveContext ec)
462 return base.CreateExpressionTree (ec);
465 public override void Emit (EmitContext ec)
467 throw new NotImplementedException ();
470 protected override void EmitMoveNextEpilogue (EmitContext ec)
472 var storey = (AsyncTaskStorey) Storey;
473 storey.EmitSetResult (ec);
476 public override void EmitStatement (EmitContext ec)
478 var storey = (AsyncTaskStorey) Storey;
479 storey.EmitInitializer (ec);
480 ec.Emit (OpCodes.Ret);
484 class AsyncTaskStorey : StateMachine
488 readonly TypeSpec return_type;
489 MethodSpec set_result;
490 MethodSpec set_exception;
491 MethodSpec builder_factory;
492 MethodSpec builder_start;
494 LocalVariable hoisted_return;
496 Dictionary<TypeSpec, List<StackField>> stack_fields;
498 public AsyncTaskStorey (IMemberContext context, AsyncInitializer initializer, TypeSpec type)
499 : base (initializer.OriginalBlock, initializer.Host, context.CurrentMemberDefinition as MemberBase, context.CurrentTypeParameters, "async", MemberKind.Class)
506 public LocalVariable HoistedReturn {
508 return hoisted_return;
512 public TypeSpec ReturnType {
518 public PropertySpec Task {
526 public Field AddAwaiter (TypeSpec type, Location loc)
528 return AddCapturedVariable ("$awaiter" + awaiters++.ToString ("X"), type);
531 public StackField AddCapturedLocalVariable (TypeSpec type)
534 type = mutator.Mutate (type);
536 List<StackField> existing_fields = null;
537 if (stack_fields == null) {
538 stack_fields = new Dictionary<TypeSpec, List<StackField>> ();
539 } else if (stack_fields.TryGetValue (type, out existing_fields)) {
540 foreach (var f in existing_fields) {
542 f.CanBeReused = false;
548 const Modifiers mod = Modifiers.COMPILER_GENERATED | Modifiers.PRIVATE;
549 var field = new StackField (this, new TypeExpression (type, Location), mod, new MemberName ("$stack" + locals_captured++.ToString ("X"), Location));
554 if (existing_fields == null) {
555 existing_fields = new List<StackField> ();
556 stack_fields.Add (type, existing_fields);
559 existing_fields.Add (field);
564 protected override bool DoDefineMembers ()
566 PredefinedType builder_type;
567 PredefinedMember<MethodSpec> bf;
568 PredefinedMember<MethodSpec> bs;
569 PredefinedMember<MethodSpec> sr;
570 PredefinedMember<MethodSpec> se;
571 PredefinedMember<MethodSpec> sm;
572 bool has_task_return_type = false;
573 var pred_members = Module.PredefinedMembers;
575 if (return_type.Kind == MemberKind.Void) {
576 builder_type = Module.PredefinedTypes.AsyncVoidMethodBuilder;
577 bf = pred_members.AsyncVoidMethodBuilderCreate;
578 bs = pred_members.AsyncVoidMethodBuilderStart;
579 sr = pred_members.AsyncVoidMethodBuilderSetResult;
580 se = pred_members.AsyncVoidMethodBuilderSetException;
581 sm = pred_members.AsyncVoidMethodBuilderSetStateMachine;
582 } else if (return_type == Module.PredefinedTypes.Task.TypeSpec) {
583 builder_type = Module.PredefinedTypes.AsyncTaskMethodBuilder;
584 bf = pred_members.AsyncTaskMethodBuilderCreate;
585 bs = pred_members.AsyncTaskMethodBuilderStart;
586 sr = pred_members.AsyncTaskMethodBuilderSetResult;
587 se = pred_members.AsyncTaskMethodBuilderSetException;
588 sm = pred_members.AsyncTaskMethodBuilderSetStateMachine;
589 task = pred_members.AsyncTaskMethodBuilderTask.Get ();
591 builder_type = Module.PredefinedTypes.AsyncTaskMethodBuilderGeneric;
592 bf = pred_members.AsyncTaskMethodBuilderGenericCreate;
593 bs = pred_members.AsyncTaskMethodBuilderGenericStart;
594 sr = pred_members.AsyncTaskMethodBuilderGenericSetResult;
595 se = pred_members.AsyncTaskMethodBuilderGenericSetException;
596 sm = pred_members.AsyncTaskMethodBuilderGenericSetStateMachine;
597 task = pred_members.AsyncTaskMethodBuilderGenericTask.Get ();
598 has_task_return_type = true;
601 set_result = sr.Get ();
602 set_exception = se.Get ();
603 builder_factory = bf.Get ();
604 builder_start = bs.Get ();
606 var istate_machine = Module.PredefinedTypes.IAsyncStateMachine;
607 var set_statemachine = sm.Get ();
609 if (!builder_type.Define () || !istate_machine.Define () || set_result == null || builder_factory == null ||
610 set_exception == null || set_statemachine == null || builder_start == null ||
611 !Module.PredefinedTypes.INotifyCompletion.Define ()) {
612 Report.Error (1993, Location,
613 "Cannot find compiler required types for asynchronous functions support. Are you targeting the wrong framework version?");
614 return base.DoDefineMembers ();
617 var bt = builder_type.TypeSpec;
620 // Inflate generic Task types
622 if (has_task_return_type) {
623 var task_return_type = return_type.TypeArguments;
625 task_return_type = mutator.Mutate (task_return_type);
627 bt = bt.MakeGenericType (Module, task_return_type);
628 set_result = MemberCache.GetMember (bt, set_result);
629 set_exception = MemberCache.GetMember (bt, set_exception);
630 set_statemachine = MemberCache.GetMember (bt, set_statemachine);
633 task = MemberCache.GetMember (bt, task);
636 builder = AddCompilerGeneratedField ("$builder", new TypeExpression (bt, Location));
638 var set_state_machine = new Method (this, new TypeExpression (Compiler.BuiltinTypes.Void, Location),
639 Modifiers.COMPILER_GENERATED | Modifiers.DEBUGGER_HIDDEN | Modifiers.PUBLIC,
640 new MemberName ("SetStateMachine"),
641 ParametersCompiled.CreateFullyResolved (
642 new Parameter (new TypeExpression (istate_machine.TypeSpec, Location), "stateMachine", Parameter.Modifier.NONE, null, Location),
643 istate_machine.TypeSpec),
646 ToplevelBlock block = new ToplevelBlock (Compiler, set_state_machine.ParameterInfo, Location);
647 block.IsCompilerGenerated = true;
648 set_state_machine.Block = block;
650 Members.Add (set_state_machine);
652 if (!base.DoDefineMembers ())
656 // Fabricates SetStateMachine method
658 // public void SetStateMachine (IAsyncStateMachine stateMachine)
660 // $builder.SetStateMachine (stateMachine);
663 var mg = MethodGroupExpr.CreatePredefined (set_statemachine, bt, Location);
664 mg.InstanceExpression = new FieldExpr (builder, Location);
666 var param_reference = block.GetParameterReference (0, Location);
667 param_reference.Type = istate_machine.TypeSpec;
668 param_reference.eclass = ExprClass.Variable;
670 var args = new Arguments (1);
671 args.Add (new Argument (param_reference));
672 set_state_machine.Block.AddStatement (new StatementExpression (new Invocation (mg, args)));
674 if (has_task_return_type) {
675 hoisted_return = LocalVariable.CreateCompilerGenerated (bt.TypeArguments[0], StateMachineMethod.Block, Location);
681 public void EmitAwaitOnCompletedDynamic (EmitContext ec, FieldExpr awaiter)
683 var critical = Module.PredefinedTypes.ICriticalNotifyCompletion;
684 if (!critical.Define ()) {
685 throw new NotImplementedException ();
688 var temp_critical = new LocalTemporary (critical.TypeSpec);
689 var label_critical = ec.DefineLabel ();
690 var label_end = ec.DefineLabel ();
693 // Special path for dynamic awaiters
695 // var awaiter = this.$awaiter as ICriticalNotifyCompletion;
696 // if (awaiter == null) {
697 // var completion = (INotifyCompletion) this.$awaiter;
698 // this.$builder.AwaitOnCompleted (ref completion, ref this);
700 // this.$builder.AwaitUnsafeOnCompleted (ref awaiter, ref this);
704 ec.Emit (OpCodes.Isinst, critical.TypeSpec);
705 temp_critical.Store (ec);
706 temp_critical.Emit (ec);
707 ec.Emit (OpCodes.Brtrue_S, label_critical);
709 var temp = new LocalTemporary (Module.PredefinedTypes.INotifyCompletion.TypeSpec);
711 ec.Emit (OpCodes.Castclass, temp.Type);
713 EmitOnCompleted (ec, temp, false);
715 ec.Emit (OpCodes.Br_S, label_end);
717 ec.MarkLabel (label_critical);
719 EmitOnCompleted (ec, temp_critical, true);
721 ec.MarkLabel (label_end);
723 temp_critical.Release (ec);
726 public void EmitAwaitOnCompleted (EmitContext ec, FieldExpr awaiter)
728 bool unsafe_version = false;
729 if (Module.PredefinedTypes.ICriticalNotifyCompletion.Define ()) {
730 unsafe_version = awaiter.Type.ImplementsInterface (Module.PredefinedTypes.ICriticalNotifyCompletion.TypeSpec, false);
733 EmitOnCompleted (ec, awaiter, unsafe_version);
736 void EmitOnCompleted (EmitContext ec, Expression awaiter, bool unsafeVersion)
738 var pm = Module.PredefinedMembers;
739 PredefinedMember<MethodSpec> predefined;
740 bool has_task_return_type = false;
741 if (return_type.Kind == MemberKind.Void) {
742 predefined = unsafeVersion ? pm.AsyncVoidMethodBuilderOnCompletedUnsafe : pm.AsyncVoidMethodBuilderOnCompleted;
743 } else if (return_type == Module.PredefinedTypes.Task.TypeSpec) {
744 predefined = unsafeVersion ? pm.AsyncTaskMethodBuilderOnCompletedUnsafe : pm.AsyncTaskMethodBuilderOnCompleted;
746 predefined = unsafeVersion ? pm.AsyncTaskMethodBuilderGenericOnCompletedUnsafe : pm.AsyncTaskMethodBuilderGenericOnCompleted;
747 has_task_return_type = true;
750 var on_completed = predefined.Resolve (Location);
751 if (on_completed == null)
754 if (has_task_return_type)
755 on_completed = MemberCache.GetMember<MethodSpec> (set_result.DeclaringType, on_completed);
757 on_completed = on_completed.MakeGenericMethod (this, awaiter.Type, ec.CurrentType);
759 var mg = MethodGroupExpr.CreatePredefined (on_completed, on_completed.DeclaringType, Location);
760 mg.InstanceExpression = new FieldExpr (builder, Location) {
761 InstanceExpression = new CompilerGeneratedThis (ec.CurrentType, Location)
764 // TODO: CompilerGeneratedThis is enough for structs
765 var temp_this = new LocalTemporary (CurrentType);
766 temp_this.EmitAssign (ec, new CompilerGeneratedThis (CurrentType, Location), false, false);
768 var args = new Arguments (2);
769 args.Add (new Argument (awaiter, Argument.AType.Ref));
770 args.Add (new Argument (temp_this, Argument.AType.Ref));
771 mg.EmitCall (ec, args);
773 temp_this.Release (ec);
776 public void EmitInitializer (EmitContext ec)
779 // Some predefined types are missing
784 var instance = (TemporaryVariableReference) Instance;
785 var builder_field = builder.Spec;
786 if (MemberName.Arity > 0) {
787 builder_field = MemberCache.GetMember (instance.Type, builder_field);
791 // Inflated factory method when task is of generic type
793 if (builder_factory.DeclaringType.IsGeneric) {
794 var task_return_type = return_type.TypeArguments;
795 var bt = builder_factory.DeclaringType.MakeGenericType (Module, task_return_type);
796 builder_factory = MemberCache.GetMember (bt, builder_factory);
797 builder_start = MemberCache.GetMember (bt, builder_start);
801 // stateMachine.$builder = AsyncTaskMethodBuilder<{task-type}>.Create();
803 instance.Emit (ec); // .AddressOf (ec, AddressOp.Store);
804 ec.Emit (OpCodes.Call, builder_factory);
805 ec.Emit (OpCodes.Stfld, builder_field);
808 // stateMachine.$builder.Start<{storey-type}>(ref stateMachine);
810 instance.Emit (ec); //.AddressOf (ec, AddressOp.Store);
811 ec.Emit (OpCodes.Ldflda, builder_field);
813 ec.Emit (OpCodes.Dup);
814 instance.AddressOf (ec, AddressOp.Store);
815 ec.Emit (OpCodes.Call, builder_start.MakeGenericMethod (Module, instance.Type));
818 // Emits return stateMachine.$builder.Task;
821 var task_get = Task.Get;
823 if (MemberName.Arity > 0) {
824 task_get = MemberCache.GetMember (builder_field.MemberType, task_get);
827 var pe_task = new PropertyExpr (Task, Location) {
828 InstanceExpression = EmptyExpression.Null, // Comes from the dup above
836 public void EmitSetException (EmitContext ec, LocalVariableReference exceptionVariable)
839 // $builder.SetException (Exception)
841 var mg = MethodGroupExpr.CreatePredefined (set_exception, set_exception.DeclaringType, Location);
842 mg.InstanceExpression = new FieldExpr (builder, Location) {
843 InstanceExpression = new CompilerGeneratedThis (ec.CurrentType, Location)
846 Arguments args = new Arguments (1);
847 args.Add (new Argument (exceptionVariable));
849 mg.EmitCall (ec, args);
852 public void EmitSetResult (EmitContext ec)
855 // $builder.SetResult ();
856 // $builder.SetResult<return-type> (value);
858 var mg = MethodGroupExpr.CreatePredefined (set_result, set_result.DeclaringType, Location);
859 mg.InstanceExpression = new FieldExpr (builder, Location) {
860 InstanceExpression = new CompilerGeneratedThis (ec.CurrentType, Location)
864 if (hoisted_return == null) {
865 args = new Arguments (0);
867 args = new Arguments (1);
868 args.Add (new Argument (new LocalVariableReference (hoisted_return, Location)));
871 mg.EmitCall (ec, args);
874 protected override TypeSpec[] ResolveBaseTypes (out FullNamedExpression base_class)
876 base_type = Compiler.BuiltinTypes.Object; // ValueType;
879 var istate_machine = Module.PredefinedTypes.IAsyncStateMachine;
880 if (istate_machine.Define ()) {
881 return new[] { istate_machine.TypeSpec };
888 class StackField : Field
890 public StackField (TypeDefinition parent, FullNamedExpression type, Modifiers mod, MemberName name)
891 : base (parent, type, mod, name, null)
895 public bool CanBeReused { get; set; }
898 class StackFieldExpr : FieldExpr
900 public StackFieldExpr (Field field)
901 : base (field, Location.Null)
905 public override void Emit (EmitContext ec)
909 var field = (StackField) spec.MemberDefinition;
910 field.CanBeReused = true;