6746907e908bb13b51610570a871a5105a9c46f1
[mono.git] / mcs / class / corlib / System.Threading.Tasks / TaskCompletionQueue.cs
1 //
2 // TaskCompletionQueue.cs
3 //
4 // Authors:
5 //    Jérémie Laval <jeremie dot laval at xamarin dot com>
6 //
7 // Copyright 2011 Xamarin Inc (http://www.xamarin.com).
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining a copy
10 // of this software and associated documentation files (the "Software"), to deal
11 // in the Software without restriction, including without limitation the rights
12 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
13 // copies of the Software, and to permit persons to whom the Software is
14 // furnished to do so, subject to the following conditions:
15 //
16 // The above copyright notice and this permission notice shall be included in
17 // all copies or substantial portions of the Software.
18 //
19 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
22 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
25 // THE SOFTWARE.
26 //
27 //
28
29 #if NET_4_0
30
31 using System;
32 using System.Threading;
33 using System.Collections.Generic;
34 using System.Collections.Concurrent;
35
36 namespace System.Threading.Tasks
37 {
38         internal struct TaskCompletionQueue<TCompletion> where TCompletion : class
39         {
40                 TCompletion single;
41                 ConcurrentOrderedList<TCompletion> completed;
42
43                 public void Add (TCompletion continuation)
44                 {
45                         if (single == null && Interlocked.CompareExchange (ref single, continuation, null) == null)
46                                 return;
47                         if (completed == null)
48                                 Interlocked.CompareExchange (ref completed, new ConcurrentOrderedList<TCompletion> (), null);
49                         completed.TryAdd (continuation);
50                 }
51
52                 public bool Remove (TCompletion continuation)
53                 {
54                         TCompletion temp = single;
55                         if (temp != null && temp == continuation && Interlocked.CompareExchange (ref single, null, continuation) == continuation)
56                                 return true;
57                         if (completed != null)
58                                 return completed.TryRemove (continuation);
59                         return false;
60                 }
61
62                 public bool HasElements {
63                         get {
64                                 return single != null || (completed != null && completed.Count != 0);
65                         }
66                 }
67
68                 public bool TryGetNextCompletion (out TCompletion continuation)
69                 {
70                         continuation = null;
71
72                         if (single != null && (continuation = Interlocked.Exchange (ref single, null)) != null)
73                                 return true;
74
75                         return completed != null && completed.TryPop (out continuation);
76                 }
77         }
78 }
79
80 #endif