| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- // SpinLock.cs
- //
- // Copyright (c) 2008 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.Runtime.ConstrainedExecution;
- using System.Runtime.InteropServices;
- #if NET_4_0
- namespace System.Threading
- {
- [StructLayout(LayoutKind.Explicit)]
- internal struct TicketType {
- [FieldOffset(0)]
- public long TotalValue;
- [FieldOffset(0)]
- public int Value;
- [FieldOffset(4)]
- public int Users;
- }
- // Implement the ticket SpinLock algorithm described on http://locklessinc.com/articles/locks/
- // This lock is usable on both endianness
- // TODO: some 32 bits platform apparently doesn't support CAS with 64 bits value
- public struct SpinLock
- {
- TicketType ticket;
- int threadWhoTookLock;
- readonly bool isThreadOwnerTrackingEnabled;
- public bool IsThreadOwnerTrackingEnabled {
- get {
- return isThreadOwnerTrackingEnabled;
- }
- }
- public bool IsHeld {
- get {
- // No need for barrier here
- long totalValue = ticket.TotalValue;
- return (totalValue >> 32) != (totalValue & 0xFFFFFFFF);
- }
- }
- public bool IsHeldByCurrentThread {
- get {
- if (isThreadOwnerTrackingEnabled)
- return IsHeld && Thread.CurrentThread.ManagedThreadId == threadWhoTookLock;
- else
- return IsHeld;
- }
- }
- public SpinLock (bool trackId)
- {
- this.isThreadOwnerTrackingEnabled = trackId;
- this.threadWhoTookLock = 0;
- this.ticket = new TicketType ();
- }
- [MonoTODO("This method is not rigorously correct. Need CER treatment")]
- public void Enter (ref bool lockTaken)
- {
- if (lockTaken)
- throw new ArgumentException ("lockTaken", "lockTaken must be initialized to false");
- if (isThreadOwnerTrackingEnabled && IsHeldByCurrentThread)
- throw new LockRecursionException ();
- int slot = Interlocked.Increment (ref ticket.Users) - 1;
- SpinWait wait;
- while (slot != ticket.Value)
- wait.SpinOnce ();
-
- lockTaken = true;
-
- threadWhoTookLock = Thread.CurrentThread.ManagedThreadId;
- }
- [MonoTODO("This method is not rigorously correct. Need CER treatment")]
- public void TryEnter (ref bool lockTaken)
- {
- TryEnter (0, ref lockTaken);
- }
- [MonoTODO("This method is not rigorously correct. Need CER treatment")]
- public void TryEnter (TimeSpan timeout, ref bool lockTaken)
- {
- TryEnter ((int)timeout.TotalMilliseconds, ref lockTaken);
- }
- [MonoTODO("This method is not rigorously correct. Need CER treatment")]
- public void TryEnter (int milliSeconds, ref bool lockTaken)
- {
- if (milliSeconds < -1)
- throw new ArgumentOutOfRangeException ("milliSeconds", "millisecondsTimeout is a negative number other than -1");
- if (lockTaken)
- throw new ArgumentException ("lockTaken", "lockTaken must be initialized to false");
- if (isThreadOwnerTrackingEnabled && IsHeldByCurrentThread)
- throw new LockRecursionException ();
- Watch sw = Watch.StartNew ();
- do {
- long u = ticket.Users;
- long totalValue = (u << 32) | u;
- long newTotalValue
- = BitConverter.IsLittleEndian ? (u << 32) | (u + 1) : ((u + 1) << 32) | u;
-
- lockTaken = Interlocked.CompareExchange (ref ticket.TotalValue, newTotalValue, totalValue) == totalValue;
-
- if (lockTaken) {
- threadWhoTookLock = Thread.CurrentThread.ManagedThreadId;
- break;
- }
- } while (milliSeconds == -1 || (milliSeconds > 0 && sw.ElapsedMilliseconds < milliSeconds));
- }
- public void Exit ()
- {
- Exit (false);
- }
- public void Exit (bool flushReleaseWrites)
- {
- if (isThreadOwnerTrackingEnabled && !IsHeldByCurrentThread)
- throw new SynchronizationLockException ("Current thread is not the owner of this lock");
- threadWhoTookLock = int.MinValue;
- if (flushReleaseWrites)
- Interlocked.Increment (ref ticket.Value);
- else
- ticket.Value++;
- }
- }
- // Wraps a SpinLock in a reference when we need to pass
- // around the lock
- internal class SpinLockWrapper
- {
- public SpinLock Lock = new SpinLock (false);
- }
- }
- #endif
|