49 logical,
public :: info
50 logical,
public :: trace
51 logical,
public :: trace_term
52 logical,
public :: trace_file
53 logical :: extra_checks
54 logical,
public :: interaction_graph
55 logical,
public :: interaction_graph_full
56 logical,
public :: propagation_graph
57 logical,
public :: instrument
59 character(len=MAX_PATH_LEN),
public :: instr_sub_name
60 integer,
public :: instr_tool
63 type(debug_t),
save :: debug
66 integer,
parameter :: unit_offset = 1000
85 type(debug_t),
intent(out) :: this
86 type(namespace_t),
intent(in) :: namespace
88 character(len=256) :: node_hook
89 logical :: file_exists, mpi_debug_hook
129 call parse_variable(namespace,
'Debug', option__debug__no, this%bits)
134 this%instr_sub_name =
''
165 write(stderr,
'(a)')
"Only single function can be instrumented!"
173 select case (this%instr_tool)
174 case (option__instrumentfunctions__verrou)
175 write(stderr,
'(a)')
"Instrumenting " // trim(this%instr_sub_name) //
" for Verrou"
176#if !defined(HAVE_VERROU)
177 write(stderr,
'(a)')
"requires VERROU but that library was not linked."
181 case (option__instrumentfunctions__fenv)
182 write(stderr,
'(a)')
"Instrumenting " // trim(this%instr_sub_name) //
" with floating-point exceptions"
188 else if (this%instrument)
then
189 write(stderr,
'(a)')
"Debug=instrument requires InstrumentFunctions block."
213 if (mpi_debug_hook)
then
216 write(stdout,
'(a,i6,a,i6.6,20x,a)')
'* I ',sec,
'.',usec,
' | MPI debug hook'
218 write(stdout,
'(a,i3,a)')
'node:',
mpi_world%rank,
' In debug hook'
220 file_exists = .false.
222 do while (.not. file_exists)
223 inquire(file=
'node_hook.'//node_hook, exist=file_exists)
225 write(stdout,
'(a,i3,a)')
'node:',
mpi_world%rank, &
226 ' - still sleeping. To release me touch: node_hook.'//trim(node_hook)
229 write(stdout,
'(a,i3,a)')
'node:',
mpi_world%rank,
' Leaving debug hook'
231 call loct_rm(
'node_hook.'//trim(node_hook))
235 write(stdout,
'(a,i6,a,i6.6,20x,a)')
'* O ', sec,
'.', usec,
' | MPI debug hook'
244 type(
debug_t),
intent(inout) :: this
248 this%trace_term = .
true.
249 this%trace_file = .
true.
250 this%interaction_graph = .
true.
251 this%interaction_graph_full = .
true.
252 this%propagation_graph = .
true.
259 type(
debug_t),
intent(inout) :: this
270 character(len=6) :: filenum
273 write(filenum,
'(i6.6)') iunit - unit_offset
275 call loct_rm(
'debug/debug_trace.node.'//filenum)
282 integer,
intent(out) :: iunit
284 character(len=6) :: filenum
287 write(filenum,
'(i6.6)') iunit - unit_offset
289 open(iunit, file =
'debug/debug_trace.node.'//filenum, &
290 action=
'write', status=
'unknown', position=
'append')
297 type(
debug_t),
intent(inout) :: this
299 this%info = (
bitand(this%bits, option__debug__info) /= 0)
300 this%trace_term = (
bitand(this%bits, option__debug__trace_term) /= 0)
301 this%trace_file = (
bitand(this%bits, option__debug__trace_file) /= 0)
302 this%trace = (
bitand(this%bits, option__debug__trace) /= 0) .or. this%trace_term .or. this%trace_file
303 this%extra_checks = (
bitand(this%bits, option__debug__extra_checks) /= 0) .or. this%trace_term .or. this%trace_file
304 this%interaction_graph = (
bitand(this%bits, option__debug__interaction_graph) /= 0)
305 this%interaction_graph_full = (
bitand(this%bits, option__debug__interaction_graph_full) /= 0)
306 this%propagation_graph = (
bitand(this%bits, option__debug__propagation_graph) /= 0)
307 this%instrument = (
bitand(this%bits, option__debug__instrument) /= 0)
314 integer,
intent(inout) :: sec
315 integer,
intent(inout) :: usec
326 subroutine time_diff(sec1, usec1, sec2, usec2)
327 integer,
intent(in) :: sec1
328 integer,
intent(in) :: usec1
329 integer,
intent(inout) :: sec2
330 integer,
intent(inout) :: usec2
335 if (usec2 - usec1 < 0)
then
336 usec2 = 1000000 + usec2
337 if (sec2 >= sec1)
then
343 if (sec2 >= sec1)
then
346 usec2 = usec2 - usec1
355 character(len=*),
intent(in) :: sub_name
357 integer,
parameter :: max_recursion_level = 50
358 integer iunit, sec, usec
360 if (debug%instrument)
then
362 select case (debug%instr_tool)
363 case (option__instrumentfunctions__verrou)
365 case (option__instrumentfunctions__fenv)
373 if (.not. debug%trace)
return
380 sub_stack(max_recursion_level) =
'debug_push_sub'
381 write(stderr,
'(a,i3,a)')
'Too many recursion levels in debug trace (max=', max_recursion_level,
')'
389 if (debug%trace_file)
then
404 integer,
intent(in) :: iunit_out
407 character(len=1000) :: tmpstr
409 write(tmpstr,
'(a,i6,a,i6.6,f20.6,i8,a)')
"* I ", &
414 write(tmpstr,
'(2a)') trim(tmpstr),
"..|"
417 write(iunit_out,
'(a)') trim(tmpstr)
426 character(len=*),
intent(in) :: sub_name
428 character(len=80) :: sub_name_short
429 integer iunit, sec, usec
431 if (debug%instrument)
then
433 select case (debug%instr_tool)
434 case (option__instrumentfunctions__verrou)
436 case (option__instrumentfunctions__fenv)
444 if (.not. debug%trace)
return
452 write(stderr,
'(a)')
'Too few recursion levels in debug trace'
462 write(stderr,
'(a)')
'Wrong sub name on pop_sub :'
463 write(stderr,
'(2a)')
' got : ', sub_name_short
469 if (debug%trace_file)
then
486 integer,
intent(in) :: iunit_out
489 character(len=1000) :: tmpstr
491 write(tmpstr,
'(a,i6,a,i6.6,f20.6,i8, a)')
"* O ", &
496 write(tmpstr,
'(2a)') trim(tmpstr),
"..|"
500 write(iunit_out,
'(a)') trim(tmpstr)
509 character(len=MAX_PATH_LEN) function debug_clean_path(filename)
result(clean_path)
510 character(len=*),
intent(in) :: filename
514 pos = index(filename,
'src/', back = .
true.)
517 clean_path = filename
520 clean_path = filename(pos+4:)
subroutine pop_sub_write(iunit_out)
subroutine push_sub_write(iunit_out)
character(len=max_path_len) function, public debug_clean_path(filename)
Prune a filename path to only include subdirectories of the "src" directory.
subroutine, public debug_enable(this)
type(debug_t), save, public debug
subroutine, public debug_pop_sub(sub_name)
Pop a routine from the debug trace.
subroutine, public debug_open_trace(iunit)
subroutine, public epoch_time_diff(sec, usec)
subroutine from_bits(this)
subroutine, public debug_init(this, namespace)
subroutine, public debug_disable(this)
subroutine time_diff(sec1, usec1, sec2, usec2)
Computes t2 <- t2-t1. sec1,2 and usec1,2 are seconds,microseconds of t1,2.
subroutine, public debug_push_sub(sub_name)
Push a routine to the debug trace.
subroutine, public debug_delete_trace()
integer, public s_epoch_sec
global epoch time (time at startup)
integer, public no_sub_stack
real(real64), dimension(50), public time_stack
character(len=80), dimension(50), public sub_stack
The stack.
integer, public s_epoch_usec
subroutine, public mpi_debug_init(rank, info)
logical function mpi_grp_is_root(grp)
Is the current MPI process of grpcomm, root.
type(mpi_grp_t), public mpi_world
type(namespace_t), public global_namespace
integer function, public parse_block(namespace, name, blk, check_varinfo_)