2 * Copyright (c) 1995, 1996, 1997, 1999, 2007 Kungliga Tekniska Högskolan
3 * (Royal Institute of Technology, Stockholm, Sweden).
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
17 * 3. Neither the name of the Institute nor the names of its contributors
18 * may be used to endorse or promote products derived from this software
19 * without specific prior written permission.
21 * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
49 #ifndef WIN32 /* don't bother with this on windows */
51 static volatile int counter
;
52 static volatile unsigned char *gdata
; /* Global data */
53 static volatile int igdata
; /* Index into global data */
61 gdata
[igdata
++] ^= counter
& 0xff;
63 #ifndef HAVE_SIGACTION
64 signal(SIGALRM
, sigALRM
); /* Reinstall SysV signal handler */
69 #ifndef HAVE_SETITIMER
71 pacemaker(struct timeval
*tv
)
79 select(1, &fds
, NULL
, NULL
, tv
);
86 /* XXX ugly hack, should perhaps use function from roken */
88 (*fake_signal(int sig
, RETSIGTYPE (*f
)(int)))(int)
90 struct sigaction sa
, osa
;
93 sigemptyset(&sa
.sa_mask
);
94 sigaction(sig
, &sa
, &osa
);
95 return osa
.sa_handler
;
97 #define signal(S, F) fake_signal((S), (F))
107 timer_seed(const void *indata
, int size
)
112 timer_bytes(unsigned char *outdata
, int size
)
117 struct itimerval tv
, otv
;
118 RETSIGTYPE (*osa
)(int);
120 #ifndef HAVE_SETITIMER
121 RETSIGTYPE (*ochld
)(int);
129 osa
= signal(SIGALRM
, sigALRM
);
132 tv
.it_value
.tv_sec
= 0;
133 tv
.it_value
.tv_usec
= 10 * 1000; /* 10 ms */
134 tv
.it_interval
= tv
.it_value
;
135 #ifdef HAVE_SETITIMER
136 setitimer(ITIMER_REAL
, &tv
, &otv
);
138 ochld
= signal(SIGCHLD
, SIG_IGN
);
141 signal(SIGCHLD
, ochld
!= SIG_ERR
? ochld
: SIG_DFL
);
142 des_not_rand_data(data
, size
);
146 pacemaker(&tv
.it_interval
);
149 for(i
= 0; i
< 4; i
++) {
150 for (igdata
= 0; igdata
< size
;) /* igdata++ in sigALRM */
152 for (j
= 0; j
< size
; j
++) /* Only use 2 bits each lap */
153 gdata
[j
] = (gdata
[j
]>>2) | (gdata
[j
]<<6);
155 #ifdef HAVE_SETITIMER
156 setitimer(ITIMER_REAL
, &otv
, 0);
159 while(waitpid(pid
, NULL
, 0) != pid
);
160 signal(SIGCHLD
, ochld
!= SIG_ERR
? ochld
: SIG_DFL
);
162 signal(SIGALRM
, osa
!= SIG_ERR
? osa
: SIG_DFL
);
174 timer_add(const void *indata
, int size
, double entropi
)
179 timer_pseudorand(unsigned char *outdata
, int size
)
181 return timer_bytes(outdata
, size
);
194 const RAND_METHOD hc_rand_timer_method
= {
204 RAND_timer_method(void)
206 return &hc_rand_timer_method
;