Skip to content

Commit

Permalink
Self-review
Browse files Browse the repository at this point in the history
  • Loading branch information
mosteo committed Jun 17, 2024
1 parent 260af12 commit f87a85c
Showing 1 changed file with 75 additions and 79 deletions.
154 changes: 75 additions & 79 deletions src/alire/alire-directories.adb
Original file line number Diff line number Diff line change
Expand Up @@ -553,102 +553,98 @@ package body Alire.Directories is
Epoch : constant Ada.Real_Time.Time :=
Ada.Real_Time.Time_Of (0, Ada.Real_Time.To_Time_Span (0.0));

-------------
-- Counter --
-------------
----------------------
-- Tempfile_Support --
----------------------

protected Counter is
procedure Get (Value : out Interfaces.Unsigned_32);
protected Tempfile_Support is
procedure Next_Name (Name : out String);
private
Next : Interfaces.Unsigned_32 := 0;
end Counter;

protected body Counter is
procedure Get (Value : out Interfaces.Unsigned_32) is
Next_Seed : Interfaces.Unsigned_32 := 0;
Used_Names : AAA.Strings.Set;
end Tempfile_Support;

protected body Tempfile_Support is

---------------
-- Next_Name --
---------------

procedure Next_Name (Name : out String) is
subtype Valid_Character is Character range 'a' .. 'z';
package Char_Random is new
Ada.Numerics.Discrete_Random (Valid_Character);
Gen : Char_Random.Generator;

-- The default random seed has a granularity of 1 second, which is
-- not enough when we run our tests with high parallelism. Increasing
-- the resolution to nanoseconds is less collision-prone. On top, we
-- add the current working directory path to the hash input, which
-- should disambiguate even further for our most usual case which is
-- during testsuite execution, and a counter to avoid clashes in the
-- same process.

-- It would be safer to use an atomic OS call that returns a unique
-- file name, but we would need native versions for all OSes we
-- support and that may be too much hassle? since GNAT.OS_Lib
-- doesn't do it either.

use Ada.Real_Time;
use type Interfaces.Unsigned_32;
begin
Value := Next;
Next := Next + 1;
end Get;
end Counter;

----------
-- Next --
----------
Nano : constant String :=
AAA.Strings.Replace (To_Duration (Clock - Epoch)'Image,
".", "");
-- This gives us an image without loss of precision and without
-- having to be worried about overflows

function Next return String is
Val : Interfaces.Unsigned_32;
begin
Counter.Get (Val);
return Val'Image;
end Next;
type Hash_Type is mod 2 ** 32;
pragma Compile_Time_Error (Hash_Type'Size > Integer'Size,
"Hash_Type is too large");

---------------
-- Temp_Name --
---------------
function Hash is new GNAT.String_Hash.Hash
(Char_Type => Character,
Key_Type => String,
Hash_Type => Hash_Type);

Used_Names : AAA.Strings.Set;
function To_Integer is
new Ada.Unchecked_Conversion (Hash_Type, Integer);
-- Ensure unsigned -> signed conversion doesn't bite us

function Temp_Name (Length : Positive := 8) return String is
subtype Valid_Character is Character range 'a' .. 'z';
package Char_Random is new
Ada.Numerics.Discrete_Random (Valid_Character);
Gen : Char_Random.Generator;

-- The default random seed has a granularity of 1 second, which is not
-- enough when we run our tests with high parallelism. Increasing the
-- resolution to nanoseconds is less collision-prone. On top, we add
-- the current working directory path to the hash input, which should
-- disambiguate even further for our most usual case which is during
-- testsuite execution, and a counter to avoid clashes in the same
-- process.
Seed : constant Hash_Type :=
Hash (Nano & " at " & Current & "#" & Next_Seed'Image);
begin
Next_Seed := Next_Seed + 1;

-- It would be safer to use an atomic OS call that returns a unique file
-- name, but we would need native versions for all OSes we support and
-- that may be too much hassle? since GNAT.OS_Lib doesn't do it either.
Char_Random.Reset (Gen, To_Integer (Seed));

use Ada.Real_Time;
loop
for I in Name'Range loop
Name (I) := Char_Random.Random (Gen);
end loop;

Nano : constant String :=
AAA.Strings.Replace (To_Duration (Clock - Epoch)'Image,
".", "");
-- This gives us an image without loss of precision and without
-- having to be worried about overflows
-- Make totally sure that not even by random chance we are reusing
-- a temporary name.

type Hash_Type is mod 2 ** 32;
pragma Compile_Time_Error (Hash_Type'Size > Integer'Size,
"Hash_Type is too large");
exit when not Used_Names.Contains (Name);
end loop;

function Hash is new GNAT.String_Hash.Hash
(Char_Type => Character,
Key_Type => String,
Hash_Type => Hash_Type);
Used_Names.Insert (Name);
end Next_Name;

function To_Integer is new Ada.Unchecked_Conversion (Hash_Type, Integer);
-- Ensure unsigned -> signed conversion doesn't bite us
end Tempfile_Support;

Seed : constant Hash_Type := Hash (Nano & " at " & Current & "#" & Next);
---------------
-- Temp_Name --
---------------

function Temp_Name (Length : Positive := 8) return String is
Result : String (1 .. Length + 4);
begin

Char_Random.Reset (Gen, To_Integer (Seed));

return Result : String (1 .. Length + 4) do
Result (1 .. 4) := "alr-";
Result (Length + 1 .. Result'Last) := ".tmp";
for I in 5 .. Length loop
Result (I) := Char_Random.Random (Gen);
end loop;

-- Make totally sure that not even by random chance we are reusing a
-- temporary name.

while Used_Names.Contains (Result) loop
Result := Temp_Name; -- Try again
end loop;

Used_Names.Insert (Result);
end return;
Result (1 .. 4) := "alr-";
Result (Length + 1 .. Result'Last) := ".tmp";
Tempfile_Support.Next_Name (Result (5 .. Length));
return Result;
end Temp_Name;

----------------
Expand Down

0 comments on commit f87a85c

Please sign in to comment.