10
0
mirror of https://github.com/LCPQ/quantum_package synced 2025-01-08 20:33:26 +01:00

Less memory in Davdison

This commit is contained in:
Anthony Scemama 2017-04-17 02:54:19 +02:00
parent 04e9918b90
commit 30d529aeb2

View File

@ -684,70 +684,105 @@ subroutine get_all_spin_singles_and_doubles(buffer, idx, spindet, Nint, size_buf
integer, intent(out) :: n_singles integer, intent(out) :: n_singles
integer, intent(out) :: n_doubles integer, intent(out) :: n_doubles
integer :: i,k integer :: i,k,ii, imax
integer(bit_kind), allocatable :: xorvec(:,:) integer, parameter :: block_size=64
integer, allocatable :: degree(:) include 'Utils/constants.include.F'
integer :: size_buffer_align integer(bit_kind) :: xorvec(block_size,N_int_max)
integer :: degree(block_size)
integer, external :: align_double integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree !DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
select case (Nint) ! select case (Nint)
case (1) ! case (1)
call get_all_spin_singles_and_doubles_1(buffer, idx, spindet(1), size_buffer, singles, doubles, n_singles, n_doubles) ! call get_all_spin_singles_and_doubles_1(buffer, idx, spindet(1), size_buffer, singles, doubles, n_singles, n_doubles)
return ! return
case (2) ! case (2)
call get_all_spin_singles_and_doubles_2(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles) ! call get_all_spin_singles_and_doubles_2(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles)
return ! return
case (3) ! case (3)
call get_all_spin_singles_and_doubles_3(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles) ! call get_all_spin_singles_and_doubles_3(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles)
return ! return
end select ! end select
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, Nint), degree(size_buffer) )
do k=1,Nint
do i=1,size_buffer
xorvec(i, k) = xor( spindet(k), buffer(k,i) )
enddo
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
do k=2,Nint
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,k) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,k))
endif
enddo
enddo
n_singles = 1 n_singles = 1
n_doubles = 1 n_doubles = 1
do i=1,size_buffer !DIR$ VECTOR ALIGNED
if ( degree(i) == 4 ) then do i=0,size_buffer-block_size, block_size
doubles(n_doubles) = idx(i)
do k=1,Nint
do ii=1,block_size
xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
enddo
enddo
do ii=1,block_size
if (xorvec(ii,1) /= 0_8) then
degree(ii) = popcnt(xorvec(ii,1))
else
degree(ii) = 0
endif
do k=2,Nint
!DIR$ VECTOR ALIGNED
if ( (degree(ii) <= 4).and.(xorvec(ii,k) /= 0_8) ) then
degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
endif
enddo
enddo
do ii=1,block_size
if ( degree(ii) == 4 ) then
doubles(n_doubles) = idx(i+ii)
n_doubles = n_doubles+1 n_doubles = n_doubles+1
endif endif
if ( degree(i) == 2 ) then if ( degree(ii) == 2 ) then
singles(n_singles) = idx(i) singles(n_singles) = idx(i+ii)
n_singles = n_singles+1 n_singles = n_singles+1
endif endif
enddo enddo
enddo
imax = size_buffer-i
do k=1,Nint
do ii=1,imax
xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
enddo
enddo
do ii=1,imax
if (xorvec(ii,1) /= 0_8) then
degree(ii) = popcnt(xorvec(ii,1))
else
degree(ii) = 0
endif
do k=2,Nint
!DIR$ VECTOR ALIGNED
if ( (degree(ii) <= 4).and.(xorvec(ii,k) /= 0_8) ) then
degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
endif
enddo
enddo
do ii=1,imax
if ( degree(ii) == 4 ) then
doubles(n_doubles) = idx(i+ii)
n_doubles = n_doubles+1
endif
if ( degree(ii) == 2 ) then
singles(n_singles) = idx(i+ii)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1 n_singles = n_singles-1
n_doubles = n_doubles-1 n_doubles = n_doubles-1
deallocate(xorvec)
end end
@ -767,63 +802,90 @@ subroutine get_all_spin_singles(buffer, idx, spindet, Nint, size_buffer, singles
integer, intent(out) :: singles(size_buffer) integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: n_singles integer, intent(out) :: n_singles
integer :: i,k integer :: i,k,ii, imax
integer(bit_kind), allocatable :: xorvec(:,:) integer, parameter :: block_size=64
integer, allocatable :: degree(:) include 'Utils/constants.include.F'
integer :: size_buffer_align integer(bit_kind) :: xorvec(block_size,N_int_max)
integer :: degree(block_size)
integer, external :: align_double integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree !DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
select case (Nint) ! select case (Nint)
case (1) ! case (1)
call get_all_spin_singles_1(buffer, idx, spindet(1), size_buffer, singles, n_singles) ! call get_all_spin_singles_1(buffer, idx, spindet(1), size_buffer, singles, n_singles)
return ! return
case (2) ! case (2)
call get_all_spin_singles_2(buffer, idx, spindet, size_buffer, singles, n_singles) ! call get_all_spin_singles_2(buffer, idx, spindet, size_buffer, singles, n_singles)
return ! return
case (3) ! case (3)
call get_all_spin_singles_3(buffer, idx, spindet, size_buffer, singles, n_singles) ! call get_all_spin_singles_3(buffer, idx, spindet, size_buffer, singles, n_singles)
return ! return
end select ! end select
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, Nint), degree(size_buffer) )
do k=1,Nint
do i=1,size_buffer
xorvec(i, k) = xor( spindet(k), buffer(k,i) )
enddo
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
do k=2,Nint
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 2).and.(xorvec(i,k) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,k))
endif
enddo
enddo
n_singles = 1 n_singles = 1
do i=1,size_buffer !DIR$ VECTOR ALIGNED
if ( degree(i) == 2 ) then do i=0,size_buffer-block_size, block_size
singles(n_singles) = idx(i)
do k=1,Nint
do ii=1,block_size
xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
enddo
enddo
do ii=1,block_size
if (xorvec(ii,1) /= 0_8) then
degree(ii) = popcnt(xorvec(ii,1))
else
degree(ii) = 0
endif
do k=2,Nint
if ( (degree(ii) <= 2).and.(xorvec(ii,k) /= 0_8) ) then
degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
endif
enddo
enddo
do ii=1,block_size
if ( degree(ii) == 2 ) then
singles(n_singles) = idx(i+ii)
n_singles = n_singles+1
endif
enddo
enddo
imax = size_buffer-i
do k=1,Nint
do ii=1,imax
xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
enddo
enddo
do ii=1,imax
if (xorvec(ii,1) /= 0_8) then
degree(ii) = popcnt(xorvec(ii,1))
else
degree(ii) = 0
endif
do k=2,Nint
if ( (degree(ii) <= 2).and.(xorvec(ii,k) /= 0_8) ) then
degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
endif
enddo
enddo
do ii=1,imax
if ( degree(ii) == 2 ) then
singles(n_singles) = idx(i+ii)
n_singles = n_singles+1 n_singles = n_singles+1
endif endif
enddo enddo
n_singles = n_singles-1 n_singles = n_singles-1
deallocate(xorvec, degree)
end end
@ -843,604 +905,98 @@ subroutine get_all_spin_doubles(buffer, idx, spindet, Nint, size_buffer, doubles
integer, intent(out) :: doubles(size_buffer) integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_doubles integer, intent(out) :: n_doubles
integer :: i,k integer :: i,k,ii, imax
integer(bit_kind), allocatable :: xorvec(:,:) integer, parameter :: block_size=64
integer, allocatable :: degree(:) include 'Utils/constants.include.F'
integer :: size_buffer_align integer(bit_kind) :: xorvec(block_size,N_int_max)
integer :: degree(block_size)
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree !DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
select case (Nint) ! select case (Nint)
case (1) ! case (1)
call get_all_spin_doubles_1(buffer, idx, spindet(1), size_buffer, doubles, n_doubles) ! call get_all_spin_doubles_1(buffer, idx, spindet(1), size_buffer, doubles, n_doubles)
return ! return
case (2) ! case (2)
call get_all_spin_doubles_2(buffer, idx, spindet, size_buffer, doubles, n_doubles) ! call get_all_spin_doubles_2(buffer, idx, spindet, size_buffer, doubles, n_doubles)
return ! return
case (3) ! case (3)
call get_all_spin_doubles_3(buffer, idx, spindet, size_buffer, doubles, n_doubles) ! call get_all_spin_doubles_3(buffer, idx, spindet, size_buffer, doubles, n_doubles)
return ! return
end select ! end select
size_buffer_align = align_double(size_buffer) n_doubles = 1
allocate( xorvec(size_buffer_align, Nint), degree(size_buffer) ) !DIR$ VECTOR ALIGNED
do i=0,size_buffer-block_size, block_size
do k=1,Nint do k=1,Nint
do i=1,size_buffer do ii=1,block_size
xorvec(i, k) = xor( spindet(k), buffer(k,i) ) xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
enddo enddo
enddo enddo
!DIR$ VECTOR ALIGNED do ii=1,block_size
do i=1,size_buffer if (xorvec(ii,1) /= 0_8) then
if (xorvec(i,1) /= 0_8) then degree(ii) = popcnt(xorvec(ii,1))
degree(i) = popcnt(xorvec(i,1))
else else
degree(i) = 0 degree(ii) = 0
endif endif
enddo
do k=2,Nint do k=2,Nint
!DIR$ VECTOR ALIGNED !DIR$ VECTOR ALIGNED
do i=1,size_buffer if ( (degree(ii) <= 4).and.(xorvec(ii,k) /= 0_8) ) then
if ( (degree(i) <= 4).and.(xorvec(i,k) /= 0_8) ) then degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
degree(i) = degree(i) + popcnt(xorvec(i,k))
endif endif
enddo enddo
enddo enddo
n_doubles = 1 do ii=1,block_size
do i=1,size_buffer if ( degree(ii) == 4 ) then
if ( degree(i) == 4 ) then doubles(n_doubles) = idx(i+ii)
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1 n_doubles = n_doubles+1
endif endif
enddo enddo
n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine get_all_spin_singles_and_doubles_1(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single and double excitations in the list of
! unique alpha determinants.
!
! /!\ : The buffer is transposed !
!
END_DOC
integer, intent(in) :: size_buffer
integer, intent(in) :: idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(size_buffer)
integer(bit_kind), intent(in) :: spindet
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_singles
integer, intent(out) :: n_doubles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:)
integer :: degree
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align) )
do i=1,size_buffer
xorvec(i) = xor( spindet, buffer(i) )
enddo enddo
n_singles = 1 imax = size_buffer-i
n_doubles = 1
do i=1,size_buffer do k=1,Nint
degree = popcnt(xorvec(i)) do ii=1,imax
if ( degree == 4 ) then xorvec(ii, k) = xor( spindet(k), buffer(k,ii+i) )
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1
endif
if ( degree == 2 ) then
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo enddo
n_singles = n_singles-1
n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine get_all_spin_singles_1(buffer, idx, spindet, size_buffer, singles, n_singles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(size_buffer)
integer(bit_kind), intent(in) :: spindet
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: n_singles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:)
allocate( xorvec(size_buffer) )
do i=1,size_buffer
xorvec(i) = xor( spindet, buffer(i) )
enddo enddo
n_singles = 1 do ii=1,imax
do i=1,size_buffer if (xorvec(ii,1) /= 0_8) then
if ( popcnt(xorvec(i)) == 2 ) then degree(ii) = popcnt(xorvec(ii,1))
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1
deallocate(xorvec)
end
subroutine get_all_spin_doubles_1(buffer, idx, spindet, size_buffer, doubles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the double excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(size_buffer)
integer(bit_kind), intent(in) :: spindet
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_doubles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:)
integer, external :: align_double
allocate( xorvec(size_buffer) )
do i=1,size_buffer
xorvec(i) = xor( spindet, buffer(i) )
enddo
n_doubles = 1
do i=1,size_buffer
if ( popcnt(xorvec(i)) == 4 ) then
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1
endif
enddo
n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine get_all_spin_singles_and_doubles_2(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single and double excitations in the list of
! unique alpha determinants.
!
! /!\ : The buffer is transposed !
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(2,size_buffer)
integer(bit_kind), intent(in) :: spindet(2)
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_singles
integer, intent(out) :: n_doubles
integer :: i
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 2), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else else
degree(i) = 0 degree(ii) = 0
endif endif
enddo
do k=2,Nint
!DIR$ VECTOR ALIGNED !DIR$ VECTOR ALIGNED
do i=1,size_buffer if ( (degree(ii) <= 4).and.(xorvec(ii,k) /= 0_8) ) then
if ( (degree(i) <= 4).and.(xorvec(i,2) /= 0_8) ) then degree(ii) = degree(ii) + popcnt(xorvec(ii,k))
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif endif
enddo enddo
n_singles = 1
n_doubles = 1
do i=1,size_buffer
if ( degree(i) == 4 ) then
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1
endif
if ( degree(i) == 2 ) then
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1
n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine get_all_spin_singles_2(buffer, idx, spindet, size_buffer, singles, n_singles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(2,size_buffer)
integer(bit_kind), intent(in) :: spindet(2)
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: n_singles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 2), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
enddo enddo
!DIR$ VECTOR ALIGNED do ii=1,imax
do i=1,size_buffer if ( degree(ii) == 4 ) then
if (xorvec(i,1) /= 0_8) then doubles(n_doubles) = idx(i+ii)
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 2).and.(xorvec(i,2) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif
enddo
n_singles = 1
do i=1,size_buffer
if ( degree(i) == 2 ) then
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1
deallocate(xorvec)
end
subroutine get_all_spin_doubles_2(buffer, idx, spindet, size_buffer, doubles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the double excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(2,size_buffer)
integer(bit_kind), intent(in) :: spindet(2)
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_doubles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 2), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,2) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif
enddo
n_doubles = 1
do i=1,size_buffer
if ( degree(i) == 4 ) then
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1 n_doubles = n_doubles+1
endif endif
enddo enddo
n_doubles = n_doubles-1 n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine get_all_spin_singles_and_doubles_3(buffer, idx, spindet, size_buffer, singles, doubles, n_singles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single and double excitations in the list of
! unique alpha determinants.
!
! /!\ : The buffer is transposed !
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(3,size_buffer)
integer(bit_kind), intent(in) :: spindet(3)
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_singles
integer, intent(out) :: n_doubles
integer :: i
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 3), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
xorvec(i, 3) = xor( spindet(3), buffer(3,i) )
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,2) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,3) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,3))
endif
enddo
n_singles = 1
n_doubles = 1
do i=1,size_buffer
if ( degree(i) == 4 ) then
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1
endif
if ( degree(i) == 2 ) then
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1
n_doubles = n_doubles-1
deallocate(xorvec)
end end
subroutine get_all_spin_singles_3(buffer, idx, spindet, size_buffer, singles, n_singles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the single excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(3,size_buffer)
integer(bit_kind), intent(in) :: spindet(3)
integer, intent(out) :: singles(size_buffer)
integer, intent(out) :: n_singles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 3), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
xorvec(i, 3) = xor( spindet(3), buffer(3,i) )
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 2).and.(xorvec(i,2) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 2).and.(xorvec(i,3) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,3))
endif
enddo
n_singles = 1
do i=1,size_buffer
if ( degree(i) == 2 ) then
singles(n_singles) = idx(i)
n_singles = n_singles+1
endif
enddo
n_singles = n_singles-1
deallocate(xorvec)
end
subroutine get_all_spin_doubles_3(buffer, idx, spindet, size_buffer, doubles, n_doubles)
use bitmasks
implicit none
BEGIN_DOC
!
! Returns the indices of all the double excitations in the list of
! unique alpha determinants.
!
END_DOC
integer, intent(in) :: size_buffer, idx(size_buffer)
integer(bit_kind), intent(in) :: buffer(3,size_buffer)
integer(bit_kind), intent(in) :: spindet(3)
integer, intent(out) :: doubles(size_buffer)
integer, intent(out) :: n_doubles
integer :: i,k
integer(bit_kind), allocatable :: xorvec(:,:)
integer, allocatable :: degree(:)
integer :: size_buffer_align
integer, external :: align_double
!DIR$ ATTRIBUTES ALIGN : $IRP_ALIGN :: xorvec, degree
size_buffer_align = align_double(size_buffer)
allocate( xorvec(size_buffer_align, 3), degree(size_buffer) )
do i=1,size_buffer
xorvec(i, 1) = xor( spindet(1), buffer(1,i) )
xorvec(i, 2) = xor( spindet(2), buffer(2,i) )
xorvec(i, 3) = xor( spindet(3), buffer(3,i) )
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if (xorvec(i,1) /= 0_8) then
degree(i) = popcnt(xorvec(i,1))
else
degree(i) = 0
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,2) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,2))
endif
enddo
!DIR$ VECTOR ALIGNED
do i=1,size_buffer
if ( (degree(i) <= 4).and.(xorvec(i,3) /= 0_8) ) then
degree(i) = degree(i) + popcnt(xorvec(i,3))
endif
enddo
n_doubles = 1
do i=1,size_buffer
if ( degree(i) == 4 ) then
doubles(n_doubles) = idx(i)
n_doubles = n_doubles+1
endif
enddo
n_doubles = n_doubles-1
deallocate(xorvec)
end
subroutine copy_psi_bilinear_to_psi(psi, isize) subroutine copy_psi_bilinear_to_psi(psi, isize)
implicit none implicit none