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 {
53 if (Exception != null)
62 string ResultAsString {
64 if ((Status & (TaskStatus.RanToCompletion)) != 0)
67 return "<value not available>";
71 public static new TaskFactory<TResult> Factory {
77 public Task (Func<TResult> function) : this (function, TaskCreationOptions.None)
82 public Task (Func<TResult> function, CancellationToken cancellationToken)
83 : this (function, cancellationToken, TaskCreationOptions.None)
88 public Task (Func<TResult> function, TaskCreationOptions creationOptions)
89 : this (function, CancellationToken.None, creationOptions)
94 public Task (Func<TResult> function, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
95 : base (emptyAction, null, cancellationToken, creationOptions)
98 throw new ArgumentNullException ("function");
100 this.simpleFunction = function;
104 public Task (Func<object, TResult> function, object state) : this (function, state, TaskCreationOptions.None)
109 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken)
110 : this (function, state, cancellationToken, TaskCreationOptions.None)
115 public Task (Func<object, TResult> function, object state, TaskCreationOptions creationOptions)
116 : this (function, state, CancellationToken.None, creationOptions)
121 public Task (Func<object, TResult> function, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions)
122 : base (emptyAction, state, cancellationToken, creationOptions)
124 if (function == null)
125 throw new ArgumentNullException ("function");
127 this.function = function;
131 internal Task (Func<object, TResult> function,
133 CancellationToken cancellationToken,
134 TaskCreationOptions creationOptions,
136 : base (emptyAction, state, cancellationToken, creationOptions, parent)
138 this.function = function;
142 internal override void InnerInvoke ()
144 if (function != null)
145 value = function (state);
146 else if (simpleFunction != null)
147 value = simpleFunction ();
150 simpleFunction = null;
154 public Task ContinueWith (Action<Task<TResult>> continuationAction)
156 return ContinueWith (continuationAction, TaskContinuationOptions.None);
159 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskContinuationOptions continuationOptions)
161 return ContinueWith (continuationAction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
164 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken)
166 return ContinueWith (continuationAction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
169 public Task ContinueWith (Action<Task<TResult>> continuationAction, TaskScheduler scheduler)
171 return ContinueWith (continuationAction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
174 public Task ContinueWith (Action<Task<TResult>> continuationAction, CancellationToken cancellationToken,
175 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
177 if (continuationAction == null)
178 throw new ArgumentNullException ("continuationAction");
179 if (scheduler == null)
180 throw new ArgumentNullException ("scheduler");
182 Task t = new Task (l => continuationAction ((Task<TResult>)l),
185 GetCreationOptions (continuationOptions),
187 ContinueWithCore (t, continuationOptions, scheduler);
192 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction)
194 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
197 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, CancellationToken cancellationToken)
199 return ContinueWith<TNewResult> (continuationFunction, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
202 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskContinuationOptions continuationOptions)
204 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, continuationOptions, TaskScheduler.Current);
207 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction, TaskScheduler scheduler)
209 return ContinueWith<TNewResult> (continuationFunction, CancellationToken.None, TaskContinuationOptions.None, scheduler);
212 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, TNewResult> continuationFunction,
213 CancellationToken cancellationToken,
214 TaskContinuationOptions continuationOptions,
215 TaskScheduler scheduler)
217 if (continuationFunction == null)
218 throw new ArgumentNullException ("continuationFunction");
219 if (scheduler == null)
220 throw new ArgumentNullException ("scheduler");
222 Task<TNewResult> t = new Task<TNewResult> ((o) => continuationFunction ((Task<TResult>)o),
225 GetCreationOptions (continuationOptions),
227 ContinueWithCore (t, continuationOptions, scheduler);
234 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state)
236 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
239 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken)
241 return ContinueWith (continuationAction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
244 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskContinuationOptions continuationOptions)
246 return ContinueWith (continuationAction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
249 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, TaskScheduler scheduler)
251 return ContinueWith (continuationAction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
254 public Task ContinueWith (Action<Task<TResult>, object> continuationAction, object state, CancellationToken cancellationToken,
255 TaskContinuationOptions continuationOptions, TaskScheduler scheduler)
257 if (continuationAction == null)
258 throw new ArgumentNullException ("continuationAction");
259 if (scheduler == null)
260 throw new ArgumentNullException ("scheduler");
262 var t = new Task (l => continuationAction (this, l),
265 GetCreationOptions (continuationOptions),
268 ContinueWithCore (t, continuationOptions, scheduler);
273 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state)
275 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, TaskScheduler.Current);
278 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, CancellationToken cancellationToken)
280 return ContinueWith<TNewResult> (continuationFunction, state, cancellationToken, TaskContinuationOptions.None, TaskScheduler.Current);
283 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskContinuationOptions continuationOptions)
285 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, continuationOptions, TaskScheduler.Current);
288 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state, TaskScheduler scheduler)
290 return ContinueWith<TNewResult> (continuationFunction, state, CancellationToken.None, TaskContinuationOptions.None, scheduler);
293 public Task<TNewResult> ContinueWith<TNewResult> (Func<Task<TResult>, object, TNewResult> continuationFunction, object state,
294 CancellationToken cancellationToken,
295 TaskContinuationOptions continuationOptions,
296 TaskScheduler scheduler)
298 if (continuationFunction == null)
299 throw new ArgumentNullException ("continuationFunction");
300 if (scheduler == null)
301 throw new ArgumentNullException ("scheduler");
303 var t = new Task<TNewResult> (l => continuationFunction (this, l),
306 GetCreationOptions (continuationOptions),
309 ContinueWithCore (t, continuationOptions, scheduler);
314 public new ConfiguredTaskAwaitable<TResult> ConfigureAwait (bool continueOnCapturedContext)
316 return new ConfiguredTaskAwaitable<TResult> (this, continueOnCapturedContext);
319 public new TaskAwaiter<TResult> GetAwaiter ()
321 return new TaskAwaiter<TResult> (this);