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
|
/* random.cpp
*
* Copyright (C) 2003 Sawtooth Consulting Ltd.
*
* This file is part of yaSSL.
*
* yaSSL is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* yaSSL 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. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
*/
/* random.cpp implements a crypto secure Random Number Generator using an OS
specific seed, switch to /dev/random for more security but may block
*/
#include "runtime.hpp"
#include "random.hpp"
#if defined(_WIN32)
#define _WIN32_WINNT 0x0400
#include <windows.h>
#include <wincrypt.h>
#else
#include <errno.h>
#include <fcntl.h>
#include <unistd.h>
#endif // _WIN32
namespace TaoCrypt {
// Get seed and key cipher
RandomNumberGenerator::RandomNumberGenerator()
{
byte key[32];
seed_.GenerateSeed(key, sizeof(key));
cipher_.SetKey(key, sizeof(key));
}
// place a generated block in output
void RandomNumberGenerator::GenerateBlock(byte* output, word32 sz)
{
cipher_.Process(output, output, sz);
}
byte RandomNumberGenerator::GenerateByte()
{
byte b;
GenerateBlock(&b, 1);
return b;
}
#if defined(_WIN32)
OS_Seed::OS_Seed()
{
if(!CryptAcquireContext(&handle_, 0, 0, PROV_RSA_FULL,
CRYPT_VERIFYCONTEXT))
error_.SetError(WINCRYPT_E);
}
OS_Seed::~OS_Seed()
{
CryptReleaseContext(handle_, 0);
}
void OS_Seed::GenerateSeed(byte* output, word32 sz)
{
if (!CryptGenRandom(handle_, sz, output))
error_.SetError(CRYPTGEN_E);
}
#else // _WIN32
OS_Seed::OS_Seed()
{
fd_ = open("/dev/urandom",O_RDONLY);
if (fd_ == -1)
{
fd_ = open("/dev/random",O_RDONLY);
if (fd_ == -1)
error_.SetError(OPEN_RAN_E);
}
}
OS_Seed::~OS_Seed()
{
close(fd_);
}
// may block
void OS_Seed::GenerateSeed(byte* output, word32 sz)
{
while (sz) {
int len = read(fd_, output, sz);
if (len == -1) {
error_.SetError(READ_RAN_E);
return;
}
sz -= len;
output += len;
if (sz)
sleep(1);
}
}
#endif // _WIN32
} // namespace
|