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 internal static new readonly Task<TResult> Canceled = new Task<TResult> (TaskStatus.Canceled);
41 static readonly TaskFactory<TResult> factory = new TaskFactory<TResult> ();
44 internal Func<object, TResult> function;
47 [System.Diagnostics.DebuggerBrowsable (System.Diagnostics.DebuggerBrowsableState.Never)]
48 public TResult Result {
52 else if (Exception != null)
61 string ResultAsString {
63 if ((Status & (TaskStatus.RanToCompletion)) != 0)
66 return "<value not available>";
70 public static new TaskFactory<TResult> Factory {
76 public Task (Func<TResult> function) : this (function, TaskCreationOptions.None)
81 public Task (Func<TResult> function, CancellationToken cancellationToken)
82 : this (function == null ? (Func<object, TResult>)null : (o) => function(), null, cancellationToken, TaskCreationOptions.None)
87 public Task (Func<TResult> function, TaskCreationOptions creationOptions)
88 : this (function == null ? (Func<object, TResult>)null : (o) => function(), null, CancellationToken.None, creationOptions)
93 public Task (Func<TResult> function, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
94 : this (function == null ? (Func<object, TResult>)null : (o) => function(), null, cancellationToken, creationOptions)
99 public Task (Func<object, TResult> function, object state) : this (function, state, TaskCreationOptions.None)
104 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken)
105 : this (function, state, cancellationToken, TaskCreationOptions.None)
110 public Task (Func<object, TResult> function, object state, TaskCreationOptions creationOptions)
111 : this (function, state, CancellationToken.None, creationOptions)
116 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
117 : base (delegate { }, state, cancellationToken, creationOptions)
119 if (function == null)
120 throw new ArgumentNullException ("function");
122 this.function = function;
126 internal Task (Func<object, TResult> function,
128 CancellationToken cancellationToken,
129 TaskCreationOptions creationOptions,
131 : base (emptyAction, state, cancellationToken, creationOptions, parent)
133 this.function = function;
138 internal Task (TaskStatus status)
143 internal override void InnerInvoke ()
145 if (function != null)
146 value = function (state);
152 public Task ContinueWith (Action<Task<TResult>> continuationAction)
154 return ContinueWith (continuationAction, TaskContinuationOptions.None);
157 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskContinuationOptions continuationOptions)
159 return ContinueWith (continuationAction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
162 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken)
164 return ContinueWith (continuationAction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
167 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskScheduler scheduler)
169 return ContinueWith (continuationAction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
172 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken,
173 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
175 if (continuationAction == null)
176 throw new ArgumentNullException ("continuationAction");
177 if (scheduler == null)
178 throw new ArgumentNullException ("scheduler");
180 Task t = new Task (l => continuationAction ((Task<TResult>)l),
183 GetCreationOptions (continuationOptions),
185 ContinueWithCore (t, continuationOptions, scheduler);
190 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction)
192 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
195 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, CancellationToken cancellationToken)
197 return ContinueWith<TNewResult> (continuationFunction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
200 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskContinuationOptions continuationOptions)
202 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
205 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskScheduler scheduler)
207 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
210 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction,
211 CancellationToken cancellationToken,
212 TaskContinuationOptions continuationOptions,
213 TaskScheduler scheduler)
215 if (continuationFunction == null)
216 throw new ArgumentNullException ("continuationFunction");
217 if (scheduler == null)
218 throw new ArgumentNullException ("scheduler");
220 Task<TNewResult> t = new Task<TNewResult> ((o) => continuationFunction ((Task<TResult>)o),
223 GetCreationOptions (continuationOptions),
225 ContinueWithCore (t, continuationOptions, scheduler);
232 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state)
234 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
237 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken)
239 return ContinueWith (continuationAction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
242 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskContinuationOptions continuationOptions)
244 return ContinueWith (continuationAction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
247 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskScheduler scheduler)
249 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
252 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken,
253 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
255 if (continuationAction == null)
256 throw new ArgumentNullException ("continuationAction");
257 if (scheduler == null)
258 throw new ArgumentNullException ("scheduler");
260 var t = new Task (l => continuationAction (this, l),
263 GetCreationOptions (continuationOptions),
266 ContinueWithCore (t, continuationOptions, scheduler);
271 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state)
273 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
276 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, CancellationToken cancellationToken)
278 return ContinueWith<TNewResult> (continuationFunction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
281 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskContinuationOptions continuationOptions)
283 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
286 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskScheduler scheduler)
288 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
291 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state,
292 CancellationToken cancellationToken,
293 TaskContinuationOptions continuationOptions,
294 TaskScheduler scheduler)
296 if (continuationFunction == null)
297 throw new ArgumentNullException ("continuationFunction");
298 if (scheduler == null)
299 throw new ArgumentNullException ("scheduler");
301 var t = new Task<TNewResult> (l => continuationFunction (this, l),
304 GetCreationOptions (continuationOptions),
307 ContinueWithCore (t, continuationOptions, scheduler);
312 public new ConfiguredTaskAwaitable<TResult> ConfigureAwait (bool continueOnCapturedContext)
314 return new ConfiguredTaskAwaitable<TResult> (this, continueOnCapturedContext);
317 public new TaskAwaiter<TResult> GetAwaiter ()
319 return new TaskAwaiter<TResult> (this);