mirror of
https://gitlab.com/scemama/qmcchem.git
synced 2024-12-22 04:13:31 +01:00
Fixed DMC
This commit is contained in:
parent
dc6d6b1637
commit
75d099a2e8
@ -630,7 +630,7 @@ end = struct
|
|||||||
let of_float x =
|
let of_float x =
|
||||||
if (x >= 100.) then
|
if (x >= 100.) then
|
||||||
failwith "DMC Projection time should be < 100.";
|
failwith "DMC Projection time should be < 100.";
|
||||||
if (x <= 0.) then
|
if (x < 0.) then
|
||||||
failwith "DMC Projection time should be positive.";
|
failwith "DMC Projection time should be positive.";
|
||||||
x
|
x
|
||||||
|
|
||||||
@ -886,6 +886,16 @@ let validate () =
|
|||||||
| _ -> ()
|
| _ -> ()
|
||||||
in
|
in
|
||||||
|
|
||||||
|
(*
|
||||||
|
(* Check Projection time is greater than time step *)
|
||||||
|
let () =
|
||||||
|
match (meth, DMC_projection_time.(read () |> to_float) ) with
|
||||||
|
| (Method.DMC,p) ->
|
||||||
|
if (p < ts) then failwith "E_ref should not be zero in DMC"
|
||||||
|
| _ -> ()
|
||||||
|
in
|
||||||
|
*)
|
||||||
|
|
||||||
(* Set block and total time*)
|
(* Set block and total time*)
|
||||||
let () =
|
let () =
|
||||||
if ( (Block_time.read ()) > Stop_time.read ()) then
|
if ( (Block_time.read ()) > Stop_time.read ()) then
|
||||||
|
@ -10,7 +10,7 @@ t = """
|
|||||||
&BEGIN_PROVIDER [ $T, $X_2_dmc_block_walk_kahan $D2 ]
|
&BEGIN_PROVIDER [ $T, $X_2_dmc_block_walk_kahan $D2 ]
|
||||||
implicit none
|
implicit none
|
||||||
BEGIN_DOC
|
BEGIN_DOC
|
||||||
! VMC averages of $X
|
! DMC averages of $X. Computed in E_loc_dmc_block_walk
|
||||||
END_DOC
|
END_DOC
|
||||||
$X_dmc_block_walk = 0.d0
|
$X_dmc_block_walk = 0.d0
|
||||||
$X_dmc_block_walk_kahan = 0.d0
|
$X_dmc_block_walk_kahan = 0.d0
|
||||||
@ -27,14 +27,15 @@ for p in properties:
|
|||||||
D1 = ", ("+p[2][1:-1]+")"
|
D1 = ", ("+p[2][1:-1]+")"
|
||||||
D2 = ", ("+p[2][1:-1]+",3)"
|
D2 = ", ("+p[2][1:-1]+",3)"
|
||||||
print t.replace("$X",p[1]).replace("$T",p[0]).replace("$D1",D1).replace("$D2",D2)
|
print t.replace("$X",p[1]).replace("$T",p[0]).replace("$D1",D1).replace("$D2",D2)
|
||||||
|
|
||||||
END_SHELL
|
END_SHELL
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
BEGIN_PROVIDER [ double precision, E_loc_dmc_block_walk ]
|
BEGIN_PROVIDER [ double precision, E_loc_dmc_block_walk ]
|
||||||
&BEGIN_PROVIDER [ double precision, E_loc_2_dmc_block_walk ]
|
&BEGIN_PROVIDER [ double precision, E_loc_2_dmc_block_walk ]
|
||||||
&BEGIN_PROVIDER [ double precision, E_loc_dmc_block_walk_kahan, (3) ]
|
&BEGIN_PROVIDER [ double precision, E_loc_dmc_block_walk_kahan, (3) ]
|
||||||
&BEGIN_PROVIDER [ double precision, E_loc_2_dmc_block_walk_kahan, (3)
|
&BEGIN_PROVIDER [ double precision, E_loc_2_dmc_block_walk_kahan, (3) ]
|
||||||
implicit none
|
implicit none
|
||||||
include '../types.F'
|
include '../types.F'
|
||||||
BEGIN_DOC
|
BEGIN_DOC
|
||||||
@ -43,19 +44,18 @@ END_SHELL
|
|||||||
|
|
||||||
real, allocatable :: elec_coord_tmp(:,:,:)
|
real, allocatable :: elec_coord_tmp(:,:,:)
|
||||||
integer :: mod_align
|
integer :: mod_align
|
||||||
double precision, allocatable :: psi_grad_psi_inv_save_tmp(:,:,:)
|
double precision :: psi_value_save(walk_num)
|
||||||
double precision :: psi_value_save_tmp(walk_num)
|
double precision :: psi_value_save_tmp(walk_num)
|
||||||
integer :: trapped_walk_tmp(walk_num)
|
integer :: trapped_walk_tmp(walk_num)
|
||||||
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: psi_grad_psi_inv_save_tmp
|
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: psi_value_save
|
||||||
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: psi_value_save_tmp
|
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: psi_value_save_tmp
|
||||||
allocate ( elec_coord_tmp(mod_align(elec_num+1),3,walk_num) )
|
allocate ( elec_coord_tmp(mod_align(elec_num+1),3,walk_num) )
|
||||||
allocate ( psi_grad_psi_inv_save_tmp(elec_num_8,3,walk_num) )
|
psi_value_save = 0.d0
|
||||||
|
|
||||||
! Initialization
|
! Initialization
|
||||||
if (vmc_algo /= t_Brownian) then
|
if (vmc_algo /= t_Brownian) then
|
||||||
call abrt(irp_here,'DMC should run with Brownian algorithm')
|
call abrt(irp_here,'DMC should run with Brownian algorithm')
|
||||||
endif
|
endif
|
||||||
PROVIDE E_loc_vmc_block_walk
|
|
||||||
|
|
||||||
integer :: k, i_walk, i_step
|
integer :: k, i_walk, i_step
|
||||||
|
|
||||||
@ -63,9 +63,13 @@ BEGIN_SHELL [ /usr/bin/python ]
|
|||||||
from properties import *
|
from properties import *
|
||||||
t = """
|
t = """
|
||||||
if (calc_$X) then
|
if (calc_$X) then
|
||||||
|
!DIR$ VECTOR ALIGNED
|
||||||
$X_dmc_block_walk = 0.d0
|
$X_dmc_block_walk = 0.d0
|
||||||
|
!DIR$ VECTOR ALIGNED
|
||||||
$X_dmc_block_walk_kahan = 0.d0
|
$X_dmc_block_walk_kahan = 0.d0
|
||||||
|
!DIR$ VECTOR ALIGNED
|
||||||
$X_2_dmc_block_walk = 0.d0
|
$X_2_dmc_block_walk = 0.d0
|
||||||
|
!DIR$ VECTOR ALIGNED
|
||||||
$X_2_dmc_block_walk_kahan = 0.d0
|
$X_2_dmc_block_walk_kahan = 0.d0
|
||||||
$X_min = huge(1.)
|
$X_min = huge(1.)
|
||||||
$X_max =-huge(1.)
|
$X_max =-huge(1.)
|
||||||
@ -75,65 +79,49 @@ for p in properties:
|
|||||||
print t.replace("$X",p[1])
|
print t.replace("$X",p[1])
|
||||||
END_SHELL
|
END_SHELL
|
||||||
|
|
||||||
double precision :: icount
|
|
||||||
|
|
||||||
icount = 0.d0
|
|
||||||
|
|
||||||
logical :: loop
|
logical :: loop
|
||||||
integer*8 :: cpu0, cpu1, cpu2, count_rate, count_max
|
integer*8 :: cpu0, cpu1, cpu2, count_rate, count_max
|
||||||
|
|
||||||
loop = .True.
|
loop = .True.
|
||||||
call system_clock(cpu0, count_rate, count_max)
|
call system_clock(cpu0, count_rate, count_max)
|
||||||
cpu2 = cpu0
|
cpu2 = cpu0
|
||||||
|
|
||||||
|
block_weight = 0.d0
|
||||||
|
|
||||||
do while (loop)
|
do while (loop)
|
||||||
! Move to the next projection step
|
|
||||||
dmc_projection_step = mod(dmc_projection_step,dmc_projection)+1
|
|
||||||
|
|
||||||
! Remove contribution of the old value of the weight at the new
|
! Every walker makes a step
|
||||||
! projection step
|
do i_walk=1,walk_num
|
||||||
pop_weight_mult *= 1.d0/pop_weight(dmc_projection_step)
|
integer :: i,j,l
|
||||||
|
do l=1,3
|
||||||
! Compute the new weight of the population
|
do i=1,elec_num+1
|
||||||
pop_weight(dmc_projection_step) = 0.d0
|
elec_coord(i,l) = elec_coord_full(i,l,i_walk)
|
||||||
do k=1,walk_num
|
enddo
|
||||||
pop_weight(dmc_projection_step) += dmc_weight(k)
|
enddo
|
||||||
enddo
|
TOUCH elec_coord
|
||||||
|
|
||||||
! Normalize the weight of the walkers by the weight of the population
|
|
||||||
do k=1,walk_num
|
|
||||||
dmc_weight(k) = dmc_weight(k)/pop_weight(dmc_projection_step)
|
|
||||||
enddo
|
|
||||||
|
|
||||||
! Normalize the weight of the population at the current projection step by
|
|
||||||
! the number of walkers
|
|
||||||
pop_weight(dmc_projection_step) = pop_weight(dmc_projection_step)/dble(walk_num)
|
|
||||||
|
|
||||||
! Update the running population weight
|
|
||||||
pop_weight_mult *= pop_weight(dmc_projection_step)
|
|
||||||
SOFT_TOUCH pop_weight_mult
|
|
||||||
|
|
||||||
BEGIN_SHELL [ /usr/bin/python ]
|
BEGIN_SHELL [ /usr/bin/python ]
|
||||||
from properties import *
|
from properties import *
|
||||||
t = """
|
t = """
|
||||||
if (calc_$X) then
|
if (calc_$X) then
|
||||||
! Kahan's summation algorithm to compute these sums reducing the rounding error:
|
! Kahan's summation algorithm to compute these sums reducing the rounding error:
|
||||||
! $X_dmc_block_walk($D) += $X * pop_weight_mult
|
! $X_dmc_block_walk += $X * pop_weight_mult
|
||||||
! $X_2_dmc_block_walk($D) += $X_2 * pop_weight_mult
|
! $X_2_dmc_block_walk += $X_2 * pop_weight_mult
|
||||||
! see http://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
! see http://en.wikipedia.org/wiki/Kahan_summation_algorithm
|
||||||
|
|
||||||
$X_dmc_block_walk_kahan($D2 3) = $X * pop_weight_mult - $X_dmc_block_walk_kahan($D2 1)
|
$X_dmc_block_walk_kahan($D2 3) = $X * pop_weight_mult - $X_dmc_block_walk_kahan($D2 1)
|
||||||
$X_dmc_block_walk_kahan($D2 2) = $X_dmc_block_walk $D1 + $X_dmc_block_walk_kahan($D2 3)
|
$X_dmc_block_walk_kahan($D2 2) = $X_dmc_block_walk $D1 + $X_dmc_block_walk_kahan($D2 3)
|
||||||
$X_dmc_block_walk_kahan($D2 1) = ($X_dmc_block_walk_kahan($D2 2) - $X_dmc_block_walk $D1 ) &
|
$X_dmc_block_walk_kahan($D2 1) = ($X_dmc_block_walk_kahan($D2 2) - $X_dmc_block_walk $D1 ) &
|
||||||
- $X_dmc_block_walk_kahan($D2 3)
|
- $X_dmc_block_walk_kahan($D2 3)
|
||||||
$X_dmc_block_walk $D1 = $X_dmc_block_walk_kahan($D2 2)
|
$X_dmc_block_walk $D1 = $X_dmc_block_walk_kahan($D2 2)
|
||||||
|
|
||||||
|
|
||||||
$X_2_dmc_block_walk_kahan($D2 3) = $X_2 * pop_weight_mult - $X_2_dmc_block_walk_kahan($D2 1)
|
$X_2_dmc_block_walk_kahan($D2 3) = $X_2 * pop_weight_mult - $X_2_dmc_block_walk_kahan($D2 1)
|
||||||
$X_2_dmc_block_walk_kahan($D2 2) = $X_2_dmc_block_walk $D1 + $X_2_dmc_block_walk_kahan($D2 3)
|
$X_2_dmc_block_walk_kahan($D2 2) = $X_2_dmc_block_walk $D1 + $X_2_dmc_block_walk_kahan($D2 3)
|
||||||
$X_2_dmc_block_walk_kahan($D2 1) = ($X_2_dmc_block_walk_kahan($D2 2) - $X_2_dmc_block_walk $D1 ) &
|
$X_2_dmc_block_walk_kahan($D2 1) = ($X_2_dmc_block_walk_kahan($D2 2) - $X_2_dmc_block_walk $D1 ) &
|
||||||
- $X_2_dmc_block_walk_kahan($D2 3)
|
- $X_2_dmc_block_walk_kahan($D2 3)
|
||||||
$X_2_dmc_block_walk $D1 = $X_2_dmc_block_walk_kahan($D2 2)
|
$X_2_dmc_block_walk $D1 = $X_2_dmc_block_walk_kahan($D2 2)
|
||||||
endif
|
endif
|
||||||
"""
|
"""
|
||||||
for p in properties:
|
for p in properties:
|
||||||
if p[2] == "":
|
if p[2] == "":
|
||||||
@ -143,25 +131,83 @@ for p in properties:
|
|||||||
D1 = "("+":"*(p[2].count(',')+1)+")"
|
D1 = "("+":"*(p[2].count(',')+1)+")"
|
||||||
D2 = ":"*(p[2].count(',')+1)+","
|
D2 = ":"*(p[2].count(',')+1)+","
|
||||||
print t.replace("$X",p[1]).replace("$D1",D1).replace("$D2",D2)
|
print t.replace("$X",p[1]).replace("$D1",D1).replace("$D2",D2)
|
||||||
|
|
||||||
END_SHELL
|
END_SHELL
|
||||||
icount += pop_weight_mult
|
|
||||||
|
|
||||||
|
double precision :: p,q
|
||||||
|
real :: delta_x
|
||||||
|
logical :: accepted
|
||||||
|
call brownian_step(p,q,accepted,delta_x)
|
||||||
|
if (accepted) then
|
||||||
|
trapped_walk(i_walk) = 0
|
||||||
|
else
|
||||||
|
trapped_walk(i_walk) += 1
|
||||||
|
endif
|
||||||
|
|
||||||
|
if ( (trapped_walk(i_walk) < trapped_walk_max).and. &
|
||||||
|
(psi_value * psi_value_save(i_walk) >= 0.d0) ) then
|
||||||
|
dmc_weight(i_walk) = dexp(dtime_step*(E_ref - E_loc))
|
||||||
|
else
|
||||||
|
dmc_weight(i_walk) = 0.d0
|
||||||
|
trapped_walk(i_walk) = 0
|
||||||
|
endif
|
||||||
|
|
||||||
|
elec_coord(elec_num+1,1) += p*time_step
|
||||||
|
elec_coord(elec_num+1,2) = E_loc
|
||||||
|
elec_coord(elec_num+1,3) = dmc_weight(i_walk)
|
||||||
|
do l=1,3
|
||||||
|
do i=1,elec_num+1
|
||||||
|
elec_coord_full(i,l,i_walk) = elec_coord(i,l)
|
||||||
|
enddo
|
||||||
|
enddo
|
||||||
|
|
||||||
|
psi_value_save(i_walk) = psi_value
|
||||||
|
|
||||||
|
enddo
|
||||||
|
|
||||||
|
! Move to the next projection step
|
||||||
|
if (dmc_projection > 0) then
|
||||||
|
dmc_projection_step = mod(dmc_projection_step,dmc_projection)+1
|
||||||
|
else
|
||||||
|
dmc_projection_step = 1
|
||||||
|
endif
|
||||||
|
|
||||||
|
! Eventually, recompute the weight of the population
|
||||||
|
if (dmc_projection_step == 1) then
|
||||||
|
pop_weight_mult = 1.d0
|
||||||
|
do k=1,dmc_projection
|
||||||
|
pop_weight_mult *= pop_weight(k)
|
||||||
|
enddo
|
||||||
|
endif
|
||||||
|
|
||||||
|
! Remove contribution of the old value of the weight at the new
|
||||||
|
! projection step
|
||||||
|
pop_weight_mult *= 1.d0/pop_weight(dmc_projection_step)
|
||||||
|
|
||||||
|
! Compute the new weight of the population
|
||||||
|
double precision :: sum_weight
|
||||||
|
sum_weight = 0.d0
|
||||||
|
do k=1,walk_num
|
||||||
|
sum_weight += dmc_weight(k)
|
||||||
|
enddo
|
||||||
|
pop_weight(dmc_projection_step) = sum_weight/dble(walk_num)
|
||||||
|
|
||||||
|
! Update the running population weight
|
||||||
|
pop_weight_mult *= pop_weight(dmc_projection_step)
|
||||||
|
|
||||||
|
block_weight += pop_weight_mult * dble(walk_num)
|
||||||
|
|
||||||
! Reconfiguration
|
! Reconfiguration
|
||||||
|
|
||||||
integer :: ipos(walk_num)
|
integer :: ipos(walk_num)
|
||||||
|
|
||||||
call reconfigure(ipos,dmc_weight)
|
call reconfigure(ipos,dmc_weight)
|
||||||
|
|
||||||
do k=1,walk_num
|
do k=1,walk_num
|
||||||
integer :: i, l
|
|
||||||
do l=1,3
|
do l=1,3
|
||||||
do i=1,elec_num+1
|
do i=1,elec_num+1
|
||||||
elec_coord_tmp(i,l,k) = elec_coord_full(i,l,k)
|
elec_coord_tmp(i,l,k) = elec_coord_full(i,l,k)
|
||||||
enddo
|
enddo
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
!DIR$ LOOP COUNT(200)
|
|
||||||
do i=1,elec_num
|
|
||||||
psi_grad_psi_inv_save_tmp(i,l,k) = psi_grad_psi_inv_save(i,l,k)
|
|
||||||
enddo
|
|
||||||
enddo
|
enddo
|
||||||
psi_value_save_tmp(k) = psi_value_save(k)
|
psi_value_save_tmp(k) = psi_value_save(k)
|
||||||
trapped_walk_tmp(k) = trapped_walk(k)
|
trapped_walk_tmp(k) = trapped_walk(k)
|
||||||
@ -174,36 +220,11 @@ END_SHELL
|
|||||||
do i=1,elec_num+1
|
do i=1,elec_num+1
|
||||||
elec_coord_full(i,l,k) = elec_coord_tmp(i,l,ipm)
|
elec_coord_full(i,l,k) = elec_coord_tmp(i,l,ipm)
|
||||||
enddo
|
enddo
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
!DIR$ LOOP COUNT(200)
|
|
||||||
do i=1,elec_num
|
|
||||||
psi_grad_psi_inv_save(i,l,k) = psi_grad_psi_inv_save_tmp(i,l,ipm)
|
|
||||||
enddo
|
|
||||||
enddo
|
enddo
|
||||||
psi_value_save(k) = psi_value_save_tmp(ipm)
|
psi_value_save(k) = psi_value_save_tmp(ipm)
|
||||||
trapped_walk(k) = trapped_walk_tmp(ipm)
|
trapped_walk(k) = trapped_walk_tmp(ipm)
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
! Set 1st walker
|
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
!DIR$ LOOP COUNT(200)
|
|
||||||
do i=1,elec_num
|
|
||||||
psi_grad_psi_inv_x(i) = psi_grad_psi_inv_save(i,1,1)
|
|
||||||
psi_grad_psi_inv_y(i) = psi_grad_psi_inv_save(i,2,1)
|
|
||||||
psi_grad_psi_inv_z(i) = psi_grad_psi_inv_save(i,3,1)
|
|
||||||
enddo
|
|
||||||
|
|
||||||
!DIR$ VECTOR UNALIGNED
|
|
||||||
!DIR$ LOOP COUNT(200)
|
|
||||||
do i=1,elec_num
|
|
||||||
elec_coord(i,1) = elec_coord_full(i,1,1)
|
|
||||||
elec_coord(i,2) = elec_coord_full(i,2,1)
|
|
||||||
elec_coord(i,3) = elec_coord_full(i,3,1)
|
|
||||||
enddo
|
|
||||||
psi_value = psi_value_save(1)
|
|
||||||
|
|
||||||
TOUCH elec_coord_full psi_value_save psi_grad_psi_inv_save psi_value psi_grad_psi_inv_x psi_grad_psi_inv_y psi_grad_psi_inv_z elec_coord
|
|
||||||
|
|
||||||
call system_clock(cpu1, count_rate, count_max)
|
call system_clock(cpu1, count_rate, count_max)
|
||||||
if (cpu1 < cpu0) then
|
if (cpu1 < cpu0) then
|
||||||
cpu1 = cpu1+cpu0
|
cpu1 = cpu1+cpu0
|
||||||
@ -216,14 +237,14 @@ END_SHELL
|
|||||||
cpu2 = cpu1
|
cpu2 = cpu1
|
||||||
endif
|
endif
|
||||||
|
|
||||||
|
SOFT_TOUCH elec_coord_full psi_value psi_grad_psi_inv_x psi_grad_psi_inv_y psi_grad_psi_inv_z elec_coord pop_weight_mult
|
||||||
|
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
double precision :: factor
|
double precision :: factor
|
||||||
factor = 1.d0/icount
|
factor = 1.d0/block_weight
|
||||||
block_weight *= icount
|
|
||||||
SOFT_TOUCH block_weight
|
SOFT_TOUCH block_weight
|
||||||
|
|
||||||
BEGIN_SHELL [ /usr/bin/python ]
|
BEGIN_SHELL [ /usr/bin/python ]
|
||||||
from properties import *
|
from properties import *
|
||||||
t = """
|
t = """
|
||||||
@ -237,7 +258,6 @@ for p in properties:
|
|||||||
END_SHELL
|
END_SHELL
|
||||||
|
|
||||||
deallocate ( elec_coord_tmp )
|
deallocate ( elec_coord_tmp )
|
||||||
deallocate ( psi_grad_psi_inv_save_tmp )
|
|
||||||
|
|
||||||
END_PROVIDER
|
END_PROVIDER
|
||||||
|
|
||||||
@ -272,12 +292,33 @@ END_PROVIDER
|
|||||||
dmc_projection_step = 0
|
dmc_projection_step = 0
|
||||||
END_PROVIDER
|
END_PROVIDER
|
||||||
|
|
||||||
BEGIN_PROVIDER [ double precision, pop_weight, (dmc_projection) ]
|
BEGIN_PROVIDER [ double precision, pop_weight, (0:dmc_projection+1) ]
|
||||||
implicit none
|
implicit none
|
||||||
BEGIN_DOC
|
BEGIN_DOC
|
||||||
! Population weight of DMC
|
! Population weight of DMC
|
||||||
END_DOC
|
END_DOC
|
||||||
pop_weight = 1.d0
|
pop_weight = 1.d0
|
||||||
pop_weight(dmc_projection) = 1.d0/dble(dmc_projection)
|
pop_weight(dmc_projection) = 1.d0/dble(size(pop_weight))
|
||||||
END_PROVIDER
|
END_PROVIDER
|
||||||
|
|
||||||
|
BEGIN_PROVIDER [ integer, trapped_walk, (walk_num_8) ]
|
||||||
|
&BEGIN_PROVIDER [ integer, trapped_walk_max ]
|
||||||
|
implicit none
|
||||||
|
BEGIN_DOC
|
||||||
|
! Number of steps when the walkers were trapped
|
||||||
|
END_DOC
|
||||||
|
trapped_walk = 0
|
||||||
|
trapped_walk_max = 20
|
||||||
|
END_PROVIDER
|
||||||
|
|
||||||
|
|
||||||
|
BEGIN_PROVIDER [ double precision, dmc_weight, (walk_num_8) ]
|
||||||
|
implicit none
|
||||||
|
BEGIN_DOC
|
||||||
|
! Weight of the walkers in the DMC algorithm: exp(-time_step*(E_loc-E_ref))
|
||||||
|
END_DOC
|
||||||
|
!DIR$ VECTOR ALIGNED
|
||||||
|
dmc_weight = 1.d0
|
||||||
|
END_PROVIDER
|
||||||
|
|
||||||
|
|
||||||
|
@ -3,44 +3,49 @@ subroutine reconfigure(ipos,w)
|
|||||||
integer, intent(inout) :: ipos(*)
|
integer, intent(inout) :: ipos(*)
|
||||||
double precision, intent(in) :: w(*)
|
double precision, intent(in) :: w(*)
|
||||||
|
|
||||||
integer :: kp, km
|
|
||||||
double precision :: accup, accum
|
|
||||||
integer :: k
|
|
||||||
|
|
||||||
double precision :: dwalk_num
|
|
||||||
dwalk_num = dble(walk_num)
|
|
||||||
|
|
||||||
integer :: kptab(walk_num), kmtab(walk_num)
|
integer :: kptab(walk_num), kmtab(walk_num)
|
||||||
double precision :: wp(walk_num), wm(walk_num)
|
double precision :: wp(walk_num), wm(walk_num)
|
||||||
double precision :: tmp
|
double precision :: tmp
|
||||||
|
|
||||||
|
double precision :: dwalk_num
|
||||||
|
|
||||||
|
tmp = 0.d0
|
||||||
do k=1,walk_num
|
do k=1,walk_num
|
||||||
ipos(k) = k
|
ipos(k) = k
|
||||||
|
tmp = tmp + w(k)
|
||||||
enddo
|
enddo
|
||||||
|
dwalk_num = dble(walk_num)/tmp
|
||||||
|
|
||||||
|
integer :: kp, km
|
||||||
kp=0
|
kp=0
|
||||||
km=0
|
km=0
|
||||||
|
|
||||||
|
double precision :: accup, accum
|
||||||
accup = 0.d0
|
accup = 0.d0
|
||||||
accum = 0.d0
|
accum = 0.d0
|
||||||
|
|
||||||
|
integer :: k
|
||||||
do k=1,walk_num
|
do k=1,walk_num
|
||||||
tmp = dwalk_num*w(k)-1.d0
|
tmp = dwalk_num*w(k)-1.d0
|
||||||
if (tmp >= 0.d0) then
|
if (tmp >= 0.d0) then
|
||||||
kp += 1
|
kp = kp+1
|
||||||
wp(kp) = abs(tmp)
|
wp(kp) = dabs(tmp)
|
||||||
accup += wp(kp)
|
accup = accup + wp(kp)
|
||||||
kptab(kp) = k
|
kptab(kp) = k
|
||||||
else
|
else
|
||||||
km += 1
|
km = km+1
|
||||||
wm(km) = abs(tmp)
|
wm(km) = dabs(tmp)
|
||||||
accum += wm(km)
|
accum = accum + wm(km)
|
||||||
kmtab(km) = k
|
kmtab(km) = k
|
||||||
endif
|
endif
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
if(kp+km /= walk_num) then
|
if(kp+km /= walk_num) then
|
||||||
print *, kp, km
|
print *, kp, km
|
||||||
call abrt(irp_here,'pb in reconfiguration +/-')
|
call abrt(irp_here,'pb in reconfiguration +/-')
|
||||||
endif
|
endif
|
||||||
if(abs(accup-accum).gt.1.d-11) then
|
|
||||||
|
if(dabs(accup-accum) > 1.d-11) then
|
||||||
print *, accup, accum
|
print *, accup, accum
|
||||||
call abrt(irp_here,'pb in reconfiguration')
|
call abrt(irp_here,'pb in reconfiguration')
|
||||||
endif
|
endif
|
||||||
@ -59,24 +64,26 @@ subroutine reconfigure(ipos,w)
|
|||||||
averageconf = accup
|
averageconf = accup
|
||||||
kcp = 1
|
kcp = 1
|
||||||
rand = rando(kcp)
|
rand = rando(kcp)
|
||||||
|
|
||||||
do while (rand < averageconf)
|
do while (rand < averageconf)
|
||||||
k=1
|
k=1
|
||||||
current=wm(k)
|
current=wm(k)
|
||||||
do while (rand > current)
|
do while (rand > current)
|
||||||
k += 1
|
k = k+1
|
||||||
current += wm(k)
|
current = current + wm(k)
|
||||||
enddo
|
enddo
|
||||||
kremove = kmtab(k)
|
kremove = kmtab(k)
|
||||||
|
|
||||||
k=1
|
k=1
|
||||||
current=wp(k)
|
current=wp(k)
|
||||||
do while (rand > current)
|
do while (rand > current)
|
||||||
k += 1
|
k = k+1
|
||||||
current += wp(k)
|
current = current + wp(k)
|
||||||
enddo
|
enddo
|
||||||
kadd = kptab(k)
|
kadd = kptab(k)
|
||||||
|
|
||||||
ipos(kremove) = kadd
|
ipos(kremove) = kadd
|
||||||
kcp += 1
|
kcp = kcp + 1
|
||||||
rand = rando(kcp)
|
rand = rando(kcp)
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
|
@ -39,7 +39,6 @@ END_SHELL
|
|||||||
|
|
||||||
PROVIDE time_step
|
PROVIDE time_step
|
||||||
|
|
||||||
|
|
||||||
BEGIN_SHELL [ /usr/bin/python ]
|
BEGIN_SHELL [ /usr/bin/python ]
|
||||||
from properties import *
|
from properties import *
|
||||||
t = """
|
t = """
|
||||||
@ -61,7 +60,6 @@ for p in properties:
|
|||||||
END_SHELL
|
END_SHELL
|
||||||
|
|
||||||
double precision :: dnorm
|
double precision :: dnorm
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
block_weight = 0.d0
|
block_weight = 0.d0
|
||||||
do i_walk=1,walk_num
|
do i_walk=1,walk_num
|
||||||
integer :: i,j,l
|
integer :: i,j,l
|
||||||
@ -71,24 +69,7 @@ END_SHELL
|
|||||||
elec_coord(i,l) = elec_coord_full(i,l,i_walk)
|
elec_coord(i,l) = elec_coord_full(i,l,i_walk)
|
||||||
enddo
|
enddo
|
||||||
enddo
|
enddo
|
||||||
endif
|
TOUCH elec_coord
|
||||||
|
|
||||||
PROVIDE psi_grad_psi_inv_save psi_value_save
|
|
||||||
|
|
||||||
if (psi_value_save(walk_num) /= 0.) then
|
|
||||||
psi_value = psi_value_save(i_walk)
|
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
!DIR$ LOOP COUNT(200)
|
|
||||||
do i=1,elec_num
|
|
||||||
psi_grad_psi_inv_x(i) = psi_grad_psi_inv_save(i,1,i_walk)
|
|
||||||
psi_grad_psi_inv_y(i) = psi_grad_psi_inv_save(i,2,i_walk)
|
|
||||||
psi_grad_psi_inv_z(i) = psi_grad_psi_inv_save(i,3,i_walk)
|
|
||||||
enddo
|
|
||||||
TOUCH psi_value psi_grad_psi_inv_x psi_grad_psi_inv_y psi_grad_psi_inv_z elec_coord
|
|
||||||
else
|
|
||||||
if (i_walk > 1) then
|
|
||||||
TOUCH elec_coord
|
|
||||||
endif
|
|
||||||
endif
|
endif
|
||||||
|
|
||||||
logical :: loop
|
logical :: loop
|
||||||
@ -101,7 +82,6 @@ END_SHELL
|
|||||||
double precision :: p,q
|
double precision :: p,q
|
||||||
real :: delta_x
|
real :: delta_x
|
||||||
logical :: accepted
|
logical :: accepted
|
||||||
double precision :: E_old
|
|
||||||
if (vmc_algo == t_Brownian) then
|
if (vmc_algo == t_Brownian) then
|
||||||
call brownian_step(p,q,accepted,delta_x)
|
call brownian_step(p,q,accepted,delta_x)
|
||||||
else if (vmc_algo == t_Langevin) then
|
else if (vmc_algo == t_Langevin) then
|
||||||
@ -109,15 +89,10 @@ END_SHELL
|
|||||||
endif
|
endif
|
||||||
elec_coord(elec_num+1,1) += p*time_step
|
elec_coord(elec_num+1,1) += p*time_step
|
||||||
elec_coord(elec_num+1,2) = E_loc
|
elec_coord(elec_num+1,2) = E_loc
|
||||||
elec_coord(elec_num+1,3) += p*time_step
|
elec_coord(elec_num+1,3) = 1.
|
||||||
if (accepted) then
|
|
||||||
trapped_walk(i_walk) = 0
|
|
||||||
else
|
|
||||||
trapped_walk(i_walk) += 1
|
|
||||||
endif
|
|
||||||
|
|
||||||
block_weight += 1.d0
|
block_weight += 1.d0
|
||||||
|
|
||||||
|
|
||||||
BEGIN_SHELL [ /usr/bin/python ]
|
BEGIN_SHELL [ /usr/bin/python ]
|
||||||
from properties import *
|
from properties import *
|
||||||
t = """
|
t = """
|
||||||
@ -153,21 +128,17 @@ for p in properties:
|
|||||||
END_SHELL
|
END_SHELL
|
||||||
|
|
||||||
|
|
||||||
if ( qmc_method == t_VMC ) then
|
call system_clock(cpu1, count_rate, count_max)
|
||||||
call system_clock(cpu1, count_rate, count_max)
|
if (cpu1 < cpu0) then
|
||||||
if (cpu1 < cpu0) then
|
cpu1 = cpu1+cpu0
|
||||||
cpu1 = cpu1+cpu0
|
endif
|
||||||
endif
|
loop = dble(cpu1-cpu0)*dble(walk_num) < dble(block_time)*dble(count_rate)
|
||||||
loop = dble(cpu1-cpu0)*dble(walk_num) < dble(block_time)*dble(count_rate)
|
if (cpu1-cpu2 > count_rate) then
|
||||||
if (cpu1-cpu2 > count_rate) then
|
integer :: do_run
|
||||||
integer :: do_run
|
call get_running(do_run)
|
||||||
call get_running(do_run)
|
loop = do_run == t_Running
|
||||||
loop = do_run == t_Running
|
cpu2 = cpu1
|
||||||
cpu2 = cpu1
|
endif
|
||||||
endif
|
|
||||||
else
|
|
||||||
loop = .False.
|
|
||||||
endif
|
|
||||||
|
|
||||||
enddo ! while (loop)
|
enddo ! while (loop)
|
||||||
|
|
||||||
@ -177,27 +148,6 @@ END_SHELL
|
|||||||
enddo
|
enddo
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
if (qmc_method == t_DMC) then
|
|
||||||
|
|
||||||
if ( (trapped_walk(i_walk) < trapped_walk_max).and. &
|
|
||||||
(psi_value * psi_value_save(i_walk) >= 0.d0) ) then
|
|
||||||
dmc_weight(i_walk) = dexp(dtime_step*(E_ref - E_loc))
|
|
||||||
else
|
|
||||||
dmc_weight(i_walk) = 0.d0
|
|
||||||
trapped_walk(i_walk) = 0
|
|
||||||
endif
|
|
||||||
|
|
||||||
psi_value_save(i_walk) = psi_value
|
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
!DIR$ LOOP COUNT (200)
|
|
||||||
do i=1,elec_num
|
|
||||||
psi_grad_psi_inv_save(i,1,i_walk) = psi_grad_psi_inv_x(i)
|
|
||||||
psi_grad_psi_inv_save(i,2,i_walk) = psi_grad_psi_inv_y(i)
|
|
||||||
psi_grad_psi_inv_save(i,3,i_walk) = psi_grad_psi_inv_z(i)
|
|
||||||
enddo
|
|
||||||
|
|
||||||
endif
|
|
||||||
|
|
||||||
enddo
|
enddo
|
||||||
|
|
||||||
double precision :: factor
|
double precision :: factor
|
||||||
@ -220,47 +170,3 @@ END_SHELL
|
|||||||
END_PROVIDER
|
END_PROVIDER
|
||||||
|
|
||||||
|
|
||||||
BEGIN_PROVIDER [ double precision, dmc_weight, (walk_num_8) ]
|
|
||||||
implicit none
|
|
||||||
BEGIN_DOC
|
|
||||||
! Weight of the walkers in the DMC algorithm: exp(-time_step*(E_loc-E_ref))
|
|
||||||
END_DOC
|
|
||||||
!DIR$ VECTOR ALIGNED
|
|
||||||
dmc_weight = 1.d0
|
|
||||||
END_PROVIDER
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
BEGIN_PROVIDER [ double precision, psi_grad_psi_inv_save, (elec_num_8,3,walk_num) ]
|
|
||||||
&BEGIN_PROVIDER [ double precision, psi_value_save, (walk_num_8) ]
|
|
||||||
implicit none
|
|
||||||
BEGIN_DOC
|
|
||||||
! psi_grad_psi_inv of the previous step to accelerate DMC
|
|
||||||
!
|
|
||||||
! updated in vmc_step
|
|
||||||
END_DOC
|
|
||||||
integer, save :: ifirst = 0
|
|
||||||
if (ifirst == 0) then
|
|
||||||
psi_grad_psi_inv_save = 0.d0
|
|
||||||
psi_value_save = 0.d0
|
|
||||||
ifirst = 1
|
|
||||||
endif
|
|
||||||
END_PROVIDER
|
|
||||||
|
|
||||||
BEGIN_PROVIDER [ integer, trapped_walk, (walk_num_8) ]
|
|
||||||
implicit none
|
|
||||||
BEGIN_DOC
|
|
||||||
! Number of steps when the walkers were trapped
|
|
||||||
END_DOC
|
|
||||||
trapped_walk = 0
|
|
||||||
END_PROVIDER
|
|
||||||
|
|
||||||
BEGIN_PROVIDER [ integer, trapped_walk_max ]
|
|
||||||
implicit none
|
|
||||||
BEGIN_DOC
|
|
||||||
! Max number of trapped MC steps before killing walker
|
|
||||||
END_DOC
|
|
||||||
trapped_walk_max = 20
|
|
||||||
END_PROVIDER
|
|
||||||
|
|
||||||
|
@ -89,59 +89,11 @@ end
|
|||||||
|
|
||||||
double precision function gauss()
|
double precision function gauss()
|
||||||
implicit none
|
implicit none
|
||||||
! include 'constants.F'
|
include '../constants.F'
|
||||||
double precision :: qmc_ranf
|
double precision :: qmc_ranf
|
||||||
! double precision :: u1,u2
|
double precision :: u1,u2
|
||||||
! u1=qmc_ranf()
|
u1=qmc_ranf()
|
||||||
! u2=qmc_ranf()
|
u2=qmc_ranf()
|
||||||
! gauss=sqrt(-2.d0*dlog(u1))*cos(dfour_pi*u2)
|
gauss=dsqrt(-2.d0*dlog(u1))*dcos(dfour_pi*u2)
|
||||||
double precision :: inverse_normal_cdf
|
|
||||||
gauss = inverse_normal_cdf(qmc_ranf())
|
|
||||||
end
|
|
||||||
|
|
||||||
double precision function inverse_normal_cdf(p)
|
|
||||||
implicit none
|
|
||||||
double precision, intent(in) :: p
|
|
||||||
double precision :: p_low,p_high
|
|
||||||
double precision :: a1,a2,a3,a4,a5,a6
|
|
||||||
double precision :: b1,b2,b3,b4,b5
|
|
||||||
double precision :: c1,c2,c3,c4,c5,c6
|
|
||||||
double precision :: d1,d2,d3,d4
|
|
||||||
double precision :: z,q,r
|
|
||||||
double precision :: qmc_ranf
|
|
||||||
a1=-39.6968302866538d0
|
|
||||||
a2=220.946098424521d0
|
|
||||||
a3=-275.928510446969d0
|
|
||||||
a4=138.357751867269d0
|
|
||||||
a5=-30.6647980661472d0
|
|
||||||
a6=2.50662827745924d0
|
|
||||||
b1=-54.4760987982241d0
|
|
||||||
b2=161.585836858041d0
|
|
||||||
b3=-155.698979859887d0
|
|
||||||
b4=66.8013118877197d0
|
|
||||||
b5=-13.2806815528857d0
|
|
||||||
c1=-0.00778489400243029d0
|
|
||||||
c2=-0.322396458041136d0
|
|
||||||
c3=-2.40075827716184d0
|
|
||||||
c4=-2.54973253934373d0
|
|
||||||
c5=4.37466414146497d0
|
|
||||||
c6=2.93816398269878d0
|
|
||||||
d1=0.00778469570904146d0
|
|
||||||
d2=0.32246712907004d0
|
|
||||||
d3=2.445134137143d0
|
|
||||||
d4=3.75440866190742d0
|
|
||||||
p_low=0.02425d0
|
|
||||||
p_high=1.d0-0.02425d0
|
|
||||||
if(p < p_low) then
|
|
||||||
q=dsqrt(-2.d0*dlog(p))
|
|
||||||
inverse_normal_cdf=(((((c1*q+c2)*q+c3)*q+c4)*q+c5)*q+c6)/((((d1*q+d2)*q+d3)*q+d4)*q+1.d0)
|
|
||||||
else if(p <= p_high) then
|
|
||||||
q=p-0.5d0
|
|
||||||
r=q*q
|
|
||||||
inverse_normal_cdf=(((((a1*r+a2)*r+a3)*r+a4)*r+a5)*r+a6)*q/(((((b1*r+b2)*r+b3)*r+b4)*r+b5)*r+1.d0)
|
|
||||||
else
|
|
||||||
q=dsqrt(-2.d0*dlog(max(tiny(1.d0),1.d0-p)))
|
|
||||||
inverse_normal_cdf=-(((((c1*q+c2)*q+c3)*q+c4)*q+c5)*q+c6)/((((d1*q+d2)*q+d3)*q+d4)*q+1)
|
|
||||||
endif
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user