You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			244 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // 
 | |
| // CSnzi.cs
 | |
| //  
 | |
| // Author:
 | |
| //       Jérémie "Garuma" Laval <jeremie.laval@gmail.com>
 | |
| // 
 | |
| // Copyright (c) 2009 Jérémie "Garuma" Laval
 | |
| // 
 | |
| // Permission is hereby granted, free of charge, to any person obtaining a copy
 | |
| // of this software and associated documentation files (the "Software"), to deal
 | |
| // in the Software without restriction, including without limitation the rights
 | |
| // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | |
| // copies of the Software, and to permit persons to whom the Software is
 | |
| // furnished to do so, subject to the following conditions:
 | |
| // 
 | |
| // The above copyright notice and this permission notice shall be included in
 | |
| // all copies or substantial portions of the Software.
 | |
| // 
 | |
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | |
| // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | |
| // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | |
| // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | |
| // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | |
| // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | |
| // THE SOFTWARE.
 | |
| 
 | |
| using System;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace Mono.Threading
 | |
| {
 | |
| 	public abstract class CSnziNode
 | |
| 	{
 | |
| 		internal abstract bool Arrive ();
 | |
| 		internal abstract bool Depart ();
 | |
| 	}
 | |
| 	
 | |
| 	public enum CSnziState
 | |
| 	{
 | |
| 		Open,
 | |
| 		Closed
 | |
| 	}
 | |
| 	
 | |
| 	internal class CSnziLeafNode : CSnziNode
 | |
| 	{
 | |
| 		int count;
 | |
| 		readonly CSnziNode parent;
 | |
| 		
 | |
| 		public CSnziLeafNode (CSnziNode parent)
 | |
| 		{
 | |
| 			this.parent = parent;
 | |
| 		}
 | |
| 
 | |
| 		#region CSnziNode implementation
 | |
| 		internal override bool Arrive ()
 | |
| 		{
 | |
| 			bool arrivedAtParent = false;
 | |
| 			int x;
 | |
| 			
 | |
| 			do {
 | |
| 				x = count;
 | |
| 				if (x == 0 && !arrivedAtParent) {
 | |
| 					if (parent.Arrive ())
 | |
| 						arrivedAtParent = true;
 | |
| 					else
 | |
| 						return false;
 | |
| 				}
 | |
| 			} while (Interlocked.CompareExchange (ref count, x + 1, x) != x);
 | |
| 			
 | |
| 			if (arrivedAtParent && x != 0)
 | |
| 				parent.Depart ();
 | |
| 			
 | |
| 			return true;
 | |
| 		}
 | |
| 		
 | |
| 		internal override bool Depart ()
 | |
| 		{
 | |
| 			int x = Interlocked.Decrement (ref count);
 | |
| 			if (x == 1)
 | |
| 				return parent.Depart ();
 | |
| 			else
 | |
| 				return true;
 | |
| 		}
 | |
| 		#endregion
 | |
| 		
 | |
| 	}
 | |
| 	
 | |
| 	internal class CSnziRootNode : CSnziNode
 | |
| 	{
 | |
| 		int root;
 | |
| 		
 | |
| 		public int Count {
 | |
| 			get {
 | |
| 				return root & 0x7FFFFFFF;
 | |
| 			}
 | |
| 		}
 | |
| 					
 | |
| 		public CSnziState State {
 | |
| 			get {
 | |
| 				return (root >> 31) > 0 ? CSnziState.Open : CSnziState.Closed;
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		public CSnziRootNode () : this (0, CSnziState.Open)
 | |
| 		{
 | |
| 			
 | |
| 		}
 | |
| 		
 | |
| 		public CSnziRootNode (int count, CSnziState state)
 | |
| 		{
 | |
| 			root = Encode (count, state);
 | |
| 		}
 | |
| 
 | |
| 		#region CSnziNode implementation
 | |
| 		internal override bool Arrive ()
 | |
| 		{
 | |
| 			int old;
 | |
| 			int c;
 | |
| 			CSnziState s;
 | |
| 			
 | |
| 			do {
 | |
| 				old = root;
 | |
| 				
 | |
| 				Decode (old, out c, out s);
 | |
| 				
 | |
| 				if (c == 0 && s == CSnziState.Closed)
 | |
| 					return false;
 | |
| 			} while (Interlocked.CompareExchange (ref root, Encode (c + 1, s), old) != old);
 | |
| 			
 | |
| 			return true;
 | |
| 		}
 | |
| 		
 | |
| 		internal override bool Depart ()
 | |
| 		{
 | |
| 			int old;
 | |
| 			int c;
 | |
| 			CSnziState s;
 | |
| 			
 | |
| 			do {
 | |
| 				old = root;	
 | |
| 				Decode (old, out c, out s); 
 | |
| 			} while (Interlocked.CompareExchange (ref root, Encode (c - 1, s), old) != old);
 | |
| 			
 | |
| 			return c != 0 && s != CSnziState.Closed;
 | |
| 		}
 | |
| 		#endregion
 | |
| 		
 | |
| 		public void Open ()
 | |
| 		{
 | |
| 			root = Encode (0, CSnziState.Open);
 | |
| 		}
 | |
| 		
 | |
| 		public bool Close ()
 | |
| 		{
 | |
| 			int old, newRoot;
 | |
| 			int c;
 | |
| 			CSnziState s;
 | |
| 			
 | |
| 			do {
 | |
| 				old = root;
 | |
| 	
 | |
| 				Decode (old, out c, out s);
 | |
| 				if (s != CSnziState.Open)
 | |
| 					return false;
 | |
| 				
 | |
| 				newRoot = Encode (c, CSnziState.Closed);
 | |
| 			} while (Interlocked.CompareExchange (ref root, newRoot, old) != old);
 | |
| 			
 | |
| 			return c == 0;
 | |
| 		}
 | |
| 		
 | |
| 		int Encode (int count, CSnziState state)
 | |
| 		{
 | |
| 			return (state == CSnziState.Open) ? (int)(((uint)count) | 0x80000000) : count & 0x7FFFFFFF;
 | |
| 		}
 | |
| 		
 | |
| 		void Decode (int code, out int count, out CSnziState state)
 | |
| 		{
 | |
| 			count = code & 0x7FFFFFFF;
 | |
| 			state = (code >> 31) > 0 ? CSnziState.Open : CSnziState.Closed;
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	public class CSnzi
 | |
| 	{
 | |
| 		CSnziRootNode root;
 | |
| 		CSnziLeafNode[] leafs;
 | |
| 		
 | |
| 		readonly int LeafCount = Environment.ProcessorCount * 2;
 | |
| 		
 | |
| 		public CSnzi ()
 | |
| 		{
 | |
| 			leafs = new CSnziLeafNode[LeafCount];
 | |
| 			root = new CSnziRootNode ();
 | |
| 			
 | |
| 			for (int i = 0; i < leafs.Length; i++) {
 | |
| 				leafs[i] = new CSnziLeafNode (root);
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		public CSnziNode Arrive ()
 | |
| 		{
 | |
| 			while (true) {
 | |
| 				if (root.State != CSnziState.Open)
 | |
| 					return null;
 | |
| 				
 | |
| 				CSnziNode leaf = leafs[GetLeafIndex ()];
 | |
| 				if (leaf.Arrive ())
 | |
| 					return leaf;
 | |
| 				else {
 | |
| 					return null;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		public bool Depart (CSnziNode node)
 | |
| 		{
 | |
| 			return node.Depart ();
 | |
| 		}
 | |
| 		
 | |
| 		public bool Close ()
 | |
| 		{
 | |
| 			return root.Close ();
 | |
| 		}
 | |
| 		
 | |
| 		public void Open ()
 | |
| 		{
 | |
| 			root.Open ();
 | |
| 		}
 | |
| 		
 | |
| 		public Tuple<bool, CSnziState> Query ()
 | |
| 		{
 | |
| 			CSnziRootNode copy = root;
 | |
| 			
 | |
| 			return Tuple.Create (copy.Count > 0, copy.State);
 | |
| 		}
 | |
| 		
 | |
| 		int GetLeafIndex ()
 | |
| 		{
 | |
| 			return (Thread.CurrentThread.ManagedThreadId - 1) % leafs.Length;
 | |
| 		}
 | |
| 	}
 | |
| }
 |