|
DataMuseum.dkPresents historical artifacts from the history of: Jet Computer Jet80 |
This is an automatic "excavation" of a thematic subset of
See our Wiki for more about Jet Computer Jet80 Excavated with: AutoArchaeologist - Free & Open Source Software. |
top - 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