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
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
|
PAGE 60,132 ;
TITLE DEBCOM2.ASM - PART2 DEBUGGER COMMANDS PC DOS
;======================= START OF SPECIFICATIONS =========================
;
; MODULE NAME: DECOM2.SAL
;
; DESCRIPTIVE NAME: DEBUGGING TOOL
;
; FUNCTION: PROVIDES USERS WITH A TOOL FOR DEBUGGING PROGRAMS.
;
; ENTRY POINT: ANY CALLED ROUTINE
;
; INPUT: NA
;
; EXIT NORMAL: NA
;
; EXIT ERROR: NA
;
; INTERNAL REFERENCES:
;
; EXTERNAL REFERENCES:
;
; ROUTINE: DEBCOM1 - CONTAINS ROUTINES CALLED BY DEBUG
; DEBCOM3 - CONTAINS ROUTINES CALLED BY DEBUG
; DEBASM - CONTAINS ROUTINES CALLED BY DEBUG
; DEBUASM - CONTAINS ROUTINES CALLED BY DEBUG
; DEBMES - CONTAINS ROUTINES CALLED BY DEBUG
;
; NOTES: THIS MODULE IS TO BE PREPPED BY SALUT WITH THE "PR" OPTIONS.
; LINK DEBUG+DEBCOM1+DEBCOM2+DEBCOM3+DEBASM+DEBUASM+DEBERR+
; DEBCONST+DEBDATA+DEBMES
;
; REVISION HISTORY:
;
; AN000 VERSION 4.00 - REVISIONS MADE RELATE TO THE FOLLOWING:
; AC000 VERSION 4.00 -
;
; - IMPLEMENT DBCS HANDLING DMS:6/17/87
; - IMPLEMENT MESSAGE RETRIEVER DMS:6/17/87
; - IMPLEMENT > 32MB SUPPORT DMS:6/17/87
;
; COPYRIGHT: "MS DOS DEBUG UTILITY"
; "VERSION 4.00 (C) COPYRIGHT 1988 Microsoft"
; "LICENSED MATERIAL - PROPERTY OF Microsoft "
;
;======================= END OF SPECIFICATIONS ===========================
; Routines to perform debugger commands except ASSEMble and UASSEMble
IF1
%OUT COMPONENT=DEBUG, MODULE=DEBCOM2
ENDIF
.XLIST
.XCREF
INCLUDE DOSSYM.INC
INCLUDE DEBEQU.ASM
.CREF
.LIST
CODE SEGMENT PUBLIC BYTE
CODE ENDS
CONST SEGMENT PUBLIC BYTE
EXTRN NOTFND_PTR:BYTE,NOROOM_PTR:BYTE,DRVLET:BYTE,ERRMES_PTR:BYTE
EXTRN NAMBAD_PTR:BYTE,NOSPACE_PTR:BYTE,TOOBIG_PTR:BYTE
EXTRN HEXERR_PTR:BYTE,HEXWRT_PTR:BYTE,ACCMES_PTR:BYTE
EXTRN EXEBAD_PTR:BYTE,EXEWRT_PTR:BYTE
EXTRN EXECEMES_PTR:BYTE,NONAMESPEC_PTR:BYTE
EXTRN FLAGTAB:WORD,EXEC_BLOCK:BYTE,COM_LINE:DWORD,COM_FCB1:DWORD
EXTRN COM_FCB2:DWORD,COM_SSSP:DWORD,COM_CSIP:DWORD,RETSAVE:WORD
EXTRN NEWEXEC:BYTE,HEADSAVE:WORD
EXTRN REGTAB:BYTE,TOTREG:BYTE,NOREGL:BYTE
EXTRN USER_PROC_PDB:WORD,STACK:BYTE,RSTACK:WORD,AXSAVE:WORD
EXTRN BXSAVE:WORD,DSSAVE:WORD,ESSAVE:WORD,CSSAVE:WORD,IPSAVE:WORD
EXTRN SSSAVE:WORD,CXSAVE:WORD,SPSAVE:WORD,FLSAVE:WORD
EXTRN SREG:BYTE,SEGTAB:WORD,REGDIF:ABS,RDFLG:BYTE
EXTRN REGTABEND:WORD
EXTRN NAMESPEC:BYTE
CONST ENDS
CSTACK SEGMENT STACK
CSTACK ENDS
DATA SEGMENT PUBLIC BYTE
EXTRN DEFDUMP:BYTE,TRANSADD:DWORD,INDEX:WORD,BUFFER:BYTE
EXTRN ASMADD:BYTE,DISADD:BYTE,NSEG:WORD
EXTRN SWITCHAR:BYTE,XNXCMD:BYTE,XNXOPT:BYTE
EXTRN AWORD:BYTE,EXTPTR:WORD,HANDLE:WORD,PARSERR:BYTE
EXTRN REG_NAME:WORD,REG_CONTENTS:WORD,REGISTER_PTR:BYTE
EXTRN ARG_BUF:BYTE,ARG_BUF_PTR:BYTE,LOC_ADD:WORD,LOC_PTR:BYTE
EXTRN BIG_CONTENTS:WORD,BIG_PTR:BYTE,LITTLE_CONTENTS:WORD,LITTLE_PTR:BYTE
EXTRN SINGLE_REG_ARG:WORD,CHANGE_FLAG_PTR:BYTE,DF_ERROR:BYTE
EXTRN BR_ERROR:BYTE,BF_ERROR:BYTE,SINGLE_REG_PTR:WORD
EXTRN WRT_ARG1:WORD,WRT_ARG2:WORD,WRTMES_PTR:BYTE,BEGSEG:WORD
EXTRN FILESTRT:WORD,FILEEND:WORD
EXTRN ERR_TYPE:BYTE ;ac000;converted to buffer
extrn rel_read_write_tab:dword ;an000;primitive I/O
extrn rel_rw_add:dword ;an000;transfer address
extrn rel_low_sec:word ;an000;low sector word
extrn rel_high_sec:word ;an000;high sector word
extrn rel_sec_num:word ;an000;# of sectors
fnd_dbcs db 0
DATA ENDS
DG GROUP CODE,CONST,CSTACK,DATA
CODE SEGMENT PUBLIC BYTE
ASSUME CS:DG,DS:DG,ES:DG,SS:DG
PUBLIC DEFIO,PREPNAME,DEBUG_FOUND
PUBLIC REG,LOAD
PUBLIC NAMED,DWRITE
PUBLIC DISPREG,ERR,DELIM1,DELIM2,delim0
public getchrup,open1,open2,open3,open4,oc_file,opnret ;an001;bgb
public delete_a_file, parse_a_file, exec_a_file, open_a_file, create_a_file ;an001;bgb
public gcur,ifhex ;an001;bgb
public comtail ;an001;bgb
extrn test_lead:near ;an001;bgb
EXTRN OUTSI:NEAR,OUTDI:NEAR,INBUF:NEAR,SCANB:NEAR,SCANP:NEAR
EXTRN COMMAND:NEAR,DISASLN:NEAR,SET_TERMINATE_VECTOR:NEAR
EXTRN RESTART:NEAR,TERMINATE:NEAR,DRVERR:NEAR
EXTRN GETHEX:NEAR,GETEOL:NEAR,SKIP_FILE:NEAR
EXTRN HEXCHK:NEAR,GETHEX1:NEAR,PRINT:NEAR
EXTRN CRLF:NEAR,BLANK:NEAR
EXTRN HEX:NEAR,DIGIT:NEAR
EXTRN FIND_DEBUG:NEAR
EXTRN ADDRESS:NEAR,PERROR:NEAR
EXTRN STD_PRINTF:NEAR,PRINTF_CRLF:NEAR
DEBCOM2:
DISPLAY_LINE:
mov ax,word ptr [si] ;an000;move reg name to ax
MOV [REG_NAME],ax ;ac000;save it in reg_name
ADD SI,3
MOV AX,[BX]
ADD BX,2
MOV [REG_CONTENTS],AX
MOV DX,OFFSET DG:REGISTER_PTR
CALL STD_PRINTF
LOOP DISPLAY_LINE
RETURN
DISPLAY_FLAGS:
MOV DI,OFFSET DG:ARG_BUF
MOV AL,CHAR_BLANK
STOSB
DISPLAY_FLAGS_2:
MOV SI,OFFSET DG:FLAGTAB
MOV CX,16
MOV DX,[FLSAVE]
DFLAGS:
LODS CS:WORD PTR [SI]
SHL DX,1
JC FLAGSET
MOV AX,CS:[SI+30]
FLAGSET:
OR AX,AX
JZ NEXT_FLAG
STOSW
MOV AL,CHAR_BLANK
STOSB
NEXT_FLAG:
LOOP DFLAGS
XOR AL,AL
STOSB
RETURN
DISPREG:
MOV SI,OFFSET DG:REGTAB
MOV DI,OFFSET DG:ARG_BUF
MOV BX,OFFSET DG:AXSAVE
MOV BYTE PTR TOTREG,CR
MOV CH,0
MOV CL,NOREGL
SET_DISPLAY:
REPEAT_DISPLAY:
SUB TOTREG,CL
CALL DISPLAY_LINE
CALL CRLF
XOR CH,CH
MOV CL,NOREGL
CMP CL,TOTREG
JB REPEAT_DISPLAY
MOV CL,TOTREG
CALL DISPLAY_LINE
CALL DISPLAY_FLAGS
MOV DX,OFFSET DG:ARG_BUF_PTR
CALL PRINTF_CRLF
MOV AX,[IPSAVE]
MOV WORD PTR [DISADD],AX
PUSH AX
MOV AX,[CSSAVE]
MOV WORD PTR [DISADD+WORD],AX
PUSH AX
MOV [NSEG],-1
CALL DISASLN
POP WORD PTR DISADD+WORD
POP WORD PTR DISADD
MOV AX,[NSEG]
CMP AL,-1
JNZ ASSEM_LIN_CONT
JMP CRLF
ASSEM_LIN_CONT:
CMP AH,-1
JZ NOOVER
XCHG AL,AH
NOOVER:
CBW
MOV BX,AX
SHL BX,1
MOV AX,WORD PTR [BX+SREG]
MOV DI,OFFSET DG:ARG_BUF
STOSB
XCHG AL,AH
STOSB
XOR AL,AL
STOSB
MOV DX,[INDEX]
MOV LOC_ADD,DX
MOV DX,OFFSET DG:LOC_PTR
CALL STD_PRINTF
MOV BX,[BX+SEGTAB]
PUSH DS
MOV DS,[BX]
MOV BX,CS:[INDEX]
MOV BX,[BX]
POP DS
MOV BIG_CONTENTS,BX
MOV DX,OFFSET DG:BIG_PTR
TEST BYTE PTR [AWORD],-1
JNZ SHOW_CHARS
XOR BH,BH
MOV LITTLE_CONTENTS,BX
MOV DX,OFFSET DG:LITTLE_PTR
SHOW_CHARS:
CALL PRINTF_CRLF
RETURN
DISPREGJ:
JMP DISPREG
; Perform register dump if no parameters or set register if a
; register designation is a parameter.
REG:
CALL SCANP
JZ DISPREGJ
MOV DL,[SI]
INC SI
MOV DH,[SI]
CMP DH,CR
JZ FLAG
INC SI
CALL GETEOL
CMP DH,CHAR_BLANK
JZ FLAG
MOV DI,OFFSET DG:REGTAB
XCHG AX,DX
PUSH CS
POP ES
XOR CX,CX
CHECK_NEXT_REG:
CMP AX,WORD PTR[ DI]
JZ REG_FOUND
ADD DI,3
INC CX
CMP DI,OFFSET DG:REGTABEND
JB CHECK_NEXT_REG
JMP BADREG
REG_FOUND:
CMP DI,OFFSET DG:REGTABEND
JNZ NOTPC
DEC DI
DEC DI
DEC DI
MOV AX,CS:[DI-WORD]
NOTPC:
PUSH DI
MOV DI,OFFSET DG:ARG_BUF
STOSB
XCHG AL,AH
STOSB
XOR AL,AL
STOSB
POP DI
PUSH DS
POP ES
LEA BX,[DI+REGDIF]
SUB BX,CX
MOV DX,[BX]
MOV SINGLE_REG_ARG,DX
MOV DX,OFFSET DG:SINGLE_REG_PTR
CALL STD_PRINTF
CALL INBUF
CALL SCANB
RETZ
push bx ;an000;save bx - we stomp it
MOV CX,4
CALL GETHEX1
pop bx ;an000;restore it
CALL GETEOL
MOV [BX],DX
RETURN
BADREG:
MOV DX,OFFSET DG:BR_ERROR ; BR ERROR
JMP ERR
FLAG:
CMP DL,UPPER_F
JNZ BADREG
MOV DI,OFFSET DG:ARG_BUF
CALL DISPLAY_FLAGS_2
MOV DX,OFFSET DG:CHANGE_FLAG_PTR
CALL STD_PRINTF
CALL INBUF
CALL SCANB
XOR BX,BX
MOV DX,[FLSAVE]
GETFLG:
LODSW
CMP AL,CR
JZ SAVCHG
CMP AH,CR
JZ FLGERR
MOV DI,OFFSET DG:FLAGTAB
MOV CX,32
PUSH CS
POP ES
REPNE SCASW
JNZ FLGERR
MOV CH,CL
AND CL,0FH
MOV AX,1
ROL AX,CL
TEST AX,BX
JNZ REPFLG
OR BX,AX
OR DX,AX
TEST CH,16
JNZ NEXFLG
XOR DX,AX
NEXFLG:
CALL SCANP
JMP SHORT GETFLG
REPFLG:
MOV DX,OFFSET DG:DF_ERROR ; DF ERROR
FERR:
CALL SAVCHG
ERR:
push si ;an000;save affected registers
push di ;an000;
push cx ;an000;
mov cx,03h ;an000;move only three bytes
mov di,offset dg:err_type ;an000;point to buffer
mov si,dx ;an000;dx holds the string
rep movsb ;an000;fill up the buffer
pop cx ;an000;restore registers
pop di ;an000;
pop si ;an000;
MOV DX,OFFSET DG:ERRMES_PTR
JMP PRINT
SAVCHG:
MOV [FLSAVE],DX
RETURN
FLGERR:
MOV DX,OFFSET DG:BF_ERROR ; BF ERROR
JMP SHORT FERR
PREPNAME:
MOV ES,DSSAVE
PUSH SI
MOV DI,81H
COMTAIL:
LODSB
STOSB
CMP AL,CR
JNZ COMTAIL
SUB DI,82H
XCHG AX,DI
MOV ES:(BYTE PTR [80H]),AL
POP SI
MOV DI,FCB ;05cH
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR SET_DRIVEID_OPTION ;AL=01H
INT 21H
MOV BYTE PTR [AXSAVE],AL ; Indicate analysis of first parm
CALL SKIP_FILE
MOV DI,6CH
MOV AX,(PARSE_FILE_DESCRIPTOR SHL 8) OR SET_DRIVEID_OPTION ;AL=01H
INT 21H
MOV BYTE PTR [AXSAVE+1],AL ; Indicate analysis of second parm
RETURN
; OPENS A XENIX PATHNAME SPECIFIED IN THE UNFORMATTED PARAMETERS
; VARIABLE [XNXCMD] SPECIFIES WHICH COMMAND TO OPEN IT WITH
; VARIABLE [HANDLE] CONTAINS THE HANDLE
; VARIABLE [EXTPTR] POINTS TO THE FILES EXTENSION
DELETE_A_FILE:
MOV BYTE PTR [XNXCMD],UNLINK
JMP SHORT OC_FILE
PARSE_A_FILE:
MOV BYTE PTR [XNXCMD],0
JMP SHORT OC_FILE
EXEC_A_FILE:
MOV BYTE PTR CS:[XNXCMD],EXEC
MOV BYTE PTR CS:[XNXOPT],1
JMP SHORT OC_FILE
OPEN_A_FILE:
MOV BYTE PTR [XNXCMD],OPEN
MOV BYTE PTR [XNXOPT],2 ; Try read write
CALL OC_FILE
RETNC
MOV BYTE PTR [XNXCMD],OPEN
MOV BYTE PTR [XNXOPT],0 ; Try read only
JMP SHORT OC_FILE
CREATE_A_FILE:
MOV BYTE PTR [XNXCMD],CREAT
OC_FILE:
PUSH DS
PUSH ES
PUSH AX
PUSH BX
PUSH CX
PUSH DX
PUSH SI
XOR AX,AX
MOV CS:[EXTPTR],AX ; INITIALIZE POINTER TO EXTENSIONS
MOV AH,CHAR_OPER
INT 21H
MOV CS:[SWITCHAR],DL ; GET THE CURRENT SWITCH CHARACTER
MOV SI,81H
open1: CALL GETCHRUP ;convert 1 byte to uppercase
CALL DELIM2 ; END OF LINE?
JZ OPEN4
CALL DELIM1 ; SKIP LEADING DELIMITERS
JZ OPEN1
MOV DX,SI ; SAVE POINTER TO BEGINNING
cmp fnd_dbcs,1
; $if z
JNZ $$IF1
dec dx ;dec it twice if dbcs
; $endif
$$IF1:
DEC DX
open2: CMP AL,CHAR_PERIOD ; LAST CHAR A "."?
JNZ OPEN3
MOV CS:[EXTPTR],SI ; SAVE POINTER TO THE EXTENSION
OPEN3:
CALL GETCHRUP
CALL DELIM1 ; LOOK FOR END OF PATHNAME
JZ OPEN4
CALL DELIM2
JNZ OPEN2
OPEN4: DEC SI ; POINT BACK TO LAST CHAR
PUSH [SI] ; SAVE TERMINATION CHAR
MOV BYTE PTR [SI],0 ; NULL TERMINATE THE STRING
MOV AL,CS:[XNXOPT]
MOV AH,CS:[XNXCMD] ; OPEN OR CREATE FILE
OR AH,AH
JZ OPNRET
MOV CS:[FILESTRT],DX ; Set values for later call on this file
MOV CS:[FILEEND],SI
PUSH CS
POP ES ; Set ES seg for EXEC_BLOCK
MOV BX,OFFSET DG:EXEC_BLOCK
XOR CX,CX
INT 21H
MOV CS:[HANDLE],AX ; SAVE ERROR CODE OR HANDLE
OPNRET:
POP [SI]
POP SI
POP DX
POP CX
POP BX
POP AX ; blow away error code...
POP ES
POP DS
RETURN
GETCHRUP: ;an001;bgb
lodsb ;get the character from [si] ;an001;bgb
call test_lead ;is it a dbcs lead byte? ;an001;bgb
; $IF C ;yes ;an001;bgb
JNC $$IF3
inc si ;bump ptr to past 2nd dbcs byte ;an001;bgb
mov fnd_dbcs,1 ;found a dbcs char
jmp gcur ;dont capitalize it ;an001;bgb
; $ENDIF ;an001;bgb
$$IF3:
; ;an001;bgb
mov fnd_dbcs,0 ;did not find a dbcs char
cmp al,lower_a ;is it >= "a" ? ;an001;bgb
jb gcur ;no - exit ;an001;bgb
;an001;bgb
cmp al,lower_z ;is it =< "z" ? ;an001;bgb
ja gcur ;no - exit ;an001;bgb
;an001;bgb
;if we get here, the char is lowercase, so change it ;an001;bgb
sub al,32 ;convert to uppercase ;an001;bgb
mov [si-1],al ;move it back (si points 1 past) ;an001;bgb
gcur: return ;an001;bgb
DELIM0:
CMP AL,CHAR_LEFT_BRACKET
RETZ
DELIM1:
CMP AL,CHAR_BLANK ; SKIP THESE GUYS
RETZ
CMP AL,CHAR_SEMICOLON
RETZ
CMP AL,CHAR_EQUAL
RETZ
CMP AL,CHAR_TAB
RETZ
CMP AL,CHAR_COMMA
RETURN
DELIM2:
CMP AL,CS:[SWITCHAR] ; STOP ON THESE GUYS
RETZ
CMP AL,CR
RETURN
NAMED:
OR [NAMESPEC],1 ; Flag a name command executed
CALL PREPNAME
MOV AL,BYTE PTR AXSAVE
MOV PARSERR,AL
PUSH ES
POP DS
PUSH CS
POP ES
MOV SI,FCB ; DS:SI points to user FCB
MOV DI,SI ; ES:DI points to DEBUG FCB
MOV CX,82
REP MOVSW
RETURN
BADNAM:
MOV DX,OFFSET DG:NAMBAD_PTR
JMP RESTART
IFHEX:
CMP BYTE PTR [PARSERR],-1 ; Invalid drive specification?
JZ BADNAM
CALL PARSE_A_FILE
MOV BX,[EXTPTR]
CMP WORD PTR DS:[BX],"EH" ; "HE"
RETNZ
CMP BYTE PTR DS:[BX+WORD],UPPER_X
RETURN
IFEXE:
PUSH BX
MOV BX,[EXTPTR]
CMP WORD PTR DS:[BX],"XE" ; "EX"
JNZ RETIF
CMP BYTE PTR DS:[BX+WORD],UPPER_E
RETIF:
POP BX
RETURN
LOAD:
MOV BYTE PTR [RDFLG],READ
JMP SHORT DSKIO
DWRITE:
MOV BYTE PTR [RDFLG],WRITE
DSKIO:
MOV BP,[CSSAVE]
CALL SCANB
JNZ PRIMIO
JMP DEFIO
PRIMIO:
CALL ADDRESS
CALL SCANB
JNZ PRMIO
JMP FILEIO
;=========================================================================
; PRMIO: This routine builds the necessary table for the new
; generic IOCtl primitive read/write logical sector function.
;
; Inputs : Binary addresses and values converted by GETHEX
;
; Outputs: REL_READ_WRITE_TAB - Table needed by IOCtl function to
; perform 32 bit sector addressing.
;
; Date : 6/17/87
;=========================================================================
PRMIO:
mov word ptr dg:[rel_rw_add],dx ;ac000;save transfer address
; in table
mov word ptr dg:[rel_rw_add+2],ax ;ac000;save segment of transfer
; address
MOV CX,2
CALL GETHEX ; Drive number must be 2 digits or less
PUSH DX ;save drive number
MOV CX,8 ;ac000;allow 32 bit addressibilty
CALL GETHEX ; Logical record number
mov word ptr dg:[rel_low_sec],dx ;ac000;save low word of logical
; sector address
mov word ptr dg:[rel_high_sec],bx ;ac000;save high word of
; logical sector address
MOV CX,3
CALL GETHEX ; Number of records
mov word ptr dg:[rel_sec_num],dx ;ac000;save number of sectors
; to read/write
CALL GETEOL
POP BX ;ac000;drive number
CBW ; Turn off verify after write
MOV BYTE PTR DRVLET,bl ;ac000;save drive in case of error
PUSH BX
MOV DL,bL ;ac000;move drive to dl
; Clean off the buffer cache for physical I/O
push ds
MOV AH,DISK_RESET
INT 21H
INC DL
MOV AH,GET_DPB
INT 21H
pop ds
or al,al ;ac000;see if an error occurred
pop ax ;ac000;restore drive
JNZ DRVERRJ
CMP CS:BYTE PTR [RDFLG],WRITE
; $if z ;an000;we will write to sector(s)
JNZ $$IF5
call ABSWRT ;an000;logical sector write
; $else ;an000;
JMP SHORT $$EN5
$$IF5:
call ABSREAD ;an000;we will read sector(s)
; $endif ;an000;
$$EN5:
ENDABS:
JNC RET0
DRVERRJ:
JMP DRVERR
RET0:
; Clean cache again...
MOV AH,DISK_RESET
INT 21H
RETURN
;called from debug.sal
DEFIO:
MOV AX,[CSSAVE] ; Default segment
MOV DX,100H ; Default file I/O offset
CALL IFHEX
JNZ EXECHK
XOR DX,DX ; If HEX file, default OFFSET is zero
HEX2BINJ:
JMP HEX2BIN
FILEIO:
; AX and DX have segment and offset of transfer, respectively
CALL IFHEX
JZ HEX2BINJ
EXECHK:
CALL IFEXE
JNZ BINFIL
CMP BYTE PTR [RDFLG],READ
JZ EXELJ
MOV DX,OFFSET DG:EXEWRT_PTR
JMP RESTART ; Can't write .EXE files
BINFIL:
CMP BYTE PTR [RDFLG],WRITE
JZ BINLOAD
CMP WORD PTR DS:[BX],4F00H + UPPER_C ;"CO"
JNZ BINLOAD
CMP BYTE PTR DS:[BX+WORD],UPPER_M
JNZ BINLOAD
EXELJ:
DEC SI
CMP DX,100H
JNZ PRER
CMP AX,[CSSAVE]
JZ OAF
PRER:
JMP PERROR
OAF:
CALL OPEN_A_FILE
JNC GDOPEN
MOV AX,ERROR_FILE_NOT_FOUND
JMP EXECERR
GDOPEN:
XOR DX,DX
XOR CX,CX
MOV BX,[HANDLE]
MOV AL,2
MOV AH,LSEEK
INT 21H
CALL IFEXE ; SUBTRACT 512 BYTES FOR EXE
JNZ BIN2 ; FILE LENGTH BECAUSE OF
SUB AX,512 ; THE HEADER
BIN2:
MOV [BXSAVE],DX ; SET UP FILE SIZE IN DX:AX
MOV [CXSAVE],AX
MOV AH,CLOSE
INT 21H
JMP EXELOAD
NO_MEM_ERR:
MOV DX,OFFSET DG:TOOBIG_PTR
CALL PRINTF_CRLF
JMP COMMAND
WRTFILEJ:
JMP WRTFILE
NOFILEJ:
JMP NOFILE
BINLOAD:
PUSH AX
PUSH DX
CMP BYTE PTR [RDFLG],WRITE
JZ WRTFILEJ
CALL OPEN_A_FILE
JC NOFILEJ
MOV BX,[HANDLE]
MOV AX,(LSEEK SHL 8) OR LSEEK_EOF_OPTION
XOR DX,DX ;CX:DX=DISTANCE (OFFSET) TO MOVE IN BYTES
MOV CX,DX
INT 21H ; GET SIZE OF FILE
MOV SI,DX
MOV DI,AX ; SIZE TO SI:DI
MOV AX,(LSEEK SHL 8) OR LSEEK_FROM_START
XOR DX,DX
MOV CX,DX
INT 21H ; RESET POINTER BACK TO BEGINNING
POP AX
POP BX
PUSH BX
PUSH AX ; TRANS ADDR TO BX:AX
ADD AX,15
RCR AX,1
MOV CL,3
MOV CL,4
SHR AX,CL
ADD BX,AX ; Start of transfer rounded up to seg
MOV DX,SI
MOV AX,DI ; DX:AX is size
cmp dx,10h
jnc no_mem_err
MOV CX,16
DIV CX
OR DX,DX
JZ NOREM
INC AX
NOREM: ; AX is number of paras in transfer
ADD AX,BX ; AX is first seg that need not exist
jc no_mem_err
CMP AX,CS:[PDB_BLOCK_LEN]
JA NO_MEM_ERR
MOV CXSAVE,DI
MOV BXSAVE,SI
POP DX
POP AX
; AX:DX is disk transfer address (segment:offset)
; SI:DI is length (32-bit number)
RDWR:
RDWRLOOP:
MOV BX,DX ; Make a copy of the offset
AND DX,000FH ; Establish the offset in 0H-FH range
MOV CL,4
SHR BX,CL ; Shift offset and
ADD AX,BX ; Add to segment register to get new Seg:offset
PUSH AX
PUSH DX ; Save AX,DX register pair
MOV WORD PTR [TRANSADD],DX
MOV WORD PTR [TRANSADD+WORD],AX
MOV CX,0FFF0H ; Keep request in segment
OR SI,SI ; Need > 64K?
JNZ BIGRDWR
MOV CX,DI ; Limit to amount requested
BIGRDWR:
PUSH DS
PUSH BX
MOV BX,[HANDLE]
MOV AH,[RDFLG]
LDS DX,[TRANSADD]
INT 21H ; Perform read or write
POP BX
POP DS
JC BADWR
CMP BYTE PTR [RDFLG],WRITE
JNZ GOODR
CMP CX,AX
JZ GOODR
BADWR:
MOV CX,AX
STC
POP DX ; READ OR WRITE BOMBED OUT
POP AX
RETURN
GOODR:
MOV CX,AX
SUB DI,CX ; Request minus amount transferred
SBB SI,0 ; Ripple carry
OR CX,CX ; End-of-file?
POP DX ; Restore DMA address
POP AX
JZ DOCLOSE
ADD DX,CX ; Bump DMA address by transfer length
MOV BX,SI
OR BX,DI ; Finished with request
JNZ RDWRLOOP
DOCLOSE:
SAVEREG <AX,BX>
MOV BX,HANDLE
MOV AH,CLOSE
INT 21H
RESTOREREG <BX,AX>
RETURN
NOFILE:
MOV DX,OFFSET DG:NOTFND_PTR
JMP RESTART
NO_NAME_GIVEN:
MOV DX,OFFSET DG:NONAMESPEC_PTR
RESTARTJMP:
JMP RESTART
WRTFILE:
CMP [NAMESPEC],0
JZ NO_NAME_GIVEN ; Hey User, you forgot to specify a name
CALL CREATE_A_FILE ; Create file we want to write to
JC CHECKREADONLY ; ARR 2.4
MOV SI,BXSAVE ; Get high order number of bytes to transfer
CMP SI,000FH
JLE WRTSIZE ; Is bx less than or equal to FH
XOR SI,SI ; Ignore BX if greater than FH - set to zero
WRTSIZE:
MOV WRT_ARG2,SI
MOV DI,CXSAVE
MOV WRT_ARG1,DI
MOV DX,OFFSET DG:WRTMES_PTR
CALL PRINTF_CRLF
POP DX
POP AX
CALL RDWR
JNC CLSFLE
CALL CLSFLE
CALL DELETE_A_FILE
MOV DX,OFFSET DG:NOSPACE_PTR
JMP RESTARTJMP
CALL CLSFLE ;is this dead code? - edk
JMP COMMAND
CHECKREADONLY: ; ARR 2.4
MOV DX,[FILESTRT]
MOV SI,[FILEEND]
PUSH [SI]
MOV BYTE PTR [SI],0
MOV AX,CHMOD SHL 8 ;AL=0,REQUEST FILE'S CURRENT
; ATTRIBUTE BE RETURNED IN CX
INT 21H
POP [SI]
MOV DX,OFFSET DG:NOROOM_PTR ; Creation error - report error
JC RESTARTJMP
TEST CX,ATTR_READ_ONLY+ATTR_HIDDEN+ATTR_SYSTEM
JZ RESTARTJMP
MOV DX,OFFSET DG:ACCMES_PTR ; Write on read only file
JMP RESTARTJMP
CLSFLE:
MOV AH,CLOSE
MOV BX,[HANDLE]
INT 21H
RETURN
EXELOAD:
POP [RETSAVE] ; Suck up return addr
INC BYTE PTR [NEWEXEC]
MOV BX,[USER_PROC_PDB]
MOV AX,BEGSEG
MOV DS,AX
ASSUME DS:NOTHING
CMP AX,BX
JZ DEBUG_CURRENT
JMP FIND_DEBUG
DEBUG_CURRENT:
MOV AX,CS:[DSSAVE]
DEBUG_FOUND:
MOV CS:BYTE PTR [NEWEXEC],0
MOV CS:[HEADSAVE],AX
PUSH CS:[RETSAVE] ; Get the return address back
PUSH AX
MOV BX,CS
SUB AX,BX
PUSH ES
MOV ES,CS:BEGSEG
ASSUME ES:NOTHING
MOV BX,AX ; size of debug in para.
ADD BX,10H
MOV AX,CS ; and the size of printf in para.
SUB AX,CS:BEGSEG
ADD BX,AX
MOV AH,SETBLOCK
INT 21H
POP ES
POP AX
MOV CS:WORD PTR [COM_LINE+WORD],AX
MOV CS:WORD PTR [COM_FCB1+WORD],AX
MOV CS:WORD PTR [COM_FCB2+WORD],AX
PUSH DS
PUSH CS
POP DS
CALL EXEC_A_FILE
POP DS
MOV AX,CS:[HANDLE]
JC EXECERR
CALL SET_TERMINATE_VECTOR ; Reset int 22
MOV AH,GET_CURRENT_PDB
INT 21H
MOV CS:[USER_PROC_PDB],BX
MOV CS:[DSSAVE],BX
MOV CS:[ESSAVE],BX
MOV ES,BX
MOV WORD PTR ES:[PDB_EXIT],OFFSET DG:TERMINATE
MOV WORD PTR ES:[PDB_EXIT+WORD],CS
LES DI,CS:[COM_CSIP]
MOV CS:[CSSAVE],ES
MOV CS:[IPSAVE],DI
MOV CS:WORD PTR [DISADD+WORD],ES
MOV CS:WORD PTR [DISADD],DI
MOV CS:WORD PTR [ASMADD+WORD],ES
MOV CS:WORD PTR [ASMADD],DI
MOV CS:WORD PTR [DEFDUMP+WORD],ES
MOV CS:WORD PTR [DEFDUMP],DI
MOV BX,DS
MOV AH,SET_CURRENT_PDB
INT 21H
LES DI,CS:[COM_SSSP]
MOV AX,ES:[DI]
INC DI
INC DI
MOV CS:[AXSAVE],AX
MOV CS:[SSSAVE],ES
MOV CS:[SPSAVE],DI
RETURN
EXECERR:
PUSH CS
POP DS
MOV DX,OFFSET DG:NOTFND_PTR
CMP AX,ERROR_FILE_NOT_FOUND
JZ GOTEXECEMES
MOV DX,OFFSET DG:ACCMES_PTR
CMP AX,ERROR_ACCESS_DENIED
JZ GOTEXECEMES
MOV DX,OFFSET DG:TOOBIG_PTR
CMP AX,ERROR_NOT_ENOUGH_MEMORY
JZ GOTEXECEMES
MOV DX,OFFSET DG:EXEBAD_PTR
CMP AX,ERROR_BAD_FORMAT
JZ GOTEXECEMES
MOV DX,OFFSET DG:EXECEMES_PTR
GOTEXECEMES:
CALL PRINTF_CRLF
JMP COMMAND
HEX2BIN:
MOV [INDEX],DX
MOV DX,OFFSET DG:HEXWRT_PTR
CMP BYTE PTR [RDFLG],WRITE
JNZ RDHEX
JMP RESTARTJ2
RDHEX:
MOV ES,AX
CALL OPEN_A_FILE
MOV DX,OFFSET DG:NOTFND_PTR
JNC HEXFND
JMP RESTART
HEXFND:
XOR BP,BP
MOV SI,OFFSET DG:(BUFFER+BUFSIZ) ; Flag input buffer as empty
READHEX:
CALL GETCH
CMP AL,CHAR_COLON ; Search for : to start line
JNZ READHEX
CALL GETBYT ; Get byte count
MOV CL,AL
MOV CH,0
JCXZ HEXDONE
CALL GETBYT ; Get high byte of load address
MOV BH,AL
CALL GETBYT ; Get low byte of load address
MOV BL,AL
ADD BX,[INDEX] ; Add in offset
MOV DI,BX
CALL GETBYT ; Throw away type byte
READLN:
CALL GETBYT ; Get data byte
STOSB
CMP DI,BP ; Check if this is the largest address so far
JBE HAVBIG
MOV BP,DI ; Save new largest
HAVBIG:
LOOP READLN
JMP SHORT READHEX
GETCH:
CMP SI,OFFSET DG:(BUFFER+BUFSIZ)
JNZ NOREAD
MOV DX,OFFSET DG:BUFFER
MOV SI,DX
MOV AH,READ
PUSH BX
PUSH CX
MOV CX,BUFSIZ
MOV BX,cs:[HANDLE]
INT 21H
POP CX
POP BX
OR AX,AX
JZ HEXDONE
NOREAD:
LODSB
CMP AL,CHAR_EOF
JZ HEXDONE
OR AL,AL
RETNZ
HEXDONE:
MOV [CXSAVE],BP
MOV BXSAVE,0
RETURN
HEXDIG:
CALL GETCH
CALL HEXCHK
RETNC
MOV DX,OFFSET DG:HEXERR_PTR
RESTARTJ2:
JMP RESTART
GETBYT:
CALL HEXDIG
MOV BL,AL
CALL HEXDIG
SHL BL,1
SHL BL,1
SHL BL,1
SHL BL,1
OR AL,BL
RETURN
;=========================================================================
; ABSREAD: This routine performs a primitive logical sector read of
; the specified drive. This routine replaces the old
; INT 25h function which only allowed 16 bit addressibility.
; The new generic IOCtl logical sector read will permit
; 32 bit addressibility on a disk device.
;
; Inputs : REL_READ_WRITE_TAB - Table provides dword sector
; addressibility.
;
; Outputs: Data located at specified transfer address.
;
; Error : Carry is set on error.
;
; Date : 6/17/87
;=========================================================================
ABSREAD proc near ;an000;read logical sector(s)
push ds ;an000;save affected regs
push cx ;an000;save affected regs
push bx ;an000;
mov cx,-1 ;an000;extended format
mov bx,offset dg:rel_read_write_tab ;an000;point to read/write table
int 25h ;an000;invoke relative sector read
pop bx ;an000;discard stack word
pop bx ;an000;restore regs
pop cx ;an000;
pop ds ;an000;
ret ;an000;return to caller
ABSREAD endp ;an000;end proc
;=========================================================================
; ABSWRT: This routine performs a primitive logical sector write of
; the specified drive. This routine replaces the old
; INT 26h function which only allowed 16 bit addressibility.
; The new generic IOCtl logical sector write will permit
; 32 bit addressibility on a disk device.
;
; Inputs : REL_READ_WRITE_TAB - Table provides dword sector
; addressibility.
;
; Outputs: Data moved from transfer address to applicable sector(s).
;
; Error : Carry is set on error.
;
; Date : 6/17/87
;=========================================================================
ABSWRT proc near ;an000;write logical sector(s)
push ds ;an000;save affected regs
push cx ;an000;
push bx ;an000;
mov cx,-1 ;an000;extended format
mov bx,offset dg:rel_read_write_tab ;an000;point to read/write table
int 26h ;an000;invoke relative sector write
pop bx ;an000;discard stack word
pop bx ;an000;restore regs
pop cx ;an000;
pop ds ;an000;
ret ;an000;return to caller
ABSWRT endp ;an000;end proc
CODE ENDS
END DEBCOM2
|