This file is indexed.

/usr/src/castle-game-engine-6.4/base/castlerandom.pas is in castle-game-engine-src 6.4+dfsg1-2.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
{
  Copyright 2016-2017 Yevhen Loza, Michalis Kamburelis.

  This file is part of "Castle Game Engine".

  "Castle Game Engine" is free software; see the file COPYING.txt,
  included in this distribution, for details about the copyright.

  "Castle Game Engine" is distributed in the hope that it will be useful,
  but WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

  ----------------------------------------------------------------------------
}

{ Custom random number generator (TCastleRandom). }
unit CastleRandom;

interface

{$I castleconf.inc}
{$I norqcheckbegin.inc} // the whole unit should be used without overflow checking, for speed

type
  { Custom, fast random number generator.
    Implementation of XorShift algorithm for random numbers generation.
    In some cases it works 2 to 3 times faster than native
    FPC random function. It also allows for multiple
    repeatable random seeds to support parallel pseudo-random sequences.
    The algorithm should not be used for cryptographic purposes, because
    it's speed-oriented (with quality ok for game, but not cryptographic use),
    and might subject to change in future. }
  TCastleRandom = class(TObject)
  public
    { Create and initialize (seed) the random generator.
      Parameter RandomSeed value 0 indicates to use a random seed
      (derived from current time and some other paramteres). }
    constructor Create(RandomSeed: LongWord = 0);
    { Initializes current seed. The seed must be a non-zero integer.
      Provide Zero value to initialize random seed based on
      current time (CPU ticks) and some other paramteres.
      This procedure is thread-safe, you'll get different random seeds
      even if initialization happens absolutely simultaneously. }
    procedure Initialize(RandomSeed: LongWord = 0);
    { Returns random float value in the 0..1 range. }
    function Random: single;
    { Returns random integer number in the 0..N-1 range. }
    function Random(N: LongInt): LongInt;
    { A relatively slow procedure to get a 64 bit integer random number. }
    function RandomInt64(N: int64): int64;
    { A simple Yes/No function that with 50% chance returns true or false.
      Something like flipping a coin... }
    function RandomBoolean: boolean;
    { Randomly provides "-1", "0" or "1" with equal chances. }
    function RandomSign: longint;
    { Returns a random number in 1 .. High(LongWord) range. }
    function Random32bit: LongWord;
  private
    Seed: LongInt;
    procedure XorShiftCycle;
    function GetRandomSeed: LongInt;
  end;

{ Implementation of MurMur2 hash algorithm
  to make a cryptographic-unsafe but highly uniform
  32bit hash from a given string very quickly.
  It may be used for better initialization of user defined random seeds
  (e.g. "My New World" sounds much better than 6592202398)
  or for better sorting/searching of strings.
  Warning: the hash is deterministic, but the result may be different
  depending on CPU architecture or endianness (test required).

  @param(InputString The input string)
  @param(Seed Optional parameter enabling initialization
    of the algorithm seed different from default)
  @returns(Unsigned 32-bit integer)
}
function StringToHash(const InputString: AnsiString; const Seed: LongWord=0): LongWord;

{ Single random instance. }
function Rand: TCastleRandom;

function Rnd: single; deprecated 'use Rand.Random';
function Rnd(N: LongInt): LongInt; deprecated 'use Rand.Random';

implementation

uses
  { Required only for randomization based on "Now" function. }
  SysUtils,
  { Required for the GetTickCount64 function (not available in FPC 2.6.x).
    Note that the CastleTimeUtils.GetTickCount64 hides the SysUtils.GetTickCount64
    in FPC 3.x. }
  CastleTimeUtils;

constructor TCastleRandom.Create(RandomSeed: LongWord);
begin
  Initialize(RandomSeed);
end;

procedure TCastleRandom.Initialize(RandomSeed: LongWord);
begin
  if RandomSeed = 0 then
  begin
    Seed := GetRandomSeed;
  end
  else Seed := LongInt(RandomSeed);
end;

//we're not using dev_urandom for now to support identical implementation for different OSes and devices
{$IFDEF UNIX}
//{$DEFINE USE_DEV_URANDOM}
{$ENDIF}

