Octopus
utils.F90
Go to the documentation of this file.
1!! Copyright (C) 2002-2006 M. Marques, A. Castro, A. Rubio, G. Bertsch
2!! Copyright (C) 2021 S. Ohlmann
3!!
4!! This program is free software; you can redistribute it and/or modify
5!! it under the terms of the GNU General Public License as published by
6!! the Free Software Foundation; either version 2, or (at your option)
7!! any later version.
8!!
9!! This program is distributed in the hope that it will be useful,
10!! but WITHOUT ANY WARRANTY; without even the implied warranty of
11!! MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12!! GNU General Public License for more details.
13!!
14!! You should have received a copy of the GNU General Public License
15!! along with this program; if not, write to the Free Software
16!! Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17!! 02110-1301, USA.
18!!
19
20#include "global.h"
21
24
25module utils_oct_m
26 use debug_oct_m
27 use global_oct_m
28 use io_oct_m
29 use iso_c_binding
30 use loct_oct_m
32 use mpi_oct_m
36 use unit_oct_m
38 use string_oct_m
39
40 implicit none
41
42 private
43 public :: &
45 index2axis, &
51 lead_dim, &
55
57 module procedure dleading_dimension_is_known, &
65 end interface leading_dimension_is_known
66
67 interface lead_dim
68 module procedure dlead_dim, zlead_dim
69 module procedure dlead_dim2, zlead_dim2
70 end interface lead_dim
71
72contains
73
74 ! ---------------------------------------------------------
75 subroutine get_divisors(nn, n_divisors, divisors)
76 integer, intent(in) :: nn
77 integer, intent(inout) :: n_divisors
78 integer, intent(out) :: divisors(:)
79
80 integer :: ii, max_d
81
82 push_sub(get_divisors)
83
84 assert(n_divisors > 1)
85 max_d = n_divisors
86
87 n_divisors = 1
88 divisors(n_divisors) = 1
89 do ii = 2, nn / 2
90 if (mod(nn, ii) == 0) then
91 n_divisors = n_divisors + 1
92
93 if (n_divisors > max_d - 1) then
94 message(1) = "Internal error in get_divisors. Please increase n_divisors"
95 call messages_fatal(1)
96 end if
97
98 divisors(n_divisors) = ii
99 end if
100 end do
101 n_divisors = n_divisors + 1
102 divisors(n_divisors) = nn
103
104 pop_sub(get_divisors)
105 end subroutine get_divisors
106
107
108 ! ---------------------------------------------------------
109 character pure function index2axis(idir) result(ch)
110 integer, intent(in) :: idir
111
112 select case (idir)
113 case (1)
114 ch = 'x'
115 case (2)
116 ch = 'y'
117 case (3)
118 ch = 'z'
119 case (4)
120 ch = 'w'
121 case default
122 write(ch,'(i1)') idir
123 end select
124
125 end function index2axis
126
127 ! ---------------------------------------------------------
128 pure function index2axisbz(idir) result(ch)
129 integer, intent(in) :: idir
130 character(len=2) :: ch
131
132 select case (idir)
133 case (1)
134 ch = "kx"
135 case (2)
136 ch = "ky"
137 case (3)
138 ch = "kz"
139 case (4)
140 ch = "kw"
141 case default
142 write(ch,'(i2)') idir
143 end select
144
145 end function index2axisbz
146
147
148 ! ---------------------------------------------------------
149 subroutine output_tensor(tensor, ndim, unit, write_average, iunit, namespace)
150 real(real64), intent(in) :: tensor(:,:)
151 integer, intent(in) :: ndim
152 type(unit_t), intent(in) :: unit
153 logical, optional, intent(in) :: write_average
154 integer, optional, intent(in) :: iunit
155 type(namespace_t), optional, intent(in) :: namespace
156
157 real(real64) :: trace
158 integer :: jj, kk
159 logical :: write_average_
161 push_sub(output_tensor)
162
163 write_average_ = optional_default(write_average, .true.)
164
165 trace = m_zero
166 message(1) = ""
167 do jj = 1, ndim
168 do kk = 1, ndim
169 write(message(1), '(a,f20.6)') trim(message(1)), units_from_atomic(unit, tensor(jj, kk))
170 end do
171 trace = trace + tensor(jj, jj)
172 call messages_info(1, iunit=iunit, namespace=namespace)
173 end do
174
175 if (write_average_) then
176 write(message(1), '(a, f20.6)') 'Isotropic average', units_from_atomic(unit, trace/real(ndim, real64) )
177 call messages_info(1, iunit=iunit, namespace=namespace)
178 end if
179
180 pop_sub(output_tensor)
181 end subroutine output_tensor
182
183
184 ! ---------------------------------------------------------
185 subroutine output_dipole(dipole, ndim, iunit, namespace)
186 real(real64), intent(in) :: dipole(:)
187 integer, intent(in) :: ndim
188 integer, optional, intent(in) :: iunit
189 type(namespace_t), optional, intent(in) :: namespace
190
191 integer :: idir
192
193 push_sub(output_dipole)
194
195 write(message(1), '(a,a20,a17)') 'Dipole:', '[' // trim(units_abbrev(units_out%length)) // ']', &
196 '[' // trim(units_abbrev(unit_debye)) // ']'
197 do idir = 1, ndim
198 write(message(1+idir), '(6x,3a,es14.5,3x,2es14.5)') '<', index2axis(idir), '> = ', &
199 units_from_atomic(units_out%length, dipole(idir)), units_from_atomic(unit_debye, dipole(idir))
200 end do
201 call messages_info(1+ndim, iunit=iunit, namespace=namespace)
203 pop_sub(output_dipole)
204 end subroutine output_dipole
205
208 subroutine print_header()
209 use iso_fortran_env
210
211 character(len=256) :: sys_name
212
213 ! Let us print our logo
214 call io_dump_file(stdout, trim(trim(conf%share) // '/logo'))
215
216 ! Let us print the version
217 message(1) = ""
218 message(2) = str_center("Running octopus", 70)
219 message(3) = ""
220 call messages_info(3)
222 message(1) = &
223 "Version : " // trim(conf%version)
224 message(2) = &
225 "Commit : "// trim(conf%git_commit)
226 message(3) = &
227 "Configuration time : "// trim(conf%config_time)
228 call messages_info(3)
229
230 message(1) = 'Configuration options : ' // trim(get_config_opts())
231 message(2) = 'Optional libraries :' // trim(get_optional_libraries())
232
233 message(3) = 'Architecture : ' + tostring(oct_arch)
234 call messages_info(3)
235
237
238 message(1) = &
239 "C compiler : "//trim(conf%cc)
240 message(2) = &
241 "C compiler flags : "//trim(conf%cflags)
242 message(3) = &
243 "C++ compiler : "//trim(conf%cxx)
244 message(4) = &
245 "C++ compiler flags : "//trim(conf%cxxflags)
246#ifdef HAVE_FC_COMPILER_VERSION
247 message(5) = "Fortran compiler : "//trim(conf%fc) //" ("//compiler_version()//")"
248#else
249 message(5) = "Fortran compiler : "//trim(conf%fc)
250#endif
251 message(6) = &
252 "Fortran compiler flags : "//trim(conf%fcflags)
253 call messages_info(6)
254
255 message(1) = ""
256 call messages_info(1)
257
258 ! Let us print where we are running
259 call loct_sysname(sys_name)
260 write(message(1), '(a)') str_center("The octopus is swimming in " // trim(sys_name), 70)
261 message(2) = ""
262 call messages_info(2)
263
264 call mpi_world%barrier()
265
266 call print_date("Calculation started on ")
267 end subroutine print_header
268
269 character(len=256) function get_config_opts()
270
271 get_config_opts = ''
272#ifdef HAVE_OPENMP
273 get_config_opts = trim(get_config_opts)//' openmp'
274#endif
275#ifdef HAVE_MPI
276 get_config_opts = trim(get_config_opts)//' mpi'
277#endif
278#ifdef HAVE_OPENCL
279 get_config_opts = trim(get_config_opts)//' opencl'
280#endif
281#ifdef HAVE_CUDA
282 get_config_opts = trim(get_config_opts)//' cuda'
283#endif
284#ifdef HAVE_LIBXC_FXC
285 get_config_opts = trim(get_config_opts)//' libxc_fxc'
286#endif
287#ifdef HAVE_LIBXC_KXC
288 get_config_opts = trim(get_config_opts)//' libxc_kxc'
289#endif
290
291 end function get_config_opts
292
293 character(len=256) function get_optional_libraries()
294
295 ! keep in alphabetical order, for ease in seeing if something is listed
297#ifdef HAVE_BERKELEYGW
299#endif
300#ifdef HAVE_CGAL
302#endif
303#ifdef HAVE_CLFFT
305#endif
306#if (defined(HAVE_CLBLAS)) || (defined(HAVE_CLBLAST))
308#endif
309#ifdef HAVE_DFTBPLUS
311#endif
312#ifdef HAVE_ELPA
314#endif
315#ifdef HAVE_ETSF_IO
317#endif
318#ifdef HAVE_GDLIB
320#endif
321#ifdef HAVE_PSOLVER
323#endif
324#ifdef HAVE_LIBVDWXC
326#endif
327#ifdef HAVE_METIS
329#endif
330#ifdef HAVE_NETCDF
332#endif
333#ifdef HAVE_NFFT
335#endif
336#ifdef HAVE_PARMETIS
338#endif
339#ifdef HAVE_PFFT
341#endif
342#ifdef HAVE_PNFFT
344#endif
345#ifdef HAVE_PSPIO
347#endif
348#ifdef HAVE_SCALAPACK
350#endif
351#ifdef HAVE_SPARSKIT
353#endif
354#ifdef HAVE_NLOPT
356#endif
357
358 end function get_optional_libraries
359
360 ! ---------------------------------------------------------
361
362 logical function dleading_dimension_is_known(array) result(known)
363 real(real64), target, intent(in) :: array(:, :)
364
365 integer(c_intptr_t) :: addr1, addr2
366
367 known = .true.
369 if (ubound(array, dim = 2) > 1) then
370 addr1 = transfer(c_loc(array(1,1)), 0_c_intptr_t)
371 addr2 = transfer(c_loc(array(1,2)), 0_c_intptr_t)
372 known = ubound(array, dim = 1) == (addr2-addr1)/c_sizeof(array(1, 1))
373 end if
374
375 end function dleading_dimension_is_known
376
378 ! ---------------------------------------------------------
379
380 logical function zleading_dimension_is_known(array) result(known)
381 complex(real64), target, intent(in) :: array(:, :)
382
383 integer(c_intptr_t) :: addr1, addr2
384
385 known = .true.
386
387 if (ubound(array, dim = 2) > 1) then
388 addr1 = transfer(c_loc(array(1,1)), 0_c_intptr_t)
389 addr2 = transfer(c_loc(array(1,2)), 0_c_intptr_t)
390 known = ubound(array, dim = 1) == (addr2-addr1)/c_sizeof(array(1, 1))
391 end if
392
393 end function zleading_dimension_is_known
394
395 ! ---------------------------------------------------------
396
397 logical function ileading_dimension_is_known(array) result(known)
398 integer, target, intent(in) :: array(:, :)
399
400 integer(c_intptr_t) :: addr1, addr2
401
402 known = .true.
403
404 if (ubound(array, dim = 2) > 1) then
405 addr1 = transfer(c_loc(array(1,1)), 0_c_intptr_t)
406 addr2 = transfer(c_loc(array(1,2)), 0_c_intptr_t)
407
408 known = ubound(array, dim = 1) == (addr2-addr1)/c_sizeof(array(1, 1))
409 end if
410
411 end function ileading_dimension_is_known
413
414 logical function lleading_dimension_is_known(array) result(known)
415 integer(int64), target, intent(in) :: array(:, :)
416
417 integer(c_intptr_t) :: addr1, addr2
418
419 known = .true.
420
421 if (ubound(array, dim = 2) > 1) then
422 addr1 = transfer(c_loc(array(1,1)), 0_c_intptr_t)
423 addr2 = transfer(c_loc(array(1,2)), 0_c_intptr_t)
424
425 known = ubound(array, dim = 1) == (addr2-addr1)/c_sizeof(array(1, 1))
426 end if
427
428 end function lleading_dimension_is_known
430
431 ! ---------------------------------------------------------
432
433 logical function dleading_dimension_is_known2(array) result(known)
434 real(real64), target, intent(in) :: array(:, :, :)
435
436 integer(c_intptr_t) :: addr1, addr2
437
438 known = .true.
439
440 if (ubound(array, dim = 2) > 1) then
441 addr1 = transfer(c_loc(array(1,1,1)), 0_c_intptr_t)
442 addr2 = transfer(c_loc(array(1,2,1)), 0_c_intptr_t)
443 known = ubound(array, dim = 1) == (addr2 - addr1)/c_sizeof(array(1, 1, 1))
444 end if
445
447
449 ! ---------------------------------------------------------
450
451 logical function zleading_dimension_is_known2(array) result(known)
452 complex(real64), target, intent(in) :: array(:, :, :)
453
454 integer(c_intptr_t) :: addr1, addr2
455
456 known = .true.
457
458 if (ubound(array, dim = 2) > 1) then
459 addr1 = transfer(c_loc(array(1,1,1)), 0_c_intptr_t)
460 addr2 = transfer(c_loc(array(1,2,1)), 0_c_intptr_t)
461 known = ubound(array, dim = 1) == (addr2 - addr1)/c_sizeof(array(1, 1, 1))
462 end if
463
465
466 ! ---------------------------------------------------------
467
468 logical function ileading_dimension_is_known2(array) result(known)
469 integer, target, intent(in) :: array(:, :, :)
470
471 integer(c_intptr_t) :: addr1, addr2
472
473 known = .true.
474
475 if (ubound(array, dim = 2) > 1) then
476 addr1 = transfer(c_loc(array(1,1,1)), 0_c_intptr_t)
477 addr2 = transfer(c_loc(array(1,2,1)), 0_c_intptr_t)
478 known = ubound(array, dim = 1) == (addr2 - addr1)/c_sizeof(array(1, 1, 1))
479 end if
480
482
484 logical function lleading_dimension_is_known2(array) result(known)
485 integer(int64), target, intent(in) :: array(:, :, :)
486
487 integer(c_intptr_t) :: addr1, addr2
488
489 known = .true.
490
491 if (ubound(array, dim = 2) > 1) then
492 addr1 = transfer(c_loc(array(1,1,1)), 0_c_intptr_t)
493 addr2 = transfer(c_loc(array(1,2,1)), 0_c_intptr_t)
494 known = ubound(array, dim = 1) == (addr2 - addr1)/c_sizeof(array(1, 1, 1))
495 end if
496
498
499 ! ---------------------------------------------------------
500
501 integer function dlead_dim(array) result(lead_dim)
502 real(real64), intent(in) :: array(:, :)
503
504 assert(leading_dimension_is_known(array))
505
506 lead_dim = ubound(array, dim = 1)
507 end function dlead_dim
508
509 ! ---------------------------------------------------------
510
511 integer function zlead_dim(array) result(lead_dim)
512 complex(real64), intent(in) :: array(:, :)
513
514 assert(leading_dimension_is_known(array))
515
516 lead_dim = ubound(array, dim = 1)
517 end function zlead_dim
518
519 ! ---------------------------------------------------------
520
521 integer function dlead_dim2(array) result(lead_dim)
522 real(real64), intent(in) :: array(:, :, :)
523
524 assert(leading_dimension_is_known(array))
525
526 lead_dim = ubound(array, dim = 1) * ubound(array, dim = 2)
527 end function dlead_dim2
528
529 ! ---------------------------------------------------------
530
531 integer function zlead_dim2(array) result(lead_dim)
532 complex(real64), intent(in) :: array(:, :, :)
533
534 assert(leading_dimension_is_known(array))
535
536 lead_dim = ubound(array, dim = 1) * ubound(array, dim = 2)
537 end function zlead_dim2
538
539 subroutine make_array_larger(array, new_size)
540 integer(int64), allocatable, intent(inout) :: array(:)
541 integer, intent(in) :: new_size
542
543 integer(int64), allocatable :: tmp(:)
544 integer :: copy_size
545
547
548 ! Use allocate here as move_alloc deallocate internally the from array
549 allocate(tmp(1:new_size))
550 copy_size = min(new_size, size(array))
551 tmp(1:copy_size) = array(1:copy_size)
552 safe_deallocate_a(array)
553 call move_alloc(tmp, array)
555 pop_sub(make_array_larger)
556 end subroutine make_array_larger
557
559 subroutine write_vectorization_level()
560 character(len=32) :: vec
561 character(kind=c_char) :: c_str(33)
562
563 c_str = c_null_char
564 call get_vectorization_level(c_str)
565 call string_c_to_f(c_str, vec)
566 message(1) = 'Vectorization level : ' // trim(vec)
567 call messages_info(1)
568
569 end subroutine write_vectorization_level
570
571end module utils_oct_m
572
573!! Local Variables:
574!! mode: f90
575!! coding: utf-8
576!! End:
Definition: io.F90:114
character(len=256), dimension(max_lines), public message
to be output by fatal, warning
Definition: messages.F90:160
subroutine, public messages_fatal(no_lines, only_root_writes, namespace)
Definition: messages.F90:414
This module contains interfaces for routines in operate.c.
Definition: operate_f.F90:117
brief This module defines the class unit_t which is used by the unit_systems_oct_m module.
Definition: unit.F90:132
This module defines the unit system, used for input and output.
This module is intended to contain simple general-purpose utility functions and procedures.
Definition: utils.F90:118
subroutine, public output_tensor(tensor, ndim, unit, write_average, iunit, namespace)
Definition: utils.F90:243
character(len=256) function, public get_config_opts()
Definition: utils.F90:363
integer function zlead_dim2(array)
Definition: utils.F90:547
subroutine write_vectorization_level()
Prints the level of vectorization used for the vectorized finite differences.
Definition: utils.F90:575
subroutine, public output_dipole(dipole, ndim, iunit, namespace)
Definition: utils.F90:279
logical function zleading_dimension_is_known2(array)
Definition: utils.F90:467
logical function lleading_dimension_is_known(array)
Definition: utils.F90:430
logical function zleading_dimension_is_known(array)
Definition: utils.F90:396
subroutine, public get_divisors(nn, n_divisors, divisors)
Definition: utils.F90:169
integer function dlead_dim(array)
Definition: utils.F90:517
subroutine, public make_array_larger(array, new_size)
Definition: utils.F90:555
logical function dleading_dimension_is_known2(array)
Definition: utils.F90:449
character(len=256) function, public get_optional_libraries()
Definition: utils.F90:369
logical function ileading_dimension_is_known(array)
Definition: utils.F90:413
integer function dlead_dim2(array)
Definition: utils.F90:537
character pure function, public index2axis(idir)
Definition: utils.F90:203
logical function lleading_dimension_is_known2(array)
Definition: utils.F90:500
logical function ileading_dimension_is_known2(array)
Definition: utils.F90:484
pure character(len=2) function, public index2axisbz(idir)
Definition: utils.F90:222
subroutine, public print_header()
This subroutine prints the logo followed by information about the compilation and the system....
Definition: utils.F90:302
integer function zlead_dim(array)
Definition: utils.F90:527
logical function dleading_dimension_is_known(array)
Definition: utils.F90:378
void get_vectorization_level(char *level)
Definition: operate.c:733
int true(void)