10
0
mirror of https://github.com/LCPQ/quantum_package synced 2025-01-13 14:29:10 +01:00
quantum_package/plugins/Full_CI_ZMQ/pt2_stoch_routines.irp.f

386 lines
10 KiB
Fortran
Raw Normal View History

2018-08-29 11:30:19 +02:00
BEGIN_PROVIDER [ integer, pt2_stoch_istate ]
implicit none
BEGIN_DOC
! State for stochatsic PT2
END_DOC
pt2_stoch_istate = 1
END_PROVIDER
BEGIN_PROVIDER [ integer, pt2_N_teeth ]
&BEGIN_PROVIDER [ integer, pt2_minDetInFirstTeeth ]
&BEGIN_PROVIDER [ integer, pt2_n_tasks_max ]
&BEGIN_PROVIDER [ integer, pt2_F, (N_det_generators) ]
2017-01-30 09:38:04 +01:00
implicit none
2018-08-29 11:30:19 +02:00
pt2_F(:) = 1
pt2_F(:N_det_generators/100 + 1) = 1
pt2_n_tasks_max = N_det_generators/100 + 1
if(N_det_generators < 256) then
pt2_minDetInFirstTeeth = 1
pt2_N_teeth = 1
else
pt2_minDetInFirstTeeth = 5
pt2_N_teeth = 16
end if
2017-01-18 08:42:17 +01:00
END_PROVIDER
2018-08-29 11:30:19 +02:00
2017-11-21 10:43:38 +01:00
subroutine ZMQ_pt2(E, pt2,relative_error, absolute_error, error)
2017-01-16 18:19:00 +01:00
use f77_zmq
use selection_types
implicit none
2017-06-26 19:12:44 +02:00
character(len=64000) :: task
2017-11-27 23:18:18 +01:00
integer(ZMQ_PTR) :: zmq_to_qp_run_socket, zmq_socket_pull
2017-01-16 18:19:00 +01:00
integer, external :: omp_get_thread_num
2017-11-20 15:19:00 +01:00
double precision, intent(in) :: relative_error, absolute_error, E(N_states)
2017-11-21 10:43:38 +01:00
double precision, intent(out) :: pt2(N_states),error(N_states)
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
integer :: i, j, k
2017-01-16 18:19:00 +01:00
2017-06-26 19:12:44 +02:00
double precision, external :: omp_get_wtime
2018-01-10 18:11:49 +01:00
double precision :: state_average_weight_save(N_states), w(N_states)
2017-11-27 23:18:18 +01:00
integer(ZMQ_PTR), external :: new_zmq_to_qp_run_socket
2017-05-09 23:11:45 +02:00
2017-11-21 17:20:18 +01:00
if (N_det < max(10,N_states)) then
pt2=0.d0
2017-06-26 19:12:44 +02:00
call ZMQ_selection(0, pt2)
2017-11-21 17:20:18 +01:00
error(:) = 0.d0
2017-06-26 19:12:44 +02:00
else
2018-01-10 18:11:49 +01:00
state_average_weight_save(:) = state_average_weight(:)
2017-11-21 10:43:38 +01:00
do pt2_stoch_istate=1,N_states
2017-11-22 16:21:16 +01:00
SOFT_TOUCH pt2_stoch_istate
2018-01-10 18:11:49 +01:00
state_average_weight(:) = 0.d0
state_average_weight(pt2_stoch_istate) = 1.d0
TOUCH state_average_weight
2017-11-27 23:18:18 +01:00
2018-08-29 11:30:19 +02:00
provide nproc pt2_F mo_bielec_integrals_in_map mo_mono_elec_integral pt2_w psi_selectors
2017-11-21 10:43:38 +01:00
print *, '========== ================= ================= ================='
print *, ' Samples Energy Stat. Error Seconds '
print *, '========== ================= ================= ================='
2017-11-27 23:18:18 +01:00
call new_parallel_job(zmq_to_qp_run_socket, zmq_socket_pull, 'pt2')
2017-11-29 15:15:10 +01:00
integer, external :: zmq_put_psi
integer, external :: zmq_put_N_det_generators
integer, external :: zmq_put_N_det_selectors
integer, external :: zmq_put_dvector
if (zmq_put_psi(zmq_to_qp_run_socket,1) == -1) then
stop 'Unable to put psi on ZMQ server'
endif
if (zmq_put_N_det_generators(zmq_to_qp_run_socket, 1) == -1) then
stop 'Unable to put N_det_generators on ZMQ server'
endif
if (zmq_put_N_det_selectors(zmq_to_qp_run_socket, 1) == -1) then
stop 'Unable to put N_det_selectors on ZMQ server'
endif
2017-11-29 13:52:52 +01:00
if (zmq_put_dvector(zmq_to_qp_run_socket,1,'energy',pt2_e0_denominator,size(pt2_e0_denominator)) == -1) then
stop 'Unable to put energy on ZMQ server'
endif
2017-11-21 10:43:38 +01:00
2017-11-29 15:15:10 +01:00
integer, external :: add_task_to_taskserver
2017-11-21 10:43:38 +01:00
2018-08-29 11:30:19 +02:00
do i=1,N_det_generators
do j=1,pt2_F(i) !!!!!!!!!!!!
write(task(1:20),'(I9,1X,I9''|'')') j, pt2_J(i)
if (add_task_to_taskserver(zmq_to_qp_run_socket,trim(task(1:20))) == -1) then
stop 'Unable to add task to task server'
2017-06-26 19:12:44 +02:00
endif
2018-08-29 11:30:19 +02:00
end do
2017-11-21 10:43:38 +01:00
end do
2017-12-01 13:27:34 +01:00
integer, external :: zmq_set_running
if (zmq_set_running(zmq_to_qp_run_socket) == -1) then
print *, irp_here, ': Failed in zmq_set_running'
endif
2017-11-21 10:43:38 +01:00
!$OMP PARALLEL DEFAULT(shared) NUM_THREADS(nproc+1) &
!$OMP PRIVATE(i)
i = omp_get_thread_num()
if (i==0) then
2018-08-29 11:30:19 +02:00
call pt2_collector(zmq_socket_pull, E(pt2_stoch_istate),relative_error, absolute_error, w, error)
2017-11-21 10:43:38 +01:00
pt2(pt2_stoch_istate) = w(pt2_stoch_istate)
else
call pt2_slave_inproc(i)
endif
!$OMP END PARALLEL
2017-11-27 23:18:18 +01:00
call end_parallel_job(zmq_to_qp_run_socket, zmq_socket_pull, 'pt2')
2017-11-21 10:43:38 +01:00
print *, '========== ================= ================= ================='
enddo
2018-01-10 18:11:49 +01:00
FREE pt2_stoch_istate
state_average_weight(:) = state_average_weight_save(:)
TOUCH state_average_weight
2017-06-26 19:12:44 +02:00
endif
2017-11-23 10:35:13 +01:00
do k=N_det+1,N_states
pt2(k) = 0.d0
2017-11-21 17:20:18 +01:00
enddo
2017-03-03 23:14:04 +01:00
2017-01-16 18:19:00 +01:00
end subroutine
subroutine pt2_slave_inproc(i)
implicit none
integer, intent(in) :: i
call run_pt2_slave(1,i,pt2_e0_denominator)
end
2018-08-29 11:30:19 +02:00
subroutine pt2_collector(zmq_socket_pull, E, relative_error, absolute_error, pt2, error)
2017-01-16 18:19:00 +01:00
use f77_zmq
use selection_types
use bitmasks
implicit none
2017-11-27 23:18:18 +01:00
integer(ZMQ_PTR), intent(in) :: zmq_socket_pull
2018-08-29 11:30:19 +02:00
double precision, intent(in) :: relative_error, absolute_error, E
double precision, intent(out) :: pt2(N_states), error(N_states)
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
double precision, allocatable :: eI(:,:), eI_task(:,:), S(:), S2(:)
2017-01-16 18:19:00 +01:00
integer(ZMQ_PTR),external :: new_zmq_to_qp_run_socket
integer(ZMQ_PTR) :: zmq_to_qp_run_socket
2018-08-29 11:30:19 +02:00
integer, external :: zmq_delete_tasks
integer, external :: pt2_find_sample
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
integer :: more, n, i, p, c, t, n_tasks, U
2017-01-16 18:19:00 +01:00
integer, allocatable :: task_id(:)
integer, allocatable :: index(:)
2018-08-29 11:30:19 +02:00
double precision, external :: omp_get_wtime
double precision :: v, x, avg, eqt, E0
double precision :: time, time0
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
integer, allocatable :: f(:)
logical, allocatable :: d(:)
2017-01-16 18:19:00 +01:00
zmq_to_qp_run_socket = new_zmq_to_qp_run_socket()
2018-08-29 11:30:19 +02:00
allocate(task_id(pt2_n_tasks_max), index(pt2_n_tasks_max), f(N_det_generators))
allocate(d(N_det_generators+1))
allocate(eI(N_states, N_det_generators), eI_task(N_states, pt2_n_tasks_max))
allocate(S(pt2_N_teeth+1), S2(pt2_N_teeth+1))
S(:) = 0d0
S2(:) = 0d0
n = 1
t = 0
U = 0
eI(:,:) = 0d0
f(:) = pt2_F(:)
d(:) = .false.
n_tasks = 0
E0 = E
2017-01-16 18:19:00 +01:00
more = 1
2018-08-29 11:30:19 +02:00
time0 = omp_get_wtime()
2017-05-02 16:18:02 +02:00
2018-08-29 11:30:19 +02:00
do while (n <= N_det_generators)
if(f(pt2_J(n)) == 0) then
d(pt2_J(n)) = .true.
do while(d(U+1))
U += 1
end do
do while(t <= pt2_N_teeth)
if(U >= pt2_n_0(t+1)) then
t=t+1
E0 = E
do i=pt2_n_0(t),1,-1
E0 += eI(pt2_stoch_istate, i)
end do
else
exit
2017-01-16 18:19:00 +01:00
end if
end do
2018-08-29 11:30:19 +02:00
c = pt2_R(n)
if(c /= 0) then
x = 0d0
do p=pt2_N_teeth, 1, -1
v = pt2_u_0 + pt2_W_T * (pt2_u(c) + dble(p-1))
i = pt2_find_sample(v, pt2_cW)
x += eI(pt2_stoch_istate, i) * pt2_W_T / pt2_w(i)
S(p) += x
S2(p) += x**2
end do
avg = S(t) / dble(c)
eqt = (S2(t) / c) - (S(t)/c)**2
eqt = sqrt(eqt / dble(c-1))
pt2(pt2_stoch_istate) = E0-E+avg
2017-11-21 10:43:38 +01:00
error(pt2_stoch_istate) = eqt
2018-08-29 11:30:19 +02:00
time = omp_get_wtime()
print '(G10.3, 2X, F16.10, 2X, G16.3, 2X, F16.4, A20)', c, avg+E0, eqt, time-time0, ''
end if
n += 1
else if(more == 0) then
exit
else
call pull_pt2_results(zmq_socket_pull, index, eI_task, task_id, n_tasks)
if (zmq_delete_tasks(zmq_to_qp_run_socket,zmq_socket_pull,task_id,n_tasks,more) == -1) then
stop 'Unable to delete tasks'
2017-01-16 18:19:00 +01:00
endif
2018-08-29 11:30:19 +02:00
do i=1,n_tasks
eI(:, index(i)) += eI_task(:, i)
f(index(i)) -= 1
end do
2017-01-16 18:19:00 +01:00
end if
2018-08-29 11:30:19 +02:00
end do
2017-01-16 18:19:00 +01:00
call end_zmq_to_qp_run_socket(zmq_to_qp_run_socket)
end subroutine
2018-08-29 11:30:19 +02:00
integer function pt2_find_sample(v, w)
2017-01-16 18:19:00 +01:00
implicit none
2018-08-29 11:30:19 +02:00
double precision, intent(in) :: v, w(0:N_det_generators)
2017-01-16 18:19:00 +01:00
integer :: i,l,h
2017-02-01 16:35:47 +01:00
integer, parameter :: block=64
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
l = 0
h = N_det_generators
2017-01-16 18:19:00 +01:00
2017-02-01 16:35:47 +01:00
do while(h-l >= block)
2017-01-30 20:15:28 +01:00
i = ishft(h+l,-1)
2017-01-16 18:19:00 +01:00
if(w(i+1) > v) then
h = i-1
else
l = i+1
end if
end do
2017-02-01 16:35:47 +01:00
!DIR$ LOOP COUNT (64)
2018-08-29 11:30:19 +02:00
do pt2_find_sample=l,h
if(w(pt2_find_sample) >= v) then
2017-02-01 16:35:47 +01:00
exit
end if
end do
2017-01-16 18:19:00 +01:00
end function
2018-08-29 11:30:19 +02:00
BEGIN_PROVIDER[ integer, pt2_J, (N_det_generators)]
&BEGIN_PROVIDER[ double precision, pt2_u, (N_det_generators)]
&BEGIN_PROVIDER[ integer, pt2_R, (N_det_generators)]
2017-01-16 18:19:00 +01:00
implicit none
2018-08-29 11:30:19 +02:00
integer :: N_c, N_j, U, t, i
double precision :: v
logical, allocatable :: d(:)
integer, external :: pt2_find_sample
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
allocate(d(N_det_generators))
2017-05-09 19:17:07 +02:00
2018-08-29 11:30:19 +02:00
pt2_R(:) = 0
N_c = 0
N_j = pt2_n_0(1)
d(:) = .false.
do i=1,N_j
d(i) = .true.
pt2_J(i) = i
2017-01-16 18:19:00 +01:00
end do
2018-08-29 11:30:19 +02:00
call RANDOM_NUMBER(pt2_u)
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
U = 0
do while(N_j < N_det_generators)
!ADD_COMB
N_c += 1
do t=0, pt2_N_teeth-1
v = pt2_u_0 + pt2_W_T * (dble(t) + pt2_u(N_c))
i = pt2_find_sample(v, pt2_cW)
if(.not. d(i)) then
N_j += 1
pt2_J(N_j) = i
d(i) = .true.
end if
end do
pt2_R(N_j) = N_c
!FILL_TOOTH
do while(U < N_det_generators)
U += 1
if(.not. d(U)) then
N_j += 1
pt2_J(N_j) = U
d(U) = .true.
exit;
end if
end do
enddo
if(N_det_generators > 1) then
pt2_R(N_det_generators-1) = 0
pt2_R(N_det_generators) = N_c
end if
2017-11-21 10:43:38 +01:00
END_PROVIDER
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
BEGIN_PROVIDER [ double precision, pt2_w, (N_det_generators) ]
&BEGIN_PROVIDER [ double precision, pt2_cW, (0:N_det_generators) ]
&BEGIN_PROVIDER [ double precision, pt2_W_T ]
&BEGIN_PROVIDER [ double precision, pt2_u_0 ]
&BEGIN_PROVIDER [ integer, pt2_n_0, (pt2_N_teeth+1) ]
2017-01-16 18:19:00 +01:00
implicit none
2018-08-29 11:30:19 +02:00
integer :: i, t
double precision, allocatable :: tilde_w(:), tilde_cW(:)
double precision :: r, tooth_width
integer, external :: pt2_find_sample
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
allocate(tilde_w(N_det_generators), tilde_cW(0:N_det_generators))
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
tilde_cW(0) = 0d0
2017-01-16 18:19:00 +01:00
2017-04-12 14:46:12 +02:00
do i=1,N_det_generators
2018-08-29 11:30:19 +02:00
tilde_w(i) = psi_coef_generators(i,pt2_stoch_istate)**2
tilde_cW(i) = tilde_cW(i-1) + tilde_w(i)
2017-04-12 14:46:12 +02:00
enddo
2017-01-16 18:19:00 +01:00
2018-08-29 11:30:19 +02:00
pt2_n_0(1) = 0
do
pt2_u_0 = tilde_cW(pt2_n_0(1))
r = tilde_cW(pt2_n_0(1) + pt2_minDetInFirstTeeth)
pt2_W_T = (1d0 - pt2_u_0) / dble(pt2_N_teeth)
if(pt2_W_T >= r - pt2_u_0) then
2017-01-16 18:19:00 +01:00
exit
end if
2018-08-29 11:30:19 +02:00
pt2_n_0(1) += 1
if(N_det_generators - pt2_n_0(1) < pt2_minDetInFirstTeeth * pt2_N_teeth) then
stop "teeth building failed"
end if
2017-01-16 18:19:00 +01:00
end do
2018-08-29 11:30:19 +02:00
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
do t=2, pt2_N_teeth
r = pt2_u_0 + pt2_W_T * dble(t-1)
pt2_n_0(t) = pt2_find_sample(r, tilde_cW)
end do
pt2_n_0(pt2_N_teeth+1) = N_det_generators
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
pt2_w(:pt2_n_0(1)) = tilde_w(:pt2_n_0(1))
do t=1, pt2_N_teeth
tooth_width = tilde_cW(pt2_n_0(t+1)) - tilde_cW(pt2_n_0(t))
do i=pt2_n_0(t)+1, pt2_n_0(t+1)
pt2_w(i) = tilde_w(i) * pt2_W_T / tooth_width
end do
2017-01-16 18:19:00 +01:00
end do
2018-08-29 11:30:19 +02:00
pt2_cW(0) = 0d0
2017-01-31 21:52:31 +01:00
do i=1,N_det_generators
2018-08-29 11:30:19 +02:00
pt2_cW(i) = pt2_cW(i-1) + pt2_w(i)
end do
pt2_n_0(pt2_N_teeth+1) = N_det_generators
2017-01-31 21:52:31 +01:00
END_PROVIDER
2017-01-16 18:19:00 +01:00