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
44 #ifndef WIN32 /* don't bother with this on windows */
46 static volatile int counter
;
47 static volatile unsigned char *gdata
; /* Global data */
48 static volatile int igdata
; /* Index into global data */
56 gdata
[igdata
++] ^= counter
& 0xff;
58 #ifndef HAVE_SIGACTION
59 signal(SIGALRM
, sigALRM
); /* Reinstall SysV signal handler */
64 #ifndef HAVE_SETITIMER
66 pacemaker(struct timeval
*tv
)
74 select(1, &fds
, NULL
, NULL
, tv
);
81 /* XXX ugly hack, should perhaps use function from roken */
83 (*fake_signal(int sig
, RETSIGTYPE (*f
)(int)))(int)
85 struct sigaction sa
, osa
;
88 sigemptyset(&sa
.sa_mask
);
89 sigaction(sig
, &sa
, &osa
);
90 return osa
.sa_handler
;
92 #define signal(S, F) fake_signal((S), (F))
102 timer_seed(const void *indata
, int size
)
107 timer_bytes(unsigned char *outdata
, int size
)
112 struct itimerval tv
, otv
;
113 RETSIGTYPE (*osa
)(int);
115 #ifndef HAVE_SETITIMER
116 RETSIGTYPE (*ochld
)(int);
124 osa
= signal(SIGALRM
, sigALRM
);
127 tv
.it_value
.tv_sec
= 0;
128 tv
.it_value
.tv_usec
= 10 * 1000; /* 10 ms */
129 tv
.it_interval
= tv
.it_value
;
130 #ifdef HAVE_SETITIMER
131 setitimer(ITIMER_REAL
, &tv
, &otv
);
133 ochld
= signal(SIGCHLD
, SIG_IGN
);
136 signal(SIGCHLD
, ochld
!= SIG_ERR
? ochld
: SIG_DFL
);
137 des_not_rand_data(data
, size
);
141 pacemaker(&tv
.it_interval
);
144 for(i
= 0; i
< 4; i
++) {
145 for (igdata
= 0; igdata
< size
;) /* igdata++ in sigALRM */
147 for (j
= 0; j
< size
; j
++) /* Only use 2 bits each lap */
148 gdata
[j
] = (gdata
[j
]>>2) | (gdata
[j
]<<6);
150 #ifdef HAVE_SETITIMER
151 setitimer(ITIMER_REAL
, &otv
, 0);
154 while(waitpid(pid
, NULL
, 0) != pid
);
155 signal(SIGCHLD
, ochld
!= SIG_ERR
? ochld
: SIG_DFL
);
157 signal(SIGALRM
, osa
!= SIG_ERR
? osa
: SIG_DFL
);
169 timer_add(const void *indata
, int size
, double entropi
)
174 timer_pseudorand(unsigned char *outdata
, int size
)
176 return timer_bytes(outdata
, size
);
189 const RAND_METHOD hc_rand_timer_method
= {
199 RAND_timer_method(void)
201 return &hc_rand_timer_method
;