ITM Grid Service Library: Fortran 90
 All Classes Files Functions Variables
itm_assert.f90
Go to the documentation of this file.
1 module itm_assert
2 
3  !> Assertions for Fortran 90, written by H.-J. Klingshirn
4 
5  !> @author
6 
7  use itm_types , itm_r8 => r8, itm_r4 => r4
8  use itm_string
9 
10  implicit none
11 
12  private
13 
15 
16  logical, save :: defaultdostop = .true.
17  integer, save :: failcount = 0
18  integer, save :: msgprefixlen = 0
19  character(256), save :: msgprefix
20 
21 contains
22 
23  !> Private subroutine that handles a failed assertion. Prints out the
24  !> fail message (msgPrefix + failmsg, depending on what is given), and
25  !> either stops of does assertion bookkeeping.
26 
27  !> @param failmsg The fail message to print
28  !> @param doStop Controls whether to stop execution. If doStop .true.,
29  !> the program is stopped. If .false., only the fail message is printed
30  !> and bookkeeping is done for delayed stopping (@see assertStopOnFailed)
31 
32  subroutine assertfail( failmsg, doStop )
33  character(*), intent(in), optional :: failmsg
34  logical, intent(in), optional :: dostop
35 
36  ! internal
37  logical :: ldostop
38 
39  ldostop = defaultdostop
40  if ( present( dostop ) ) ldostop = dostop
41 
42  if ( present( failmsg ) ) then
43  if ( msgprefixlen > 0 ) then
44  write (*,*) 'itm_assert: '//msgprefix(1:msgprefixlen)//' '//failmsg
45  else
46  write (*,*) 'itm_assert: '//failmsg
47  end if
48  else
49  if ( msgprefixlen > 0 ) then
50  write (*,*) 'itm_assert: '//msgprefix(1:msgprefixlen)
51  end if
52  end if
53 
54  if ( ldostop ) then
55  stop "itm_assert: an assertion failed, stopping immediately."
56  else
57  failcount = failcount + 1
58  write(*,*) "itm_assert: an assertion failed, continuing."
59  end if
60 
61  end subroutine assertfail
62 
63 
64  !> Set the default assertion stop behaviour.
65 
66  !> @param doStop .true. means immediate
67  !> stop on fail, .false. means don't stop on fail (for use with
68  !> delayed stopping, @see assertStopOnFailed). The default value is .true.,
69  !> it is set if doStop is omitted.
70  subroutine assertsetstopmode( doStop )
71  logical, intent(in), optional :: dostop
72 
73  if ( present( dostop ) ) then
74  defaultdostop = dostop
75  else
76  ! if no argument given, just set default again
77  defaultdostop = .true.
78  end if
79 
80  end subroutine assertsetstopmode
81 
82 
83  !> Set a prefix for the assertion fail message.
84  !> @param prefix The prefix string for the fail messages. Optional. If
85  !> omitted, clear the prefix string (i.e. no prefix is used anymore)
86  subroutine assertsetmsgprefix( prefix )
87  character(len=*), intent(in), optional :: prefix
88 
89  if ( present( prefix ) ) then
90  msgprefixlen = min( len(prefix), len( msgprefix ) )
91  msgprefix(1:msgprefixlen) = prefix(1:msgprefixlen)
92  else
93  msgprefixlen = 0
94  end if
95 
96  end subroutine assertsetmsgprefix
97 
98 
99  !> Reset the assertion module to its default state.
100  !>
101  !> Reset stop mode and message prefix to their default values.
102  !> Also resets the fail counter, i.e. any previous failed assertions are forgotton.
103  subroutine assertreset()
104  call assertsetstopmode()
105  call assertsetmsgprefix()
106  failcount = 0
107  end subroutine assertreset
108 
109 
110  !> Stop if a previously called assertion failed (delayed stop).
111 
112  !> This is useful if the
113  !> default stop mode is set to continue on failed assertions (which
114  !> can be enabled by call assertSetStopMode( .false. ).
115 
116  !> @param failmsg Message to print if assertion(s) failed.
117  !> @param doStop Controls stop behaviour, @see assert. Overrides the default
118  !> set with @see assertSetStopMode
119 
120  subroutine assertstoponfailed( failmsg, doStop )
121  character(*), intent(in), optional :: failmsg
122  logical, intent(in), optional :: dostop
123 
124  ! internal
125  logical :: ldostop
126 
127  ldostop = .true. ! default for this routine is to stop, regardless of current module default
128  if ( present( dostop ) ) ldostop = dostop ! ...but can be overridden again with optional argument
129 
130  if ( present( failmsg ) ) then
131  ! ...use given message
132  call assert( failcount == 0, failmsg, ldostop )
133  else
134  ! ...or substitute generic message
135  call assert( failcount == 0, itmint2str( failcount )//' assertions failed', ldostop )
136  end if
137  failcount = 0
138 
139  end subroutine assertstoponfailed
140 
141 
142  !> A generic assertion, tests a given logical expression.
143 
144  !> If it evaluates to .false.,
145  !> print the fail message and possibly stop execution.
146  !>
147  !> @param test The logical expression to test.
148  !> @param failmsg The message to print on fail. If omitted, a generic message is printed. Can be modified with a prefix (see assertSetMsgPrefix)
149  !> @param doStop Controls whether to stop execution. If doStop .true.,
150  !> the program is stopped. If .false., only the fail message is printed
151  !> and bookkeeping is done for delayed stopping (see assertStopOnFailed).
152  !> If given, overrides the default behaviour set by assertSetStopMode.
153 
154  !> @see assertStopOnFailed
155  !> @see assertSetStopMode
156  !> @see assertSetMsgPrefix
157 
158  !> @author H.-J. Klingshirn
159  !> @version 1.0
160 
161  subroutine assert( test, failmsg, doStop )
162  logical, intent(in) :: test
163  character(*), intent(in), optional :: failmsg
164  logical, intent(in), optional :: dostop
165 
166  if ( .not. test ) then
167  call assertfail( failmsg, dostop )
168  end if
169 
170  end subroutine assert
171 
172 
173  !> Test double precision floating point numbers for equality.
174 
175  !> @param x1, x2 The values to test
176  !> @param failmsg Same as for assert
177  !> @param doStop Same as for assert
178  !> @see assert
179 
180  subroutine assertequal( x1, x2, failmsg, doStop )
181  real(ITM_R8), intent(in) :: x1, x2
182  character(*), intent(in), optional :: failmsg
183  logical, intent(in), optional :: dostop
184 
185  ! x1 is reference value
186  ! x2 is actual value
187 
188  if ( .not. ( ( ( x1 + 2 * spacing( real( x1 ) ) ) >= x2 ) &
189  & .and. ( x1 - 2 * spacing( real( x1 ) ) ) <= x2 ) ) then
190 
191  call assertfail( failmsg, dostop )
192 
193  end if
194 
195  end subroutine assertequal
196 
197 
198 end module itm_assert