diff options
-rw-r--r-- | libgfortran/ChangeLog | 5 | ||||
-rw-r--r-- | libgfortran/intrinsics/random.c | 384 | ||||
-rw-r--r-- | libgfortran/io/format.c | 175 |
3 files changed, 20 insertions, 544 deletions
diff --git a/libgfortran/ChangeLog b/libgfortran/ChangeLog index 44ad2a3..0fee798 100644 --- a/libgfortran/ChangeLog +++ b/libgfortran/ChangeLog @@ -1,3 +1,8 @@ +2005-12-04 Francois-Xavier Coudert <coudert@clipper.ens.fr> + + * io/format.c: Removing unused code. + * intrinsics/random.c: Likewise. + 2005-12-02 Francois-Xavier Coudert <coudert@clipper.ens.fr> PR libfortran/25116 diff --git a/libgfortran/intrinsics/random.c b/libgfortran/intrinsics/random.c index ef7ed760..d77a381 100644 --- a/libgfortran/intrinsics/random.c +++ b/libgfortran/intrinsics/random.c @@ -51,383 +51,30 @@ static __gthread_mutex_t random_lock = __GTHREAD_MUTEX_INIT; static __gthread_mutex_t random_lock; #endif -#if 0 - -/* The Mersenne Twister code is currently commented out due to - - (1) Simple user specified seeds lead to really bad sequences for - nearly 100000 random numbers. - (2) open(), read(), and close() are not properly declared via header - files. - (3) The global index i is abused and causes unexpected behavior with - GET and PUT. - (4) See PR 15619. - - The algorithm was taken from the paper : +/* libgfortran previously had a Mersenne Twister, taken from the paper: + Mersenne Twister: 623-dimensionally equidistributed uniform pseudorandom generator. - by: Makoto Matsumoto - Takuji Nishimura - - Which appeared in the: ACM Transactions on Modelling and Computer + by Makoto Matsumoto & Takuji Nishimura + which appeared in the: ACM Transactions on Modelling and Computer Simulations: Special Issue on Uniform Random Number - Generation. ( Early in 1998 ). */ - - -#include <stdio.h> -#include <stdlib.h> -#include <sys/types.h> -#include <sys/stat.h> -#include <fcntl.h> - -#ifdef HAVE_UNISTD_H -#include <unistd.h> -#endif - -/*Use the 'big' generator by default ( period -> 2**19937 ). */ - -#define MT19937 - -/* Define the necessary constants for the algorithm. */ - -#ifdef MT19937 -enum constants -{ - N = 624, M = 397, R = 19, TU = 11, TS = 7, TT = 15, TL = 17 -}; -#define M_A 0x9908B0DF -#define T_B 0x9D2C5680 -#define T_C 0xEFC60000 -#else -enum constants -{ - N = 351, M = 175, R = 19, TU = 11, TS = 7, TT = 15, TL = 17 -}; -#define M_A 0xE4BD75F5 -#define T_B 0x655E5280 -#define T_C 0xFFD58000 -#endif - -static int i = N; -static unsigned int seed[N]; - -/* This is the routine which handles the seeding of the generator, - and also reading and writing of the seed. */ - -void -random_seed (GFC_INTEGER_4 *size, gfc_array_i4 *put, gfc_array_i4 *get) -{ - __gthread_mutex_lock (&random_lock); - - /* Initialize the seed in system dependent manner. */ - if (get == NULL && put == NULL && size == NULL) - { - int fd; - fd = open ("/dev/urandom", O_RDONLY); - if (fd < 0) - { - /* We dont have urandom. */ - GFC_UINTEGER_4 s = (GFC_UINTEGER_4) seed; - for (i = 0; i < N; i++) - { - s = s * 29943829 - 1; - seed[i] = s; - } - } - else - { - /* Using urandom, might have a length issue. */ - read (fd, &seed[0], sizeof (GFC_UINTEGER_4) * N); - close (fd); - i = N; - } - goto return_unlock; - } - - /* Return the size of the seed */ - if (size != NULL) - { - *size = N; - goto return_unlock; - } - - /* if we have gotten to this pount we have a get or put - * now we check it the array fulfills the demands in the standard . - */ - - /* Set the seed to PUT data */ - if (put != NULL) - { - /* if the rank of the array is not 1 abort */ - if (GFC_DESCRIPTOR_RANK (put) != 1) - abort (); - - /* if the array is too small abort */ - if (((put->dim[0].ubound + 1 - put->dim[0].lbound)) < N) - abort (); - - /* If this is the case the array is a temporary */ - if (put->dim[0].stride == 0) - goto return_unlock; - - /* This code now should do correct strides. */ - for (i = 0; i < N; i++) - seed[i] = put->data[i * put->dim[0].stride]; - } - - /* Return the seed to GET data */ - if (get != NULL) - { - /* if the rank of the array is not 1 abort */ - if (GFC_DESCRIPTOR_RANK (get) != 1) - abort (); - - /* if the array is too small abort */ - if (((get->dim[0].ubound + 1 - get->dim[0].lbound)) < N) - abort (); - - /* If this is the case the array is a temporary */ - if (get->dim[0].stride == 0) - goto return_unlock; - - /* This code now should do correct strides. */ - for (i = 0; i < N; i++) - get->data[i * get->dim[0].stride] = seed[i]; - } - - random_unlock: - __gthread_mutex_unlock (&random_lock); -} -iexport(random_seed); - -/* Here is the internal routine which generates the random numbers - in 'batches' based upon the need for a new batch. - It's an integer based routine known as 'Mersenne Twister'. - This implementation still lacks 'tempering' and a good verification, - but gives very good metrics. */ - -static void -random_generate (void) -{ - /* 32 bits. */ - GFC_UINTEGER_4 y; - - /* Generate batch of N. */ - int k, m; - for (k = 0, m = M; k < N - 1; k++) - { - y = (seed[k] & (-1 << R)) | (seed[k + 1] & ((1u << R) - 1)); - seed[k] = seed[m] ^ (y >> 1) ^ (-(GFC_INTEGER_4) (y & 1) & M_A); - if (++m >= N) - m = 0; - } - - y = (seed[N - 1] & (-1 << R)) | (seed[0] & ((1u << R) - 1)); - seed[N - 1] = seed[M - 1] ^ (y >> 1) ^ (-(GFC_INTEGER_4) (y & 1) & M_A); - i = 0; -} - -/* A routine to return a REAL(KIND=4). */ - -void -random_r4 (GFC_REAL_4 * harv) -{ - __gthread_mutex_lock (&random_lock); - - /* Regenerate if we need to. */ - if (i >= N) - random_generate (); - - /* Convert uint32 to REAL(KIND=4). */ - *harv = (GFC_REAL_4) ((GFC_REAL_4) (GFC_UINTEGER_4) seed[i++] / - (GFC_REAL_4) (~(GFC_UINTEGER_4) 0)); - __gthread_mutex_unlock (&random_lock); -} -iexport(random_r4); - -/* A routine to return a REAL(KIND=8). */ - -void -random_r8 (GFC_REAL_8 * harv) -{ - __gthread_mutex_lock (&random_lock); - - /* Regenerate if we need to, may waste one 32-bit value. */ - if ((i + 1) >= N) - random_generate (); - - /* Convert two uint32 to a REAL(KIND=8). */ - *harv = ((GFC_REAL_8) ((((GFC_UINTEGER_8) seed[i+1]) << 32) + seed[i])) / - (GFC_REAL_8) (~(GFC_UINTEGER_8) 0); - i += 2; - __gthread_mutex_unlock (&random_lock); -} -iexport(random_r8); + Generation. ( Early in 1998 ). -/* Code to handle arrays will follow here. */ + The Mersenne Twister code was replaced due to -/* REAL(KIND=4) REAL array. */ - -void -arandom_r4 (gfc_array_r4 * harv) -{ - index_type count[GFC_MAX_DIMENSIONS]; - index_type extent[GFC_MAX_DIMENSIONS]; - index_type stride[GFC_MAX_DIMENSIONS]; - index_type stride0; - index_type dim; - GFC_REAL_4 *dest; - int n; - - dest = harv->data; - - if (harv->dim[0].stride == 0) - harv->dim[0].stride = 1; - - dim = GFC_DESCRIPTOR_RANK (harv); - - for (n = 0; n < dim; n++) - { - count[n] = 0; - stride[n] = harv->dim[n].stride; - extent[n] = harv->dim[n].ubound + 1 - harv->dim[n].lbound; - if (extent[n] <= 0) - return; - } - - stride0 = stride[0]; - - __gthread_mutex_lock (&random_lock); - - while (dest) - { - /* Set the elements. */ - - /* regenerate if we need to */ - if (i >= N) - random_generate (); - - /* Convert uint32 to float in a hopefully g95 compiant manner */ - *dest = (GFC_REAL_4) ((GFC_REAL_4) (GFC_UINTEGER_4) seed[i++] / - (GFC_REAL_4) (~(GFC_UINTEGER_4) 0)); - - /* Advance to the next element. */ - dest += stride0; - count[0]++; - /* Advance to the next source element. */ - n = 0; - while (count[n] == extent[n]) - { - /* When we get to the end of a dimension, - reset it and increment - the next dimension. */ - count[n] = 0; - /* We could precalculate these products, - but this is a less - frequently used path so proabably not worth it. */ - dest -= stride[n] * extent[n]; - n++; - if (n == dim) - { - dest = NULL; - break; - } - else - { - count[n]++; - dest += stride[n]; - } - } - } - - __gthread_mutex_unlock (&random_lock); -} - -/* REAL(KIND=8) array. */ - -void -arandom_r8 (gfc_array_r8 * harv) -{ - index_type count[GFC_MAX_DIMENSIONS]; - index_type extent[GFC_MAX_DIMENSIONS]; - index_type stride[GFC_MAX_DIMENSIONS]; - index_type stride0; - index_type dim; - GFC_REAL_8 *dest; - int n; - - dest = harv->data; - - if (harv->dim[0].stride == 0) - harv->dim[0].stride = 1; - - dim = GFC_DESCRIPTOR_RANK (harv); - - for (n = 0; n < dim; n++) - { - count[n] = 0; - stride[n] = harv->dim[n].stride; - extent[n] = harv->dim[n].ubound + 1 - harv->dim[n].lbound; - if (extent[n] <= 0) - return; - } - - stride0 = stride[0]; - - __gthread_mutex_lock (&random_lock); - - while (dest) - { - /* Set the elements. */ - - /* regenerate if we need to, may waste one 32-bit value */ - if ((i + 1) >= N) - random_generate (); - - /* Convert two uint32 to a REAL(KIND=8). */ - *dest = ((GFC_REAL_8) ((((GFC_UINTEGER_8) seed[i+1]) << 32) + seed[i])) / - (GFC_REAL_8) (~(GFC_UINTEGER_8) 0); - i += 2; - - /* Advance to the next element. */ - dest += stride0; - count[0]++; - /* Advance to the next source element. */ - n = 0; - while (count[n] == extent[n]) - { - /* When we get to the end of a dimension, - reset it and increment - the next dimension. */ - count[n] = 0; - /* We could precalculate these products, - but this is a less - frequently used path so proabably not worth it. */ - dest -= stride[n] * extent[n]; - n++; - if (n == dim) - { - dest = NULL; - break; - } - else - { - count[n]++; - dest += stride[n]; - } - } - } - - __gthread_mutex_unlock (&random_lock); -} - -#else + (1) Simple user specified seeds lead to really bad sequences for + nearly 100000 random numbers. + (2) open(), read(), and close() were not properly declared via header + files. + (3) The global index i was abused and caused unexpected behavior with + GET and PUT. + (4) See PR 15619. -/* George Marsaglia's KISS (Keep It Simple Stupid) random number generator. - This PRNG combines: + libgfortran currently uses George Marsaglia's KISS (Keep It Simple Stupid) + random number generator. This PRNG combines: (1) The congruential generator x(n)=69069*x(n-1)+1327217885 with a period of 2^32, @@ -733,7 +380,6 @@ random_seed (GFC_INTEGER_4 *size, gfc_array_i4 *put, gfc_array_i4 *get) } iexport(random_seed); -#endif /* mersenne twister */ #ifndef __GTHREAD_MUTEX_INIT static void __attribute__((constructor)) diff --git a/libgfortran/io/format.c b/libgfortran/io/format.c index 1d7e15b..23ea317 100644 --- a/libgfortran/io/format.c +++ b/libgfortran/io/format.c @@ -1117,178 +1117,3 @@ unget_format (st_parameter_dt *dtp, const fnode *f) dtp->u.p.fmt->saved_format = f; } - - - -#if 0 - -static void dump_format1 (fnode * f); - -/* dump_format0()-- Dump a single format node */ - -void -dump_format0 (fnode * f) -{ - char *p; - int i; - - switch (f->format) - { - case FMT_COLON: - st_printf (" :"); - break; - case FMT_SLASH: - st_printf (" %d/", f->u.r); - break; - case FMT_DOLLAR: - st_printf (" $"); - break; - case FMT_T: - st_printf (" T%d", f->u.n); - break; - case FMT_TR: - st_printf (" TR%d", f->u.n); - break; - case FMT_TL: - st_printf (" TL%d", f->u.n); - break; - case FMT_X: - st_printf (" %dX", f->u.n); - break; - case FMT_S: - st_printf (" S"); - break; - case FMT_SS: - st_printf (" SS"); - break; - case FMT_SP: - st_printf (" SP"); - break; - - case FMT_LPAREN: - if (f->repeat == 1) - st_printf (" ("); - else - st_printf (" %d(", f->repeat); - - dump_format1 (f->u.child); - st_printf (" )"); - break; - - case FMT_STRING: - st_printf (" '"); - p = f->u.string.p; - for (i = f->u.string.length; i > 0; i--) - st_printf ("%c", *p++); - - st_printf ("'"); - break; - - case FMT_P: - st_printf (" %dP", f->u.k); - break; - case FMT_I: - st_printf (" %dI%d.%d", f->repeat, f->u.integer.w, f->u.integer.m); - break; - - case FMT_B: - st_printf (" %dB%d.%d", f->repeat, f->u.integer.w, f->u.integer.m); - break; - - case FMT_O: - st_printf (" %dO%d.%d", f->repeat, f->u.integer.w, f->u.integer.m); - break; - - case FMT_Z: - st_printf (" %dZ%d.%d", f->repeat, f->u.integer.w, f->u.integer.m); - break; - - case FMT_BN: - st_printf (" BN"); - break; - case FMT_BZ: - st_printf (" BZ"); - break; - case FMT_D: - st_printf (" %dD%d.%d", f->repeat, f->u.real.w, f->u.real.d); - break; - - case FMT_EN: - st_printf (" %dEN%d.%dE%d", f->repeat, f->u.real.w, f->u.real.d, - f->u.real.e); - break; - - case FMT_ES: - st_printf (" %dES%d.%dE%d", f->repeat, f->u.real.w, f->u.real.d, - f->u.real.e); - break; - - case FMT_F: - st_printf (" %dF%d.%d", f->repeat, f->u.real.w, f->u.real.d); - break; - - case FMT_E: - st_printf (" %dE%d.%dE%d", f->repeat, f->u.real.w, f->u.real.d, - f->u.real.e); - break; - - case FMT_G: - st_printf (" %dG%d.%dE%d", f->repeat, f->u.real.w, f->u.real.d, - f->u.real.e); - break; - - case FMT_L: - st_printf (" %dL%d", f->repeat, f->u.w); - break; - case FMT_A: - st_printf (" %dA%d", f->repeat, f->u.w); - break; - - default: - st_printf (" ???"); - break; - } -} - - -/* dump_format1()-- Dump a string of format nodes */ - -static void -dump_format1 (fnode * f) -{ - for (; f; f = f->next) - dump_format1 (f); -} - -/* dump_format()-- Dump the whole format node tree */ - -void -dump_format (void) -{ - st_printf ("format = "); - dump_format0 (&array[0]); - st_printf ("\n"); -} - - -void -next_test (st_parameter_dt *dtp) -{ - fnode *f; - int i; - - for (i = 0; i < 20; i++) - { - f = next_format (dtp); - if (f == NULL) - { - st_printf ("No format!\n"); - break; - } - - dump_format1 (f); - st_printf ("\n"); - } -} - -#endif |