2 * Copyright (c) 1999, 2000-2001 Apple Computer, Inc. All rights reserved.
4 * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
6 * This file contains Original Code and/or Modifications of Original Code
7 * as defined in and that are subject to the Apple Public Source License
8 * Version 2.0 (the 'License'). You may not use this file except in
9 * compliance with the License. The rights granted to you under the License
10 * may not be used to create, or enable the creation or redistribution of,
11 * unlawful or unlicensed copies of an Apple operating system, or to
12 * circumvent, violate, or enable the circumvention or violation of, any
13 * terms of an Apple operating system software license agreement.
15 * Please obtain a copy of the License at
16 * http://www.opensource.apple.com/apsl/ and read it before using this file.
18 * The Original Code and all software distributed under the License are
19 * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22 * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23 * Please see the License for the specific language governing rights and
24 * limitations under the License.
26 * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
32 Contains: Core routines for the Counterpane Yarrow PRNG.
34 Written by: Counterpane, Inc.
36 Copyright: (c) 2000 by Apple Computer, Inc., all rights reserved.
38 Change History (most recent first):
40 02/10/99 dpm Created, based on Counterpane source.
46 Core routines for the Counterpane PRNG
48 #include "userdefines.h"
49 #include "assertverify.h"
50 #include "dev/random/YarrowCoreLib/include/yarrowUtils.h"
52 #if defined(macintosh) || defined(__APPLE__)
53 /* FIXME - this file needs to be in a platform-independent place */
56 #endif /* macintosh */
59 #include "entropysources.h"
61 #include "dev/random/YarrowCoreLib/include/yarrow.h"
66 #define _MAX(a,b) (((a)>(b))?(a):(b))
67 #define _MIN(a,b) (((a)<(b))?(a):(b))
69 #if defined(macintosh) || defined(__APPLE__)
71 * No mutexes in this module for Macintosh/OSX. We handle the
72 * required locking elsewhere.
74 #define MUTEX_ENABLE 0
76 #include <string.h> /* memcpy, etc. */
77 #if TARGET_API_MAC_OSX
78 #include <sys/time.h> /* for timespec */
79 #elif TARGET_API_MAC_CARBON
80 #include <Timer.h> /* Microseconds */
85 #error Unknown TARGET_API
86 #endif /* TARGET_API */
88 #define MUTEX_ENABLE 1
89 #endif /* macintosh */
92 static HANDLE Statmutex
= NULL
;
93 static DWORD mutexCreatorId
= 0;
97 #pragma mark * * * Static Utility functions * * *
99 /* All error checking should be done in the function that calls these */
102 * out := SHA1(IV | out)
105 prng_do_SHA1(GEN_CTX
*ctx
)
110 YSHA1Update(&sha
,ctx
->IV
,20);
111 YSHA1Update(&sha
,ctx
->out
,20);
112 YSHA1Final(ctx
->out
,&sha
);
120 * Called from init, prngForceReseed(), and prngOutput()
121 * as anti-backtracking mechanism.
124 prng_make_new_state(GEN_CTX
*ctx
,BYTE
*newState
)
128 memcpy(ctx
->IV
,newState
,20);
130 YSHA1Update(&sha
,ctx
->IV
,20);
131 YSHA1Final(ctx
->out
,&sha
);
139 /* Initialize the secret state with a slow poll */
140 /* Currently only called from prngInitialize */
142 #define SPLEN 65536 /* 64K */
145 prng_slow_init(PRNG
*p
)
146 /* This fails silently and must be fixed. */
148 YSHA1_CTX
* ctx
= NULL
;
149 MMPTR mmctx
= MM_NULL
;
151 MMPTR mmbigbuf
= MM_NULL
;
153 MMPTR mmbuf
= MM_NULL
;
156 mmbigbuf
= mmMalloc(SPLEN
);
157 if(mmbigbuf
== MM_NULL
) {goto cleanup_slow_init
;}
158 bigbuf
= (BYTE
*)mmGetPtr(mmbigbuf
);
160 mmbuf
= mmMalloc(20);
161 if(mmbuf
== MM_NULL
) {goto cleanup_slow_init
;}
162 buf
= (BYTE
*)mmGetPtr(mmbuf
);
164 mmctx
= mmMalloc(sizeof(YSHA1_CTX
));
165 if(mmctx
== MM_NULL
) {goto cleanup_slow_init
;}
166 ctx
= (YSHA1_CTX
*)mmGetPtr(mmctx
);
169 /* Initialize the secret state. */
170 /* Init entropy pool */
172 /* Init output generator */
173 polllength
= prng_slow_poll(bigbuf
,SPLEN
);
175 YSHA1Update(ctx
,bigbuf
,polllength
);
177 prng_make_new_state(&p
->outstate
, buf
);
187 #endif /* SLOW_POLL_ENABLE */
189 /* In-place modifed bubble sort */
191 bubbleSort( UINT
*data
, LONG len
)
202 if(data
[i
+1] > data
[i
])
215 #pragma mark * * * Public functions * * *
217 /* Set up the PRNG */
219 prngInitialize(PrngRef
*prng
)
222 comp_error_status resp
;
223 prng_error_status retval
= PRNG_ERR_LOW_MEMORY
;
230 /* Create the mutex */
231 /* NOTE: on return the mutex should bve held, since our caller (prngInitialize)
234 if(mutexCreatorId
!=0) {return PRNG_ERR_REINIT
;}
235 Statmutex
= CreateMutex(NULL
,TRUE
,NULL
);
236 if(Statmutex
== NULL
) {mutexCreatorId
= 0; return PRNG_ERR_MUTEX
;}
237 DuplicateHandle(GetCurrentProcess(),Statmutex
,GetCurrentProcess(),&mutex
,SYNCHRONIZE
,FALSE
,0);
238 mutexCreatorId
= GetCurrentProcessId();
239 #endif /* MUTEX_ENABLE */
242 mmp
= mmMalloc(sizeof(PRNG
));
249 p
= (PRNG
*)mmGetPtr(mmp
);
250 memset(p
, 0, sizeof(PRNG
));
253 /* Initialize Variables */
254 for(i
=0;i
<TOTAL_SOURCES
;i
++)
257 p
->poolEstBits
[i
] = 0;
261 /* Setup security on the registry so that remote users cannot predict the slow pool */
262 prng_set_NT_security();
265 /* Initialize the secret state. */
266 /* FIXME - might want to make this an option here and have the caller
267 * do it after we return....? */
270 prng_slow_init(p
); /* Does a slow poll and then calls prng_make_state(...) */
273 prng_do_SHA1(&p
->outstate
);
274 prng_make_new_state(&p
->outstate
, p
->outstate
.out
);
275 #endif /* SLOW_POLL_ENABLE */
277 /* Initialize compression routines */
278 for(i
=0;i
<COMP_SOURCES
;i
++)
280 resp
= comp_init((p
->comp_state
)+i
);
281 if(resp
!=COMP_SUCCESS
) {retval
= PRNG_ERR_COMPRESSION
; goto cleanup_init
;}
284 p
->ready
= PRNG_READY
;
290 /* Program failed on one of the mmmallocs */
295 CloseHandle(Statmutex
);
300 return retval
; /* default PRNG_ERR_LOW_MEMORY */
305 prngOutput(PRNG
*p
, BYTE
*outbuf
,UINT outbuflen
)
308 GEN_CTX
*ctx
= &p
->outstate
;
313 chASSERT(BACKTRACKLIMIT
> 0);
315 for(i
=0;i
<outbuflen
;i
++,ctx
->index
++,ctx
->numout
++)
317 /* Check backtracklimit */
318 if(ctx
->numout
> BACKTRACKLIMIT
)
321 prng_make_new_state(ctx
, ctx
->out
);
323 /* Check position in IV */
329 outbuf
[i
] = (ctx
->out
)[ctx
->index
];
336 /* Cause the PRNG to reseed now regardless of entropy pool */
337 /* Should this be public? */
339 prngForceReseed(PRNG
*p
, LONGLONG ticks
)
343 FILETIME a
,b
,c
,usertime
;
347 #if defined(macintosh) || defined(__APPLE__)
348 #if (defined(TARGET_API_MAC_OSX) || defined(KERNEL_BUILD))
350 int64_t endTime
, curTime
;
351 #else /* TARGET_API_MAC_CARBON */
352 UnsignedWide uwide
; /* struct needed for Microseconds() */
362 /* Set up start and end times */
363 #if defined(macintosh) || defined(__APPLE__)
364 #if (defined(TARGET_API_MAC_OSX) || defined(KERNEL_BUILD))
365 /* note we can't loop for more than a million microseconds */
369 gettimeofday(&tv
, NULL
);
371 endTime
= (int64_t)tv
.tv_sec
*1000000LL + (int64_t)tv
.tv_usec
+ ticks
;
372 #else /* TARGET_API_MAC_OSX */
373 Microseconds(&uwide
);
374 start
= UnsignedWideToUInt64(uwide
);
375 #endif /* TARGET_API_xxx */
376 #endif /* macintosh */
379 /* Do a couple of iterations between time checks */
380 prngOutput(p
, buf
,64);
381 YSHA1Update(&p
->pool
,buf
,64);
382 prngOutput(p
, buf
,64);
383 YSHA1Update(&p
->pool
,buf
,64);
384 prngOutput(p
, buf
,64);
385 YSHA1Update(&p
->pool
,buf
,64);
386 prngOutput(p
, buf
,64);
387 YSHA1Update(&p
->pool
,buf
,64);
388 prngOutput(p
, buf
,64);
389 YSHA1Update(&p
->pool
,buf
,64);
391 #if defined(macintosh) || defined(__APPLE__)
392 #if defined(TARGET_API_MAC_OSX) || defined(KERNEL_BUILD)
393 #ifdef TARGET_API_MAC_OSX
394 gettimeofday(&tv
, NULL
);
397 curTime
= (int64_t)tv
.tv_sec
*1000000LL + (int64_t)tv
.tv_usec
;
399 } while(curTime
< endTime
);
401 Microseconds(&uwide
);
402 now
= UnsignedWideToUInt64(uwide
);
403 } while ( (now
-start
) < ticks
) ;
406 } while ( (now
-start
) < ticks
) ;
408 YSHA1Final(dig
,&p
->pool
);
409 YSHA1Update(&p
->pool
,dig
,20);
410 YSHA1Final(dig
,&p
->pool
);
412 /* Reset secret state */
414 prng_make_new_state(&p
->outstate
,dig
);
416 /* Clear counter variables */
417 for(i
=0;i
<TOTAL_SOURCES
;i
++)
420 p
->poolEstBits
[i
] = 0;
424 trashMemory(dig
,20*sizeof(char));
425 trashMemory(buf
,64*sizeof(char));
431 /* Input a state into the PRNG */
433 prngProcessSeedBuffer(PRNG
*p
, BYTE
*buf
,LONGLONG ticks
)
439 /* Put the data into the entropy, add some data from the unknown state, reseed */
440 YSHA1Update(&p
->pool
,buf
,20); /* Put it into the entropy pool */
441 prng_do_SHA1(&p
->outstate
); /* Output 20 more bytes and */
442 YSHA1Update(&p
->pool
,p
->outstate
.out
,20);/* add it to the pool as well. */
443 prngForceReseed(p
, ticks
); /* Do a reseed */
444 return prngOutput(p
, buf
,20); /* Return the first 20 bytes of output in buf */
448 /* Take some "random" data and make more "random-looking" data from it */
449 /* note: this routine has no context, no mutex wrapper */
451 prngStretch(BYTE
*inbuf
,UINT inbuflen
,BYTE
*outbuf
,UINT outbuflen
) {
459 if(inbuflen
>= outbuflen
)
461 memcpy(outbuf
,inbuf
,outbuflen
);
464 else /* Extend using SHA1 hash of inbuf */
467 YSHA1Update(&ctx
,inbuf
,inbuflen
);
468 YSHA1Final(dig
,&ctx
);
469 for(prev
=0,left
=outbuflen
;left
>0;prev
+=20,left
-=20)
471 YSHA1Update(&ctx
,dig
,20);
472 YSHA1Final(dig
,&ctx
);
473 memcpy(outbuf
+prev
,dig
,(left
>20)?20:left
);
475 trashMemory(dig
,20*sizeof(BYTE
));
480 return PRNG_ERR_PROGRAM_FLOW
;
484 /* Add entropy to the PRNG from a source */
486 prngInput(PRNG
*p
, BYTE
*inbuf
,UINT inbuflen
,UINT poolnum
, __unused UINT estbits
)
488 #ifndef YARROW_KERNEL
489 comp_error_status resp
;
495 if(poolnum
>= TOTAL_SOURCES
) {return PRNG_ERR_OUT_OF_BOUNDS
;}
497 /* Add to entropy pool */
498 YSHA1Update(&p
->pool
,inbuf
,inbuflen
);
500 #ifndef YARROW_KERNEL
501 /* skip this step for the kernel */
503 /* Update pool size, pool user estimate and pool compression context */
504 p
->poolSize
[poolnum
] += inbuflen
;
505 p
->poolEstBits
[poolnum
] += estbits
;
506 if(poolnum
<COMP_SOURCES
)
508 resp
= comp_add_data((p
->comp_state
)+poolnum
,inbuf
,inbuflen
);
509 if(resp
!=COMP_SUCCESS
) {return PRNG_ERR_COMPRESSION
;}
511 #endif /* YARROW_KERNEL */
518 /* If we have enough entropy, allow a reseed of the system */
520 prngAllowReseed(PRNG
*p
, LONGLONG ticks
)
522 UINT temp
[TOTAL_SOURCES
];
530 comp_error_status resp
;
535 for(i
=0;i
<ENTROPY_SOURCES
;i
++)
537 /* Make sure that compression-based entropy estimates are current */
538 #ifndef KERNEL_BUILD // floating point in a kernel is BAD!
539 resp
= comp_get_ratio((p
->comp_state
)+i
,&ratio
);
540 if(resp
!=COMP_SUCCESS
) {return PRNG_ERR_COMPRESSION
;}
541 /* Use 4 instead of 8 to half compression estimate */
542 temp
[i
] = (int)(ratio
*p
->poolSize
[i
]*4);
544 temp
[i
] = p
->poolSize
[i
] * 4;
548 /* Use minumum of user and compression estimate for compressed sources */
549 for(i
=ENTROPY_SOURCES
;i
<COMP_SOURCES
;i
++)
552 /* Make sure that compression-based entropy estimates are current */
553 resp
= comp_get_ratio((p
->comp_state
)+i
,&ratio
);
554 if(resp
!=COMP_SUCCESS
) {return PRNG_ERR_COMPRESSION
;}
555 /* Use 4 instead of 8 to half compression estimate */
556 temp
[i
] = _MIN((int)(ratio
*p
->poolSize
[i
]*4),(int)p
->poolEstBits
[i
]);
558 temp
[i
] = _MIN (p
->poolSize
[i
] * 4, p
->poolEstBits
[i
]);
562 /* Use user estimate for remaining sources */
563 for(i
=COMP_SOURCES
;i
<TOTAL_SOURCES
;i
++) {temp
[i
] = p
->poolEstBits
[i
];}
566 /* pointless if we're not ignoring any sources */
567 bubbleSort(temp
,TOTAL_SOURCES
);
569 for(i
=K
,sum
=0;i
<TOTAL_SOURCES
;sum
+=temp
[i
++]); /* Stupid C trick */
571 return prngForceReseed(p
, ticks
);
573 return PRNG_ERR_NOT_ENOUGH_ENTROPY
;
575 return PRNG_ERR_PROGRAM_FLOW
;
579 /* Call a slow poll and insert the data into the entropy pool */
580 static prng_error_status
581 prngSlowPoll(PRNG
*p
, UINT pollsize
)
585 prng_error_status retval
;
589 buf
= (BYTE
*)malloc(pollsize
);
590 if(buf
==NULL
) {return PRNG_ERR_LOW_MEMORY
;}
591 len
= prng_slow_poll(buf
,pollsize
); /* OS specific call */
592 retval
= prngInput(p
, buf
,len
,SLOWPOLLSOURCE
, len
* 8);
593 trashMemory(buf
,pollsize
);
598 #endif /* SLOW_POLL_ENABLE */
601 /* Delete the PRNG */
608 if(GetCurrentProcessId()!=mutexCreatorId
) {return PRNG_ERR_WRONG_CALLER
;}
610 if(p
==NULL
) {return PRNG_SUCCESS
;} /* Well, there is nothing to destroy... */
612 p
->ready
= PRNG_NOT_READY
;
614 for(i
=0;i
<COMP_SOURCES
;i
++)
616 comp_end((p
->comp_state
)+i
);
620 CloseHandle(Statmutex
);