summaryrefslogtreecommitdiff
path: root/klibc/klibc/lrand48.c
blob: 4d05de2e895802d5d39f634a00e6c86b6af2558c (plain)
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
/*
 * lrand48.c
 */

#include <stdlib.h>
#include <stdint.h>

unsigned short __rand48_seed[3];

long jrand48(unsigned short xsubi[3])
{
  uint64_t x;

  /* The xsubi[] array is littleendian by spec */
  x = (uint64_t)xsubi[0] +
    ((uint64_t)xsubi[1] << 16) +
    ((uint64_t)xsubi[2] << 32);

  x = (0x5deece66dULL * x) + 0xb;
  
  xsubi[0] = (unsigned short)x;
  xsubi[1] = (unsigned short)(x >> 16);
  xsubi[2] = (unsigned short)(x >> 32);

  return (long)(int32_t)(x >> 16);
}

long mrand48(void)
{
  return jrand48(__rand48_seed);
}

long nrand48(unsigned short xsubi[3])
{
  return (long)((uint32_t)jrand48(xsubi) >> 1);
}

long lrand48(void)
{
  return (long)((uint32_t)(mrand48() >> 1));
}