{$IFDEF USE_DEV_URANDOM}
function DEV_URANDOM: longint;
var DevRnd: file of integer;
begin
  { algorithm according to http://wiki.freepascal.org/Dev_random
   /dev/urandom is a native *nix very high-quality random number generator.
   it's 1000 times slower than CastleRandom,
   but provides a perfect seed initialization. }
  AssignFile(DevRnd, '/dev/urandom');
  Reset(DevRnd);
  repeat
    Read(DevRnd, Result);
  until Result <> 0; // xorshift can't get 0 as a random seed so, we just read /dev/urandom until its not zero
  CloseFile(DevRnd);
end;
{$ELSE}

{ This procedure is relatively complex. However I was trying to solve
  a whole set of problems of random seeding. Including possible
  semi-simultaneous seeding requests by threads. On the other hand, there are
  more comments than the code itself :)
  I hope I've made everything right :) At least formal tests show it is so.}
var Store64bitSeed: QWord = 0; //this variable stores 64 bit seed for reusing
   WaitForSeed: boolean = false;
function Get_Randomseed: longint;
const DateMultiplier: QWord = 30000000;  // approximate accuracy of the date
      DateOrder: QWord = 80000 * 30000000; // order of the "now*date_multiplier" variable
      {p.s. date_order will be true until year ~2119}
var c64: QWord; //current seed;
    b64: QWord; //additional seed for multi-threading safety
  procedure XorShift64; //{$IFDEF SUPPORTS_INLINE} inline; {$ENDIF} //we're using it too many times to inline
  begin
    c64 := c64 xor (c64 shl 12);
    c64 := c64 xor (c64 shr 25);
    c64 := c64 xor (c64 shl 27);
  end;
