2 * Bittorrent Client using Qt and libtorrent.
3 * Copyright (C) 2017 Mike Tzou
5 * This program is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU General Public License
7 * as published by the Free Software Foundation; either version 2
8 * of the License, or (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
19 * In addition, as a special exception, the copyright holders give permission to
20 * link this program with the OpenSSL project's "OpenSSL" library (or with
21 * modified versions of it that use the same license as the "OpenSSL" library),
22 * and distribute the linked executables. You must obey the GNU General Public
23 * License in all respects for all of the code used other than "OpenSSL". If you
24 * modify file(s), you may extend this exception to your version of the file(s),
25 * but you are not obligated to do so. If you do not wish to do so, delete this
26 * exception statement from your version.
34 #include <QtSystemDetection>
38 #include "base/global.h"
39 #include "base/utils/os.h"
40 #elif defined(Q_OS_LINUX)
43 #include <sys/random.h>
55 // need to satisfy UniformRandomBitGenerator requirements
57 using result_type
= uint32_t;
60 : m_processPrng
{Utils::OS::loadWinAPI
<PPROCESSPRNG
>(u
"BCryptPrimitives.dll"_s
, "ProcessPrng")}
63 qFatal("Failed to load ProcessPrng().");
66 static constexpr result_type
min()
68 return std::numeric_limits
<result_type
>::min();
71 static constexpr result_type
max()
73 return std::numeric_limits
<result_type
>::max();
76 result_type
operator()()
79 const bool result
= m_processPrng(reinterpret_cast<PBYTE
>(&buf
), sizeof(buf
));
81 qFatal("ProcessPrng() failed.");
87 using PPROCESSPRNG
= BOOL (WINAPI
*)(PBYTE
, SIZE_T
);
88 const PPROCESSPRNG m_processPrng
;
90 #elif defined(Q_OS_LINUX)
93 // need to satisfy UniformRandomBitGenerator requirements
95 using result_type
= uint32_t;
101 static constexpr result_type
min()
103 return std::numeric_limits
<result_type
>::min();
106 static constexpr result_type
max()
108 return std::numeric_limits
<result_type
>::max();
111 result_type
operator()()
113 const int RETRY_MAX
= 3;
115 for (int i
= 0; i
< RETRY_MAX
; ++i
)
118 const ssize_t result
= ::getrandom(&buf
, sizeof(buf
), 0);
119 if (result
== sizeof(buf
)) // success
123 qFatal("getrandom() error. Reason: %s. Error code: %d.", std::strerror(errno
), errno
);
126 qFatal("getrandom() failed. Reason: too many retries.");
132 // need to satisfy UniformRandomBitGenerator requirements
134 using result_type
= uint32_t;
137 : m_randDev
{fopen("/dev/urandom", "rb")}
140 qFatal("Failed to open /dev/urandom. Reason: %s. Error code: %d.", std::strerror(errno
), errno
);
148 static constexpr result_type
min()
150 return std::numeric_limits
<result_type
>::min();
153 static constexpr result_type
max()
155 return std::numeric_limits
<result_type
>::max();
158 result_type
operator()() const
161 if (fread(&buf
, sizeof(buf
), 1, m_randDev
) != 1)
162 qFatal("Read /dev/urandom error. Reason: %s. Error code: %d.", std::strerror(errno
), errno
);
168 FILE *m_randDev
= nullptr;
173 uint32_t Utils::Random::rand(const uint32_t min
, const uint32_t max
)
175 static RandomLayer layer
;
177 // new distribution is cheap: https://stackoverflow.com/a/19036349
178 std::uniform_int_distribution
<uint32_t> uniform(min
, max
);
180 return uniform(layer
);