mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-26 10:30:22 -04:00 
			
		
		
		
	git-svn-id: svn+ssh://svn.code.sf.net/p/wsjt/wsjt/branches/wsjtx@6273 ab8295b8-cf94-4d9e-aec4-7959e3be5d79
		
			
				
	
	
		
			56 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			Fortran
		
	
	
	
	
	
			
		
		
	
	
			56 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			Fortran
		
	
	
	
	
	
| !
 | |
| ! Generate a seed for the RANDOM_NUMBER PRNG that is guaranteed to be
 | |
| ! unique even if many processes are started simultaneously
 | |
| !
 | |
| subroutine init_random_seed()
 | |
|   use iso_fortran_env, only: int64
 | |
|   implicit none
 | |
|   integer, allocatable :: seed(:)
 | |
|   integer :: i, n, un, istat, dt(8), pid
 | |
|   integer(int64) :: t
 | |
| 
 | |
|   call random_seed(size = n)
 | |
|   allocate(seed(n))
 | |
|   ! First try if the OS provides a random number generator
 | |
|   open(newunit=un, file="/dev/urandom", access="stream", &
 | |
|        form="unformatted", action="read", status="old", iostat=istat)
 | |
|   if (istat == 0) then
 | |
|      read(un) seed
 | |
|      close(un)
 | |
|   else
 | |
|      ! Fallback to XOR:ing the current time and pid. The PID is
 | |
|      ! useful in case one launches multiple instances of the same
 | |
|      ! program in parallel.
 | |
|      call system_clock(t)
 | |
|      if (t == 0) then
 | |
|         call date_and_time(values=dt)
 | |
|         t = (dt(1) - 1970) * 365_int64 * 24 * 60 * 60 * 1000 &
 | |
|              + dt(2) * 31_int64 * 24 * 60 * 60 * 1000 &
 | |
|              + dt(3) * 24_int64 * 60 * 60 * 1000 &
 | |
|              + dt(5) * 60 * 60 * 1000 &
 | |
|              + dt(6) * 60 * 1000 + dt(7) * 1000 &
 | |
|              + dt(8)
 | |
|      end if
 | |
|      pid = getpid()
 | |
|      t = ieor(t, int(pid, kind(t)))
 | |
|      do i = 1, n
 | |
|         seed(i) = lcg(t)
 | |
|      end do
 | |
|   end if
 | |
|   call random_seed(put=seed)
 | |
| contains
 | |
|   ! This simple PRNG might not be good enough for real work, but is
 | |
|   ! sufficient for seeding a better PRNG.
 | |
|   function lcg(s)
 | |
|     integer :: lcg
 | |
|     integer(int64) :: s
 | |
|     if (s == 0) then
 | |
|        s = 104729
 | |
|     else
 | |
|        s = mod(s, 4294967296_int64)
 | |
|     end if
 | |
|     s = mod(s * 279470273_int64, 4294967291_int64)
 | |
|     lcg = int(mod(s, int(huge(0), int64)), kind(0))
 | |
|   end function lcg
 | |
| end subroutine init_random_seed
 |