begin
  {We add an additional semi-random variable based on local c64 variable
   address. The only profit we have here is that this address will be
   different for different threads, therefore no 2 threads can be initialized
   with equal seed even if they are absolutely simultaneous}
  c64 := PtrUInt(@(c64));

  { TODO: use TCriticalSession }
  while WaitForSeed do XorShift64; //do something nearly useful while randomization is buisy

  WaitForSeed := true;     //prevents another randomization to start until this one is finished

  XorShift64; XorShift64; XorShift64; XorShift64; XorShift64; XorShift64;
  {I've made it 6 times, because sometimes values returned by
   xorshift algorithm are not too different,
   but we want them really independent for random seed initialization.
   So, multiple xorshift64 will take this "little" difference and eventually
   transform it into truly unpredictable number in 1..high(QWORD) range}

  b64 := c64;   //and use our another random seed based on current thread memory allocation
  {can this actually damage randomness in case of randomization happens only once
   as b64 will be constant then?}

  {basically we don't care if threads accidently will pass "wait_for_seed" lock
   because thanks to b64 still we shall get different random values. Just the
   algorithm would not be as optimal as it might be}

  if Store64bitSeed = 0 then  //if this is the first randomization
  begin

    {This random seed initialization follows SysUtils random.
     Actually it is a relatively bad initialization for random numbers
     comparing to *nix /dev/urandom.
     It provides a limited amount of random numbers and it has a step of
     15 or 16 ms, so it's not continuous. Moreover it has just 5 mlns of
     possible values per 24 hours while xorshift32 supports for high(LongWord) -
     i.e. we get ~800 times less variants or 2400 times less
     as a "normal" user doesn't run computer for longer than 8 hours.
     And even less than that in case the player runs the game near the time
     the computer starts - just 200 thousands of combinations for 1 hour.

     On the other hand that's relatively enough for a computer game.

     Another, much more serious problem is that initializing 2 random generators
     semi-simultaneously will seed them with EQUAL seeds
     which we obviously don't want to.}

    {so let's start by getting tick count as SysUtils does}
    { TODO: Use CastleTimeUtils.Timer }
    {$PUSH}{$WARN 5066 OFF}
    {Yes, we are using a deprecated function, it's ok here,
     because its goal is not convenient time measurement
     but getting a semi-random number as fine as possible.
     We're using CastleTimeUtils.GetTickCount64,
     as SysUtils.GetTickCount64 is not available in FPC 2.6.4.
     We will switch to SysUtils.GetTickCount64 implementation when the engine will
     no longer need to support FPC 2.6.4. }
    c64 := GetTickCount64;
    {$POP}
    {just to make sure it's not zero. It's not really important here.}
    if c64 = 0 then
      c64 := 2903758934725;

    {"Trying to solve the problem" we do the following:}

    {make a 64-bit xorshift cycle several times
     to kill any possible link to GetTickCount64}
    XorShift64; XorShift64; XorShift64; XorShift64; XorShift64; XorShift64;
    {the same note on quantity of xorshift's as above}

    {now we have to make sure adding "now" won't overflow our c64 variable
     and add a few xorshift64-cycles just for fun in case it will.}
    while (c64 > High(QWord) - DateOrder) do XorShift64;

    {to kill a random discretness introduced by gettickcount64 we add "now".
     "now" and gettickcount64 are not independent and, in fact, change
     synchronously. But after several xorshift64-s c64 has no information
     left off gettickcount64 and therefore we introduce an additional
     semi-independent shift into the random seed}
    c64 += QWord(Round(Now * DateMultiplier));
    {now we are sure that the player will get a different random seed even
     in case he/she launches the game exactly at the same milisecond since
     the OS startup - different date&time will shift the random seed...
     unless he/she deliberately sets the date&time&tick to some specific value}

    {and another 64-bit XorShift cycle to kill everything left off "now"}
    XorShift64;
  end else
    c64 := Store64bitSeed; //else - just grab the last seed.

  {Now we cycle xorshift64 as we have a decent random c64 variable}
  XorShift64;
  {and merge another random-variable based on current thread memory allocation}
  c64 := c64 xor b64;

  {and finally...}
  repeat
    {cycle everything one more time}
    XorShift64;
    {leave higher 32-bits of c64 as a true random seed}
    Result := longint(c64 shr 32);
  until Result <> 0;
  {and strictly demand it's not zero!
   adding a few XorShift64-cycles in case it does.}

  {Eventually, store the final and truly random 64 bit seed for reusing}
  Store64bitSeed := c64;
  {and release the next thread to continue if any pending...}
  WaitForSeed := false;
end;
{$ENDIF}

function TCastleRandom.GetRandomSeed: longint;
begin
  {$IFDEF USE_DEV_URANDOM}
    { guarantees initialization with absolutely random number provided by
      native *nix algorithm.
      Yes, /dev/URandom is not as good as /dev/Random in cryptographic understanding
      but is faster and perfectly enough for a game-oriented random initalization. }
    Result := DEV_URANDOM;
  {$ELSE}
    { Castle's own random initialization algorithm. Thread-safe and 64-bit quality. }
    Result := Get_Randomseed;
  {$ENDIF}
end;

procedure TCastleRandom.XorShiftCycle; {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
begin
  { such implementation works a tiny bit faster (+4%) due to better optimization
    by compiler (uses CPU registers instead of a variable) }
  Seed := ((Seed xor (Seed shl 1)) xor ((Seed xor (Seed shl 1)) shr 15)) xor
         (((Seed xor (Seed shl 1)) xor ((Seed xor (Seed shl 1)) shr 15)) shl 4);
  {Seed := Seed xor (Seed shl 1);
  Seed := Seed xor (Seed shr 15);
  Seed := Seed xor (Seed shl 4); }
end;

{ This procedure is slow so it is better to use XorShiftCycle + direct access
  to seed private field instead }
function TCastleRandom.Random32bit: LongWord; {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
begin
  XorShiftCycle;
  Result := LongWord(Seed);
end;

function TCastleRandom.Random: single;
const Divisor: single = 1 / MaxInt;
begin
  XorShiftCycle;
  Result := Divisor * LongInt(Seed shr 1);       // note: we discard 1 bit of accuracy to gain speed
  //Result := Divisor * LongInt(XorShift shr 1);    // works slower
end;

{Result := LongWord((int64(Seed)*N) shr 32)// := seed mod N; works slower
//Result := Longint((int64(XorShift)*N) shr 32) // works slower}

// Adding  {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF} makes this procedure
//  +35% effective. But I don't think it's a good idea
function TCastleRandom.Random(N: LongInt): LongInt;
begin
  XorShiftCycle;
  if N > 1 then
    Result := LongInt((Int64(LongWord(Seed)) * N) shr 32)
  else
    Result := 0
end;

{ Works >10 times slower comparing to 32 bit version. And even slower than float version.
  Another problem is that it cycles the seed twice which might cause
  strange results if exact reproduction of the random sequence is required }
function TCastleRandom.RandomInt64(N: int64): int64;
var c64: QWord;
  procedure XorShift64; {$IFDEF SUPPORTS_INLINE} inline; {$ENDIF}
  begin
    c64 := c64 xor (c64 shl 12);
    c64 := c64 xor (c64 shr 25);
    c64 := c64 xor (c64 shl 27);
  end;
begin
  {we need to do it even if N=0..1 to cycle 32bit random seed twice as expected}
  c64 := qword(Random32bit) or (qword(Random32bit) shl 32);
  if N > 1 then begin
    {adding a XorShift64 cycle guarantees us that c64 is truly random
     in range 1..high(QWORD)
     but slows down execution by ~10%}
    XorShift64;
    {in contrast to SysUtils we make it a true 64-bit random, not a fake 63 bit :)
     There can be no overflow here, because N is int64 and it can't be
     larger than (High(QWORD) div 2), i.e. we can never get "negative" result
     as the first bit of the result will be always zero}
    Result := Int64(QWord(c64) mod QWord(N))
  end
  else
    Result := 0;
end;

function TCastleRandom.RandomBoolean: boolean;
begin
  XorShiftCycle;
  Result := Seed and %1 = 0   //can be %11 to provide for 1/4, %111 - 1/8 probability ...
end;

function TCastleRandom.RandomSign: longint;
begin
  XorShiftCycle;
  Result := LongInt((Int64(LongWord(Seed))*3) shr 32)-1
end;

{hashing}

{ MurMur algorithm works on any memory region at pointer Data
  of length Len, and the result differs depending on Seed. }
function MurMur2(const Data: pointer; const Len: Integer; const Seed: LongWord): LongWord;
var h, k: LongWord; //MurMur variables
    p: Pointer;
    i: Integer;
  procedure CycleHash(var x: LongWord); {$IFDEF SUPPORTS_INLINE}inline;{$ENDIF}
  const m = $5bd1e995; //MurMur "magic" cycling constant
        MaxLongWord = $FFFFFFFF;
  begin
    x := QWord(x * m) and MaxLongWord //prevent overflows during multiplication;
  end;
begin
  i := Len;
  p := Data;
  h := Seed xor i; //init the deterministic seed

  //cycle through all bytes of the string in 32 bit blocks
  while (i >= 4) do begin
    k := PLongWord(p)^;   //get next 4 bytes of data and process them
    CycleHash(k);
    k := k xor (k shr 24);
    CycleHash(k);

    CycleHash(h);
    h := h xor k;         //merge data into hash

    inc(p, 4); //advance to next character
    dec(i, 4); //to gain some speed we don't use p>pmax-4 check
  end;

  //upmix 0..3 final bytes of data to hash
  if i  = 3 then h := h xor (PByte(p + 2)^ shl 16);
  if i >= 2 then h := h xor (PByte(p + 1)^ shl 8);
  if i >= 1 then begin
                   h := h xor PByte(p)^;
                   CycleHash(h);
                 end;

  //and add a few final mixes
  h := h xor (h shr 13);
  CycleHash(h);
  h := h xor (h shr 15);

  Result := h;
end;

function StringToHash(const InputString: AnsiString; const Seed: LongWord = 0): LongWord;
begin
  Result := MurMur2(Pointer(InputString), Length(InputString), Seed);
  //Pointer(InputString) is an untyped pointer to the first character of the string
end;

var GlobalRandom: TCastleRandom;

function Rand: TCastleRandom;
begin
  if GlobalRandom = nil then GlobalRandom := TCastleRandom.Create;
  Result := GlobalRandom;
end;

function Rnd: Single;
begin
  if GlobalRandom = nil then
    GlobalRandom := TCastleRandom.Create;
  Result := GlobalRandom.Random;
end;

function Rnd(N: LongInt): LongInt;
begin
  if GlobalRandom = nil then
    GlobalRandom := TCastleRandom.Create;
  Result := GlobalRandom.Random(N);
end;

{$I norqcheckend.inc}

finalization
  FreeAndNil(GlobalRandom);

end.