forked from microsoft/GW-BASIC
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCALL86.ASM
95 lines (86 loc) · 3.13 KB
/
CALL86.ASM
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
; [ This translation created 10-Feb-83 by Version 4.3 ]
.RADIX 8 ; To be safe
CSEG SEGMENT PUBLIC 'CODESG'
ASSUME CS:CSEG
INCLUDE OEM.H
TITLE CALL86 8086 CALL Statement
.SALL
.RADIX 8
EXTRN VMOVFM:NEAR,FRQINT:NEAR,PTRGET:NEAR,GETSTK:NEAR
EXTRN CHRGTR:NEAR,SYNCHR:NEAR,CHRGT2:NEAR,FCERR:NEAR
DSEG SEGMENT PUBLIC 'DATASG'
ASSUME DS:DSEG
EXTRN SUBFLG:WORD,SARYFL:WORD,TEMP:WORD,TEMPA:WORD,SAVSEG:WORD
EXTRN ARYTAB:WORD
DSEG ENDS
; This is the CALL <simple var>[(<simple var>[,<simple var>]..)]
; Stragegy:
;
; 1.) Make sure suboutine name is simple var, get value & save it
;
; 2.) Evaluate params & stuff pointers on stack
;
; 3.) CALL suboutine with return address on stack
;
; The CALLS statement is the same as CALL, except for each parameter
; it pushes Segment and Offset, not just Offset.
PUBLIC CALLS,CALLSL
CALLSL: CLC ;Clear-carry, (PUSH Segment adr of parms)
JMP SHORT CALLS1
CALLS: STC ;Set-carry, (don't PUSH Segment adr of parms)
CALLS1: PUSHF
EXTRN PRODIR:NEAR ;Don't allow CALL as direct statement in
CALL PRODIR ;protected environment
MOV BYTE PTR SUBFLG,LOW 200O ;say we want to scan only a simple
CALL PTRGET ;scan var and search symbol table
MOV BYTE PTR SARYFL,LOW 0 ;clear Scanned-Array-Element-Flag
PUSH BX ;save text pointer
MOV BX,DX ;get pointer to var in BX
CALL VMOVFM ;load variable into FAC
CALL FRQINT ;make it an integer
MOV WORD PTR TEMPA,BX ;save text pointer
MOV CL,LOW 32D ;get max # of parameters
CALL GETSTK ;see if there is that much stack space
POP BX ;get back text pointer
CALL CHRGT2 ;eat character after var name
JZ CALLST ;end of statement, no parameter list
CALL SYNCHR ;check for open paren
DB 50O ; (
GETPAR: PUSH WORD PTR ARYTAB ;save pointer to start of array var data
CALL PTRGET ;scan parameter variable
POP CX ;[CX]=old value of ARYTAB
CMP CX,WORD PTR ARYTAB ;if = old, no undefined simples were referenced
JE NONEWS ;Branch if New Simple Var not encountered
CMP BYTE PTR SARYFL,LOW 0 ;Z-FLAG if no Array elements have been parsed
; by PTRGET
JZ NONEWS
JMP FCERR ;Undefined scalers can't be passed after array
;elements since the addr of the array element
;changes when the new scaler is added.
NONEWS: POPF ;restore Carry=CALL/CALLS flag
JB SHTPRM ;Branch if CALL (not CALLS)
PUSH DS ;save Segment of Parameter on stack
SHTPRM: PUSH DX ;save Offset of parameter on stack
PUSHF ;re-save CALL/CALLS flag
MOV AL,BYTE PTR 0[BX] ;get terminator
CMP AL,LOW "," ;comma?
JNZ ENDPAR ;no, must be end of the param list
CALL CHRGTR ;scan next char
JMP SHORT GETPAR ;scan next parm
ENDPAR: CALL SYNCHR ;check for terminating right paren
DB 51O ; )
CALLST: MOV WORD PTR TEMP,BX ;save text pointer
POPF ;discard CALL/CALLS flag
PUSH CS ;save BASIC code segment
MOV AX,OFFSET CALLRT ;where to return to
PUSH AX
PUSH WORD PTR SAVSEG ;save subroutine segment
PUSH WORD PTR TEMPA ;save subroutine address
DB 313O ; Do a long return to call the subroutine
;
CALLRT: MOV BX,WORD PTR TEMP ;get back text pointer
RET ;return to newstt
;
CSEG ENDS
END