X-Git-Url: https://git.sesse.net/?a=blobdiff_plain;f=src%2Fmisc%2Frand.c;h=82c67c3d7f4626ce10b8495d7dc6c83f51b65705;hb=c12acbd7d48431d7b5029d765e69e87378aca7a2;hp=b49087171ab086cd561773ba2a07a5a809daf601;hpb=8a0b8bd8d22cf77b80e370fb018b45946002e2fe;p=vlc diff --git a/src/misc/rand.c b/src/misc/rand.c index b49087171a..82c67c3d7f 100644 --- a/src/misc/rand.c +++ b/src/misc/rand.c @@ -19,7 +19,11 @@ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/ -#include +#ifdef HAVE_CONFIG_H +# include "config.h" +#endif + +#include #include #ifndef WIN32 @@ -31,6 +35,7 @@ #include #include #include +#include #include @@ -53,7 +58,7 @@ static void vlc_rand_init (void) uint8_t key[BLOCK_SIZE]; /* Get non-predictible value as key for HMAC */ - int fd = open (randfile, O_RDONLY); + int fd = vlc_open (randfile, O_RDONLY); if (fd == -1) return; /* Uho! */ @@ -87,19 +92,21 @@ void vlc_rand_bytes (void *buf, size_t len) uint64_t val; struct md5_s mdi, mdo; + InitMD5 (&mdi); + InitMD5 (&mdo); + pthread_mutex_lock (&lock); if (counter == 0) vlc_rand_init (); val = counter++; - pthread_mutex_unlock (&lock); - InitMD5 (&mdi); AddMD5 (&mdi, ikey, sizeof (ikey)); + AddMD5 (&mdo, okey, sizeof (okey)); + pthread_mutex_unlock (&lock); + AddMD5 (&mdi, &stamp, sizeof (stamp)); AddMD5 (&mdi, &val, sizeof (val)); EndMD5 (&mdi); - InitMD5 (&mdo); - AddMD5 (&mdo, okey, sizeof (okey)); AddMD5 (&mdo, mdi.p_digest, sizeof (mdi.p_digest)); EndMD5 (&mdo); @@ -123,34 +130,115 @@ void vlc_rand_bytes (void *buf, size_t len) { HCRYPTPROV hProv; size_t count = len; + uint8_t *p_buf = (uint8_t *)buf; /* fill buffer with pseudo-random data */ while (count > 0) { unsigned int val; - val = rand(); - if (count < sizeof (val)) - { - memcpy (buf, &val, count); - break; - } - - memcpy (buf, &val, sizeof (val)); - count -= sizeof (val); + val = rand(); + if (count < sizeof (val)) + { + memcpy (p_buf, &val, count); + break; + } + + memcpy (p_buf, &val, sizeof (val)); + count -= sizeof (val); + p_buf += sizeof (val); } /* acquire default encryption context */ if( CryptAcquireContext( - &hProv, // Variable to hold returned handle. - NULL, // Use default key container. - MS_DEF_PROV, // Use default CSP. - PROV_RSA_FULL, // Type of provider to acquire. - 0) ) + &hProv, // Variable to hold returned handle. + NULL, // Use default key container. + MS_DEF_PROV, // Use default CSP. + PROV_RSA_FULL, // Type of provider to acquire. + 0) ) { /* fill buffer with pseudo-random data, intial buffer content - is used as auxillary random seed */ + is used as auxillary random seed */ CryptGenRandom(hProv, len, buf); - CryptReleaseContext(hProv, 0); + CryptReleaseContext(hProv, 0); } } #endif + +static struct +{ + bool init; + unsigned short subi[3]; + vlc_mutex_t lock; +} rand48 = { false, { 0, 0, 0, }, VLC_STATIC_MUTEX, }; + +static void init_rand48 (void) +{ + if (!rand48.init) + { + vlc_rand_bytes (rand48.subi, sizeof (rand48.subi)); +#if 0 // short would be more than 16-bits ? + for (unsigned i = 0; i < 3; i++) + subi[i] &= 0xffff; +#endif + } +} + +/** + * PRNG uniformly distributed between 0.0 and 1.0 with 48-bits precision. + * + * @note Contrary to POSIX drand48(), this function is thread-safe. + * @warning Series generated by this function are not reproducible. + * Use erand48() if you need reproducible series. + * + * @return a double value within [0.0, 1.0] inclusive + */ +double vlc_drand48 (void) +{ + double ret; + + vlc_mutex_lock (&rand48.lock); + init_rand48 (); + ret = erand48 (rand48.subi); + vlc_mutex_unlock (&rand48.lock); + return ret; +} + +/** + * PRNG uniformly distributed between 0 and 2^32 - 1. + * + * @note Contrary to POSIX lrand48(), this function is thread-safe. + * @warning Series generated by this function are not reproducible. + * Use nrand48() if you need reproducible series. + * + * @return an integral value within [0.0, 2^32-1] inclusive + */ +long vlc_lrand48 (void) +{ + long ret; + + vlc_mutex_lock (&rand48.lock); + init_rand48 (); + ret = nrand48 (rand48.subi); + vlc_mutex_unlock (&rand48.lock); + return ret; +} + +/** + * PRNG uniformly distributed between -2^32 and 2^32 - 1. + * + * @note Contrary to POSIX mrand48(), this function is thread-safe. + * @warning Series generated by this function are not reproducible. + * Use jrand48() if you need reproducible series. + * + * @return an integral value within [-2^32, 2^32-1] inclusive + */ +long vlc_mrand48 (void) +{ + long ret; + + vlc_mutex_lock (&rand48.lock); + init_rand48 (); + ret = jrand48 (rand48.subi); + vlc_mutex_unlock (&rand48.lock); + return ret; +}