Jo Shields 3c1f479b9d Imported Upstream version 4.0.0~alpha1
Former-commit-id: 806294f5ded97629b74c85c09952f2a74fe182d9
2015-04-07 09:35:12 +01:00

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;
}
}
}