-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrksuite.cpp
5119 lines (5083 loc) · 201 KB
/
rksuite.cpp
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
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
#include "rksuite.h"
#include <stdlib.h>
inline double sign(double a, double b)
{
return (b >= 0.0 ? fabs(a) : -fabs(a));
}
inline double max(double a, double b)
{
return (a >= b ? a : b);
}
inline double min(double a, double b)
{
return (a <= b ? a : b);
}
inline int sign(int a, int b)
{
return (b >= 0 ? abs(a) : -abs(a));
}
inline int max(int a, int b)
{
return (a >= b ? a : b);
}
inline int min(int a, int b)
{
return (a <= b ? a : b);
}
void RKSUITE::setup(int neq, double tstart, double ystart[], double tend,
double tol, double thres[], int method, char* task,
bool errass, double hstart, double work[], int lenwrk,
bool mesage)
{
// SUBROUTINE SETUP(NEQ,TSTART,YSTART,TEND,TOL,THRES,METHOD,TASK,
// & ERRASS,HSTART,WORK,LENWRK,MESAGE)
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C
//C If you are not familiar with the code SETUP and how it is used in
//C conjunction with UT or CT to solve initial value problems, you should study
//C the document file rksuite.doc carefully before attempting to use the code.
//C The following "Brief Reminder" is intended only to remind you of the
//C meaning, type, and size requirements of the arguments.
//C
//C The environmental parameters OUTCH, MCHEPS, and DWARF are used in the
//C following description. To find out their values
//C
//C CALL ENVIRN(OUTCH,MCHEPS,DWARF)
//C
//C INPUT VARIABLES
//C
//C NEQ - INTEGER
//C The number of differential equations in the system.
//C Constraint: NEQ >= 1
//C TSTART - double PRECISION
//C The initial value of the independent variable.
//C YSTART(*) - double PRECISION array of length NEQ
//C The vector of initial values of the solution components.
//C TEND - double PRECISION
//C The integration proceeds from TSTART in the direction of
//C TEND. You cannot go past TEND.
//C Constraint: TEND must be clearly distinguishable from TSTART
//C in the precision available.
//C TOL - double PRECISION
//C The relative error tolerance.
//C Constraint: 0.01D0 >= TOL >= 10*MCHEPS
//C THRES(*) - double PRECISION array of length NEQ
//C THRES(L) is the threshold for the Ith solution component.
//C Constraint: THRES(L) >= SQRT(DWARF)
//C METHOD - INTEGER
//C Specifies which Runge-Kutta pair is to be used.
//C = 1 - use the (2,3) pair
//C = 2 - use the (4,5) pair
//C = 3 - use the (7,8) pair
//C TASK - CHARACTER*(*)
//C Only the first character of TASK is significant.
//C TASK(1:1) = 'U' or 'u' - UT is to be used
//C = 'C' or 'c' - CT is to be used
//C Constraint: TASK(1:1) = 'U'or 'u' or'C' or 'c'
//C ERRASS - LOGICAL
//C = .FALSE. - do not attempt to assess the true error.
//C = .TRUE. - assess the true error. Costs roughly twice
//C as much as the integration with METHODs 2 and
//C 3, and three times with METHOD = 1.
//C HSTART - double PRECISION
//C 0.0D0 - select automatically the first step size.
//C non-zero - try HSTART for the first step.
//C
//C WORKSPACE
//C
//C WORK(*) - double PRECISION array of length LENWRK
//C Do not alter the contents of this array after calling SETUP.
//C
//C INPUT VARIABLES
//C
//C LENWRK - INTEGER
//C Length of WORK(*): How big LENWRK must be depends
//C on the task and how it is to be solved.
//C
//C LENWRK = 32*NEQ is sufficient for all cases.
//C
//C If storage is a problem, the least storage possible
//C in the various cases is:
//C
//C If TASK = 'U' or 'u', then
//C if ERRASS = .FALSE. and
//C METHOD = 1, LENWRK must be at least 10*NEQ
//C = 2 20*NEQ
//C = 3 16*NEQ
//C if ERRASS = .TRUE. and
//C METHOD = 1, LENWRK must be at least 15*NEQ
//C = 2 32*NEQ
//C = 3 21*NEQ
//C
//C If TASK = 'C' or 'c', then
//C if ERRASS = .FALSE. and
//C METHOD = 1, LENWRK must be at least 10*NEQ
//C = 2 14*NEQ
//C = 3 16*NEQ
//C if ERRASS = .TRUE. and
//C METHOD = 1, LENWRK must be at least 15*NEQ
//C = 2 26*NEQ
//C = 3 21*NEQ
//C
//C Warning: To exploit the interpolation capability
//C of METHODs 1 and 2, you have to call INTRP. This
//C subroutine requires working storage in addition to
//C that specified here.
//C
//C MESAGE - LOGICAL
//C Specifies whether you want informative messages written to
//C the standard output channel OUTCH.
//C = .TRUE. - provide messages
//C = .FALSE. - do not provide messages
//C
//C In the event of a "catastrophic" failure to call SETUP correctly, the
//C nature of the catastrophe is reported on the standard output channel,
//C regardless of the value of MESAGE. Unless special provision was made
//C in advance (see rksuite.doc), the computation then comes to a STOP.
//C
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C .. Scalar Arguments ..
//C .. Common Block for Problem Definition ..
// double PRECISION TSTRT, TND, DIR, HSTRT, TOLR
// INTEGER NEQN
// COMMON /RKCOM1/ TSTRT, TND, DIR, HSTRT, TOLR, NEQN
// SAVE /RKCOM1/
//C .. Common Block to hold Problem Status ..
// double PRECISION T, H, TOLD, HOLD
// INTEGER NFCN, SVNFCN, OKSTP, FLSTP
// LOGICAL FIRST, LAST
// COMMON /RKCOM2/ T, H, TOLD, HOLD, NFCN, SVNFCN, OKSTP, FLSTP,
// & FIRST, LAST
// SAVE /RKCOM2/
//C .. Common Block for General Workspace Pointers ..
// INTEGER PRTHRS, PRERST, PRWT, PRYOLD, PRSCR, PRY, PRYP,
// & PRSTGS, PRINTP, LNINTP
// COMMON /RKCOM3/ PRTHRS, PRERST, PRWT, PRYOLD, PRSCR, PRY, PRYP,
// & PRSTGS, PRINTP, LNINTP
// SAVE /RKCOM3/
//C .. Common Block to hold Formula Definitions ..
// double PRECISION A(13,13), B(13), C(13), BHAT(13), R(11,6),
// & E(7)
// INTEGER PTR(13), NSTAGE, METHD, MINTP
// LOGICAL INTP
// COMMON /RKCOM4/ A, B, C, BHAT, R, E, PTR, NSTAGE, METHD,
// & MINTP, INTP
// SAVE /RKCOM4/
//C .. Common Block to hold Formula Characterisitcs ..
// double PRECISION TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4
// INTEGER ORDER, LSTSTG, MAXTRY, NSEC
// LOGICAL FSAL
// COMMON /RKCOM5/ TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4, ORDER, LSTSTG, MAXTRY,
// & NSEC, FSAL
// SAVE /RKCOM5/
//C .. Common Block for Global Error Assessment ..
// double PRECISION MAXERR, LOCMAX
// INTEGER GNFCN, PRZSTG, PRZY, PRZYP, PRZERS, PRZERR,
// & PRZYNU
// LOGICAL ERASON, ERASFL
// COMMON /RKCOM6/ MAXERR, LOCMAX, GNFCN, PRZSTG, PRZY, PRZYP,
// & PRZERS, PRZERR, PRZYNU, ERASON, ERASFL
// SAVE /RKCOM6/
//C .. Common Block for Environment Parameters ..
// double PRECISION MCHEPS, DWARF, RNDOFF, SQRRMC, CUBRMC, TINY
// INTEGER OUTCH
// COMMON /RKCOM7/ MCHEPS, DWARF, RNDOFF, SQRRMC, CUBRMC, TINY,
// & OUTCH
// SAVE /RKCOM7/
//C .. Common Block for Integrator Options ..
// LOGICAL MSG, UTASK
// COMMON /RKCOM8/ MSG, UTASK
// SAVE /RKCOM8/
//C .. Common Block for Error Message ..
// CHARACTER*80 REC(10)
// COMMON /RKCOM9/ REC
// SAVE /RKCOM9/
//C .. Parameters ..
// CHARACTER*6 SRNAME
// PARAMETER (SRNAME='SETUP')
// INTEGER MINUS1
// LOGICAL TELL
// PARAMETER (MINUS1=-1,TELL=.FALSE.)
// double PRECISION ONE, ZERO, PT01
// PARAMETER (ONE=1.0D+0,ZERO=0.0D+0,PT01=0.01D+0)
const char srname[] = "SETUP";
const int minus1 = -1;
const bool tell = false;
const double one = 1.0;
const double zero = 0.0;
const double pt01 = 0.01;
//C .. Local Scalars ..
double hmin;
int flag, freepr, ier, l, lintpl, lreq, nrec, vecstg;
bool legalt, reqstg;
char task1;
//C .. External Subroutines ..
// EXTERNAL CONST, MCONST, RKMSG, RKSIT
//C .. Intrinsic Functions ..
// INTRINSIC ABS, MAX, SIGN
//C .. Executable Statements ..
//C
//C Clear previous flag values of subprograms in the suite.
//C
ier = minus1;
rksit(tell,srname,ier);
//C
ier = 1;
nrec = 0;
//C
//C Fetch output channel and machine constants; initialise common
//C block /RKCOM7/
//C
mconst(method);
//C
//C Check for valid input of trivial arguments
task1 = task[0];
legalt = task1 == 'U' || task1 == 'u' ||
task1 == 'C' || task1 == 'c';
if (!legalt) {
ier = 911;
nrec = 3;
sprintf(&rkcom9.rec[0][0]," ** You have set the first character of");
sprintf(&rkcom9.rec[1][0]," ** TASK to be '%c'. It must be one of",task1);
sprintf(&rkcom9.rec[2][0]," ** 'U','u','C' or 'c'.");
}
else if (neq < 1) {
ier = 911;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** You have set NEQ = %6d which is less than 1.",neq);
}
else if (method < 1 || method > 3) {
ier = 911;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** You have set METHOD = %6d which is not 1, 2, or 3.",method);
}
else if (tstart == tend) {
ier = 911;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** You have set TSTART = TEND = %13.5lf.",tstart);
}
else if ((tol > pt01) || (tol < rkcom7.rndoff)) {
ier = 911;
nrec = 2;
sprintf(&rkcom9.rec[0][0]," ** You have set TOL = %13.5lf which is not permitted. The",tol);
sprintf(&rkcom9.rec[1][0]," ** range of permitted values is (%13.5lf,0.01D0).",rkcom7.rndoff);
}
else {
l = 0;
label20:
if (thres[l] < rkcom7.tiny) {
ier = 911;
nrec = 2;
sprintf(&rkcom9.rec[0][0]," ** You have set THRES(%6d) to be %13.5lf which is",l,thres[l]);
sprintf(&rkcom9.rec[1][0]," ** less than the permitted minimum,%13.5lf.",rkcom7.tiny);
}
l++;
if (ier != 911 && l < neq) goto label20;
}
//C
//C Return if error detected
//C
if (ier != 1) goto label80;
//C
//C Set formula definitions and characteristics by means of arguments
//C in the call list and COMMON blocks /RKCOM4/ and /RKCOM5/
//C
rkconst(method,vecstg,reqstg,lintpl);
//C
//C Set options in /RKCOM8/
rkcom8.utask = task1 == 'U' || task1 == 'u';
rkcom8.msg = mesage;
//C
//C Initialise problem status in /RKCOM1/ and /RKCOM2/
rkcom1.neqn = neq;
rkcom1.tstrt = tstart;
rkcom1.tnd = tend;
rkcom2.t = tstart;
rkcom2.told = tstart;
rkcom1.dir = sign(one,tend-tstart);
//C
//C In CT the first step taken will have magnitude H. If HSTRT = ABS(HSTART)
//C is not equal to zero, H = HSTRT. If HSTRT is equal to zero, the code is
//C to find an on-scale initial step size H. To start this process, H is set
//C here to an upper bound on the first step size that reflects the scale of
//C the independent variable. UT has some additional information, namely the
//C first output point, that is used to refine this bound in UT when UTASK
//C is .TRUE.. If HSTRT is not zero, but it is either too big or too small,
//C the input HSTART is ignored and HSTRT is set to zero to activate the
//C automatic determination of an on-scale initial step size.
//C
rkcom1.hstrt = fabs(hstart);
hmin = max(rkcom7.tiny,rkcom5.toosml*max(fabs(tstart),fabs(tend)));
if (rkcom1.hstrt > fabs(tend-tstart) || rkcom1.hstrt < hmin) rkcom1.hstrt = zero;
if (rkcom1.hstrt == zero) {
rkcom2.h = max(fabs(tend-tstart)/rkcom5.rs3,hmin);
}
else {
rkcom2.h = rkcom1.hstrt;
}
rkcom2.hold = zero;
rkcom1.tolr = tol;
rkcom2.nfcn = 0;
rkcom2.svnfcn = 0;
rkcom2.okstp = 0;
rkcom2.flstp = 0;
rkcom2.first = true;
rkcom2.last = false;
//C
//C WORK(*) is partioned into a number of arrays using pointers. These
//C pointers are set in /RKCOM3/.
rkcom3.prthrs = 0;
//C the threshold values
rkcom3.prerst = rkcom3.prthrs + neq;
//C the error estimates
rkcom3.prwt = rkcom3.prerst + neq;
//C the weights used in the local error test
rkcom3.pryold = rkcom3.prwt + neq;
//C the previous value of the solution
rkcom3.prscr = rkcom3.pryold + neq;
//C scratch array used for the higher order
//C approximate solution and for the previous
//C value of the derivative of the solution
rkcom3.pry = rkcom3.prscr + neq;
//C the dependent variables
rkcom3.pryp = rkcom3.pry + neq;
//C the derivatives
rkcom3.prstgs = rkcom3.pryp + neq;
//C intermediate stages held in an internal
//C array STAGES(NEQ,VECSTG)
//C
freepr = rkcom3.prstgs + vecstg*neq;
//C
//C Allocate storage for interpolation if the TASK = 'U' or 'u' was
//C specified. INTP and LINTPL returned by CONST indicate whether there
//C is an interpolation scheme associated with the pair and how much
//C storage is required.
//C
rkcom3.printp = 1;
rkcom3.lnintp = 1;
if (rkcom8.utask) {
if (rkcom4.intp) {
rkcom3.lnintp = lintpl*neq;
if (reqstg) {
rkcom3.printp = freepr;
freepr = rkcom3.printp + rkcom3.lnintp;
}
else {
rkcom3.printp = rkcom3.prstgs;
freepr = max(rkcom3.printp+vecstg*neq,rkcom3.printp+rkcom3.lnintp);
}
}
}
//C
//C Initialise state and allocate storage for global error assessment
//C using /RKCOM6/
rkcom6.gnfcn = 0;
rkcom6.maxerr = zero;
rkcom6.locmax = tstart;
rkcom6.erason = errass;
rkcom6.erasfl = false;
if (errass) {
//C
//C Storage is required for the stages of a secondary integration. The
//C stages of the primary intergration can only be overwritten in the
//C cases where there is no interpolant or the interpolant does not
//C require information about the stages (e.g. METHOD 3 and METHOD 1,
//C respectively).
if (!reqstg) {
rkcom6.przstg = rkcom3.prstgs;
}
else {
rkcom6.przstg = freepr;
freepr = rkcom6.przstg + vecstg*neq;
}
rkcom6.przy = freepr;
rkcom6.przyp = rkcom6.przy + neq;
rkcom6.przers = rkcom6.przyp + neq;
rkcom6.przerr = rkcom6.przers + neq;
rkcom6.przynu = rkcom6.przerr + neq;
freepr = rkcom6.przynu + neq;
}
else {
rkcom6.przstg = 1;
rkcom6.przy = 1;
rkcom6.przyp = 1;
rkcom6.przers = 1;
rkcom6.przerr = 1;
rkcom6.przynu = 1;
}
//C
lreq = freepr - 1;
//C
//C Check for enough workspace and suitable range of integration
//C
if (lenwrk < lreq) {
ier = 911;
nrec = 2;
sprintf(&rkcom9.rec[0][0]," ** You have not supplied enough workspace. You gave LENWRK");
sprintf(&rkcom9.rec[1][0]," ** as%6d, but it must be at least %6d.",lenwrk,lreq);
}
else {
hmin = max(rkcom7.tiny,rkcom5.toosml*max(fabs(tstart),fabs(tend)));
if (fabs(tend-tstart) < hmin) {
ier = 911;
nrec = 4;
sprintf(&rkcom9.rec[0][0]," ** You have set values for TEND and TSTART that are not");
sprintf(&rkcom9.rec[1][0]," ** clearly distinguishable for the method and the precision");
sprintf(&rkcom9.rec[2][0]," ** of the computer being used. ABS(TEND-TSTART) is %13.5lf",fabs(tend-tstart));
sprintf(&rkcom9.rec[3][0]," ** but should be at least %13.5lf.",hmin);
}
}
//C
//C Return if error detected
//C
if (ier != 1) goto label80;
//C
//C Initialize elements of the workspace
for (l = 0; l < neq; l++) {
work[rkcom3.prthrs+l] = thres[l];
work[rkcom3.pry+l] = ystart[l];
}
//C
//C Initialize the global error to zero when ERRASS = .TRUE.
if (errass) {
for (l = 0; l < neq; l++) {
work[rkcom6.przerr+l] = zero;
}
}
//C
label80:
//C
rkmsg(ier,srname,nrec,flag);
//C
return;
}
void RKSUITE::ut(void (*f)(double, double*, double*), double twant, double& tgot,
double ygot[], double ypgot[], double ymax[], double work[], int& uflag)
{
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C
//C If you are not familiar with the code UT and how it is used in
//C conjunction with SETUP to solve initial value problems, you should study
//C the document file rksuite.doc carefully before proceeding further. The
//C following "Brief Reminder" is intended only to remind you of the meaning,
//C type, and size requirements of the arguments.
//C
//C NAME DECLARED IN AN EXTERNAL STATEMENT IN THE CALLING PROGRAM:
//C
//C F - name of the subroutine for evaluating the differential
//C equations.
//C
//C The subroutine F must have the form
//C
//C SUBROUTINE F(T,Y,YP)
//C double PRECISION T,Y(*),YP(*)
//C Given input values of the independent variable T and the solution
//C components Y(*), for each L = 1,2,...,NEQ evaluate the differential
//C equation for the derivative of the Ith solution component and place the
//C value in YP(L). Do not alter the input values of T and Y(*).
//C RETURN
//C END
//C
//C INPUT VARIABLE
//C
//C TWANT - double PRECISION
//C The next value of the independent variable where a
//C solution is desired.
//C
//C Constraints: TWANT must lie between the previous value
//C of TGOT (TSTART on the first call) and TEND. TWANT can be
//C equal to TEND, but it must be clearly distinguishable from
//C the previous value of TGOT (TSTART on the first call) in
//C the precision available.
//C
//C OUTPUT VARIABLES
//C
//C TGOT - double PRECISION
//C A solution has been computed at this value of the
//C independent variable.
//C YGOT(*) - double PRECISION array of length NEQ
//C Approximation to the true solution at TGOT. Do not alter
//C the contents of this array
//C YPGOT(*) - double PRECISION array of length NEQ
//C Approximation to the first derivative of the true
//C solution at TGOT.
//C YMAX(*) - double PRECISION array of length NEQ
//C YMAX(L) is the largest magnitude of YGOT(L) computed at any
//C time in the integration from TSTART to TGOT. Do not alter
//C the contents of this array.
//C
//C WORKSPACE
//C
//C WORK(*) - double PRECISION array as used in SETUP
//C Do not alter the contents of this array.
//C
//C OUTPUT VARIABLE
//C
//C UFLAG - INTEGER
//C
//C SUCCESS. TGOT = TWANT.
//C = 1 - Complete success.
//C
//C "SOFT" FAILURES
//C = 2 - Warning: You are using METHOD = 3 inefficiently
//C by computing answers at many values of TWANT. If
//C you really need answers at so many specific points,
//C it would be more efficient to compute them with
//C METHOD = 2. To do this you would need to restart
//C from TGOT, YGOT(*) by a call to SETUP. If you wish
//C to continue as you are, you may.
//C = 3 - Warning: A considerable amount of work has been
//C expended. If you wish to continue on to TWANT, just
//C call UT again.
//C = 4 - Warning: It appears that this problem is "stiff".
//C You really should change to another code that is
//C intended for such problems, but if you insist, you can
//C continue with UT by calling it again.
//C
//C "HARD" FAILURES
//C = 5 - You are asking for too much accuracy. You cannot
//C continue integrating this problem.
//C = 6 - The global error assessment may not be reliable beyond
//C the current point in the integration. You cannot
//C continue integrating this problem.
//C
//C "CATASTROPHIC" FAILURES
//C = 911 - The nature of the catastrophe is reported on
//C the standard output channel. Unless special
//C provision was made in advance (see rksuite.doc),
//C the computation then comes to a STOP.
//C
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C .. Scalar Arguments ..
// double PRECISION TGOT, TWANT
// INTEGER UFLAG
//C .. Array Arguments ..
// double PRECISION WORK(*), YGOT(*), YMAX(*), YPGOT(*)
//C .. Subroutine Arguments ..
// EXTERNAL F
//C .. Common Block for Problem Definition ..
// double PRECISION TSTRT, TND, DIR, HSTRT, TOLR
// INTEGER NEQN
// COMMON /RKCOM1/ TSTRT, TND, DIR, HSTRT, TOLR, NEQN
// SAVE /RKCOM1/
//C .. Common Block to hold Problem Status ..
// double PRECISION T, H, TOLD, HOLD
// INTEGER NFCN, SVNFCN, OKSTP, FLSTP
// LOGICAL FIRST, LAST
// COMMON /RKCOM2/ T, H, TOLD, HOLD, NFCN, SVNFCN, OKSTP, FLSTP,
// & FIRST, LAST
// SAVE /RKCOM2/
//C .. Common Block for General Workspace Pointers ..
// INTEGER PRTHRS, PRERST, PRWT, PRYOLD, PRSCR, PRY, PRYP,
// & PRSTGS, PRINTP, LNINTP
// COMMON /RKCOM3/ PRTHRS, PRERST, PRWT, PRYOLD, PRSCR, PRY, PRYP,
// & PRSTGS, PRINTP, LNINTP
// SAVE /RKCOM3/
//C .. Common Block to hold Formula Definitions ..
// double PRECISION A(13,13), B(13), C(13), BHAT(13), R(11,6),
// & E(7)
// INTEGER PTR(13), NSTAGE, METHD, MINTP
// LOGICAL INTP
// COMMON /RKCOM4/ A, B, C, BHAT, R, E, PTR, NSTAGE, METHD,
// & MINTP, INTP
// SAVE /RKCOM4/
//C .. Common Block to hold Formula Characterisitcs ..
// double PRECISION TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4
// INTEGER ORDER, LSTSTG, MAXTRY, NSEC
// LOGICAL FSAL
// COMMON /RKCOM5/ TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4, ORDER, LSTSTG, MAXTRY,
// & NSEC, FSAL
// SAVE /RKCOM5/
//C .. Common Block for Environment Parameters ..
// double PRECISION MCHEPS, DWARF, RNDOFF, SQRRMC, CUBRMC, TINY
// INTEGER OUTCH
// COMMON /RKCOM7/ MCHEPS, DWARF, RNDOFF, SQRRMC, CUBRMC, TINY,
// & OUTCH
// SAVE /RKCOM7/
//C .. Common Block for Integrator Options ..
// LOGICAL MSG, UTASK
// COMMON /RKCOM8/ MSG, UTASK
// SAVE /RKCOM8/
//C .. Common Block for Error Message ..
// CHARACTER*80 REC(10)
// COMMON /RKCOM9/ REC
// SAVE /RKCOM9/
//C .. Parameters ..
// CHARACTER*6 SRNAME
// PARAMETER (SRNAME='UT')
const char srname[] = "UT";
// LOGICAL ASK, TELL
// PARAMETER (ASK=.TRUE.,TELL=.FALSE.)
const bool ask = true, tell = false;
// INTEGER MINUS1, MINUS2
// PARAMETER (MINUS1=-1,MINUS2=-2)
const int minus1 = -1, minus2 = -2;
// double PRECISION ZERO
// PARAMETER (ZERO=0.0D+0)
const double zero = 0.0;
//C .. Local Scalars ..
double hmin, tnow;
int cflag, ier, l, nrec, state;
bool baderr, goback;
//C .. External Subroutines ..
// EXTERNAL CHKFL, CT, INTRP, RESET, RKMSG, RKSIT
//C .. Intrinsic Functions ..
// INTRINSIC ABS, MAX, MIN
//C .. Save statement ..
// SAVE UTEND, TLAST
static double utend, tlast;
//C .. Executable Statements ..
ier = 1;
nrec = 0;
goback = false;
baderr = false;
//C
//C Is it permissible to call UT?
//C
rksit(ask,"SETUP",state);
if (state == 911) {
ier = 912;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** A catastrophic error has already been detected elsewhere.");
goto label100;
}
if (state == minus1) {
ier = 911;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** You have not called SETUP, so you cannot use UT.");
goto label100;
}
if (!rkcom8.utask) {
ier = 911;
nrec = 2;
sprintf(&rkcom9.rec[0][0]," ** You have called UT after you specified in SETUP that");
sprintf(&rkcom9.rec[1][0]," ** you were going to use CT. This is not permitted.");
goto label100;
}
rksit(ask,srname,state);
if (state == 5 || state == 6) {
ier = 911;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** This routine has already returned with a hard failure.");
sprintf(&rkcom9.rec[1][0]," ** You must call SETUP to start another problem.");
goto label100;
}
state = minus2;
rksit(tell,srname,state);
//C
if (rkcom2.first) {
//C
//C First call.
//C
//C A value of TND is specified in SETUP. When INTP = .FALSE., as with
//C METHD = 3, output is obtained at the specified TWANT by resetting TND
//C to TWANT. At this point, before the integration gets started, this can
//C be done with a simple assignment. Later it is done with a call to RESET.
//C The original TND is SAVEd as a local variable UTEND.
//C
utend = rkcom1.tnd;
if (!rkcom4.intp) rkcom1.tnd = twant;
//C
//C The last TGOT returned is SAVEd in the variable TLAST. T (a variable
//C passed through the common block RKCOM2) records how far the integration
//C has advanced towards the specified TND. When output is obtained by
//C interpolation, the integration goes past the TGOT returned (T is closer
//C to the specified TND than TGOT). Initialize these variables and YMAX(*).
tlast = rkcom1.tstrt;
tgot = rkcom1.tstrt;
for (l = 0; l < rkcom1.neqn; l++) {
ymax[l] = fabs(work[rkcom3.pry+l]);
}
//C
//C If the code is to find an on-scale initial step size H, a bound was placed
//C on H in SETUP. Here the first output point is used to refine this bound.
if (rkcom1.hstrt == zero) {
rkcom2.h = min(fabs(rkcom2.h),fabs(twant-rkcom1.tstrt));
hmin = max(rkcom7.tiny,rkcom5.toosml*max(fabs(rkcom1.tstrt),fabs(rkcom1.tnd)));
rkcom2.h = max(rkcom2.h,hmin);
}
//C
}
else {
//C
//C Subsequent call.
//C
if (tlast == utend) {
ier = 911;
nrec = 3;
sprintf(&rkcom9.rec[0][0]," ** You have called UT after reaching TEND. (Your last");
sprintf(&rkcom9.rec[1][0]," ** call to UT resulted in TGOT = TEND.) To start a new");
sprintf(&rkcom9.rec[2][0]," ** problem, you will need to call SETUP.");
goto label100;
}
//C
}
//C
//C Check for valid TWANT.
//C
if (rkcom1.dir*(twant-tlast) <= zero) {
ier = 911;
nrec = 4;
sprintf(&rkcom9.rec[0][0]," ** You have made a call to UT with a TWANT that does");
sprintf(&rkcom9.rec[1][0]," ** not lie between the previous value of TGOT (TSTART");
sprintf(&rkcom9.rec[2][0]," ** on the first call) and TEND. This is not permitted.");
sprintf(&rkcom9.rec[3][0]," ** Check your program carefully.");
goto label100;
}
if (rkcom1.dir*(twant-utend) > zero) {
hmin = max(rkcom7.tiny,rkcom5.toosml*max(fabs(twant),fabs(utend)));
if (fabs(twant-utend) < hmin) {
ier = 911;
nrec = 5;
sprintf(&rkcom9.rec[0][0]," ** You have made a call to UT with a TWANT that does");
sprintf(&rkcom9.rec[1][0]," ** not lie between the previous value of TGOT (TSTART on");
sprintf(&rkcom9.rec[2][0]," ** the first call) and TEND. This is not permitted. TWANT");
sprintf(&rkcom9.rec[3][0]," ** is very close to TEND, so you may have meant to set");
sprintf(&rkcom9.rec[4][0]," ** it to be TEND exactly. Check your program carefully.");
}
else {
ier = 911;
nrec = 4;
sprintf(&rkcom9.rec[0][0]," ** You have made a call to UT with a TWANT that does");
sprintf(&rkcom9.rec[1][0]," ** not lie between the previous value of TGOT (TSTART");
sprintf(&rkcom9.rec[2][0]," ** on the first call) and TEND. This is not permitted.");
sprintf(&rkcom9.rec[3][0]," ** Check your program carefully.");
}
goto label100;
}
if (!rkcom4.intp) {
hmin = max(rkcom7.tiny,rkcom5.toosml*max(fabs(tlast),fabs(twant)));
if (fabs(twant-tlast) < hmin) {
ier = 911;
nrec = 4;
sprintf(&rkcom9.rec[0][0]," ** You have made a call to UT with a TWANT that is not");
sprintf(&rkcom9.rec[1][0]," ** sufficiently different from the last value of TGOT");
sprintf(&rkcom9.rec[2][0]," ** (TSTART on the first call). When using METHOD = 3,");
sprintf(&rkcom9.rec[3][0]," ** it must differ by at least %13.5lf.",hmin);
goto label100;
}
//C
//C We have a valid TWANT. There is no interpolation with this METHD and
//C therefore we step to TWANT exactly by resetting TND with a call to RESET.
//C On the first step this matter is handled differently as explained above.
//C
if (!rkcom2.first) {
reset(twant);
chkfl(ask,baderr);
if (baderr) goto label100;
}
}
//C
//C Process output, decide whether to take another step.
//C
label40:
//C
if (rkcom4.intp) {
//C
//C Interpolation is possible with this METHD. The integration has
//C already reached T. If this is past TWANT, GOBACK is set .TRUE. and
//C the answers are obtained by interpolation.
//C
goback = rkcom1.dir*(rkcom2.t-twant) >= zero;
if (goback) {
intrp(twant,"Both solution and derivative",rkcom1.neqn,ygot,
ypgot,f,work,&work[rkcom3.printp],rkcom3.lnintp);
chkfl(ask,baderr);
if (baderr) goto label100;
tgot = twant;
}
}
else {
//C
//C Interpolation is not possible with this METHD, so output is obtained
//C by integrating to TWANT = TND. Both YGOT(*) and YPGOT(*) are then
//C already loaded with the solution at TWANT by CT.
//C
goback = rkcom2.t == twant;
if (goback) tgot = twant;
}
//C
//C Updating of YMAX(*) is done here to account for the fact that when
//C interpolation is done, the integration goes past TGOT. Note that YGOT(*)
//C is not defined until CT is called. YMAX(*) was initialized at TSTRT
//C from values stored in WORK(*), so only needs to be updated for T
//C different from TSTRT.
if (rkcom2.t != rkcom1.tstrt) {
for (l = 0; l < rkcom1.neqn; l++) {
ymax[l] = max(ymax[l],fabs(ygot[l]));
}
}
//C
//C If done, go to the exit point.
if (goback) goto label100;
//C
//C Take a step with CT in the direction of TND. On exit, the solution is
//C advanced to TNOW. The way CT is written, the approximate solution at
//C TNOW is available in both YGOT(*) and in WORK(*). If output is obtained by
//C stepping to the end (TNOW = TWANT = TND), YGOT(*) can be returned directly.
//C If output is obtained by interpolation, the subroutine INTRP that does this
//C uses the values in WORK(*) for its computations and places the approximate
//C solution at TWANT in the array YGOT(*) for return to the calling program.
//C The approximate derivative is handled in the same way. TNOW is output from
//C CT and is actually a copy of T declared above in a common block.
//C
ct(f,tnow,ygot,ypgot,work,cflag);
ier = cflag;
//C
//C A successful step by CT is indicated by CFLAG = 1 or = 2.
if (cflag == 1) {
goto label40;
}
else if (cflag == 2) {
//C
//C Supplement the warning message written in CT.
nrec = 3;
sprintf(&rkcom9.rec[0][0]," ** The last message was produced on a call to CT from UT.");
sprintf(&rkcom9.rec[1][0]," ** In UT the appropriate action is to change to METHOD = 2,");
sprintf(&rkcom9.rec[2][0]," ** or, if insufficient memory is available, to METHOD = 1.");
}
else if (cflag <= 6) {
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** The last message was produced on a call to CT from UT.");
}
else {
baderr = true;
}
tgot = rkcom2.t;
//C
//C Update YMAX(*) before the return.
for (l = 0; l < rkcom1.neqn; l++) {
ymax[l] = max(ymax[l],fabs(ygot[l]));
}
//C
//C Exit point for UT.
//C
label100:
//C
if (baderr) {
ier = 911;
nrec = 4;
sprintf(&rkcom9.rec[0][0]," ** An internal call by UT to a subroutine resulted in an");
sprintf(&rkcom9.rec[1][0]," ** error that should not happen. Check your program");
sprintf(&rkcom9.rec[2][0]," ** carefully for array sizes, correct number of arguments,");
sprintf(&rkcom9.rec[3][0]," ** type mismatches ... .");
}
//C
tlast = tgot;
//C
//C All exits are done here after a call to RKMSG to report
//C what happened and set UFLAG.
//C
rkmsg(ier,srname,nrec,uflag);
//C
return;
}
void RKSUITE::stat(int& totfcn, int& stpcst, double& waste, int& stpsok, double& hnext)
{
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C
//C If you are not familiar with the code STAT and how it is used in
//C conjunction with the integrators CT and UT, you should study the
//C document file rksuite.doc carefully before attempting to use the code.
//C The following "Brief Reminder" is intended only to remind you of the
//C meaning, type, and size requirements of the arguments.
//C
//C STAT is called to obtain some details about the integration.
//C
//C OUTPUT VARIABLES
//C
//C TOTFCN - INTEGER
//C Total number of calls to F in the integration so far --
//C a measure of the cost of the integration.
//C STPCST - INTEGER
//C Cost of a typical step with this METHOD measured in
//C calls to F.
//C WASTE - double PRECISION
//C The number of attempted steps that failed to meet the
//C local error requirement divided by the total number of
//C steps attempted so far in the integration.
//C STPSOK - INTEGER
//C The number of accepted steps.
//C HNEXT - double PRECISION
//C The step size the integrator plans to use for the next step.
//C
//C$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$
//C .. Scalar Arguments ..
// double PRECISION HNEXT, WASTE
// INTEGER STPCST, STPSOK, TOTFCN
//C .. Common Block to hold Problem Status ..
// double PRECISION T, H, TOLD, HOLD
// INTEGER NFCN, SVNFCN, OKSTP, FLSTP
// LOGICAL FIRST, LAST
// COMMON /RKCOM2/ T, H, TOLD, HOLD, NFCN, SVNFCN, OKSTP, FLSTP,
// & FIRST, LAST
// SAVE /RKCOM2/
//C .. Common Block to hold Formula Characterisitcs ..
// double PRECISION TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4
// INTEGER ORDER, LSTSTG, MAXTRY, NSEC
// LOGICAL FSAL
// COMMON /RKCOM5/ TOOSML, COST, SAFETY, EXPON, STBRAD, TANANG,
// & RS, RS1, RS2, RS3, RS4, ORDER, LSTSTG, MAXTRY,
// & NSEC, FSAL
// SAVE /RKCOM5/
//C .. Common Block for Global Error Assessment ..
// double PRECISION MAXERR, LOCMAX
// INTEGER GNFCN, PRZSTG, PRZY, PRZYP, PRZERS, PRZERR,
// & PRZYNU
// LOGICAL ERASON, ERASFL
// COMMON /RKCOM6/ MAXERR, LOCMAX, GNFCN, PRZSTG, PRZY, PRZYP,
// & PRZERS, PRZERR, PRZYNU, ERASON, ERASFL
// SAVE /RKCOM6/
//C .. Common Block for Integrator Options ..
// LOGICAL MSG, UTASK
// COMMON /RKCOM8/ MSG, UTASK
// SAVE /RKCOM8/
//C .. Common Block for Error Message ..
// CHARACTER*80 REC(10)
// COMMON /RKCOM9/ REC
// SAVE /RKCOM9/
//C .. Parameters ..
// CHARACTER*6 SRNAME
// PARAMETER (SRNAME='STAT')
const char srname[] = "STAT";
// LOGICAL ASK
// INTEGER MINUS1, MINUS2
// PARAMETER (ASK=.TRUE.,MINUS1=-1,MINUS2=-2)
const bool ask = true;
const int minus1 = -1, minus2 = -2;
// double PRECISION ZERO
// PARAMETER (ZERO=0.0D+0)
const double zero = 0.0;
//C .. Local Scalars ..
int flag, ier, nrec, state;
//C .. External Subroutines ..
// EXTERNAL RKMSG, RKSIT
//C .. Intrinsic Functions ..
// INTRINSIC DBLE
//C .. Executable Statements ..
//C
ier = 1;
nrec = 0;
//C
//C Is it permissible to call STAT?
//C
rksit(ask,srname,state);
if (state == 911) {
ier = 912;
nrec = 1;
sprintf(&rkcom9.rec[0][0]," ** A catastrophic error has already been detected elsewhere.");
goto label20;
}
if (state == minus2) {
ier = 911;
nrec = 3;
sprintf(&rkcom9.rec[0][0]," ** You have already made a call to STAT after a hard");
sprintf(&rkcom9.rec[1][0]," ** failure was reported from the integrator. You cannot");
sprintf(&rkcom9.rec[2][0]," ** call STAT again.");
goto label20;
}
rksit(ask,"CT",state);
if (state == minus1) {
ier = 911;
nrec = 1;
if (rkcom8.utask) {