-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathED_SPARSE_MAP.f90
281 lines (250 loc) · 8.83 KB
/
ED_SPARSE_MAP.f90
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
MODULE ED_SPARSE_MAP
USE SF_IOTOOLS, only: str,free_unit
USE ED_INPUT_VARS
!#ifdef _MPI
! USE SF_MPI >>> TODO: write the MPI version
! USE MPI of each function! <<<
!#endif
implicit none
private
!SPARSE IMP-BATH MAP AS AN OBJECT
type sparse_row
integer :: size
integer :: bath_state_min
integer :: bath_state_max
integer,dimension(:),allocatable :: bath_state
integer,dimension(:),allocatable :: sector_indx
end type sparse_row
type sparse_map
type(sparse_row),dimension(:),pointer :: imp_state
integer :: Nimp_state
logical :: status=.false.
end type sparse_map
public :: sparse_map
public :: sp_init_map
public :: sp_delete_map
public :: sp_insert_state
public :: sp_return_intersection
public :: sp_print_map
!PRIVATE AUX
interface add_to
module procedure :: add_to_I
module procedure :: add_to_D
module procedure :: add_to_Z
end interface add_to
contains
!+------------------------------------------------------------------+
!PURPOSE: initialize the sparse map
!+------------------------------------------------------------------+
subroutine sp_init_map(sparse,Nstates)
type(sparse_map),intent(inout) :: sparse
integer :: Nstates
integer :: i
!
if(sparse%status)stop "sp_init_map: already allocated can not init"
!
sparse%Nimp_state=Nstates
!
allocate(sparse%imp_state(0:Nstates-1))
do i=0,Nstates-1
sparse%imp_state(i)%size=0
sparse%imp_state(i)%bath_state_min=huge(1)
sparse%imp_state(i)%bath_state_max=0
allocate(sparse%imp_state(i)%bath_state(0))
allocate(sparse%imp_state(i)%sector_indx(0))
end do
!
sparse%status=.true.
!
end subroutine sp_init_map
!+------------------------------------------------------------------+
!PURPOSE: delete an entire sparse map
!+------------------------------------------------------------------+
subroutine sp_delete_map(sparse)
type(sparse_map),intent(inout) :: sparse
integer :: i
!
if(.not.sparse%status)return
!
do i=0,sparse%Nimp_state-1
deallocate(sparse%imp_state(i)%bath_state)
deallocate(sparse%imp_state(i)%sector_indx)
sparse%imp_state(i)%Size = 0
enddo
deallocate(sparse%imp_state)
!
sparse%Nimp_state=0
sparse%status=.false.
end subroutine sp_delete_map
!+------------------------------------------------------------------+
!PURPOSE: insert an element in the sparse map
!+------------------------------------------------------------------+
subroutine sp_insert_state(sparse,imp_state,bath_state,sector_indx)
type(sparse_map),intent(inout) :: sparse
integer,intent(in) :: imp_state
integer,intent(in) :: bath_state
integer,intent(in) :: sector_indx
type(sparse_row),pointer :: row
integer :: column,pos
!
!
if(imp_state < 0) stop "sp_insert_state error: imp_state < 0 "
if(imp_state > sparse%Nimp_state-1) stop "sp_insert_state error: imp_state > map%Nimp_state 2^Nimp-1"
row => sparse%imp_state(imp_state)
if(any(row%bath_state == bath_state))stop "sp_insert_state error: bath_state already present for this imp_state"
!
call add_to(row%bath_state,bath_state)
call add_to(row%sector_indx,sector_indx)
if(bath_state < row%bath_state_min)row%bath_state_min=bath_state
if(bath_state > row%bath_state_max)row%bath_state_max=bath_state
row%Size = row%Size + 1
!
end subroutine sp_insert_state
!+------------------------------------------------------------------+
!PURPOSE: compute the set of bath-states intersecating Iimp and Jimp
!+------------------------------------------------------------------+
subroutine sp_return_intersection(sparse,Iimp,Jimp,array,Narray)
type(sparse_map) :: sparse
integer,intent(in) :: Iimp,Jimp
integer,intent(out),dimension(:),allocatable :: array
type(sparse_row),pointer :: rowI,rowJ
integer :: i
integer,intent(out) :: Narray
!
if(allocated(array))deallocate(array)
if((Iimp<0) .OR. (Jimp<0)) stop "sp_return_intersection error: Iimp OR Jimp < 0 "
if( (Iimp>sparse%Nimp_state-1).OR. (Jimp>sparse%Nimp_state-1)) &
stop "sp_return_intersection error: Iimp OR Jimp > 2^Nimp-1"
rowI => sparse%imp_state(Iimp)
rowJ => sparse%imp_state(Jimp)
Narray=0
if(rowI%size < rowJ%size)then
do i = 1,rowI%size
if( any(rowJ%bath_state == rowI%bath_state(i)) )then
call add_to(array,rowI%bath_state(i))
Narray=Narray+1
endif
enddo
else
do i = 1,rowJ%size
if( any(rowI%bath_state == rowJ%bath_state(i)) )then
call add_to(array,rowJ%bath_state(i))
Narray=Narray+1
endif
enddo
endif
!
end subroutine sp_return_intersection
!##################################################################
!##################################################################
! AUXILIARY COMPUTATIONAL ROUTINES
!##################################################################
!##################################################################
!
!Integer add_to overloading
!
subroutine add_to_I(vec,val)
integer,dimension(:),allocatable,intent(inout) :: vec
integer,intent(in) :: val
integer,dimension(:),allocatable :: tmp
integer :: n
!
if (allocated(vec)) then
n = size(vec)
allocate(tmp(n+1))
tmp(:n) = vec
call move_alloc(tmp,vec)
n = n + 1
else
n = 1
allocate(vec(n))
end if
!
!Put val as last entry:
vec(n) = val
!
if(allocated(tmp))deallocate(tmp)
end subroutine add_to_I
!
!Real add_to overloading
!
subroutine add_to_D(vec,val)
real(8),dimension(:),allocatable,intent(inout) :: vec
real(8),intent(in) :: val
real(8),dimension(:),allocatable :: tmp
integer :: n
!
if (allocated(vec)) then
n = size(vec)
allocate(tmp(n+1))
tmp(:n) = vec
call move_alloc(tmp,vec)
n = n + 1
else
n = 1
allocate(vec(n))
end if
!
!Put val as last entry:
vec(n) = val
!
if(allocated(tmp))deallocate(tmp)
end subroutine add_to_D
!
!Complex add_to overloading
!
subroutine add_to_Z(vec,val)
complex(8),dimension(:),allocatable,intent(inout) :: vec
complex(8),intent(in) :: val
complex(8),dimension(:),allocatable :: tmp
integer :: n
!
if (allocated(vec)) then
n = size(vec)
allocate(tmp(n+1))
tmp(:n) = vec
call move_alloc(tmp,vec)
n = n + 1
else
n = 1
allocate(vec(n))
end if
!
!Put val as last entry:
vec(n) = val
!
if(allocated(tmp))deallocate(tmp)
end subroutine add_to_Z
!##################################################################
!##################################################################
! AUXILIARY PRINTING ROUTINES
!##################################################################
!##################################################################
subroutine sp_print_imp_state(sparse,imp_state)
type(sparse_map),intent(inout) :: sparse
integer,intent(in) :: imp_state
type(sparse_row),pointer :: row
integer :: i
!
if(imp_state < 0) stop "sp_insert_state error: imp_state < 0 "
if(imp_state > sparse%Nimp_state-1) stop "sp_insert_state error: imp_state > map%Nimp_state 2^Nimp-1"
row => sparse%imp_state(imp_state)
write(*,"(A10,I5)")"Imp State:",imp_state
write(*,"(A10,I5)")" size:",row%size
write(*,"(A10,2I5)")" min,max:",row%bath_state_min,row%bath_state_max
write(*,"(A10,"//str(row%size)//"I5)")"bath state",(row%bath_state(i),i=1,row%size)
write(*,"(A10,"//str(row%size)//"I5)")"sect indxs",(row%sector_indx(i),i=1,row%size)
write(*,"(A1)")""
write(*,"(A1)")""
!
end subroutine sp_print_imp_state
subroutine sp_print_map(sparse)
type(sparse_map),intent(inout) :: sparse
integer :: i
!
do i=0,sparse%Nimp_state-1
call sp_print_imp_state(sparse,i)
enddo
!
end subroutine sp_print_map
end module ED_SPARSE_MAP