// // System.Threading.Interlocked.cs // // Author: // Patrik Torstensson (patrik.torstensson@labs2.com) // Dick Porter (dick@ximian.com) // // (C) Ximian, Inc. http://www.ximian.com // // // Copyright (C) 2004, 2005 Novell, Inc (http://www.novell.com) // // 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.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.ConstrainedExecution; namespace System.Threading { public static partial class Interlocked { [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static int CompareExchange(ref int location1, int value, int comparand); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] internal extern static int CompareExchange(ref int location1, int value, int comparand, ref bool succeeded); [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] extern static void CompareExchange (ref object location1, ref object value, ref object comparand, ref object result); [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] public static object CompareExchange (ref object location1, object value, object comparand) { // This avoids coop handles, esp. on the output which would be particularly inefficient. // Passing everything by ref is equivalent to coop handles -- ref to locals at least. // // location1's treatment is unclear. But note that passing it by handle would be incorrect, // as it would use a local alias, which the coop marshaling does, to avoid the unclarity here, // that of a ref being to a managed frame vs. a native frame. Perhaps that could be revisited. // // So there a hole here, that of calling this function with location1 being in a native frame. // Usually it will be to a field, static or not, and not even to managed stack. // // This is usually intrinsified. Ideally it is always intrinisified. // object result = null; CompareExchange (ref location1, ref value, ref comparand, ref result); return result; } [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static float CompareExchange(ref float location1, float value, float comparand); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static int Decrement(ref int location); [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long Decrement(ref long location); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static int Increment(ref int location); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long Increment(ref long location); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static int Exchange(ref int location1, int value); [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] extern static void Exchange (ref object location1, ref object value, ref object result); [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] public static object Exchange (ref object location1, object value) { // See CompareExchange(object) for comments. // object result = null; Exchange (ref location1, ref value, ref result); return result; } [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static float Exchange(ref float location1, float value); [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long CompareExchange(ref long location1, long value, long comparand); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static IntPtr CompareExchange(ref IntPtr location1, IntPtr value, IntPtr comparand); [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static double CompareExchange(ref double location1, double value, double comparand); [ComVisible (false)] [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] [Intrinsic] public static T CompareExchange (ref T location1, T value, T comparand) where T : class { unsafe { if (Unsafe.AsPointer (ref location1) == null) throw new NullReferenceException (); } // Besides avoiding coop handles for efficiency, // and correctness, this also appears needed to // avoid an assertion failure in the runtime, related to // coop handles over generics. // // See CompareExchange(object) for more comments. // // This is not entirely convincing due to lack of volatile. // T result = null; // T : class so call the object overload. CompareExchange (ref Unsafe.As (ref location1), ref Unsafe.As(ref value), ref Unsafe.As(ref comparand), ref Unsafe.As(ref result)); return result; } [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long Exchange(ref long location1, long value); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static IntPtr Exchange(ref IntPtr location1, IntPtr value); [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static double Exchange(ref double location1, double value); [ComVisible (false)] [ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)] [Intrinsic] public static T Exchange (ref T location1, T value) where T : class { unsafe { if (Unsafe.AsPointer (ref location1) == null) throw new NullReferenceException (); } // See CompareExchange(T) for comments. // // This is not entirely convincing due to lack of volatile. // T result = null; // T : class so call the object overload. Exchange (ref Unsafe.As(ref location1), ref Unsafe.As(ref value), ref Unsafe.As(ref result)); return result; } [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long Read(ref long location); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static int Add(ref int location1, int value); [ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)] [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static long Add(ref long location1, long value); public static void MemoryBarrier () { Thread.MemoryBarrier (); } [MethodImplAttribute(MethodImplOptions.InternalCall)] public extern static void MemoryBarrierProcessWide (); } }