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> ();
41 static readonly Action<object> emptyAction = delegate (object o) {};
44 Func<object, TResult> function;
45 Func<TResult> simpleFunction;
48 [System.Diagnostics.DebuggerBrowsable (System.Diagnostics.DebuggerBrowsableState.Never)]
49 public TResult Result {
54 throw new AggregateException (new TaskCanceledException (this));
55 if (Exception != null)
64 string ResultAsString {
66 if ((Status & (TaskStatus.RanToCompletion)) != 0)
69 return "<value not available>";
73 public static new TaskFactory<TResult> Factory {
79 public Task (Func<TResult> function) : this (function, TaskCreationOptions.None)
84 public Task (Func<TResult> function, CancellationToken cancellationToken)
85 : this (function, cancellationToken, TaskCreationOptions.None)
90 public Task (Func<TResult> function, TaskCreationOptions creationOptions)
91 : this (function, CancellationToken.None, creationOptions)
96 public Task (Func<TResult> function, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
97 : base (emptyAction, null, cancellationToken, creationOptions)
100 throw new ArgumentNullException ("function");
102 this.simpleFunction = function;
106 public Task (Func<object, TResult> function, object state) : this (function, state, TaskCreationOptions.None)
111 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken)
112 : this (function, state, cancellationToken, TaskCreationOptions.None)
117 public Task (Func<object, TResult> function, object state, TaskCreationOptions creationOptions)
118 : this (function, state, CancellationToken.None, creationOptions)
123 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
124 : base (emptyAction, state, cancellationToken, creationOptions)
126 if (function == null)
127 throw new ArgumentNullException ("function");
129 this.function = function;
133 internal Task (Func<object, TResult> function,
135 CancellationToken cancellationToken,
136 TaskCreationOptions creationOptions,
138 : base (emptyAction, state, cancellationToken, creationOptions, parent)
140 this.function = function;
144 internal override void InnerInvoke ()
146 if (function != null)
147 value = function (state);
148 else if (simpleFunction != null)
149 value = simpleFunction ();
152 simpleFunction = null;
156 public Task ContinueWith (Action<Task<TResult>> continuationAction)
158 return ContinueWith (continuationAction, TaskContinuationOptions.None);
161 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskContinuationOptions continuationOptions)
163 return ContinueWith (continuationAction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
166 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken)
168 return ContinueWith (continuationAction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
171 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskScheduler scheduler)
173 return ContinueWith (continuationAction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
176 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken,
177 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
179 if (continuationAction == null)
180 throw new ArgumentNullException ("continuationAction");
181 if (scheduler == null)
182 throw new ArgumentNullException ("scheduler");
184 Task t = new Task (l => continuationAction ((Task<TResult>)l),
187 GetCreationOptions (continuationOptions),
189 ContinueWithCore (t, continuationOptions, scheduler);
194 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction)
196 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
199 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, CancellationToken cancellationToken)
201 return ContinueWith<TNewResult> (continuationFunction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
204 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskContinuationOptions continuationOptions)
206 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
209 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskScheduler scheduler)
211 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
214 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction,
215 CancellationToken cancellationToken,
216 TaskContinuationOptions continuationOptions,
217 TaskScheduler scheduler)
219 if (continuationFunction == null)
220 throw new ArgumentNullException ("continuationFunction");
221 if (scheduler == null)
222 throw new ArgumentNullException ("scheduler");
224 Task<TNewResult> t = new Task<TNewResult> ((o) => continuationFunction ((Task<TResult>)o),
227 GetCreationOptions (continuationOptions),
229 ContinueWithCore (t, continuationOptions, scheduler);
236 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state)
238 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
241 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken)
243 return ContinueWith (continuationAction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
246 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskContinuationOptions continuationOptions)
248 return ContinueWith (continuationAction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
251 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskScheduler scheduler)
253 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
256 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken,
257 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
259 if (continuationAction == null)
260 throw new ArgumentNullException ("continuationAction");
261 if (scheduler == null)
262 throw new ArgumentNullException ("scheduler");
264 var t = new Task (l => continuationAction (this, l),
267 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> (l => continuationFunction (this, l),
308 GetCreationOptions (continuationOptions),
311 ContinueWithCore (t, continuationOptions, scheduler);
316 public new ConfiguredTaskAwaitable<TResult> ConfigureAwait (bool continueOnCapturedContext)
318 return new ConfiguredTaskAwaitable<TResult> (this, continueOnCapturedContext);
321 public new TaskAwaiter<TResult> GetAwaiter ()
323 return new TaskAwaiter<TResult> (this);