3
4
5
6
7
8
9 CHARACTER TRANS, UPLO
10 INTEGER IB, INFO, JA, LAF, LWORK, N, NRHS
11
12
13 INTEGER DESCA( * ), DESCB( * )
14 COMPLEX*16 AF( * ), B( * ), E( * ), WORK( * )
15 DOUBLE PRECISION D( * )
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 DOUBLE PRECISION ONE, ZERO
380 parameter( one = 1.0d+0 )
381 parameter( zero = 0.0d+0 )
382 COMPLEX*16 CONE, CZERO
383 parameter( cone = ( 1.0d+0, 0.0d+0 ) )
384 parameter( czero = ( 0.0d+0, 0.0d+0 ) )
385 INTEGER INT_ONE
386 parameter( int_one = 1 )
387 INTEGER DESCMULT, BIGNUM
388 parameter(descmult = 100, bignum = descmult * descmult)
389 INTEGER BLOCK_CYCLIC_2D, CSRC_, CTXT_, DLEN_, DTYPE_,
390 $ LLD_, MB_, M_, NB_, N_, RSRC_
391 parameter( block_cyclic_2d = 1, dlen_ = 9, dtype_ = 1,
392 $ ctxt_ = 2, m_ = 3, n_ = 4, mb_ = 5, nb_ = 6,
393 $ rsrc_ = 7, csrc_ = 8, lld_ = 9 )
394
395
396 INTEGER CSRC, FIRST_PROC, ICTXT, ICTXT_NEW, ICTXT_SAVE,
397 $ IDUM1, IDUM2, IDUM3, JA_NEW, LEVEL_DIST, LLDA,
398 $ LLDB, MYCOL, MYROW, MY_NUM_COLS, NB, NP, NPCOL,
399 $ NPROW, NP_SAVE, ODD_SIZE, PART_OFFSET,
400 $ PART_SIZE, RETURN_CODE, STORE_M_B, STORE_N_A,
401 $ TEMP, WORK_SIZE_MIN
402
403
404 INTEGER DESCA_1XP( 7 ), DESCB_PX1( 7 ),
405 $ PARAM_CHECK( 16, 3 )
406
407
408 EXTERNAL blacs_get, blacs_gridexit, blacs_gridinfo,
410 $ zgerv2d, zgesd2d, zlamov,
zmatadd, ztbtrs,
411 $ ztrmm, ztrtrs
412
413
414 LOGICAL LSAME
415 INTEGER NUMROC
417
418
419 INTRINSIC ichar,
min, mod
420
421
422
423
424
425 info = 0
426
427
428
429
430 desca_1xp( 1 ) = 501
431 descb_px1( 1 ) = 502
432
433 temp = desca( dtype_ )
434 IF( temp .EQ. 502 ) THEN
435
436 desca( dtype_ ) = 501
437 ENDIF
438
440
441 desca( dtype_ ) = temp
442
443 IF( return_code .NE. 0) THEN
444 info = -( 8*100 + 2 )
445 ENDIF
446
448
449 IF( return_code .NE. 0) THEN
450 info = -( 11*100 + 2 )
451 ENDIF
452
453
454
455
456 IF( desca_1xp( 2 ) .NE. descb_px1( 2 ) ) THEN
457 info = -( 11*100 + 2 )
458 ENDIF
459
460
461
462
463
464 IF( desca_1xp( 4 ) .NE. descb_px1( 4 ) ) THEN
465 info = -( 11*100 + 4 )
466 ENDIF
467
468
469
470 IF( desca_1xp( 5 ) .NE. descb_px1( 5 ) ) THEN
471 info = -( 11*100 + 5 )
472 ENDIF
473
474
475
476 ictxt = desca_1xp( 2 )
477 csrc = desca_1xp( 5 )
478 nb = desca_1xp( 4 )
479 llda = desca_1xp( 6 )
480 store_n_a = desca_1xp( 3 )
481 lldb = descb_px1( 6 )
482 store_m_b = descb_px1( 3 )
483
484
485
486
487 CALL blacs_gridinfo( ictxt, nprow, npcol, myrow, mycol )
488 np = nprow * npcol
489
490
491
492 IF(
lsame( uplo,
'U' ) )
THEN
493 idum1 = ichar( 'U' )
494 ELSE IF (
lsame( uplo,
'L' ) )
THEN
495 idum1 = ichar( 'L' )
496 ELSE
497 info = -1
498 END IF
499
500 IF(
lsame( trans,
'N' ) )
THEN
501 idum2 = ichar( 'N' )
502 ELSE IF (
lsame( trans,
'C' ) )
THEN
503 idum2 = ichar( 'C' )
504 ELSE
505 info = -2
506 END IF
507
508 IF( lwork .LT. -1) THEN
509 info = -15
510 ELSE IF ( lwork .EQ. -1 ) THEN
511 idum3 = -1
512 ELSE
513 idum3 = 1
514 ENDIF
515
516 IF( n .LT. 0 ) THEN
517 info = -3
518 ENDIF
519
520 IF( n+ja-1 .GT. store_n_a ) THEN
521 info = -( 8*100 + 6 )
522 ENDIF
523
524 IF( n+ib-1 .GT. store_m_b ) THEN
525 info = -( 11*100 + 3 )
526 ENDIF
527
528 IF( lldb .LT. nb ) THEN
529 info = -( 11*100 + 6 )
530 ENDIF
531
532 IF( nrhs .LT. 0 ) THEN
533 info = -4
534 ENDIF
535
536
537
538 IF( ja .NE. ib) THEN
539 info = -7
540 ENDIF
541
542
543
544 IF( nprow .NE. 1 ) THEN
545 info = -( 8*100+2 )
546 ENDIF
547
548 IF( n .GT. np*nb-mod( ja-1, nb )) THEN
549 info = -( 3 )
551 $ 'PZPTTRSV, D&C alg.: only 1 block per proc',
552 $ -info )
553 RETURN
554 ENDIF
555
556 IF((ja+n-1.GT.nb) .AND. ( nb.LT.2*int_one )) THEN
557 info = -( 8*100+4 )
559 $ 'PZPTTRSV, D&C alg.: NB too small',
560 $ -info )
561 RETURN
562 ENDIF
563
564
565 work_size_min =
566 $ int_one*nrhs
567
568 work( 1 ) = work_size_min
569
570 IF( lwork .LT. work_size_min ) THEN
571 IF( lwork .NE. -1 ) THEN
572 info = -15
574 $ 'PZPTTRSV: worksize error',
575 $ -info )
576 ENDIF
577 RETURN
578 ENDIF
579
580
581
582 param_check( 16, 1 ) = descb(5)
583 param_check( 15, 1 ) = descb(4)
584 param_check( 14, 1 ) = descb(3)
585 param_check( 13, 1 ) = descb(2)
586 param_check( 12, 1 ) = descb(1)
587 param_check( 11, 1 ) = ib
588 param_check( 10, 1 ) = desca(5)
589 param_check( 9, 1 ) = desca(4)
590 param_check( 8, 1 ) = desca(3)
591 param_check( 7, 1 ) = desca(1)
592 param_check( 6, 1 ) = ja
593 param_check( 5, 1 ) = nrhs
594 param_check( 4, 1 ) = n
595 param_check( 3, 1 ) = idum3
596 param_check( 2, 1 ) = idum2
597 param_check( 1, 1 ) = idum1
598
599 param_check( 16, 2 ) = 1105
600 param_check( 15, 2 ) = 1104
601 param_check( 14, 2 ) = 1103
602 param_check( 13, 2 ) = 1102
603 param_check( 12, 2 ) = 1101
604 param_check( 11, 2 ) = 10
605 param_check( 10, 2 ) = 805
606 param_check( 9, 2 ) = 804
607 param_check( 8, 2 ) = 803
608 param_check( 7, 2 ) = 801
609 param_check( 6, 2 ) = 7
610 param_check( 5, 2 ) = 4
611 param_check( 4, 2 ) = 3
612 param_check( 3, 2 ) = 14
613 param_check( 2, 2 ) = 2
614 param_check( 1, 2 ) = 1
615
616
617
618
619
620 IF( info.GE.0 ) THEN
621 info = bignum
622 ELSE IF( info.LT.-descmult ) THEN
623 info = -info
624 ELSE
625 info = -info * descmult
626 END IF
627
628
629
630 CALL globchk( ictxt, 16, param_check, 16,
631 $ param_check( 1, 3 ), info )
632
633
634
635
636 IF( info.EQ.bignum ) THEN
637 info = 0
638 ELSE IF( mod( info, descmult ) .EQ. 0 ) THEN
639 info = -info / descmult
640 ELSE
641 info = -info
642 END IF
643
644 IF( info.LT.0 ) THEN
645 CALL pxerbla( ictxt,
'PZPTTRSV', -info )
646 RETURN
647 END IF
648
649
650
651 IF( n.EQ.0 )
652 $ RETURN
653
654 IF( nrhs.EQ.0 )
655 $ RETURN
656
657
658
659
660
661 part_offset = nb*( (ja-1)/(npcol*nb) )
662
663 IF ( (mycol-csrc) .LT. (ja-part_offset-1)/nb ) THEN
664 part_offset = part_offset + nb
665 ENDIF
666
667 IF ( mycol .LT. csrc ) THEN
668 part_offset = part_offset - nb
669 ENDIF
670
671
672
673
674
675
676
677 first_proc = mod( ( ja-1 )/nb+csrc, npcol )
678
679
680
681 ja_new = mod( ja-1, nb ) + 1
682
683
684
685 np_save = np
686 np = ( ja_new+n-2 )/nb + 1
687
688
689
690 CALL reshape( ictxt, int_one, ictxt_new, int_one,
691 $ first_proc, int_one, np )
692
693
694
695 ictxt_save = ictxt
696 ictxt = ictxt_new
697 desca_1xp( 2 ) = ictxt_new
698 descb_px1( 2 ) = ictxt_new
699
700
701
702 CALL blacs_gridinfo( ictxt, nprow, npcol, myrow, mycol )
703
704
705
706 IF( myrow .LT. 0 ) THEN
707 GOTO 1234
708 ENDIF
709
710
711
712
713
714
715 part_size = nb
716
717
718
719 my_num_cols =
numroc( n, part_size, mycol, 0, npcol )
720
721
722
723 IF ( mycol .EQ. 0 ) THEN
724 part_offset = part_offset+mod( ja_new-1, part_size )
725 my_num_cols = my_num_cols - mod(ja_new-1, part_size )
726 ENDIF
727
728
729
730 odd_size = my_num_cols
731 IF ( mycol .LT. np-1 ) THEN
732 odd_size = odd_size - int_one
733 ENDIF
734
735
736
737
738
739 IF (
lsame( uplo,
'L' ) )
THEN
740
741 IF (
lsame( trans,
'N' ) )
THEN
742
743
744
745
746
747
748
749
750
751
752 CALL zpttrsv( uplo,
'N', odd_size, nrhs, d( part_offset+1 ),
753 $ e( part_offset+1 ), b( part_offset+1 ), lldb,
754 $ info )
755
756
757 IF ( mycol .LT. np-1 ) THEN
758
759
760
761 CALL zaxpy( nrhs, -e( part_offset+odd_size ),
762 $ b( part_offset+odd_size ), lldb,
763 $ b( part_offset+odd_size+1 ), lldb )
764
765 ENDIF
766
767
768 IF ( mycol .NE. 0 ) THEN
769
770
771
772 CALL zgemm( 'C', 'N', 1, nrhs, odd_size, -cone, af( 1 ),
773 $ odd_size, b( part_offset+1 ), lldb, czero,
774 $ work( 1+int_one-1 ), int_one )
775 ENDIF
776
777
778
779
780
781
782
783
784
785 IF( mycol .GT. 0) THEN
786
787 CALL zgesd2d( ictxt, int_one, nrhs,
788 $ work( 1 ), int_one,
789 $ 0, mycol - 1 )
790
791 ENDIF
792
793
794
795 IF( mycol .LT. npcol-1) THEN
796
797 CALL zgerv2d( ictxt, int_one, nrhs,
798 $ work( 1 ), int_one,
799 $ 0, mycol + 1 )
800
801
802
803 CALL zmatadd( int_one, nrhs, cone,
804 $ work( 1 ), int_one, cone,
805 $ b( part_offset+odd_size + 1 ), lldb )
806
807 ENDIF
808
809
810
811
812 IF( mycol .EQ. npcol-1 ) THEN
813 GOTO 14
814 ENDIF
815
816
817
818
819
820
821
822 level_dist = 1
823
824
825
826 12 CONTINUE
827 IF( mod( (mycol+1)/level_dist, 2) .NE. 0 ) GOTO 11
828
829
830
831 IF( mycol-level_dist .GE. 0 ) THEN
832
833 CALL zgerv2d( ictxt, int_one, nrhs,
834 $ work( 1 ),
835 $ int_one, 0, mycol-level_dist )
836
837 CALL zmatadd( int_one, nrhs, cone,
838 $ work( 1 ), int_one, cone,
839 $ b( part_offset+odd_size + 1 ), lldb )
840
841 ENDIF
842
843
844
845 IF( mycol+level_dist .LT. npcol-1 ) THEN
846
847 CALL zgerv2d( ictxt, int_one, nrhs,
848 $ work( 1 ),
849 $ int_one, 0, mycol+level_dist )
850
851 CALL zmatadd( int_one, nrhs, cone,
852 $ work( 1 ), int_one, cone,
853 $ b( part_offset+odd_size + 1 ), lldb )
854
855 ENDIF
856
857 level_dist = level_dist*2
858
859 GOTO 12
860 11 CONTINUE
861
862
863
864
865
866
867
868
869
870 CALL ztrtrs( 'L', 'N', 'U', int_one, nrhs, af( odd_size+2 ),
871 $ int_one, b( part_offset+odd_size+1 ), lldb, info )
872
873 IF( info.NE.0 ) THEN
874 GO TO 1000
875 ENDIF
876
877
878
879
880 IF( mycol/level_dist .LE. (npcol-1)/level_dist-2 )THEN
881
882
883
884 CALL zgemm( 'C', 'N', int_one, nrhs, int_one, -cone,
885 $ af( (odd_size)*1+1 ),
886 $ int_one,
887 $ b( part_offset+odd_size+1 ),
888 $ lldb, czero,
889 $ work( 1 ),
890 $ int_one )
891
892
893
894 CALL zgesd2d( ictxt, int_one, nrhs,
895 $ work( 1 ),
896 $ int_one, 0, mycol+level_dist )
897
898 ENDIF
899
900
901
902 IF( (mycol/level_dist .GT. 0 ).AND.
903 $ ( mycol/level_dist .LE. (npcol-1)/level_dist-1 ) ) THEN
904
905
906
907
908
909 CALL zgemm( 'N', 'N', int_one, nrhs, int_one, -cone,
910 $ af( odd_size*1+2+1 ),
911 $ int_one,
912 $ b( part_offset+odd_size+1 ),
913 $ lldb, czero,
914 $ work( 1 ),
915 $ int_one )
916
917
918
919 CALL zgesd2d( ictxt, int_one, nrhs,
920 $ work( 1 ),
921 $ int_one, 0, mycol-level_dist )
922
923 ENDIF
924
925
926 14 CONTINUE
927
928 ELSE
929
930
931
932
933
934
935
936
937
938
939
940 IF( mycol .EQ. npcol-1 ) THEN
941 GOTO 24
942 ENDIF
943
944
945
946 level_dist = 1
947 27 CONTINUE
948 IF( mod( (mycol+1)/level_dist, 2) .NE. 0 ) GOTO 26
949
950 level_dist = level_dist*2
951
952 GOTO 27
953 26 CONTINUE
954
955
956 IF( (mycol/level_dist .GT. 0 ).AND.
957 $ ( mycol/level_dist .LE. (npcol-1)/level_dist-1 ) ) THEN
958
959
960
961 CALL zgerv2d( ictxt, int_one, nrhs,
962 $ work( 1 ),
963 $ int_one, 0, mycol-level_dist )
964
965
966
967
968
969 CALL zgemm( 'C', 'N', int_one, nrhs, int_one, -cone,
970 $ af( odd_size*1+2+1 ),
971 $ int_one,
972 $ work( 1 ),
973 $ int_one, cone,
974 $ b( part_offset+odd_size+1 ),
975 $ lldb )
976 ENDIF
977
978
979
980 IF( mycol/level_dist .LE. (npcol-1)/level_dist-2 )THEN
981
982
983
984 CALL zgerv2d( ictxt, int_one, nrhs,
985 $ work( 1 ),
986 $ int_one, 0, mycol+level_dist )
987
988
989
990 CALL zgemm( 'N', 'N', int_one, nrhs, int_one, -cone,
991 $ af( (odd_size)*1+1 ),
992 $ int_one,
993 $ work( 1 ),
994 $ int_one, cone,
995 $ b( part_offset+odd_size+1 ),
996 $ lldb )
997
998 ENDIF
999
1000
1001
1002
1003
1004 CALL ztrtrs( 'L', 'C', 'U', int_one, nrhs, af( odd_size+2 ),
1005 $ int_one, b( part_offset+odd_size+1 ), lldb, info )
1006
1007 IF( info.NE.0 ) THEN
1008 GO TO 1000
1009 ENDIF
1010
1011
1012
1013
1014
1015 22 CONTINUE
1016 IF( level_dist .EQ. 1 ) GOTO 21
1017
1018 level_dist = level_dist/2
1019
1020
1021
1022 IF( mycol+level_dist .LT. npcol-1 ) THEN
1023
1024 CALL zgesd2d( ictxt, int_one, nrhs,
1025 $ b( part_offset+odd_size+1 ),
1026 $ lldb, 0, mycol+level_dist )
1027
1028 ENDIF
1029
1030
1031
1032 IF( mycol-level_dist .GE. 0 ) THEN
1033
1034 CALL zgesd2d( ictxt, int_one, nrhs,
1035 $ b( part_offset+odd_size+1 ),
1036 $ lldb, 0, mycol-level_dist )
1037
1038 ENDIF
1039
1040 GOTO 22
1041 21 CONTINUE
1042
1043
1044 24 CONTINUE
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054 IF( mycol .LT. npcol-1) THEN
1055
1056 CALL zgesd2d( ictxt, int_one, nrhs,
1057 $ b( part_offset+odd_size+1 ), lldb,
1058 $ 0, mycol +1 )
1059
1060 ENDIF
1061
1062
1063
1064 IF( mycol .GT. 0) THEN
1065
1066 CALL zgerv2d( ictxt, int_one, nrhs,
1067 $ work( 1 ), int_one,
1068 $ 0, mycol - 1 )
1069
1070 ENDIF
1071
1072
1073
1074
1075
1076
1077
1078 IF ( mycol .NE. 0 ) THEN
1079
1080
1081
1082 CALL zgemm( 'N', 'N', odd_size, nrhs, 1, -cone, af( 1 ),
1083 $ odd_size, work( 1+int_one-1 ), int_one, cone,
1084 $ b( part_offset+1 ), lldb )
1085
1086 ENDIF
1087
1088
1089 IF ( mycol .LT. np-1 ) THEN
1090
1091
1092
1093 CALL zaxpy( nrhs, -dconjg( e( part_offset+odd_size ) ),
1094 $ b( part_offset+odd_size+1 ), lldb,
1095 $ b( part_offset+odd_size ), lldb )
1096
1097 ENDIF
1098
1099
1100
1101 CALL zpttrsv( uplo,
'C', odd_size, nrhs, d( part_offset+1 ),
1102 $ e( part_offset+1 ), b( part_offset+1 ), lldb,
1103 $ info )
1104
1105 ENDIF
1106
1107
1108
1109 ELSE
1110
1111
1112
1113 IF (
lsame( trans,
'C' ) )
THEN
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124 CALL zpttrsv( uplo,
'C', odd_size, nrhs, d( part_offset+1 ),
1125 $ e( part_offset+1 ), b( part_offset+1 ), lldb,
1126 $ info )
1127
1128
1129 IF ( mycol .LT. np-1 ) THEN
1130
1131
1132
1133 CALL zaxpy( nrhs, -dconjg( e( part_offset+odd_size ) ),
1134 $ b( part_offset+odd_size ), lldb,
1135 $ b( part_offset+odd_size+1 ), lldb )
1136
1137 ENDIF
1138
1139
1140 IF ( mycol .NE. 0 ) THEN
1141
1142
1143
1144 CALL zgemm( 'T', 'N', 1, nrhs, odd_size, -cone, af( 1 ),
1145 $ odd_size, b( part_offset+1 ), lldb, czero,
1146 $ work( 1+int_one-1 ), int_one )
1147 ENDIF
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157 IF( mycol .GT. 0) THEN
1158
1159 CALL zgesd2d( ictxt, int_one, nrhs,
1160 $ work( 1 ), int_one,
1161 $ 0, mycol - 1 )
1162
1163 ENDIF
1164
1165
1166
1167 IF( mycol .LT. npcol-1) THEN
1168
1169 CALL zgerv2d( ictxt, int_one, nrhs,
1170 $ work( 1 ), int_one,
1171 $ 0, mycol + 1 )
1172
1173
1174
1175 CALL zmatadd( int_one, nrhs, cone,
1176 $ work( 1 ), int_one, cone,
1177 $ b( part_offset+odd_size + 1 ), lldb )
1178
1179 ENDIF
1180
1181
1182
1183
1184 IF( mycol .EQ. npcol-1 ) THEN
1185 GOTO 44
1186 ENDIF
1187
1188
1189
1190
1191
1192
1193
1194 level_dist = 1
1195
1196
1197
1198 42 CONTINUE
1199 IF( mod( (mycol+1)/level_dist, 2) .NE. 0 ) GOTO 41
1200
1201
1202
1203 IF( mycol-level_dist .GE. 0 ) THEN
1204
1205 CALL zgerv2d( ictxt, int_one, nrhs,
1206 $ work( 1 ),
1207 $ int_one, 0, mycol-level_dist )
1208
1209 CALL zmatadd( int_one, nrhs, cone,
1210 $ work( 1 ), int_one, cone,
1211 $ b( part_offset+odd_size + 1 ), lldb )
1212
1213 ENDIF
1214
1215
1216
1217 IF( mycol+level_dist .LT. npcol-1 ) THEN
1218
1219 CALL zgerv2d( ictxt, int_one, nrhs,
1220 $ work( 1 ),
1221 $ int_one, 0, mycol+level_dist )
1222
1223 CALL zmatadd( int_one, nrhs, cone,
1224 $ work( 1 ), int_one, cone,
1225 $ b( part_offset+odd_size + 1 ), lldb )
1226
1227 ENDIF
1228
1229 level_dist = level_dist*2
1230
1231 GOTO 42
1232 41 CONTINUE
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242 CALL ztrtrs( 'L', 'N', 'U', int_one, nrhs, af( odd_size+2 ),
1243 $ int_one, b( part_offset+odd_size+1 ), lldb, info )
1244
1245 IF( info.NE.0 ) THEN
1246 GO TO 1000
1247 ENDIF
1248
1249
1250
1251
1252 IF( mycol/level_dist .LE. (npcol-1)/level_dist-2 )THEN
1253
1254
1255
1256 CALL zgemm( 'T', 'N', int_one, nrhs, int_one, -cone,
1257 $ af( (odd_size)*1+1 ),
1258 $ int_one,
1259 $ b( part_offset+odd_size+1 ),
1260 $ lldb, czero,
1261 $ work( 1 ),
1262 $ int_one )
1263
1264
1265
1266 CALL zgesd2d( ictxt, int_one, nrhs,
1267 $ work( 1 ),
1268 $ int_one, 0, mycol+level_dist )
1269
1270 ENDIF
1271
1272
1273
1274 IF( (mycol/level_dist .GT. 0 ).AND.
1275 $ ( mycol/level_dist .LE. (npcol-1)/level_dist-1 ) ) THEN
1276
1277
1278
1279
1280
1281 CALL zgemm( 'C', 'N', int_one, nrhs, int_one, -cone,
1282 $ af( odd_size*1+2+1 ),
1283 $ int_one,
1284 $ b( part_offset+odd_size+1 ),
1285 $ lldb, czero,
1286 $ work( 1 ),
1287 $ int_one )
1288
1289
1290
1291 CALL zgesd2d( ictxt, int_one, nrhs,
1292 $ work( 1 ),
1293 $ int_one, 0, mycol-level_dist )
1294
1295 ENDIF
1296
1297
1298 44 CONTINUE
1299
1300 ELSE
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312 IF( mycol .EQ. npcol-1 ) THEN
1313 GOTO 54
1314 ENDIF
1315
1316
1317
1318 level_dist = 1
1319 57 CONTINUE
1320 IF( mod( (mycol+1)/level_dist, 2) .NE. 0 ) GOTO 56
1321
1322 level_dist = level_dist*2
1323
1324 GOTO 57
1325 56 CONTINUE
1326
1327
1328 IF( (mycol/level_dist .GT. 0 ).AND.
1329 $ ( mycol/level_dist .LE. (npcol-1)/level_dist-1 ) ) THEN
1330
1331
1332
1333 CALL zgerv2d( ictxt, int_one, nrhs,
1334 $ work( 1 ),
1335 $ int_one, 0, mycol-level_dist )
1336
1337
1338
1339
1340
1341 CALL zgemm( 'T', 'N', int_one, nrhs, int_one, -cone,
1342 $ af( odd_size*1+2+1 ),
1343 $ int_one,
1344 $ work( 1 ),
1345 $ int_one, cone,
1346 $ b( part_offset+odd_size+1 ),
1347 $ lldb )
1348 ENDIF
1349
1350
1351
1352 IF( mycol/level_dist .LE. (npcol-1)/level_dist-2 )THEN
1353
1354
1355
1356 CALL zgerv2d( ictxt, int_one, nrhs,
1357 $ work( 1 ),
1358 $ int_one, 0, mycol+level_dist )
1359
1360
1361
1362 CALL zgemm( 'C', 'N', int_one, nrhs, int_one, -cone,
1363 $ af( (odd_size)*1+1 ),
1364 $ int_one,
1365 $ work( 1 ),
1366 $ int_one, cone,
1367 $ b( part_offset+odd_size+1 ),
1368 $ lldb )
1369
1370 ENDIF
1371
1372
1373
1374
1375
1376 CALL ztrtrs( 'L', 'C', 'U', int_one, nrhs, af( odd_size+2 ),
1377 $ int_one, b( part_offset+odd_size+1 ), lldb, info )
1378
1379 IF( info.NE.0 ) THEN
1380 GO TO 1000
1381 ENDIF
1382
1383
1384
1385
1386
1387 52 CONTINUE
1388 IF( level_dist .EQ. 1 ) GOTO 51
1389
1390 level_dist = level_dist/2
1391
1392
1393
1394 IF( mycol+level_dist .LT. npcol-1 ) THEN
1395
1396 CALL zgesd2d( ictxt, int_one, nrhs,
1397 $ b( part_offset+odd_size+1 ),
1398 $ lldb, 0, mycol+level_dist )
1399
1400 ENDIF
1401
1402
1403
1404 IF( mycol-level_dist .GE. 0 ) THEN
1405
1406 CALL zgesd2d( ictxt, int_one, nrhs,
1407 $ b( part_offset+odd_size+1 ),
1408 $ lldb, 0, mycol-level_dist )
1409
1410 ENDIF
1411
1412 GOTO 52
1413 51 CONTINUE
1414
1415
1416 54 CONTINUE
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426 IF( mycol .LT. npcol-1) THEN
1427
1428 CALL zgesd2d( ictxt, int_one, nrhs,
1429 $ b( part_offset+odd_size+1 ), lldb,
1430 $ 0, mycol +1 )
1431
1432 ENDIF
1433
1434
1435
1436 IF( mycol .GT. 0) THEN
1437
1438 CALL zgerv2d( ictxt, int_one, nrhs,
1439 $ work( 1 ), int_one,
1440 $ 0, mycol - 1 )
1441
1442 ENDIF
1443
1444
1445
1446
1447
1448
1449
1450 IF ( mycol .NE. 0 ) THEN
1451
1452
1453
1454 CALL zgemm( 'C', 'N', odd_size, nrhs, 1, -cone, af( 1 ),
1455 $ int_one, work( 1 ), int_one, cone,
1456 $ b( part_offset+1 ), lldb )
1457
1458 ENDIF
1459
1460
1461 IF ( mycol .LT. np-1 ) THEN
1462
1463
1464
1465 CALL zaxpy( nrhs, -e( part_offset+odd_size ),
1466 $ b( part_offset+odd_size+1 ), lldb,
1467 $ b( part_offset+odd_size ), lldb )
1468
1469 ENDIF
1470
1471
1472
1473 CALL zpttrsv( uplo,
'N', odd_size, nrhs, d( part_offset+1 ),
1474 $ e( part_offset+1 ), b( part_offset+1 ), lldb,
1475 $ info )
1476
1477 ENDIF
1478
1479
1480
1481 ENDIF
1482
1483 1000 CONTINUE
1484
1485
1486
1487
1488 IF( ictxt_save .NE. ictxt_new ) THEN
1489 CALL blacs_gridexit( ictxt_new )
1490 ENDIF
1491
1492 1234 CONTINUE
1493
1494
1495
1496 ictxt = ictxt_save
1497 np = np_save
1498
1499
1500
1501 work( 1 ) = work_size_min
1502
1503
1504 RETURN
1505
1506
1507
subroutine desc_convert(desc_in, desc_out, info)
integer function numroc(n, nb, iproc, isrcproc, nprocs)
subroutine globchk(ictxt, n, x, ldx, iwork, info)
subroutine pxerbla(ictxt, srname, info)
void reshape(Int *context_in, Int *major_in, Int *context_out, Int *major_out, Int *first_proc, Int *nprow_new, Int *npcol_new)
subroutine zmatadd(m, n, alpha, a, lda, beta, c, ldc)
subroutine zpttrsv(uplo, trans, n, nrhs, d, e, b, ldb, info)