Line data Source code
1 : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 : /*
3 : * This file is part of the LibreOffice project.
4 : *
5 : * This Source Code Form is subject to the terms of the Mozilla Public
6 : * License, v. 2.0. If a copy of the MPL was not distributed with this
7 : * file, You can obtain one at http://mozilla.org/MPL/2.0/.
8 : *
9 : * This file incorporates work covered by the following license notice:
10 : *
11 : * Licensed to the Apache Software Foundation (ASF) under one or more
12 : * contributor license agreements. See the NOTICE file distributed
13 : * with this work for additional information regarding copyright
14 : * ownership. The ASF licenses this file to you under the Apache
15 : * License, Version 2.0 (the "License"); you may not use this file
16 : * except in compliance with the License. You may obtain a copy of
17 : * the License at http://www.apache.org/licenses/LICENSE-2.0 .
18 : */
19 :
20 : #include <sal/types.h>
21 : #include <osl/thread.h>
22 : #include <osl/time.h>
23 : #include <rtl/alloc.h>
24 : #include <rtl/digest.h>
25 : #include <rtl/random.h>
26 :
27 : /*========================================================================
28 : *
29 : * rtlRandom internals.
30 : *
31 : *======================================================================*/
32 : #define RTL_RANDOM_RNG_1(a) ((a) * 16807L)
33 : #define RTL_RANDOM_RNG_2(a) ((a) * 65539L)
34 :
35 : #define RTL_RANDOM_RNG(x, y, z) \
36 : { \
37 : (x) = 170 * ((x) % 178) - 63 * ((x) / 178); \
38 : if ((x) < 0) (x) += 30328L; \
39 : \
40 : (y) = 171 * ((y) % 177) - 2 * ((y) / 177); \
41 : if ((y) < 0) (y) += 30269L; \
42 : \
43 : (z) = 172 * ((z) % 176) - 35 * ((z) / 176); \
44 : if ((z) < 0) (z) += 30307L; \
45 : }
46 :
47 : /** RandomData_Impl.
48 : */
49 : struct RandomData_Impl
50 : {
51 : sal_Int16 m_nX;
52 : sal_Int16 m_nY;
53 : sal_Int16 m_nZ;
54 : };
55 :
56 : /** __rtl_random_data.
57 : */
58 : static double __rtl_random_data (RandomData_Impl *pImpl);
59 :
60 : /** RandomPool_Impl.
61 : */
62 : #define RTL_RANDOM_DIGEST rtl_Digest_AlgorithmMD5
63 : #define RTL_RANDOM_SIZE_DIGEST RTL_DIGEST_LENGTH_MD5
64 : #define RTL_RANDOM_SIZE_POOL 1023
65 :
66 : struct RandomPool_Impl
67 : {
68 : rtlDigest m_hDigest;
69 : sal_uInt8 m_pDigest[RTL_RANDOM_SIZE_DIGEST];
70 : sal_uInt8 m_pData[RTL_RANDOM_SIZE_POOL + 1];
71 : sal_uInt32 m_nData;
72 : sal_uInt32 m_nIndex;
73 : sal_uInt32 m_nCount;
74 : };
75 :
76 : /** __rtl_random_initPool.
77 : */
78 : static sal_Bool __rtl_random_initPool (
79 : RandomPool_Impl *pImpl);
80 :
81 : /** __rtl_random_seedPool.
82 : */
83 : static void __rtl_random_seedPool (
84 : RandomPool_Impl *pImpl, const sal_uInt8 *pBuffer, sal_Size nBufLen);
85 :
86 : /** __rtl_random_readPool.
87 : */
88 : static void __rtl_random_readPool (
89 : RandomPool_Impl *pImpl, sal_uInt8 *pBuffer, sal_Size nBufLen);
90 :
91 : /*
92 : * __rtl_random_data.
93 : */
94 69300 : static double __rtl_random_data (RandomData_Impl *pImpl)
95 : {
96 : double random;
97 :
98 69300 : RTL_RANDOM_RNG (pImpl->m_nX, pImpl->m_nY, pImpl->m_nZ);
99 138600 : random = (((double)(pImpl->m_nX) / 30328.0) +
100 69300 : ((double)(pImpl->m_nY) / 30269.0) +
101 69300 : ((double)(pImpl->m_nZ) / 30307.0) );
102 :
103 69300 : random -= ((double)((sal_uInt32)(random)));
104 69300 : return (random);
105 : }
106 :
107 : /*
108 : * __rtl_random_initPool.
109 : */
110 550 : static sal_Bool __rtl_random_initPool (RandomPool_Impl *pImpl)
111 : {
112 550 : pImpl->m_hDigest = rtl_digest_create (RTL_RANDOM_DIGEST);
113 550 : if (pImpl->m_hDigest)
114 : {
115 : oslThreadIdentifier id;
116 : TimeValue tv;
117 : RandomData_Impl rd;
118 : double seed;
119 :
120 : /* The use of uninitialized stack variables as a way to
121 : * enhance the entropy of the random pool triggers
122 : * memory checkers like purify and valgrind.
123 : */
124 :
125 : /*
126 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&id, sizeof(id));
127 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&tv, sizeof(tv));
128 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&rd, sizeof(rd));
129 : */
130 :
131 550 : id = osl_getThreadIdentifier (NULL);
132 550 : id = RTL_RANDOM_RNG_2(RTL_RANDOM_RNG_1(id));
133 550 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&id, sizeof(id));
134 :
135 550 : osl_getSystemTime (&tv);
136 550 : tv.Seconds = RTL_RANDOM_RNG_2(tv.Seconds);
137 550 : tv.Nanosec = RTL_RANDOM_RNG_2(tv.Nanosec);
138 550 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&tv, sizeof(tv));
139 :
140 550 : rd.m_nX = (sal_Int16)(((id >> 1) << 1) + 1);
141 550 : rd.m_nY = (sal_Int16)(((tv.Seconds >> 1) << 1) + 1);
142 550 : rd.m_nZ = (sal_Int16)(((tv.Nanosec >> 1) << 1) + 1);
143 550 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&rd, sizeof(rd));
144 :
145 70400 : while (pImpl->m_nData < RTL_RANDOM_SIZE_POOL)
146 : {
147 69300 : seed = __rtl_random_data (&rd);
148 69300 : __rtl_random_seedPool (pImpl, (sal_uInt8*)&seed, sizeof(seed));
149 : }
150 550 : return sal_True;
151 : }
152 0 : return sal_False;
153 : }
154 :
155 : /*
156 : * __rtl_random_seedPool.
157 : */
158 71184 : static void __rtl_random_seedPool (
159 : RandomPool_Impl *pImpl, const sal_uInt8 *pBuffer, sal_Size nBufLen)
160 : {
161 : sal_Size i;
162 : sal_sSize j, k;
163 :
164 142368 : for (i = 0; i < nBufLen; i += RTL_RANDOM_SIZE_DIGEST)
165 : {
166 71184 : j = nBufLen - i;
167 71184 : if (j > RTL_RANDOM_SIZE_DIGEST)
168 0 : j = RTL_RANDOM_SIZE_DIGEST;
169 :
170 : rtl_digest_update (
171 71184 : pImpl->m_hDigest, pImpl->m_pDigest, RTL_RANDOM_SIZE_DIGEST);
172 :
173 71184 : k = (pImpl->m_nIndex + j) - RTL_RANDOM_SIZE_POOL;
174 71184 : if (k > 0)
175 : {
176 : rtl_digest_update (
177 550 : pImpl->m_hDigest, &(pImpl->m_pData[pImpl->m_nIndex]), j - k);
178 : rtl_digest_update (
179 550 : pImpl->m_hDigest, &(pImpl->m_pData[0]), k);
180 : }
181 : else
182 : {
183 : rtl_digest_update (
184 70634 : pImpl->m_hDigest, &(pImpl->m_pData[pImpl->m_nIndex]), j);
185 : }
186 :
187 71184 : rtl_digest_update (pImpl->m_hDigest, pBuffer, j);
188 71184 : pBuffer += j;
189 :
190 : rtl_digest_get (
191 71184 : pImpl->m_hDigest, pImpl->m_pDigest, RTL_RANDOM_SIZE_DIGEST);
192 637356 : for (k = 0; k < j; k++)
193 : {
194 566172 : pImpl->m_pData[pImpl->m_nIndex++] ^= pImpl->m_pDigest[k];
195 566172 : if (pImpl->m_nIndex >= RTL_RANDOM_SIZE_POOL)
196 : {
197 550 : pImpl->m_nData = RTL_RANDOM_SIZE_POOL;
198 550 : pImpl->m_nIndex = 0;
199 : }
200 : }
201 : }
202 :
203 71184 : if (pImpl->m_nIndex > pImpl->m_nData)
204 70400 : pImpl->m_nData = pImpl->m_nIndex;
205 71184 : }
206 :
207 : /*
208 : * __rtl_random_readPool.
209 : */
210 5105 : static void __rtl_random_readPool (
211 : RandomPool_Impl *pImpl, sal_uInt8 *pBuffer, sal_Size nBufLen)
212 : {
213 : sal_Int32 j, k;
214 :
215 17963 : while (nBufLen > 0)
216 : {
217 7753 : j = nBufLen;
218 7753 : if (j > RTL_RANDOM_SIZE_DIGEST/2)
219 2648 : j = RTL_RANDOM_SIZE_DIGEST/2;
220 7753 : nBufLen -= j;
221 :
222 : rtl_digest_update (
223 : pImpl->m_hDigest,
224 : &(pImpl->m_pDigest[RTL_RANDOM_SIZE_DIGEST/2]),
225 7753 : RTL_RANDOM_SIZE_DIGEST/2);
226 :
227 7753 : k = (pImpl->m_nIndex + j) - pImpl->m_nData;
228 7753 : if (k > 0)
229 : {
230 : rtl_digest_update (
231 11 : pImpl->m_hDigest, &(pImpl->m_pData[pImpl->m_nIndex]), j - k);
232 : rtl_digest_update (
233 11 : pImpl->m_hDigest, &(pImpl->m_pData[0]), k);
234 : }
235 : else
236 : {
237 : rtl_digest_update (
238 7742 : pImpl->m_hDigest, &(pImpl->m_pData[pImpl->m_nIndex]), j);
239 : }
240 :
241 : rtl_digest_get (
242 7753 : pImpl->m_hDigest, pImpl->m_pDigest, RTL_RANDOM_SIZE_DIGEST);
243 65548 : for (k = 0; k < j; k++)
244 : {
245 57795 : if (pImpl->m_nIndex >= pImpl->m_nData) pImpl->m_nIndex = 0;
246 57795 : pImpl->m_pData[pImpl->m_nIndex++] ^= pImpl->m_pDigest[k];
247 57795 : *pBuffer++ = pImpl->m_pDigest[k + RTL_RANDOM_SIZE_DIGEST/2];
248 : }
249 : }
250 :
251 5105 : pImpl->m_nCount++;
252 : rtl_digest_update (
253 5105 : pImpl->m_hDigest, &(pImpl->m_nCount), sizeof(pImpl->m_nCount));
254 : rtl_digest_update (
255 5105 : pImpl->m_hDigest, pImpl->m_pDigest, RTL_RANDOM_SIZE_DIGEST);
256 : rtl_digest_get (
257 5105 : pImpl->m_hDigest, pImpl->m_pDigest, RTL_RANDOM_SIZE_DIGEST);
258 5105 : }
259 :
260 : /*========================================================================
261 : *
262 : * rtlRandom implementation.
263 : *
264 : *======================================================================*/
265 : /*
266 : * rtl_random_createPool.
267 : */
268 550 : rtlRandomPool SAL_CALL rtl_random_createPool() SAL_THROW_EXTERN_C()
269 : {
270 550 : RandomPool_Impl *pImpl = (RandomPool_Impl*)NULL;
271 550 : pImpl = (RandomPool_Impl*)rtl_allocateZeroMemory (sizeof(RandomPool_Impl));
272 550 : if (pImpl)
273 : {
274 550 : if (!__rtl_random_initPool (pImpl))
275 : {
276 0 : rtl_freeZeroMemory (pImpl, sizeof(RandomPool_Impl));
277 0 : pImpl = (RandomPool_Impl*)NULL;
278 : }
279 : }
280 550 : return ((rtlRandomPool)pImpl);
281 : }
282 :
283 : /*
284 : * rtl_random_destroyPool.
285 : */
286 293 : void SAL_CALL rtl_random_destroyPool (rtlRandomPool Pool) SAL_THROW_EXTERN_C()
287 : {
288 293 : RandomPool_Impl *pImpl = (RandomPool_Impl *)Pool;
289 293 : if (pImpl)
290 : {
291 293 : rtl_digest_destroy (pImpl->m_hDigest);
292 293 : rtl_freeZeroMemory (pImpl, sizeof (RandomPool_Impl));
293 : }
294 293 : }
295 :
296 : /*
297 : * rtl_random_addBytes.
298 : */
299 234 : rtlRandomError SAL_CALL rtl_random_addBytes (
300 : rtlRandomPool Pool, const void *Buffer, sal_Size Bytes) SAL_THROW_EXTERN_C()
301 : {
302 234 : RandomPool_Impl *pImpl = (RandomPool_Impl *)Pool;
303 234 : const sal_uInt8 *pBuffer = (const sal_uInt8 *)Buffer;
304 :
305 234 : if ((pImpl == NULL) || (pBuffer == NULL))
306 0 : return rtl_Random_E_Argument;
307 :
308 234 : __rtl_random_seedPool (pImpl, pBuffer, Bytes);
309 234 : return rtl_Random_E_None;
310 : }
311 :
312 : /*
313 : * rtl_random_getBytes.
314 : */
315 5105 : rtlRandomError SAL_CALL rtl_random_getBytes (
316 : rtlRandomPool Pool, void *Buffer, sal_Size Bytes) SAL_THROW_EXTERN_C()
317 : {
318 5105 : RandomPool_Impl *pImpl = (RandomPool_Impl *)Pool;
319 5105 : sal_uInt8 *pBuffer = (sal_uInt8 *)Buffer;
320 :
321 5105 : if ((pImpl == NULL) || (pBuffer == NULL))
322 0 : return rtl_Random_E_Argument;
323 :
324 5105 : __rtl_random_readPool (pImpl, pBuffer, Bytes);
325 5105 : return rtl_Random_E_None;
326 : }
327 :
328 : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|