2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
// 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
|
|
|
|
{
|
2019-04-12 14:10:50 +00:00
|
|
|
public static partial class Interlocked
|
2014-08-13 10:39:27 +01:00
|
|
|
{
|
|
|
|
[ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)]
|
|
|
|
[MethodImplAttribute(MethodImplOptions.InternalCall)]
|
|
|
|
public extern static int CompareExchange(ref int location1, int value, int comparand);
|
|
|
|
|
2015-04-07 09:35:12 +01:00
|
|
|
[ReliabilityContractAttribute (Consistency.WillNotCorruptState, Cer.Success)]
|
|
|
|
[MethodImplAttribute(MethodImplOptions.InternalCall)]
|
|
|
|
internal extern static int CompareExchange(ref int location1, int value, int comparand, ref bool succeeded);
|
|
|
|
|
2019-12-10 18:00:56 +00:00
|
|
|
[ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)]
|
2014-08-13 10:39:27 +01:00
|
|
|
[MethodImplAttribute(MethodImplOptions.InternalCall)]
|
2019-12-10 18:00:56 +00:00
|
|
|
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;
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[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);
|
|
|
|
|
2019-12-10 18:00:56 +00:00
|
|
|
[ReliabilityContract (Consistency.WillNotCorruptState, Cer.Success)]
|
2014-08-13 10:39:27 +01:00
|
|
|
[MethodImplAttribute(MethodImplOptions.InternalCall)]
|
2019-12-10 18:00:56 +00:00
|
|
|
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;
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[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)]
|
2019-12-10 18:00:56 +00:00
|
|
|
[Intrinsic]
|
|
|
|
public static T CompareExchange<T> (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<T, object> (ref location1), ref Unsafe.As<T, object>(ref value), ref Unsafe.As<T, object>(ref comparand), ref Unsafe.As<T, object>(ref result));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[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)]
|
2019-12-10 18:00:56 +00:00
|
|
|
[Intrinsic]
|
|
|
|
public static T Exchange<T> (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<T, object>(ref location1), ref Unsafe.As<T, object>(ref value), ref Unsafe.As<T, object>(ref result));
|
|
|
|
return result;
|
|
|
|
}
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
[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 ();
|
|
|
|
}
|
2019-07-26 19:53:28 +00:00
|
|
|
|
|
|
|
[MethodImplAttribute(MethodImplOptions.InternalCall)]
|
|
|
|
public extern static void MemoryBarrierProcessWide ();
|
2014-08-13 10:39:27 +01:00
|
|
|
}
|
|
|
|
}
|