Skip to content

Commit e50b41e

Browse files
authored
Remove another reference field from async state machines (#83737)
The async state machine Task-derived type currently adds three fields: - The StateMachine - An Action field for caching any delegate created to MoveNext - The ExecutionContext to flow to the next MoveNext invocation The other pending PR gets rid of the Action field by using the unused Action field from the base Task for that purpose. This PR gets rid of the ExecutionContext field by using the unused state object field from the base Task for that purpose. The field is exposed via the public AsyncState property, so this also uses a bit from the state flags field to prevent this state object from being returned from that property. The combination of removing those two fields shaves 16 bytes off of every `async Task` state machine box on 64-bit. The only remaining field added by the state machine type is for the state machine itself, which is required.
1 parent 07afbce commit e50b41e

File tree

3 files changed

+47
-7
lines changed

3 files changed

+47
-7
lines changed

src/libraries/System.Private.CoreLib/src/System/Runtime/CompilerServices/AsyncTaskMethodBuilderT.cs

+25-3
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33

44
using System.Diagnostics;
55
using System.Diagnostics.CodeAnalysis;
6-
using System.Runtime.CompilerServices;
76
using System.Threading;
87
using System.Threading.Tasks;
98

@@ -290,12 +289,35 @@ private static void ExecutionContextCallback(object? s)
290289
private Action? _moveNextAction;
291290
/// <summary>The state machine itself.</summary>
292291
public TStateMachine? StateMachine; // mutable struct; do not make this readonly. SOS DumpAsync command depends on this name.
293-
/// <summary>Captured ExecutionContext with which to invoke <see cref="MoveNextAction"/>; may be null.</summary>
294-
public ExecutionContext? Context;
292+
293+
public AsyncStateMachineBox()
294+
{
295+
// The async state machine uses the base Task's state object field to store the captured execution context.
296+
// Ensure that state object isn't published out for others to see.
297+
Debug.Assert((m_stateFlags & (int)InternalTaskOptions.PromiseTask) != 0, "Expected state flags to already be configured.");
298+
Debug.Assert(m_stateObject is null, "Expected to be able to use the state object field for ExecutionContext.");
299+
m_stateFlags |= (int)InternalTaskOptions.HiddenState;
300+
}
295301

296302
/// <summary>A delegate to the <see cref="MoveNext()"/> method.</summary>
297303
public Action MoveNextAction => _moveNextAction ??= new Action(MoveNext);
298304

305+
/// <summary>Captured ExecutionContext with which to invoke <see cref="MoveNextAction"/>; may be null.</summary>
306+
/// <remarks>
307+
/// This uses the base Task.m_stateObject field to store the context, as that field is otherwise unused for state machine boxes.
308+
/// This *must* not be set to anything other than null or an ExecutionContext, or it will result in a type safety hole.
309+
/// We also don't want this ExecutionContext exposed out to consumers of the Task via Task.AsyncState, so
310+
/// the ctor sets the HiddenState option to prevent this from leaking out.
311+
/// </remarks>
312+
public ref ExecutionContext? Context
313+
{
314+
get
315+
{
316+
Debug.Assert(m_stateObject is null || m_stateObject is ExecutionContext, $"{nameof(m_stateObject)} must only be for ExecutionContext but contained {m_stateObject}.");
317+
return ref Unsafe.As<object?, ExecutionContext?>(ref m_stateObject);
318+
}
319+
}
320+
299321
internal sealed override void ExecuteFromThreadPool(Thread threadPoolThread) => MoveNext(threadPoolThread);
300322

301323
/// <summary>Calls MoveNext on <see cref="StateMachine"/></summary>

src/libraries/System.Private.CoreLib/src/System/Threading/Tasks/Task.cs

+9-4
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@
99
//
1010
// =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-
1111

12-
using System.Buffers;
1312
using System.Collections.Generic;
1413
using System.Diagnostics;
1514
using System.Diagnostics.CodeAnalysis;
@@ -120,7 +119,7 @@ public class Task : IAsyncResult, IDisposable
120119
[ThreadStatic]
121120
internal static Task? t_currentTask; // The currently executing task.
122121

123-
internal static int s_taskIdCounter; // static counter used to generate unique task IDs
122+
private static int s_taskIdCounter; // static counter used to generate unique task IDs
124123

125124
private int m_taskId; // this task's unique ID. initialized only if it is ever requested
126125

@@ -132,7 +131,7 @@ public class Task : IAsyncResult, IDisposable
132131
// the completion event which will be set when the Future class calls Finish().
133132
// But the event would now be signalled if Cancel() is called
134133

135-
internal object? m_stateObject; // A state object that can be optionally supplied, passed to action.
134+
private protected object? m_stateObject; // A state object that can be optionally supplied, passed to action.
136135
internal TaskScheduler? m_taskScheduler; // The task scheduler this task runs under.
137136

138137
internal volatile int m_stateFlags; // SOS DumpAsync command depends on this name
@@ -566,6 +565,7 @@ internal void TaskConstructorCore(Delegate? action, object? state, CancellationT
566565
int illegalInternalOptions =
567566
(int)(internalOptions &
568567
~(InternalTaskOptions.PromiseTask |
568+
InternalTaskOptions.HiddenState |
569569
InternalTaskOptions.ContinuationTask |
570570
InternalTaskOptions.LazyCancellation |
571571
InternalTaskOptions.QueuedByRuntime));
@@ -1446,7 +1446,7 @@ WaitHandle IAsyncResult.AsyncWaitHandle
14461446
/// Gets the state object supplied when the <see cref="Task">Task</see> was created,
14471447
/// or null if none was supplied.
14481448
/// </summary>
1449-
public object? AsyncState => m_stateObject;
1449+
public object? AsyncState => (m_stateFlags & (int)InternalTaskOptions.HiddenState) == 0 ? m_stateObject : null;
14501450

14511451
/// <summary>
14521452
/// Gets an indication of whether the asynchronous operation completed synchronously.
@@ -6717,6 +6717,11 @@ internal enum InternalTaskOptions
67176717
ContinuationTask = 0x0200,
67186718
PromiseTask = 0x0400,
67196719

6720+
/// <summary>
6721+
/// The state object should not be returned from the AsyncState property.
6722+
/// </summary>
6723+
HiddenState = 0x0800,
6724+
67206725
/// <summary>
67216726
/// Store the presence of TaskContinuationOptions.LazyCancellation, since it does not directly
67226727
/// translate into any TaskCreationOptions.

src/libraries/System.Threading.Tasks/tests/System.Runtime.CompilerServices/AsyncTaskMethodBuilderTests.cs

+13
Original file line numberDiff line numberDiff line change
@@ -662,6 +662,19 @@ public void AsyncTaskMethodBuilder_Completed_RemovedFromTracking()
662662
}).Dispose();
663663
}
664664

665+
[Fact]
666+
public void AsyncTaskMethodBuilder_NullStateEvenAfterSuspend()
667+
{
668+
Task t = AwaitSomething();
669+
Assert.Null(t.AsyncState);
670+
671+
static async Task AwaitSomething()
672+
{
673+
Assert.NotNull(ExecutionContext.Capture());
674+
await new TaskCompletionSource().Task;
675+
}
676+
}
677+
665678
#region Helper Methods / Classes
666679

667680
[MethodImpl(MethodImplOptions.NoInlining)]

0 commit comments

Comments
 (0)