-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmephit_conf.f90
544 lines (449 loc) · 23.5 KB
/
mephit_conf.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
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
module mephit_conf
use iso_fortran_env, only: dp => real64
implicit none
private
! types and associated procedures
public :: config_t, config_read, config_export_hdf5
public :: config_delayed_t
public :: logger_t
! utility procedures
public :: decorate_filename
! module variables
public :: conf, conf_arr, logger
public :: longlines, cmplx_fmt, nl_fmt, basename_suffix, datafile, &
runmode_single, runmode_direct, runmode_precon, &
pres_prof_eps, pres_prof_par, pres_prof_geqdsk, &
curr_prof_ps, curr_prof_rot, curr_prof_geqdsk, &
q_prof_flux, q_prof_rot, q_prof_geqdsk, &
vac_src_nemov, vac_src_gpec, vac_src_fourier, &
currn_model_mhd, currn_model_kilca, &
refinement_scheme_geometric, refinement_scheme_gaussian
character(len = *), parameter :: cmplx_fmt = 'es24.16e3, 1x, sp, es24.16e3, s, " i"'
character(len = *), parameter :: nl_fmt = '"' // new_line('A') // '"'
character(len = 1024) :: basename_suffix = ''
character(len = 1024) :: datafile = 'mephit.h5'
integer, parameter :: longlines = 1024
integer, parameter :: runmode_single = 0 !< single iteration mode
integer, parameter :: runmode_direct = 1 !< direct iteration mode
integer, parameter :: runmode_precon = 2 !< preconditioned iteration mode
integer, parameter :: pres_prof_eps = 0 !< pressure profile from EPS paper
integer, parameter :: pres_prof_par = 1 !< parabolic pressure profile
integer, parameter :: pres_prof_geqdsk = 2 !< pressure profile from G EQDSK file
integer, parameter :: curr_prof_ps = 0 !< only Pfirsch-Schlueter current
integer, parameter :: curr_prof_rot = 1 !< current via Ampere's law
integer, parameter :: curr_prof_geqdsk = 2 !< current profile from G EQDSK file
integer, parameter :: q_prof_flux = 0 !< q profile from flux between flux surfaces
integer, parameter :: q_prof_rot = 1 !< q profile from rotational transform
integer, parameter :: q_prof_geqdsk = 2 !< q profile from G EQDSK file
integer, parameter :: vac_src_nemov = 0 !< vacuum field perturbation from Viktor Nemov's code
integer, parameter :: vac_src_gpec = 1 !< vacuum field perturbation from GPEC
integer, parameter :: vac_src_fourier = 2 !< vacuum field perturbation from precomputed Fourier modes
integer, parameter :: currn_model_mhd = 0 !< response current from iMHD model
integer, parameter :: currn_model_kilca = 1 !< response current from KiLCA model
integer, parameter :: refinement_scheme_geometric = 0 !< radial refinement via geometric series
integer, parameter :: refinement_scheme_gaussian = 1 !< radial refinement via sum of Gaussians
type :: config_t
!> Name of the file the configuration is read from.
character(len = 1024) :: config_file
!> Verbosity of logging. Possible values are 0 (none), 1 (errors), 2 (warnings),
!> 3 (info), and 4 (debug, default).
integer :: log_level = 4
!> Suppress log messages to stdout. Defaults to false.
logical :: quiet = .false.
!> Kind of iterations to run. Possible values are #runmode_single, #runmode_direct,
!> and #runmode_precon (default).
integer :: runmode = runmode_precon
!> Source of pressure profile. Possible values are #pres_prof_eps, #pres_prof_par,
!> and #pres_prof_geqdsk (default).
integer :: pres_prof = pres_prof_geqdsk
!> Source of toroidal equilibrium current. Possible values are #curr_prof_ps,
!> #curr_prof_rot, and #curr_prof_geqdsk (default).
integer :: curr_prof = curr_prof_geqdsk
!> Source of safety factor profile. Possible values are #q_prof_flux, #q_prof_rot
!> (default), and #q_prof_geqdsk
integer :: q_prof = q_prof_rot
!> Source of vacuum field perturbation. Possible values are #vac_src_nemov,
!> #vac_src_gpec, and #vac_src_fourier (default).
integer :: vac_src = vac_src_fourier
!> Method to compute response current. Possible values are #currn_model_mhd (default)
!> and #currn_model_kilca.
integer :: currn_model = currn_model_mhd
!> Method used for radial refinement. Possible values are #refinement_scheme_geometric
!> (default) and #refinement_scheme_gaussian.
integer :: refinement_scheme = refinement_scheme_geometric
!> Generate non-resonant vacuum perturbation for testing. Defaults to false.
logical :: nonres = .false.
!> Average over quadrilaterals for non-resonant test case. Defaults to true.
logical :: quad_avg = .true.
!> Prefactor in Biot-Savart law, e.g. 5 windings
!> and conversion from A to statA / c_0.
real(dp) :: Biot_Savart_prefactor = 5.0d-1
!> Mass number of ions. Defaults to 2.
real(dp) :: m_i = 2d0
!> Charge number of ions. Defaults to 1.
real(dp) :: Z_i = 1d0
!> File containing coil currents. Defaults to 'coil.dat'.
character(len = 1024) :: currents_file = 'coil.dat'
!> File containing coil geometries. Defaults to 'AUG_B_coils.h5'.
character(len = 1024) :: coil_file = 'AUG_B_coils.h5'
!> File containing equilibrium density profile. Defaults to 'n.dat'.
character(len = 1024) :: dens_file = 'n.dat'
!> File containing equilibrium electron temperature profile. Defaults to 'Te.dat'.
character(len = 1024) :: temp_e_file = 'Te.dat'
!> File containing equilibrium ion temperature profile. Defaults to 'Ti.dat'.
character(len = 1024) :: temp_i_file = 'Ti.dat'
!> File containing equilibrium radial electric field profile. Defaults to 'Er.dat'.
character(len = 1024) :: E_r_file = 'Er.dat'
!> Maximum number of iterations. Does not apply when #runmode equals
!> #runmode_single. Defaults to 50.
integer :: niter = 50
!> Acceptable relative error to consider L^2 integral in fixed-point
!> iterations converged. Defaults to 1.0e-12.
real(dp) :: iter_rel_err = 1d-12
!> Maxmimum number of iterations in Arnoldi method. Only applies when
!> #runmode equals #runmode_precon. Defaults to 100.
integer :: nkrylov = 100
!> Threshold for eigenvalues' magnitude calculated via Arnoldi method.
!> Only applies when #runmode equals #runmode_precon. Defaults to 0.5.
real(dp) :: ritz_threshold = 0.5d0
!> Acceptable relative error to consider eigenvalues converged in Arnoldi method.
!> Only applies when #runmode equals #runmode_precon. Defaults to 1.0e-12.
real(dp) :: ritz_rel_err = 1d-12
!> Index of toroidal harmonics of perturbation. Defaults to 2.
integer :: n = 2
!> Maximum poloidal mode number for Fourier transform of results. Defaults to 24.
integer :: m_max = 24
!> Maximum number of points per flux surface. Defaults to 0 (no maximum imposed).
integer :: pol_max = 0
!> Minimum number of points per flux surface. Defaults to 0 (no minimum imposed).
integer :: pol_min = 0
!> Ignore resonance position where q = 1, which is usually spurious. Defaults to true.
logical :: ignore_q1_res = .true.
!> Maximum distance between flux surfaces along \f$ \theta = 0 \f$. Defaults
!> to 1 cm.
real(dp) :: max_Delta_rad = 1.0d0
!> Minimum temperature. Does not apply when #pres_prof equals #pres_prof_geqdsk.
!> Defaults to 20 eV.
real(dp) :: temp_min = 2d1
! Minimum density. Does not apply when #pres_prof equals #pres_prof_geqdsk. Defaults
!> to 2.0e+11 cm^-3
real(dp) :: dens_min = 2d11
!> Temperature at magnetic axis. Does not apply when #pres_prof equals
!> #pres_prof_geqdsk. Defaults to 3.0e+03 eV.
real(dp) :: temp_max = 3d3
!> Density at magnetic axis. Does not apply when #pres_prof equals #pres_prof_geqdsk.
!> Defaults to 5.0e+13 cm^-3.
real(dp) :: dens_max = 5d13
!> Enable damping of the Pfirsch-Schlueter current. Defaults to true.
logical :: damp = .true.
!> Number of points in sweep over electrical resonance. Defaults to 0 (sweep not performed).
integer :: resonance_sweep = 0
!> Offset added to radial electric field. Defaults to 0.
real(dp) :: offset_E_r = 0d0
!> Enable debugging of initial iterations without plasma response. Defaults to true.
logical :: debug_initial = .true.
!> Enable MFEM interface. Defaults to false.
logical :: debug_mfem = .false.
!> Single poloidal mode used in comparison with KiLCA code. Defaults to 0 (ASDEX
!> geometry).
integer :: kilca_pol_mode = 0
!> KiLCA HDF5 output file from which to infer coefficients of vacuum perturbation.
character(len = 1024) :: kilca_vac_output = ''
!> Scaling factor used for comparison with results from KiLCA code. Defaults to 0
!> (ASDEX geometry).
!>
!> If non-zero, #n and #r0 are scaled by this factor to approximate the cylindrical
!> topology assumed in KiLCA.
integer :: kilca_scale_factor = 0
!> Maximum number of eigenvectors to be exported for plotting. Defaults to 10.
integer :: max_eig_out = 10
!> Check mesh in mephit_test.f90 (expensive)
logical :: check_mesh = .false.
real(dp) :: debug_pol_offset = 0.5d0
logical :: debug_kilca_geom_theta = .false.
logical :: debug_projection = .false.
end type config_t
type :: config_delayed_t
integer :: m_min, m_max
!> Width of refined flux surfaces around resonances in cm.
real(dp), dimension(:), allocatable :: Delta_rad_res
!> Number of additional fine flux surfaces. Defaults to 0.
integer, dimension(:), allocatable :: add_fine
!> Width ratio of neighbouring refined flux surfaces. Defaults to 0 (no refinement).
real(dp), dimension(:), allocatable :: refinement
!> Free parameters setting the magnitudes of sheet currents. Defaults to 0.
real(dp), dimension(:), allocatable :: sheet_current_factor
contains
procedure :: read => config_delayed_read
procedure :: export_hdf5 => config_delayed_export_hdf5
procedure :: import_hdf5 => config_delayed_import_hdf5
procedure :: deinit => config_delayed_deinit
end type config_delayed_t
type :: logger_t
private
character(len = 1024), public :: msg = ''
character(len = 1024), public :: filename = ''
integer :: fid
integer, public :: level
logical, public :: err, warn, info, debug, quiet
contains
procedure :: init => logger_init
procedure :: write_msg => logger_write_msg
procedure :: msg_arg_size => logger_msg_arg_size
procedure :: deinit => logger_deinit
end type logger_t
type(config_t) :: conf
type(config_delayed_t) :: conf_arr
type(logger_t) :: logger
contains
!> Read configuration namelist.
subroutine config_read(config, filename)
type(config_t), intent(inout) :: config
character(len = *), intent(in) :: filename
integer :: fid
namelist /scalars/ config
open(newunit = fid, file = filename)
read(fid, nml = scalars)
close(fid)
! override if erroneously set in namelist
config%config_file = trim(filename)
end subroutine config_read
subroutine config_export_hdf5(config, file, group)
use hdf5_tools, only: HID_T, h5_open_rw, h5_create_parent_groups, h5_add, h5_close
type(config_t), intent(in) :: config
character(len = *), intent(in) :: file
character(len = *), intent(in) :: group
character(len = len_trim(group)) :: grp
integer(HID_T) :: h5id_root
grp = trim(group)
call h5_open_rw(file, h5id_root)
call h5_create_parent_groups(h5id_root, grp // '/')
call h5_add(h5id_root, grp // '/runmode', config%runmode)
call h5_add(h5id_root, grp // '/pres_prof', config%pres_prof)
call h5_add(h5id_root, grp // '/curr_prof', config%curr_prof)
call h5_add(h5id_root, grp // '/q_prof', config%q_prof)
call h5_add(h5id_root, grp // '/vac_src', config%vac_src)
call h5_add(h5id_root, grp // '/currn_model', config%currn_model)
call h5_add(h5id_root, grp // '/refinement_scheme', config%refinement_scheme)
call h5_add(h5id_root, grp // '/nonres', config%nonres)
call h5_add(h5id_root, grp // '/quad_avg', config%quad_avg)
call h5_add(h5id_root, grp // '/m_i', config%m_i, &
comment = 'ion mass number')
call h5_add(h5id_root, grp // '/Z_i', config%Z_i, &
comment = 'ion charge number')
call h5_add(h5id_root, grp // '/dens_file', config%dens_file)
call h5_add(h5id_root, grp // '/temp_e_file', config%temp_e_file)
call h5_add(h5id_root, grp // '/temp_i_file', config%temp_i_file)
call h5_add(h5id_root, grp // '/E_r_file', config%E_r_file)
call h5_add(h5id_root, grp // '/niter', config%niter, &
comment = 'maximum number of iterations')
call h5_add(h5id_root, grp // '/iter_rel_err', config%iter_rel_err, &
comment = 'convergence threshold in L2 integral in fixed-point iteration')
call h5_add(h5id_root, grp // '/nkrylov', config%nkrylov, &
comment = 'maximum number of iterations in Arnoldi method')
call h5_add(h5id_root, grp // '/ritz_threshold', config%ritz_threshold, &
comment = 'threshold for eigenvalues'' magnitude in Arnoldi method')
call h5_add(h5id_root, grp // '/ritz_rel_err', config%ritz_rel_err, &
comment = 'relative error for eigenvalues in Arnoldi method')
call h5_add(h5id_root, grp // '/n', config%n, &
comment = 'index of toroidal harmonics of perturbation')
call h5_add(h5id_root, grp // '/m_max', config%m_max, &
comment = 'maximum poloidal mode number for Fourier transform of results')
call h5_add(h5id_root, grp // '/pol_max', config%pol_max, &
comment = 'maximum number of points per flux surface')
call h5_add(h5id_root, grp // '/pol_min', config%pol_min, &
comment = 'minimum number of points per flux surface')
call h5_add(h5id_root, grp // '/max_Delta_rad', config%max_Delta_rad, &
comment = 'maximum distance between flux surfaces along theta = 0', unit = 'cm')
call h5_add(h5id_root, grp // '/temp_min', config%temp_min, &
comment = 'minimum temperature', unit = 'eV')
call h5_add(h5id_root, grp // '/dens_min', config%dens_min, &
comment = 'minimum density', unit = 'cm^-3')
call h5_add(h5id_root, grp // '/temp_max', config%temp_max, &
comment = 'maximum temperature', unit = 'eV')
call h5_add(h5id_root, grp // '/dens_max', config%dens_max, &
comment = 'maximum density', unit = 'cm^-3')
call h5_add(h5id_root, grp // '/damp', config%damp, &
comment = 'enable damping of Pfirsch-Schlueter current')
call h5_add(h5id_root, grp // '/resonance_sweep', config%resonance_sweep, &
comment = 'number of points for sweep over electrical resonance')
call h5_add(h5id_root, grp // '/offset_E_r', config%offset_E_r, &
comment = 'offset added to radial electric field', unit = 'statV cm^-1')
call h5_add(h5id_root, grp // '/debug_initial', config%debug_initial, &
comment = 'enable debugging of initial runs without plasma response')
call h5_add(h5id_root, grp // '/debug_MFEM', config%debug_MFEM, &
comment = 'enable MFEM interface')
call h5_add(h5id_root, grp // '/kilca_pol_mode', config%kilca_pol_mode, &
comment = 'single poloidal mode used in comparison with KiLCA code')
call h5_add(h5id_root, grp // '/kilca_scale_factor', config%kilca_scale_factor, &
comment = 'scaling factor used for comparison with results from KiLCA code')
call h5_add(h5id_root, grp // '/max_eig_out', config%max_eig_out, &
comment = 'maximum number of eigenvectors exported for plotting')
call h5_add(h5id_root, grp // '/debug_pol_offset', config%debug_pol_offset)
call h5_add(h5id_root, grp // '/debug_kilca_geom_theta', config%debug_kilca_geom_theta)
call h5_add(h5id_root, grp // '/debug_projection', config%debug_projection)
call h5_close(h5id_root)
end subroutine config_export_hdf5
!> Read configuration arrays from namelist.
subroutine config_delayed_read(config, filename, m_min, m_max)
class(config_delayed_t), intent(inout) :: config
character(len = *), intent(in) :: filename
integer, intent(in) :: m_min, m_max
integer :: fid
integer, dimension(m_min:m_max) :: add_fine
real(dp), dimension(m_min:m_max) :: Delta_rad_res, refinement, sheet_current_factor
namelist /arrays/ Delta_rad_res, add_fine, refinement, sheet_current_factor
config%m_min = m_min
config%m_max = m_max
Delta_rad_res = 0d0
add_fine = 0
refinement = 0d0
sheet_current_factor = 0d0
open(newunit = fid, file = filename)
read(fid, nml = arrays)
close(fid)
if (allocated(config%Delta_rad_res)) deallocate(config%Delta_rad_res)
allocate(config%Delta_rad_res(m_min:m_max))
config%Delta_rad_res(:) = Delta_rad_res
if (allocated(config%add_fine)) deallocate(config%add_fine)
allocate(config%add_fine(m_min:m_max))
config%add_fine(:) = add_fine
if (allocated(config%refinement)) deallocate(config%refinement)
allocate(config%refinement(m_min:m_max))
config%refinement(:) = refinement
if (allocated(config%sheet_current_factor)) deallocate(config%sheet_current_factor)
allocate(config%sheet_current_factor(m_min:m_max))
config%sheet_current_factor(:) = sheet_current_factor
end subroutine config_delayed_read
subroutine config_delayed_export_hdf5(config, file, dataset)
use hdf5_tools, only: HID_T, h5_open_rw, h5_create_parent_groups, h5_add, h5_close
class(config_delayed_t), intent(in) :: config
character(len = *), intent(in) :: file, dataset
integer(HID_T) :: h5id_root
call h5_open_rw(file, h5id_root)
call h5_create_parent_groups(h5id_root, trim(adjustl(dataset)) // '/')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/m_min', config%m_min, &
comment = 'minimum poloidal mode number')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/m_max', config%m_max, &
comment = 'maximum poloidal mode number')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/Delta_rad_res', config%Delta_rad_res, &
lbound(config%Delta_rad_res), ubound(config%Delta_rad_res), &
comment = 'width of refined flux surfaces around resonances', unit = 'cm')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/add_fine', config%add_fine, &
lbound(config%add_fine), ubound(config%add_fine), &
comment = 'number of additional fine flux surfaces')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/refinement', config%refinement, &
lbound(config%refinement), ubound(config%refinement), &
comment = 'width ratio of neighbouring refined flux surfaces')
call h5_add(h5id_root, trim(adjustl(dataset)) // '/sheet_current_factor', config%sheet_current_factor, &
lbound(config%sheet_current_factor), ubound(config%sheet_current_factor), &
comment = 'free parameters setting the magnitudes of sheet currents')
call h5_close(h5id_root)
end subroutine config_delayed_export_hdf5
subroutine config_delayed_import_hdf5(config, file, dataset)
use hdf5_tools, only: HID_T, h5_open, h5_get, h5_close
class(config_delayed_t), intent(inout) :: config
character(len = *), intent(in) :: file, dataset
integer(HID_T) :: h5id_root
call h5_open(file, h5id_root)
call config_delayed_deinit(config)
call h5_get(h5id_root, trim(adjustl(dataset)) // '/m_min', config%m_min)
call h5_get(h5id_root, trim(adjustl(dataset)) // '/m_max', config%m_max)
allocate(config%Delta_rad_res(config%m_min:config%m_max))
allocate(config%add_fine(config%m_min:config%m_max))
allocate(config%refinement(config%m_min:config%m_max))
allocate(config%sheet_current_factor(config%m_min:config%m_max))
call h5_get(h5id_root, trim(adjustl(dataset)) // '/Delta_rad_res', config%Delta_rad_res)
call h5_get(h5id_root, trim(adjustl(dataset)) // '/add_fine', config%add_fine)
call h5_get(h5id_root, trim(adjustl(dataset)) // '/refinement', config%refinement)
call h5_get(h5id_root, trim(adjustl(dataset)) // '/sheet_current_factor', config%sheet_current_factor)
call h5_close(h5id_root)
end subroutine config_delayed_import_hdf5
subroutine config_delayed_deinit(config)
class(config_delayed_t), intent(inout) :: config
config%m_min = 0
config%m_max = 0
if (allocated(config%Delta_rad_res)) deallocate(config%Delta_rad_res)
if (allocated(config%add_fine)) deallocate(config%add_fine)
if (allocated(config%refinement)) deallocate(config%refinement)
if (allocated(config%sheet_current_factor)) deallocate(config%sheet_current_factor)
end subroutine config_delayed_deinit
!> Associate logfile and open if necessary.
subroutine logger_init(this, filename, log_level, quiet)
use iso_fortran_env, only: output_unit
class(logger_t), intent(inout) :: this
character(len = *), intent(in) :: filename
integer, intent(in) :: log_level
logical, intent(in) :: quiet
this%filename = filename
this%quiet = quiet
this%level = log_level
this%err = .false.
this%warn = .false.
this%info = .false.
this%debug = .false.
if (this%level > 0) this%err = .true.
if (this%level > 1) this%warn = .true.
if (this%level > 2) this%info = .true.
if (this%level > 3) this%debug = .true.
if (trim(this%filename) == '-') then
this%fid = output_unit
else
open(newunit = this%fid, file = this%filename, status = 'replace', form = 'formatted', action = 'write')
end if
end subroutine logger_init
!> Close logfile if necessary.
subroutine logger_deinit(this)
use iso_fortran_env, only: output_unit
class(logger_t), intent(inout) :: this
if (this%fid /= output_unit) close(this%fid)
end subroutine logger_deinit
!> Generate timestamp
function timestamp()
character(len=25) :: timestamp
character(len=*), parameter :: ISO_8601 = &
'(ss, i4, 2("-", i2.2), "T", i2.2, 2(":", i2.2), sp, i3.2, ":", ss, i2.2, s)'
integer :: values(8)
call date_and_time(values = values)
write (timestamp, ISO_8601) values(1:3), values(5:7), &
values(4) / 60, mod(values(4), 60)
end function timestamp
!> Write to logfile and flush if necessary.
subroutine logger_write_msg(this)
use iso_fortran_env, only: output_unit
class(logger_t), intent(inout) :: this
character(len = 28 + len_trim(logger%msg)) :: full_msg
write (full_msg, '("[", a25, "] ", a)') timestamp(), trim(this%msg)
write (this%fid, '(a)') full_msg
if (this%fid /= output_unit .and. .not. this%quiet) write (output_unit, '(a)') full_msg
flush(this%fid)
end subroutine logger_write_msg
subroutine logger_msg_arg_size(this, funcname, name1, name2, value1, value2)
class(logger_t), intent(inout) :: this
character(len = *), intent(in) :: funcname, name1, name2
integer, intent(in) :: value1, value2
write (this%msg, '("Argument size mismatch in ", a, ": ", a, " = ", i0, ' // &
'", ", a, " = ", i0)') funcname, name1, value1, name2, value2
end subroutine logger_msg_arg_size
!> Generates a new filename from a given template.
!>
!> @param in_name undecorated filename
!> @param prefix string to be prefixed to the file basename
!> @param postfix string to be postfixed to the file basename
!>
!> No attempt is made to check whether the given filename is valid or accessible.
pure function decorate_filename(in_name, prefix, postfix) result(out_name)
character(len = *), intent(in) :: in_name
character(len = *), intent(in) :: prefix
character(len = *), intent(in) :: postfix
character(len = len_trim(in_name) + len_trim(prefix) + len_trim(postfix)) :: out_name
integer :: basename_start, basename_end, name_end
name_end = len_trim(in_name)
basename_start = scan(in_name, '/', .true.) + 1
basename_end = basename_start - 2 + scan(in_name(basename_start:), '.', .true.)
if (basename_end - basename_start == -1) basename_start = basename_start + 1 ! dotfile
if (basename_end < basename_start) basename_end = name_end
out_name = in_name(:basename_start-1) // trim(prefix) // in_name(basename_start: &
basename_end) // trim(postfix) // in_name(basename_end+1:name_end)
end function decorate_filename
end module mephit_conf