|
|
DataMuseum.dkPresents historical artifacts from the history of: CP/M |
This is an automatic "excavation" of a thematic subset of
See our Wiki for more about CP/M Excavated with: AutoArchaeologist - Free & Open Source Software. |
top - metrics - download
Length: 16384 (0x4000)
Types: TextFile
Names: »CCC.ASM«
└─⟦1275f6521⟧ Bits:30005823 BD Software C Compiler v1.50a
└─⟦this⟧ »CCC.ASM«
;
; CCC.ASM: BDS C Run-Time Package (C.CCC) v1.50, 11/9/82
; Copyright (c) 1982 by Leor Zolman
;
; NOTE: If you are running under MP/M II, be sure to set the MPM2
; equate to 1.
;
; This is the BDS C run-time package. Normally, it resides at
; the start of the TPA (at BASE+100h). The code generated
; by BDS C always sits immediately on top of this run-time
; package code.
;
; Equate statements in CAPITAL letters may be customized by the
; user in order to change a) the origin of the run-time package,
; b) the origin of the run-time package's local r/w area, and c) the
; RST vector used to interface with Kirkland's CDB debugger. If
; you will be generating code to run in a non-CP/M environment,
; set the CPM equate to zero and make sure to set the ORIGIN, RAM
; and EXITAD equates to fit your custom run-time configuration.
;
FALSE: EQU 0
TRUE: EQU NOT FALSE
CPM: EQU TRUE ;True if to be run under CP/M or MP/M
MPM2: EQU FALSE ;True ONLY if running under MP/M II
DMAVIO: EQU TRUE ;True if using DMA video library routines and
;need parameters initialized
USERST: EQU FALSE ;True to use a restart vector for CDB interfacing
RSTNUM: EQU 6 ;Use "RST n" as default debugger vector. Has no
;effect if USERST is false.
rstloc: equ RSTNUM*8 ;Memory address where "RST n" vector falls
IF CPM
nfcbs: equ 9 ;maximum # of files open at one time
base: equ 0 ;start of ram in system (either 0 or 4200h)
bdos: equ base+5 ;the rest of these do not vary between CP/M systems.
tpa: equ base+100h
tbuff: equ base+80h
origin: equ tpa
exitad: equ base ;warm boot location
conin: equ 1 ;BDOS call codes...console input
cstat: equ 11 ;interrogate console status
closec: equ 16 ;close file
gsuser: equ 32 ;get/set user code
ENDIF
IF NOT CPM ;fill in the appropriate values...
ORIGIN: EQU NEWBASE ;Address at which programs are to run
RAM: EQU WHATEVER ;R/W memory area for non-CP/M configurations
; (default: just after C.CCC under CP/M)
EXITAD: EQU WHENDONE ;where to go when done executing
ENDIF
;
; The location of the jump vectors and utility routines must remain
; constant relative to the beginning of this run-time module.
;
; Do NOT change ANYTHING between here and the start of the
; "init" routine!!!!!!!!
;
org origin
;
; The "lxi sp,0" instruction at the start of the code is changed by
; CLINK, if the "-t" option is NOT used, into:
; lhld base+6
; sphl
;
; If "-t <addr>" is used, then the sequence becomes:
; lxi sp,<addr>
; nop
;
; If "-n" is used, to indicate no-warm-boot, then the the sequence becomes:
; jmp snobsp
; nop
;
lxi sp,0 ;These two instructions change depending on whether
nop ;or not the CLINK "-t" or "-n" options are given.
nop
nop
jmp skpfex ;skip over the following vector (don't ask...)
fexitv: jmp exitad ;final exit vector. If "-n" used, this
;becomes address of the "nobret" routine.
skpfex: call init ;do ARGC & ARGV processing, plus misc. initializations
call main ;go crunch!!!!
jmp vexit ;close open files and reboot
extrns: ds 2 ;set by CLINK to external data base address
cccsiz: dw main-origin ;size of this code (for use by CLINK)
codend: ds 2 ;set by CLINK to (last addr of code + 1)
freram: ds 2 ;set by CLINK to (last addr of externals + 1)
;
; Jump vectors to some file i/o utility routines:
;
error: jmp verror ;loads -1 into HL and returns
exit: jmp vexit ;close all open files and reboot
IF CPM
close: jmp vclose ;close a file
setfcb: jmp vsetfcb ;set up fcb at HL given filename at DE
fgfd: jmp vfgfd ;return C set if file fd in A not open
fgfcb: jmp vfgfcb ;compute address of internal fcb for fd in A
setfcu: jmp vsetfcu ;set up FCB and process user number prefix
setusr: jmp vsetusr ;set user area to upper 5 bits of A, save previous
rstusr: jmp vrstusr ;restore user area to what it was before setusr call
snobsp: jmp vsnobsp ;set up SP for non-boot ("-tn") CLINK option
nobret: jmp vnobret ;return to CCP when non-boot ("-tn") in effect.
khack: jmp vkhack ;Kirkland interrupt vector initialization
clrex: jmp vclrex ;routine to clear external data area
ENDIF
IF NOT CPM ;if not under CP/M, file I/O routines
jmp verror ;are not used.
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
jmp verror
ENDIF
ds 9 ;reserved
;
; The following routines fetch a variable value from either
; the local stack frame or the external area, given the relative
; offset of the datum required immediately following the call;
; for the "long displacement" routines, the offset must be 16 bits,
; for the "short displacement" routines, the offset must be 8 bits.
;
;
; long-displacement, double-byte external indirection:
;
; format: call ldei ; get 16-bit value in HL
; dw offset_from_extrns ; >= 256
;
ldei: pop h ;get address of offset
mov e,m ;put offset in DE
inx h
mov d,m
inx h
push h ;save return address
lhld extrns ;add offset to external area base
dad d
mov a,m ;and get the value into HL
inx h
mov h,m
mov l,a
ret
;
; short-displacement, double-byte external indirection:
;
; format: call sdei ; get 16-bit value in L
; db offset_from_extrns ; < 256
;
sdei: pop h
mov e,m
inx h
push h
mvi d,0
lhld extrns
dad d
mov a,m
inx h
mov h,m
mov l,a
ret
;
; long-displacement, single-byte external indirection:
;
; format: call lsei ; get 8-bit value in L
; dw offset_from_extrns ; >= 256
;
lsei: pop h
mov e,m
inx h
mov d,m
inx h
push h
lhld extrns
dad d
mov l,m
ret
;
; short-displacement, single-byte external indirection:
;
; format: call ssei ; get 8-bit value in L
; db offset_from_externs ; < 256
;
ssei: pop h
mov e,m
inx h
push h
mvi d,0
lhld extrns
dad d
mov l,m
ret
;
; long-displacement, double-byte local indirection:
;
; format: call ldli ; get 16-bit value in HL
; dw offset_from_BC ; >= 256
;
ldli: pop h
mov e,m
inx h
mov d,m
inx h
push h
xchg
dad b
mov a,m
inx h
mov h,m
mov l,a
ret
;
; short-displacement, double-byte local indirection:
;
; format: call sdli ; get 16-bit value in HL
; db offset_from_BC ; < 256
;
sdli: pop h
mov e,m
inx h
push h
xchg
mvi h,0
dad b
mov a,m
inx h
mov h,m
mov l,a
ret
;
; Flag conversion routines:
;
pzinh: lxi h,1 ;return HL = true if Z set
rz
dcx h
ret
pnzinh: lxi h,0 ;return HL = false if Z set
rz
inx h
ret
pcinh: lxi h,1 ;return HL = true if C set
rc
dcx h
ret
pncinh: lxi h,0 ;return HL = false if C set
rc
inx h
ret
ppinh: lxi h,1 ;return HL = true if P (plus) flag set
rp
dcx h
ret
pminh: lxi h,1 ;return HL = true if M (minus) flag set
rm
dcx h
ret
pzind: lxi d,1 ;return DE = true if Z set
rz
dcx d
ret
pnzind: lxi d,0 ;return DE = false if Z set
rz
inx d
ret
pcind: lxi d,1 ;return DE = true if C set
rc
dcx d
ret
pncind: lxi d,0 ;return DE = false if C set
rc
inx d
ret
ppind: lxi d,1 ;return DE = true if P (plus) flag set
rp
dcx d
ret
pmind: lxi d,1 ;return DE = true if M (minus) flag set
rm
dcx d
ret
;
; Relational operator routines: take args in DE and HL,
; and return a flag bit either set or reset.
;
; ==, >, < :
;
eqwel: mov a,l ;return Z if HL == DE, else NZ
cmp e
rnz ;if L <> E, then HL <> DE
mov a,h ;else HL == DE only if H == D
cmp d
ret
blau: xchg ;return C if HL < DE, unsigned
albu: mov a,d ;return C if DE < HL, unsigned
cmp h
rnz ;if D <> H, C is set correctly
mov a,e ;else compare E with L
cmp l
ret
bgau: xchg ;return C if HL > DE, unsigned
agbu: mov a,h ;return C if DE > HL, unsigned
cmp d
rnz ;if H <> D, C is set correctly
mov a,l ;else compare L with E
cmp e
ret
blas: xchg ;return C if HL < DE, signed
albs: mov a,h ;return C if DE < HL, signed
xra d
jp albu ;if same sign, do unsigned compare
mov a,d
ora a
rp ;else return NC if DE is positive and HL is negative
stc ;else set carry, since DE is negative and HL is pos.
ret
bgas: xchg ;return C if HL > DE, signed
agbs: mov a,h ;return C if DE > HL, signed
xra d
jp agbu ;if same sign, go do unsigned compare
mov a,h
ora a
rp ;else return NC is HL is positive and DE is negative
stc
ret ;else return C, since HL is neg and DE is pos
;
; Multiplicative operators: *, /, and %:
;
smod: mov a,d ;signed MOD routine: return (DE % HL) in HL
push psw ;save high bit of DE as sign of result
call tstn ;get absolute value of args
xchg
call tstn
xchg
call usmod ;do unsigned mod
pop psw ;was DE negative?
ora a ;if not,
rp ; all done
mov a,h ;else make result negative
cma
mov h,a
mov a,l
cma
mov l,a
inx h
ret
nop ;maintain address compatibility with some
nop ; pre-release v1.4's.
usmod: mov a,h ;unsigned MOD: return (DE % HL) in HL
ora l
rz
push d
push h
call usdiv
pop d
call usmul
mov a,h
cma
mov h,a
mov a,l
cma
mov l,a
inx h
pop d
dad d
ret
smul: xra a ;signed multiply: return (DE * HL) in HL
sta tmp
call tstn
xchg
call tstn
call usmul
smul2: lda tmp
rar
rnc
mov a,h
cma
mov h,a
mov a,l
cma
mov l,a
inx h
ret
tstn: mov a,h
ora a
rp
cma
mov h,a
mov a,l
cma
mov l,a
inx h
lda tmp
inr a
sta tmp
ret
usmul: push b ;unsigned multiply: return (DE * HL) in HL
call usm2
pop b
ret
usm2: mov b,h
mov c,l
lxi h,0
usm3: mov a,b
ora c
rz
mov a,b
rar
mov b,a
mov a,c
rar
mov c,a
jnc usm4
dad d
usm4: xchg
dad h
xchg
jmp usm3
usdiv: mov a,h ;unsigned divide: return (DE / HL) in HL
ora l ;return 0 if HL is 0
rz
push b
call usd1
mov h,b
mov l,c
pop b
ret
usd1: mvi b,1
usd2: mov a,h
ora a
jm usd3
dad h
inr b
jmp usd2
usd3: xchg
usd4: mov a,b
lxi b,0
usd5: push psw
usd6: call cmphd
jc usd7
inx b
push d
mov a,d
cma
mov d,a
mov a,e
cma
mov e,a
inx d
dad d
pop d
usd7: xra a
mov a,d
rar
mov d,a
mov a,e
rar
mov e,a
pop psw
dcr a
rz
push psw
mov a,c
ral
mov c,a
mov a,b
ral
mov b,a
jmp usd6
sdiv: xra a ;signed divide: return (DE / HL) in HL
sta tmp
call tstn
xchg
call tstn
xchg
call usdiv
jmp smul2
cmphd: mov a,h ;this returns C if HL < DE
cmp d ; (unsigned compare only used
rc ; within C.CCC, not from C)
rnz
mov a,l
cmp e
ret
;
; Shift operators << and >>:
;
sderbl: xchg ;shift DE right by L bits
shlrbe: inr e ;shift HL right by E bits
shrbe2: dcr e
rz
xra a
mov a,h
rar
mov h,a
mov a,l
rar
mov l,a
jmp shrbe2
sdelbl: xchg ;shift DE left by L bits
shllbe: inr e ;shift HL left by E bits
shlbe2: dcr e
rz
dad h
jmp shlbe2
;
; Routines to 2's complement HL and DE:
;
cmh: mov a,h
cma
mov h,a
mov a,l
cma
mov l,a
inx h
ret
cmd: mov a,d
cma
mov d,a
mov a,e
cma
mov e,a
inx d
ret
;
; The following routines yank a formal parameter value off the stack
; and place it in both HL and A (low byte), assuming the caller
; hasn't done anything to its stack pointer since IT was called.
;
; The mnemonics are "Move Arg #n To HL",
; where arg #1 is the third thing on the stack (where the first
; and second things are, respectively, the return address of the
; routine making the call to here, and the previous return
; address to the routine which actually pushed the args on the
; stack.) Thus, a call to "ma1toh" would return with the first
; passed parameter in HL and A; "ma2toh" would return the second,
; etc. Note that if the caller has pushed ÆnÅ items on the stack
; before calling "ma ÆxÅ toh", then the Æx-nÅth formal parameter
; value will be returned, not the ÆxÅth.
;
ma1toh: lxi h,4 ;get first arg
ma0toh: dad sp
mov a,m
inx h
mov h,m
mov l,a
ret
ma2toh: lxi h,6 ;get 2nd arg
jmp ma0toh
ma3toh: lxi h,8 ;get 3rd arg
jmp ma0toh
ma4toh: lxi h,10 ;get 4th arg
jmp ma0toh
ma5toh: lxi h,12 ;get 5th arg
jmp ma0toh
ma6toh: lxi h,14 ;get 6th arg
jmp ma0toh
ma7toh: lxi h,16 ;get 7th arg
jmp ma0toh
;
; This routine takes the first 7 args on the stack
; and places them contiguously at the "args" ram area.
; This allows a library routine to make one call to arghak
; and henceforth have all it's args available directly
; through lhld's instead of having to hack the stack as it
; grows and shrinks. Note that arghak should be called as the
; VERY FIRST THING a function does, before even pushing BC.
;
arghak: lxi d,args ;destination for block move in DE
lxi h,4 ;pass over two return address
dad sp ;source for block move in HL
push b ;save BC
mvi b,14 ;countdown in B
arghk2: mov a,m ;copy loop
stax d
inx h
inx d
dcr b
jnz arghk2
pop b ;restore BC
ret
;
; ABSOLUTELY NO CHANGES SHOULD EVER BE MADE TO THE CODE BEFORE
; THIS POINT IN THIS SOURCE FILE (except for customizing the EQU
; statements at the beginning of the file).
;
;
; The following two routines are used when the "-tn" CLINK option
; was given, in order to preserve the SP value passed to the transient
; command by the CCP and return to the CCP after execution without
; performing a warm-boot.
;
IF CPM
vsnobsp:
lxi h,0 ;get CCP's SP value in HL
dad sp
shld spsav ;save it for later
lhld base+6 ;get BIOS pointer
lxi d,-2100 ;subtract size of CCP plus a fudge
dad d
sphl ;make that the new SP value
jmp tpa+3 ;and get things under way...
vnobret:
lhld spsav ;restore CCP's SP
sphl
ret ;return to CCP
ENDIF
;
; This routine is called first to do argc & argv processing (if
; running under CP/M) and some odds and ends initializations:
;
init: pop h ;store return address
shld tmp2 ; somewhere safe for the time being
IF CPM
lxi h,arglst-2 ;set up "argv" for the C main program
ENDIF
IF NOT CPM
lxi h,0
ENDIF
push h
;Initialize storage allocation pointers:
lhld freram ;get address after end of externals
shld allocp ;store at allocation pointer (for "sbrk.")
lxi h,1000 ;default safety space between stack and
shld alocmx ; highest allocatable address in memory
; (for use by "sbrk".).
;Initialize random seed:
lxi h,59dch ;let's stick something wierd into the
shld rseed ;first 16 bits of the random-number seed
;Initialize I/O hack locations:
mvi a,0dbh ;"in" op, for "in xx; ret" subroutine
sta iohack
mvi a,0d3h ;"out" op for "out xx; ret" subroutine
sta iohack+3
mvi a,0c9h ;"ret" for above sobroutines
sta iohack+2 ;the port number is filled in by the
sta iohack+5 ;"inp" and "outp" library routines.
IF CPM
call khack ;initialize Kirkland debugger vector
ENDIF
IF CPM ;initialize raw I/O parameters
xra a
sta freeze ;clear freeze (^S) flag
sta pending ;no pending input yet
mvi a,1fh
sta mode ;tty mode: all features enabled
mvi a,'C'-64
sta quitc ;this is the standard interrupt char
ENDIF
;Initialize DMA video parameters:
IF DMAVIO ;if we're using DMA video routines,
lxi h,0cc00h ;set up default values (may be changed
shld pbase ;to whatever suits). Video board address,
lxi h,16
shld xsize ;# of lines,
lxi h,64
shld ysize ;# of columns,
lxi h,1024
shld psize ;and total # of characters on screen
ENDIF
IF CPM ;under CP/M: clear console, process ARGC & ARGV:
mvi c,cstat ;interrogate console status to see if there
call bdos ; happens to be a stray character there...
ora a ;(used to be `ani 1'...they tell me this works
nop ; better for certain bizarre CP/M-"like" systems)
jz initzz
mvi c,conin ;if input present, clear it
call bdos
initzz: lxi h,tbuff ;if arguments given, process them.
lxi d,comlin ;get ready to copy command line
mov b,m ;first get length of it from loc. base+80h
inx h
mov a,b
ora a ;if no arguments, don't parse for argv
jnz initl
lxi d,1 ;set argc to 1 in such a case.
jmp i5
initl: mov a,m ;ok, there are arguments. parse...
stax d ;first copy command line to comlin
inx h
inx d
dcr b
jnz initl
xra a ;pl