2 * Copyright (c) 1996, David Mazieres <dm@uun.org>
3 * Copyright (c) 2008, Damien Miller <djm@openbsd.org>
5 * Permission to use, copy, modify, and distribute this software for any
6 * purpose with or without fee is hereby granted, provided that the above
7 * copyright notice and this permission notice appear in all copies.
9 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
19 * Arc4 random number generator for OpenBSD.
21 * This code is derived from section 17.1 of Applied Cryptography,
22 * second edition, which describes a stream cipher allegedly
23 * compatible with RSA Labs "RC4" cipher (the actual description of
24 * which is a trade secret). The same algorithm is used as a stream
25 * cipher called "arcfour" in Tatu Ylonen's ssh package.
27 * RC4 is a registered trademark of RSA Laboratories.
31 #include "CryptographicallyRandomNumber.h"
33 #include "MainThread.h"
34 #include "OSRandomSource.h"
35 #include "StdLibExtras.h"
36 #include "ThreadingPrimitives.h"
40 #if USE(OS_RANDOMNESS)
53 class ARC4RandomNumberGenerator
{
55 ARC4RandomNumberGenerator();
57 uint32_t randomNumber();
58 void randomValues(void* buffer
, size_t length
);
61 inline void addRandomData(unsigned char *data
, int length
);
64 inline uint8_t getByte();
65 inline uint32_t getWord();
69 #if ENABLE(WTF_MULTIPLE_THREADS)
74 ARC4Stream::ARC4Stream()
76 for (int n
= 0; n
< 256; n
++)
82 ARC4RandomNumberGenerator::ARC4RandomNumberGenerator()
87 void ARC4RandomNumberGenerator::addRandomData(unsigned char* data
, int length
)
90 for (int n
= 0; n
< 256; n
++) {
92 uint8_t si
= m_stream
.s
[m_stream
.i
];
93 m_stream
.j
+= si
+ data
[n
% length
];
94 m_stream
.s
[m_stream
.i
] = m_stream
.s
[m_stream
.j
];
95 m_stream
.s
[m_stream
.j
] = si
;
97 m_stream
.j
= m_stream
.i
;
100 void ARC4RandomNumberGenerator::stir()
102 unsigned char randomness
[128];
103 size_t length
= sizeof(randomness
);
104 cryptographicallyRandomValuesFromOS(randomness
, length
);
105 addRandomData(randomness
, length
);
107 // Discard early keystream, as per recommendations in:
108 // http://www.wisdom.weizmann.ac.il/~itsik/RC4/Papers/Rc4_ksa.ps
109 for (int i
= 0; i
< 256; i
++)
114 void ARC4RandomNumberGenerator::stirIfNeeded()
120 uint8_t ARC4RandomNumberGenerator::getByte()
123 uint8_t si
= m_stream
.s
[m_stream
.i
];
125 uint8_t sj
= m_stream
.s
[m_stream
.j
];
126 m_stream
.s
[m_stream
.i
] = sj
;
127 m_stream
.s
[m_stream
.j
] = si
;
128 return (m_stream
.s
[(si
+ sj
) & 0xff]);
131 uint32_t ARC4RandomNumberGenerator::getWord()
134 val
= getByte() << 24;
135 val
|= getByte() << 16;
136 val
|= getByte() << 8;
141 uint32_t ARC4RandomNumberGenerator::randomNumber()
143 #if ENABLE(WTF_MULTIPLE_THREADS)
144 MutexLocker
locker(m_mutex
);
146 ASSERT(isMainThread());
154 void ARC4RandomNumberGenerator::randomValues(void* buffer
, size_t length
)
156 #if ENABLE(WTF_MULTIPLE_THREADS)
157 MutexLocker
locker(m_mutex
);
159 ASSERT(isMainThread());
162 unsigned char* result
= reinterpret_cast<unsigned char*>(buffer
);
167 result
[length
] = getByte();
171 ARC4RandomNumberGenerator
& sharedRandomNumberGenerator()
173 DEFINE_STATIC_LOCAL(ARC4RandomNumberGenerator
, randomNumberGenerator
, ());
174 return randomNumberGenerator
;
179 uint32_t cryptographicallyRandomNumber()
181 return sharedRandomNumberGenerator().randomNumber();
184 void cryptographicallyRandomValues(void* buffer
, size_t length
)
186 sharedRandomNumberGenerator().randomValues(buffer
, length
);