------------------------------------------------------------------------------ -- -- -- GNAT RUN-TIME COMPONENTS -- -- -- -- System.Atomic_Operations.Integer_Arithmetic -- -- -- -- B o d y -- -- -- -- Copyright (C) 2019-2024, Free Software Foundation, Inc. -- -- -- -- GNAT is free software; you can redistribute it and/or modify it under -- -- terms of the GNU General Public License as published by the Free Soft- -- -- ware Foundation; either version 3, or (at your option) any later ver- -- -- sion. GNAT is distributed in the hope that it will be useful, but WITH- -- -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY -- -- or FITNESS FOR A PARTICULAR PURPOSE. -- -- -- -- As a special exception under Section 7 of GPL version 3, you are granted -- -- additional permissions described in the GCC Runtime Library Exception, -- -- version 3.1, as published by the Free Software Foundation. -- -- -- -- You should have received a copy of the GNU General Public License and -- -- a copy of the GCC Runtime Library Exception along with this program; -- -- see the files COPYING3 and COPYING.RUNTIME respectively. If not, see -- -- . -- -- -- -- GNAT was originally developed by the GNAT team at New York University. -- -- Extensive contributions were provided by Ada Core Technologies Inc. -- -- -- ------------------------------------------------------------------------------ with System.Atomic_Primitives; use System.Atomic_Primitives; with System.Atomic_Operations.Exchange; with Interfaces.C; package body System.Atomic_Operations.Integer_Arithmetic is package Exchange is new System.Atomic_Operations.Exchange (Atomic_Type); ---------------- -- Atomic_Add -- ---------------- procedure Atomic_Add (Item : aliased in out Atomic_Type; Value : Atomic_Type) is Ignore : constant Atomic_Type := Atomic_Fetch_And_Add (Item, Value); begin null; end Atomic_Add; --------------------- -- Atomic_Subtract -- --------------------- procedure Atomic_Subtract (Item : aliased in out Atomic_Type; Value : Atomic_Type) is Ignore : constant Atomic_Type := Atomic_Fetch_And_Subtract (Item, Value); begin null; end Atomic_Subtract; -------------------------- -- Atomic_Fetch_And_Add -- -------------------------- function Atomic_Fetch_And_Add (Item : aliased in out Atomic_Type; Value : Atomic_Type) return Atomic_Type is pragma Warnings (Off); function Atomic_Fetch_Add (Ptr : System.Address; Val : Atomic_Type; Model : Mem_Model := Seq_Cst) return Atomic_Type; pragma Import (Intrinsic, Atomic_Fetch_Add, "__atomic_fetch_add"); pragma Warnings (On); begin -- Use the direct intrinsics when possible, and fallback to -- compare-and-exchange otherwise. if Atomic_Type'Base'Last = Atomic_Type'Last and then Atomic_Type'Base'First = Atomic_Type'First and then Atomic_Type'Last = 2**(Atomic_Type'Object_Size - 1) - 1 then if Atomic_Type'Object_Size in 8 | 16 | 32 | 64 then return Atomic_Fetch_Add (Item'Address, Value); else raise Program_Error; end if; else declare Old_Value : aliased Atomic_Type := Item; New_Value : Atomic_Type := Old_Value + Value; begin -- Keep iterating until the exchange succeeds while not Exchange.Atomic_Compare_And_Exchange (Item, Old_Value, New_Value) loop New_Value := Old_Value + Value; end loop; return Old_Value; end; end if; end Atomic_Fetch_And_Add; ------------------------------- -- Atomic_Fetch_And_Subtract -- ------------------------------- function Atomic_Fetch_And_Subtract (Item : aliased in out Atomic_Type; Value : Atomic_Type) return Atomic_Type is pragma Warnings (Off); function Atomic_Fetch_Sub (Ptr : System.Address; Val : Atomic_Type; Model : Mem_Model := Seq_Cst) return Atomic_Type; pragma Import (Intrinsic, Atomic_Fetch_Sub, "__atomic_fetch_sub"); pragma Warnings (On); begin -- Use the direct intrinsics when possible, and fallback to -- compare-and-exchange otherwise. if Atomic_Type'Base'Last = Atomic_Type'Last and then Atomic_Type'Base'First = Atomic_Type'First and then Atomic_Type'Last = 2**(Atomic_Type'Object_Size - 1) - 1 then if Atomic_Type'Object_Size in 8 | 16 | 32 | 64 then return Atomic_Fetch_Sub (Item'Address, Value); else raise Program_Error; end if; else declare Old_Value : aliased Atomic_Type := Item; New_Value : Atomic_Type := Old_Value - Value; begin -- Keep iterating until the exchange succeeds while not Exchange.Atomic_Compare_And_Exchange (Item, Old_Value, New_Value) loop New_Value := Old_Value - Value; end loop; return Old_Value; end; end if; end Atomic_Fetch_And_Subtract; ------------------ -- Is_Lock_Free -- ------------------ function Is_Lock_Free (Item : aliased Atomic_Type) return Boolean is pragma Unreferenced (Item); use type Interfaces.C.size_t; begin return Atomic_Always_Lock_Free (Atomic_Type'Object_Size / 8); end Is_Lock_Free; end System.Atomic_Operations.Integer_Arithmetic;