5 // Marek Safar <marek.safar@gmail.com>
7 // Copyright (c) 2008 Jérémie "Garuma" Laval
8 // Copyright 2011 Xamarin Inc.
10 // Permission is hereby granted, free of charge, to any person obtaining a copy
11 // of this software and associated documentation files (the "Software"), to deal
12 // in the Software without restriction, including without limitation the rights
13 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
14 // copies of the Software, and to permit persons to whom the Software is
15 // furnished to do so, subject to the following conditions:
17 // The above copyright notice and this permission notice shall be included in
18 // all copies or substantial portions of the Software.
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
21 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
22 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
23 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
24 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
25 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
32 using System.Runtime.CompilerServices;
34 namespace System.Threading.Tasks
36 [System.Diagnostics.DebuggerDisplay ("Id = {Id}, Status = {Status}, Result = {ResultAsString}")]
37 [System.Diagnostics.DebuggerTypeProxy (typeof (TaskDebuggerView))]
38 public class Task<TResult> : Task
40 static readonly TaskFactory<TResult> factory = new TaskFactory<TResult> ();
44 [System.Diagnostics.DebuggerBrowsable (System.Diagnostics.DebuggerBrowsableState.Never)]
45 public TResult Result {
50 throw new AggregateException (new TaskCanceledException (this));
51 if (Exception != null)
60 string ResultAsString {
62 if ((Status & (TaskStatus.RanToCompletion)) != 0)
65 return "<value not available>";
69 public static new TaskFactory<TResult> Factory {
75 public Task (Func<TResult> function)
76 : this (function, TaskCreationOptions.None)
81 public Task (Func<TResult> function, CancellationToken cancellationToken)
82 : this (function, cancellationToken, TaskCreationOptions.None)
87 public Task (Func<TResult> function, TaskCreationOptions creationOptions)
88 : this (function, CancellationToken.None, creationOptions)
93 public Task (Func<TResult> function, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
94 : base (TaskActionInvoker.Create (function), null, cancellationToken, creationOptions)
97 throw new ArgumentNullException ("function");
100 public Task (Func<object, TResult> function, object state)
101 : this (function, state, TaskCreationOptions.None)
106 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken)
107 : this (function, state, cancellationToken, TaskCreationOptions.None)
112 public Task (Func<object, TResult> function, object state, TaskCreationOptions creationOptions)
113 : this (function, state, CancellationToken.None, creationOptions)
118 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
119 : base (TaskActionInvoker.Create (function), state, cancellationToken, creationOptions)
121 if (function == null)
122 throw new ArgumentNullException ("function");
125 internal Task (TaskActionInvoker invoker, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions, Task parent, Task contAncestor = null)
126 : base (invoker, state, cancellationToken, creationOptions, parent, contAncestor)
130 public Task ContinueWith (Action<Task<TResult>> continuationAction)
132 return ContinueWith (continuationAction, TaskContinuationOptions.None);
135 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskContinuationOptions continuationOptions)
137 return ContinueWith (continuationAction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
140 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken)
142 return ContinueWith (continuationAction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
145 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskScheduler scheduler)
147 return ContinueWith (continuationAction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
150 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken,
151 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
153 if (continuationAction == null)
154 throw new ArgumentNullException ("continuationAction");
155 if (scheduler == null)
156 throw new ArgumentNullException ("scheduler");
158 Task t = new Task (TaskActionInvoker.Create (continuationAction),
161 GetCreationOptions (continuationOptions),
164 ContinueWithCore (t, continuationOptions, scheduler);
169 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction)
171 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
174 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, CancellationToken cancellationToken)
176 return ContinueWith<TNewResult> (continuationFunction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
179 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskContinuationOptions continuationOptions)
181 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
184 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskScheduler scheduler)
186 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
189 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction,
190 CancellationToken cancellationToken,
191 TaskContinuationOptions continuationOptions,
192 TaskScheduler scheduler)
194 if (continuationFunction == null)
195 throw new ArgumentNullException ("continuationFunction");
196 if (scheduler == null)
197 throw new ArgumentNullException ("scheduler");
199 var t = new Task<TNewResult> (TaskActionInvoker.Create (continuationFunction),
202 GetCreationOptions (continuationOptions),
205 ContinueWithCore (t, continuationOptions, scheduler);
210 internal bool TrySetResult (TResult result)
215 if (!executing.TryRelaxedSet ()) {
216 var sw = new SpinWait ();
223 Status = TaskStatus.Running;
226 Thread.MemoryBarrier ();
235 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state)
237 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
240 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken)
242 return ContinueWith (continuationAction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
245 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskContinuationOptions continuationOptions)
247 return ContinueWith (continuationAction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
250 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskScheduler scheduler)
252 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
255 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken,
256 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
258 if (continuationAction == null)
259 throw new ArgumentNullException ("continuationAction");
260 if (scheduler == null)
261 throw new ArgumentNullException ("scheduler");
263 var t = new Task (TaskActionInvoker.Create (continuationAction),
266 GetCreationOptions (continuationOptions),
270 ContinueWithCore (t, continuationOptions, scheduler);
275 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state)
277 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
280 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, CancellationToken cancellationToken)
282 return ContinueWith<TNewResult> (continuationFunction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
285 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskContinuationOptions continuationOptions)
287 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
290 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskScheduler scheduler)
292 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
295 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state,
296 CancellationToken cancellationToken,
297 TaskContinuationOptions continuationOptions,
298 TaskScheduler scheduler)
300 if (continuationFunction == null)
301 throw new ArgumentNullException ("continuationFunction");
302 if (scheduler == null)
303 throw new ArgumentNullException ("scheduler");
305 var t = new Task<TNewResult> (TaskActionInvoker.Create (continuationFunction),
308 GetCreationOptions (continuationOptions),
312 ContinueWithCore (t, continuationOptions, scheduler);
317 public new ConfiguredTaskAwaitable<TResult> ConfigureAwait (bool continueOnCapturedContext)
319 return new ConfiguredTaskAwaitable<TResult> (this, continueOnCapturedContext);
322 public new TaskAwaiter<TResult> GetAwaiter ()
324 return new TaskAwaiter<TResult> (this);