Mercurial > hg > Members > kono > nitros9-code
view level2/modules/cc3io.asm @ 1904:fd159d660df5
Changes for cobber
author | afra |
---|---|
date | Thu, 03 Nov 2005 01:37:41 +0000 |
parents | 7a973074acbe |
children |
line wrap: on
line source
******************************************************************** * CC3IO - CoCo 3 I/O driver * * $Id$ * * NOTE: CODE ISSUES FOUND!! * "Animate Palette? This obviously isn't implemented yet" * Look at this code. Why is this calling an entry point in * SNDDRV??? * * Edt/Rev YYYY/MM/DD Modified by * Comment * ------------------------------------------------------------------ * 16 1986/??/?? * Original OS-9 L2 Tandy distribution. * * 26r3 1998/10/12 * Added support for obtaining monitor type from the init module. * * 26r4 1998/10/23 * Added support for obtaining key repeat info from the init module. * * 26r5 2002/07/24 * Added support for obtaining mouse info from the init module. * * 27 2003/08/18 Boisy G. Pitre * Forward ported to NitrOS-9. * * 2003/11/16 Robert Gault * Corrected several lines for keyboard mouse. * Corrected several lines in SSMOUSE where MS.Side used incorrectly. * * 2003/12/02 Boisy G. Pitre * Keyboard mouse is now either global or local to window, depending * on whether GLOBALKEYMOUSE is defined. * * 2004/08/14 Boisy G. Pitre * Fixed a bug where the last deiniz of the last window device caused * an infinite loop. The problem was that IOMan clears the static * storage of a device whose use count has reached zero (in the * case of a hard detach). See Note below. nam CC3IO ttl CoCo 3 I/O driver * Disassembled 98/09/09 08:29:24 by Disasm v1.6 (C) 1988 by RML ifp1 use defsfile endc tylg set Drivr+Objct atrv set ReEnt+rev rev set 0 edition set 27 * Comment out next line for global keyboard mouse; otherwise, it's on/off * on a per-window basis. GLOBALKEYMOUSE equ 1 mod eom,name,tylg,atrv,start,CC3DSiz fcb EXEC.+UPDAT. name fcs /CC3IO/ fcb edition start lbra Init lbra Read lbra Write lbra GetStat lbra SetStat * Term * * Entry: * U = address of device memory area * * Exit: * CC = carry set on error * B = error code * Term equ * ldx <D.CCMem get ptr to CC memory cmpu G.CurDev,x device to be terminated is current? * cmpu >WGlobal+G.CurDev device to be terminated is current? bne noterm no, execute terminate routine in co-module lbsr SHFTCLR get last window memory pointer cmpu G.CurDev,x device to be terminated is current? * cmpu >WGlobal+G.CurDev we the only window left? bne noterm no, execute terminate routine in co-module * We are last device that CC3IO has active; terminate ourself pshs cc orcc #IRQMask IFNE H6309 clrd ELSE clra clrb ENDC std G.CurDev,x * std >WGlobal+G.CurDev * ldx <D.Clock change altirq routine to go to clock ldx G.OrgAlt,x get original D.AltIRQ address stx <D.AltIRQ puls cc restore IRQs pshs u,x ldx #(WGlobal+G.JoyEnt) bsr TermSub ldx #(WGlobal+G.SndEnt) bsr TermSub ldx #(WGlobal+G.KeyEnt) bsr TermSub puls u,x noterm ldb #$0C branch table offset for terminate lbra CallCo go to terminate in co-module * Call terminate routine in subroutine module (KeyDrv/JoyDrv/SndDrv) * X = addr in statics of entry TermSub leau 2,x point U to static area for sub module ldx ,x get entry pointer at ,X jmp $03,x call term routine in sub module * Init * * Entry: * Y = address of device descriptor * U = address of device memory area * * Exit: * CC = carry set on error * B = error code * Init ldx <D.CCMem get ptr to CC mem ldd <G.CurDev,x has CC3IO itself been initialized? lbne PerWinInit yes, don't bother doing it again * CC3IO initialization code - done on the first init of ANY cc3io device * leax >CC3Irq,pcr set up AltIRQ vector in DP * stx <D.AltIRQ leax >SHFTCLR,pcr point to SHIFT-CLEAR subroutine pshs x save it on stack leax >setmouse,pcr get address of setmouse routine tfr x,d ldx <D.CCMem get ptr to CC mem std >G.MsInit,x puls b,a get address of SHIFT-CLEAR subroutine std >G.WindBk,x save its vector stu <G.CurDev,x lbsr setmouse initialize mouse lda #$02 sta G.CurTik,x save # ticks between cursor(s)updates inc <G.Mouse+Pt.Valid,x set mouse packet to invalid ldd #$0178 default to right mouse/time out value std <G.Mouse+Pt.Actv,x ldd #$FFFF initialize keyboard values std <G.LKeyCd,x last keyboard code & key repeat counter inactive std <G.2Key2,x ldd <D.Proc get cur proc desc ptr in D pshs u,y,x,b,a save regs * Added to allow patching for RGB/CMP/Mono and Key info - BGP * Uses new init module format to get monitor type and key info ldy <D.Init get init module ptr lda MonType,y get monitor type byte 0,1,2 sta <G.MonTyp,x save off ldd MouseInf,y get mouse information sta <G.Mouse+Pt.Res,x save off hi-res/lo-res flag stb <G.Mouse+Pt.Actv,x save off left/right ldd KeyRptS,y get key repeat start/delay constant sta <G.KyRept,x set first delay std <G.KyDly,x set initial and 2ndary constants ldd <D.SysPrc get system process desc ptr std <D.Proc and make current proc leax >KeyDrv,pcr point to keyboard driver sub module name bsr LinkSys link to it * U = ptr to CC mem sty >G.KeyEnt,u and save the entry point leau >G.KeyMem,u point U to keydrv statics jsr ,y call init routine of sub module leax >JoyDrv,pcr point to joystick driver sub module name bsr LinkSys link to it * U = ptr to CC mem sty >G.JoyEnt,u and save the entry point leau >G.JoyMem,u point U to joydrv statics jsr ,y call init routine of sub module leax >SndDrv,pcr point to sound driver sub module name bsr LinkSys link to it * U = ptr to CC mem sty >G.SndEnt,u and save the entry point leau >G.SndMem,u point U to sound statics jsr ,y call init routine of sub module puls u,y,x,b,a restore saved regs std <D.Proc and restore current process ldx <D.AltIRQ get original D.AltIRQ address stx >WGlobal+G.OrgAlt save in window globals for later leax >CC3Irq,pcr set up AltIRQ vector in DP stx <D.AltIRQ * This code is executed on init of every window * U = device memory area PerWinInit ldd #$0078 set default SS.Mouse parameters std <V.MSmpl,u (Mouse sample rate & fire button timeout value) ldd <IT.PAR,y get parity/baud bytes from dev desc std <V.DevPar,u save it off in our static *** Find CC3GfxInt * pshs u,y,a ..else VDG * lda #$02 get code for VDG type window * sta <V.WinType,u save it * leax <CC3GfxInt,pcr point to CC3GfxInt name * lbsr L08D4 link to it if it exists * puls u,y,a restore regs & return *** lbra FindCoMod go find and init co-module KeyDrv fcs /KeyDrv/ JoyDrv fcs /JoyDrv/ SndDrv fcs /SndDrv/ LinkSys lda #Systm+Objct system module os9 F$Link link to it ldu <D.CCMem get ptr to CC mem rts * Read * * NOTE: * This just reads keys from the buffer. The physical reading * of keys is done by the IRQ routine. * * Entry: * Y = address of path descriptor * U = address of device memory area * * Exit: * A = character read * CC = carry set on error * B = error code * Read lda V.PAUS,u device paused? bpl read1 no, do normal read (should this be bne?) * Device is paused; check for mouse button press lda >(WGlobal+G.Mouse+Pt.CBSA) if paused, check mouse button 1 beq read1 button isn't pressed, do normal read clra clear carry (no error) rts return read1 lda <V.SSigID,u data ready signal trap set up? lbne NotReady no, exit with not ready error * Data ready signal trap set up leax >ReadBuf,u point to keyboard buffer ldb <V.InpPtr,u get current position in keyboard buffer orcc #IRQMask disable IRQs cmpb <V.EndPtr,u same as end of buffer ptr (no keys in buffer)? beq ReadSlp yes, no new chars waiting, sleep/scan for them * Character(s) waiting in buffer abx move ptr to character lda ,x get character from buffer incb inc keyboard buffer ptr bpl bumpdon if it hasn't wrapped 128 bytes, go save it * bsr ChkWrap check for wrap clrb bumpdon stb <V.InpPtr,u save updated keyboard buffer ptr andcc #^(IRQMask!Carry) restore IRQ and clear carry rts return with A containing char read * Nothing is in input buffer so wait for it ReadSlp lda V.BUSY,u get active process id # sta V.WAKE,u save as process id # to wake up when data read andcc #^IRQMask restore IRQ ldx #$0000 sleep till data ready os9 F$Sleep clr V.WAKE,u signal gotten, disable process # to wake up ldx <D.Proc get current proc desc ptr ldb <P$Signal,x signal pending? beq Read no, go read char * Signal was pending already, check it out IFNE H6309 tim #Condem,P$State,x are we condemend? ELSE lda P$State,x bita #Condem ENDC bne ReadErr yes, exit with error flag set back to SCF cmpb #S$Window window change or higher signal? bcc Read yes, read the char since it won't change ReadErr coma major signal, return with error flag rts (Keyboard abort/interrupt) * Check wraparound of keyboard buffer (could be inlined) *ChkWrap incb inc keyboard buffer pointer * cmpb #$7F wrapped around? * bls L015F branch if not * clrb else reset pointer to 0 *L015F rts return * Keyboard mouse coordinate deltas L0160 fcb 8,1 right arrow (normal, shifted) fdb MaxRows-1 right arrow (control) fcb -8,-1 left arrow (normal, shifted) fdb 0 left arrow (control) fcb 8,1 down arrow (normal, shifted) fdb MaxLine down arrow (control) fcb -8,-1 up arrow (normal, shifted) fdb 0 up arrow (control) * Check mouse coordinate * Entry: D=Maximum allowed coordinate for current axis being checked * Y=Ptr to current coordinate in mouse packet being checked L0170 cmpd ,y past maximum allowed coordinate? blt L017B ldd ,y below zero? bpl L017D no, return IFNE H6309 clrd set it to minimum coordinate (zero) ELSE clra clrb ENDC L017B std ,y set it to maximum coordinate L017D rts return * Main keyboard scan (after PIA has been read) * Check keyboard mouse arrows * Entry: U=Global mem ptr * X=??? * A=Key that was pressed * Exit: E=0 if key was pressed, 1 if none pressed * Updated for localized keyboard mouse similiar to TC9IO * L017E ldb #$01 flag pshs u,y,x,b,a save registers used & flag ldb <G.KyMse,u get keyboard mouse flag beq L01E6 branch if off * Keyboard mouse is on lda <G.KySns,u bita #%01111000 any arrow key pressed? beq L01DF clr $01,s clear flag to indicate update lda #$01 sta <G.MseMv,u flag a mouse coord change ldd #$0803 start at up arrow and up arrow table pshs b,a entries & save them leax >L0160,pcr point to keyboard mouse deltas leay <G.Mouse+Pt.AcY,u point to mouse coords * Update keyboard mouse co-ordinates according to arrow key pressed L01A2 bita <G.KySns,u desired arrow key down? beq L01C5 no, move to next key lslb multiply * 4 (size of each set) lslb to point to start of proper arrow entry tst <G.ShftDn,u shift key down? beq L01B1 no, go on incb move ptr to <shifted> offset bra L01BC get delta L01B1 tst <G.CntlDn,u control key down? beq L01BC no, go on * <CTRL>-arrow addb #$02 move ptr to <CTRL> offset ldd b,x get control coordinate bra L01C1 go store it in mouse packet * <arrow> or <SHIFT>-<arrow> L01BC ldb b,x get offset to present mouse coordinate sex make into 16 bit offset (keep sign) addd ,y add it to current coordinate L01C1 std ,y save updated coordinate ldd ,s get key count L01C5 lsla move to next key bit decb decrement key count cmpb #$01 down to X coordinates? bne L01CD no, continue leay -$02,y move to mouse X coordinate L01CD std ,s save key count & key bpl L01A2 keep trying until all keys checked puls b,a purge stack of key and delta offset ldd #MaxRows-1 get maximum X coordinate bsr L0170 check X coordinate leay $02,y move to Y coordinate ldd #MaxLine get maximum Y coordinate bsr L0170 check it L01DF lda <G.KyButt,u key button down? bne L0223 yes, return lda ,s get back character read L01E6 tst <G.Clear,u clear key down? beq L0225 yes, return clr <G.Clear,u clear out clear key flag * Check CTRL-0 (CAPS-Lock) cmpa #$81 control-0? bne L01FF no, keep checking ldb <G.KySame,u same key pressed? bne L0223 ldx <G.CurDev,u get dev mem pointer IFNE H6309 eim #CapsLck,<V.ULCase,x ELSE ldb <V.ULCase,x eorb #CapsLck reverse current CapsLock status stb <V.ULCase,x ENDC bra L0223 return * Check CLEAR key L01FF cmpa #$82 was it clear key? bne L0208 no, keep going lbsr CLEAR find next window bra L0223 return * Check SHIFT-CLEAR L0208 cmpa #$83 was it shift clear? bne L0211 no, keep checking lbsr SHFTCLR yes, find back window bra L0223 return * Check CTRL-CLEAR L0211 cmpa #$84 keyboard mouse toggle key? bne L0225 no, return ldb <G.KySame,u same key pressed? bne L0223 yes, return IFNE GLOBALKEYMOUSE com <G.KyMse,u ELSE ldx <G.CurDev,u clra assume no keyboard mouse IFNE H6309 eim #KeyMse,<V.ULCase,x ELSE ldb <V.ULCase,x eorb #KeyMse reverse current Keyboard Mouse status stb <V.ULCase,x ENDC beq KeyMOff branch if off deca else A = $FF KeyMOff sta <G.KyMse,u save window's keyboard mouse flag in global ENDC L0223 clr $01,s L0225 ldb $01,s puls pc,u,y,x,b,a restore regs L0229 pshs x,b save external mouse button status & PIA addr leax <G.Mouse,u mouse point to mouse packet tst Pt.ToTm,x timed value zero? lbeq L02C8 branch if so leas -$05,s make a buffer for locals tfr a,b move keyboard button flags to B tst <G.KyMse,u keyboard mouse activated? bne L024E yes, go on ldb #%00000101 mask for button 1 & 2 on right mouse/joystick lda Pt.Actv,x get active mouse side anda #%00000010 clear all but left side select sta ,s save result beq L0248 if 0 (off or right side), skip ahead lslb otherwise, change button 1 & 2 mask for left moue L0248 andb $05,s check with external mouse button status type tsta right side? beq L024E yes, skip ahead lsrb left side, shift over so we can use same routine * Bits 0 & 2 of B contain external mouse buttons that are pressed (doesn't * matter which side) L024E clra clear out A lsrb shift out LSBit of B rola put into LSBit of A lsrb shift out another bit of B std $01,s store fire button info bne L0276 fire button(s) pressed, go on lda Pt.TTTo,x timeout occur? beq L02C6 yes, exit bsr L02CA fire buttons change? beq L0262 no, decrement timeout count bsr L02D3 go update fire button click & timeout info beq L02AB if neither button state changed, skip ahead L0262 dec Pt.TTTo,x decrement timeout count bne L02AB not timed out, go update last state counts IFNE H6309 clrd clrw ELSE clra clrb ENDC sta >G.MsSig,u clear read flag std Pt.TSSt,x clear time since counter start IFNE H6309 stq Pt.CCtA,x clear button click count & time this state ELSE std Pt.CCtA,x clear button click count & time this state std Pt.TTSA,x ENDC std Pt.TLSA,x clear button time last state bra L02C6 exit L0276 lda Pt.ToTm,x get timeout initial value sta Pt.TTTo,x reset count bsr L02CA fire buttons change? beq L02AB no, update last state counts bsr L02D3 update fire button info inc >WGlobal+G.MsSig flag mouse signal IFNE H6309 ldq <Pt.AcX,x get actual X & Y coordinates stq <Pt.BDX,x copy it to button down X & Y coordinates ELSE ldd <Pt.AcX,x get actual X coordinate std <Pt.BDX,x copy it to button down X coordinate ldd <Pt.AcY,x get actual Y coordinate std <Pt.BDY,x copy it to button down Y coordinate ENDC pshs u save ptr to CC mem ldu <G.CurDev,u get dev mem ptr lda <V.MSigID,u get process ID requesting mouse signal beq L02A9 branch if none ldb <V.MSigSg,u else get signal code to send os9 F$Send and send it bcs L02A5 branch if error clr <V.MSigID,u clear signal ID (one shot) L02A5 clr >WGlobal+G.MsSig clear read flag L02A9 puls u recover pointer to CC mem L02AB ldd Pt.TTSA,x get button A&B time last state cmpa #$FF limit? beq L02B2 yes, go on inca increment state L02B2 cmpb #$FF limit? beq L02B7 yes, store them incb increment B state L02B7 std Pt.TTSA,x save updated states ldd Pt.TSST,x get time since start IFNE H6309 incd increment beq L02C6 branch if zero ELSE * cmpd #$FFFF check upper bound * beq L02C4 branch if so * addd #$0001 else increment addd #1 beq L02C6 ENDC L02C4 std Pt.TSST,x save updated state count L02C6 leas $05,s purge locals L02C8 puls pc,x,b restore & return L02CA ldd Pt.CBSA,x get button states IFNE H6309 eord $03,s flip fire 1 & 2 ELSE eora $03,s eorb $04,s ENDC std $05,s save 'em rts return * Update mouse button clock counts & timeouts L02D3 ldd Pt.TTSA,x get button time this state tst $05,s button A change? beq L02E9 no, go check B sta Pt.TLSA,x save button A time last state lda $03,s button A pressed? bne L02E8 yes, skip increment lda Pt.CCtA,x get click count for A inca bump up click count beq L02E9 branch if wrapped sta Pt.CCtA,x save button A click count L02E8 clra clear button A time this state L02E9 tst 6,s button B change? beq L02FD no, go save time this state stb Pt.TLSB,x save button B time last state count ldb $04,s button B pressed? bne L02FC yes, skip increment ldb Pt.CCtB,x get b click count incb bump up click count beq L02FD brach if wrapped to zero stb Pt.CCtB,x save B click count L02FC clrb clear button B time this state L02FD std Pt.TTSA,x save button time this state counts ldd $03,s get new fire buttons std Pt.CBSA,x save 'em ldd $05,s get button A & B change flags NullIRQ rts return * CC3IO IRQ routine - Entered from Clock every 1/60th of a second CC3Irq ldu <D.CCMem get ptr to CC mem ldy <G.CurDev,u get current device's static lbeq L044E branch if none lda <G.TnCnt,u get tone counter beq L0319 branch if zero deca else decrement sta <G.TnCnt,u and save back * Check for any change on screen * U=Unused now (sitting as NullIRQ ptr) - MAY WANT TO CHANGE TO CUR DEV PTR * Y=Current Device mem ptr L0319 leax <NullIRQ,pcr set AltIRQ to do nothing routine so other IRQs stx <D.AltIRQ can fall through to IOMan polling routine andcc #^(IntMasks) re-enable interrupts ldb <V.ScrChg,y check screen update request flag (cur screen) beq L0337 no update needed, skip ahead lda V.TYPE,y device a window? bpl L032F no, must be VDGInt, so go on lda G.GfBusy,u 0 = GrfDrv free, 1 = GrfDrv busy ora G.WIBusy,u 0 = WindInt free, 1 = WindInt busy bne L034F one of the two is busy, can't update, skip *L032F lda #$00 L032F clra special function: select new active window lbsr L05DA go execute co-module clr <V.ScrChg,y clear screen change flag in device mem * * CHECK IF GFX/TEXT CURSORS NEED TO BE UPDATED * G.GfBusy = 1 Grfdrv is busy processing something else * G.WIBusy = 1 Windint is busy processing something else * g0000 = # of clock ticks/cursor update constant (2) for 3 ticks: 2,1,0 * G.CntTik = current clock tick for cursor update * L0337 ldb G.CntTik,u get current clock tick count for cursor updates beq L034F if 0, no update required decb decrement the tick count stb G.CntTik,u if still not 0, don't do update bne L034F lda G.GfBusy,u get GrfDrv busy flag ora G.WIBusy,u merge with WindInt busy flag beq L034A if both not busy, go update cursors inc G.CntTik,u otherwise bump tick count up again bra L034F and don't update L034A lda #$02 update cursors sub-function code lbsr L05DA go update cursors through co-module * Check for mouse update L034F equ * * Major error here. Used regU which points to D.CCMem not G.CurDev. RG IFNE GLOBALKEYMOUSE tst <G.KyMse,u keyboard mouse? ELSE IFNE H6309 tim #KeyMse,<V.ULCase,y keyboard mouse? ELSE lda <V.ULCase,y keyboard mouse? bita #KeyMse ENDC ENDC bne L0369 branch if so lda <G.MSmpRt,u get # ticks until next mouse read beq L0369 0 means shut off, don't bother deca decrement # ticks bne L0366 still not yet, save tick counter & skip mouse pshs u,y,x save dev mem ptr and others lbsr L0739 go update mouse packet puls u,y,x restore regs lda <G.MSmpRV,u get # ticks/mouse read reset value L0366 sta <G.MSmpRt,u save updated tick count * Check keyboard L0369 equ * IFNE H6309 clrd initialize keysense & same key flag ELSE clra clrb ENDC std <G.KySns,u initialize keysense & same key flag * Major error here. Was regU; see above. RG IFNE GLOBALKEYMOUSE tst <G.KyMse,u ELSE IFNE H6309 tim #KeyMse,>V.ULCase,y ELSE pshs a lda >V.ULCase,y is the keyboard mouse enabled? bita #KeyMse puls a ENDC ENDC beq L0381 no, try joystick ldx >WGlobal+G.KeyEnt else get ptr to keydrv leau >G.KeyMem,u and ptr to its statics jsr K$FnKey,x call into it ldu <D.CCMem get ptr to CC mem sta <G.KyButt,u save key button L0381 ldx >WGlobal+G.JoyEnt get ptr to joydrv leau >G.JoyMem,u and ptr to its statics jsr J$MsBtn,x get mouse button info ldu <D.CCMem get ptr to CC mem lda #%10000010 A = $82 cmpb #%10000000 clear flag set? beq L0397 branch if so inca A now = $83 cmpb #%11000000 shift clear flag set? bne L039C branch if not L0397 inc <G.Clear,u bra L03C8 L039C tst V.PAUS,y pause screen on? bpl L03A8 branch if not bitb #%00000011 any mouse buttons down? beq L03A8 branch if not lda #C$CR load A with carriage return bra L03C8 L03A8 lda <G.KyButt,u lbsr L0229 tstb lbne L044E pshs u,y,x ldx >WGlobal+G.KeyEnt leau >G.KeyMem,u jsr K$RdKey,x call Read Key routine puls u,y,x bpl L03C8 branch if valid char received clr <G.LastCh,u else clear last character var lbra L044E L03C8 cmpa <G.LastCh,u is current ASCII code same as last one pressed? bne L03DF no, no keyboard repeat, skip ahead ldb <G.KyRept,u get repeat delay constant beq L044E if keyboard repeat shut off, skip repeat code decb repeat delay up? beq L03DA branch if so and reset L03D5 stb <G.KyRept,u update delay bra L044E return L03DA ldb <G.KySpd,u get reset value for repeat delay bra L03ED go update it L03DF sta <G.LastCh,u store last keyboard character ldb <G.KyDly,u get keyboard delay speed tst <G.KySame,u same key as last time? bne L03D5 no, go reset repeat delay ldb <G.KyDly,u get time remaining L03ED stb <G.KyRept,u save updated repeat delay lbsr L017E beq L044E ldb #$01 This may be wrong because regB was created in sub RG stb >g00BF,u menu keypress flag ldu <G.CurDev,u get ptr to statics in U ldb <V.EndPtr,u leax >ReadBuf,u point to keyboard buffer abx move to proper offset incb inc keyboard buffer ptr bpl bumpdon2 hasn't wrapped, skip ahead clrb reset pointer * lbsr ChkWrap check for wrap-around bumpdon2 cmpb <V.InpPtr,u same as start? beq L0411 yep, go on stb <V.EndPtr,u save updated pointer L0411 sta ,x save key in buffer beq L0431 go on if it was 0 * Check for special characters cmpa V.PCHR,u pause character? bne L0421 no, keep checking ldx V.DEV2,u is there an output path? beq L0443 no, wake up the process sta V.PAUS,x set immediate pause request on device bra L0443 wake up the process L0421 ldb #S$Intrpt get signal code for key interrupt cmpa V.INTR,u is key an interrupt? beq L042D branch if so (go send signal) ldb #S$Abort get signal code for key abort cmpa V.QUIT,u is it a key abort? bne L0431 no, check data ready signal L042D lda V.LPRC,u get last process ID bra L0447 go send the signal L0431 lda <V.SSigID,u send signal on data ready? beq L0443 no, just go wake up process ldb <V.SSigSg,u else get signal code os9 F$Send bcs L044E clr <V.SSigID,u clear signal ID bra L044E return L0443 ldb #S$Wake get signal code for wakeup lda V.WAKE,u get process ID to wake up L0447 beq L044E no process to wake, return clr V.WAKE,u clear it os9 F$Send send the signal L044E ldu <D.CCMem get ptr to CC mem lda <G.AutoMs,u auto mouse flag set? beq L046B branch if not lda <G.MseMv,u get mouse moved flag ora <G.Mouse+Pt.CBSA,u beq L046B lda G.GfBusy,u check for GrfDrv busy ora G.WIBusy,u OR with WindInt busy bne L046B branch if they are busy lda #$03 lbsr L05DA clr <G.MseMv,u clear mouse move flag L046B orcc #IntMasks mask interrupts leax >CC3Irq,pcr get CC3Irq vector stx <D.AltIRQ and store in AltIRQ rts return org 4 f.nbyte rmb 1 # of bytes to next entry in table (signed #) f.tblend rmb 2 ptr to end of device table + 1 f.ptrstr rmb 2 start of search ptr (if backwards, -1 entry) f.ptrend rmb 2 end of search ptr (if backwards, -1 entry) *f.ptrcur rmb 2 ptr to current device's device table entry f.ptrdrv rmb 2 ptr to current device's driver f.ptrchk rmb 2 ptr to the device table entry we are currently checking f.numdve rmb 1 number of device table entries in device table f.end equ . * Prepare for Window search in Device Table * Point to end of device table WinSearchInit stb f.nbyte+2,s save # bytes to next (neg or pos) ldx <D.Init get pointer to init module lda DevCnt,x get max # of devices allowed sta f.numdve+2,s ldb #DEVSIZ get size of each device table entry mul calculate total size of device table ldy <D.DevTbl get device table ptr leax d,y point X to end of devtable + 1 stx f.tblend+2,s save the ptr & return rts * CLEAR processor CLEAR pshs u,y,x,b,a preserve registers leas <-f.end,s make a buffer on stack ldb #DEVSIZ get # of bytes to move to next entry (forward) bsr WinSearchInit get pointer to devtable stx f.ptrend,s save end of devtable sty f.ptrstr,s save beginning of devtable bra FindWin * Shift-CLEAR processor SHFTCLR pshs u,y,x,b,a preserve registers leas <-f.end,s make a buffer on the stack ldb #-DEVSIZ # of bytes to move next entry (backwards) bsr WinSearchInit make ptrs to devtable * Here, Y points to first entry of device table * and X points to last entry of device table + 1 leay -DEVSIZ,y bump Y back by 1 entry (for start of loop) sty f.ptrend,s save it leax -DEVSIZ,x bump X back for start of loop stx f.ptrstr,s save it * FindWin - Find the next (or previous) window in the device table * * The search takes place just before or after the current window's * device table entry. * * NOTE: SS.OPEN for current window has changed V.PORT to be the ptr to the * current window's entry in the device table FindWin ldx <D.CCMem get ptr to CC mem ldu <G.CurDev,x get active device's static mem ptr lbeq L0546 if none (no screens), exit without error ldx V.PORT,u get device table ptr for current device * stx f.ptrcur,s save it on stack stx f.ptrchk,s save as default we are checking ldd V$DRIV,x get ptr to current device driver's module std f.ptrdrv,s save it on stack * Main search loop L04BA ldx f.ptrchk,s get ptr to device tbl entry we are checking L04BC ldb f.nbyte,s get # of bytes to next entry (signed) dec f.numdve,s + have we exhausted all entries? bmi L0541 + yes, end leax b,x point to next entry (signed add) cmpx f.ptrend,s did we hit end of search table? bne L04C6 no, go check if it is a screen device ldx f.ptrstr,s otherwise wrap around to start of search ptr * Check device table entry (any entry we can switch to has to have CC3IO as * the driver) L04C6 stx f.ptrchk,s save new device table ptr we are checking ldd V$DRIV,x get ptr to driver cmpd f.ptrdrv,s same driver as us? (CC3IO) bne L04BC no, try next one * NOTE: The next two lines are moved down two lines, past the check * for our own device table pointer. This fixes a bug where the last * deiniz of the last window device caused an infinite loop. The problem * was that IOMan clears the static storage of a device whose use count * has reached zero (in the case of a hard detach), and we were testing for * a V$STAT of zero BEFORE seeing if we reached our own device table entry. * Next two lines moved... * ldu V$STAT,x get ptr to static storage for tbl entry * beq L04BC there is none, try next one * cmpx f.ptrcur,s is this our own (have we come full circle)? * beq L0541 yes, obviously nowhere else to switch to * ...to here. ldu V$STAT,x get ptr to static storage for tbl entry beq L04BC there is none, try next one * Found an initialized device controlled by CC3IO that is not current device lda <V.InfVld,u is the extra window data in static mem valid? beq L04BA no, not good enough, try next one ldx <V.PDLHd,u get ptr to list of open paths on device beq L0536 no open paths, so switch to that device lda V.LPRC,u get last active process ID # that used device beq L0536 * Path's open to device & there is a last process # for that path ldy <D.PrcDBT get process descriptor table ptr lda a,y get MSB of ptr to process descriptor last on it beq L0536 process now gone, so switch to device clrb move process desc ptr to Y tfr d,y lda >P$SelP,y get the path # that outputs to the window leay <P$Path,y move to the path table local to the process sta ,s pshs x L04FA ldb #NumPaths for every possible path... lda ,x get system path into A L04FE decb decrement cmpa b,y same? beq L050F branch if so tstb are we at start of paths? bne L04FE branch if not ldx <PD.PLP,x get ptr to next path dsc. list (linked list) bne L04FA branch if valid puls x else restore X bra L0536 L050F puls x lda ,s L0513 sta ,s cmpa #$02 is selected path one of the 3 std paths? bhi L051F not one of the std 3 paths, skip ahead ldb #$02 standard error path lda b,y get system path # for local error path bra L0522 L051F lda a,y get system path # for local path clrb standard in * X=Ptr to linked list of open paths on device * A=System path # * B=Local (to process) path # * Check if any paths to device are open, if they are we can switch to it L0522 cmpa ,x path we are checking same as path already open beq L0536 on device? yes, go switch to it decb bump local path # down bmi L052D if no more paths to check, skip ahead lda b,y get system path # for new local path to check bra L0522 check if it is already open on device L052D lda ,s get local path # we started on ldx <PD.PLP,x get ptr to path dsc. list (linked list) bne L0513 there is no path desc list, try next path bra L04BA can't switch to it, go to next device tbl entry L0536 ldx <D.CCMem get ptr to CC mem stu <G.CurDev,x save new active device clr g000A,x flag that we are not on active device anymore clr >g00BF,x clear WindInt's key was pressed flag (new window) * If there is only one window, it comes here to allow the text/mouse cursors * to blink so you know you hit CLEAR or SHIFT-CLEAR L0541 inc <V.ScrChg,u flag device for a screen change bsr setmouse check mouse L0546 leas <f.end,s purge stack buffer clrb clear carry puls pc,u,y,x,b,a restore regs and return * Initialize mouse * Also called when CLEARing to a new window. setmouse pshs x save register used ldd <V.MSmpl,u get sample and timeout from win devmem ldx <D.CCMem get ptr to CC mem sta <G.MSmpRt,x set sample tick count in global mem sta <G.MSmpRV,x set sample rate in global mem stb <G.Mouse+Pt.ToTm,x set timeout constant in mouse packet ldb <V.MAutoF,u get auto follow flag from win devmem stb <G.AutoMs,x and set auto follow flag in global mem lda V.TYPE,u get device type sta <G.WinType,x set it IFEQ GLOBALKEYMOUSE * Added: get window's keyboard mouse flag and update global keyboard mouse IFNE H6309 tim #KeyMse,<V.ULCase,u keyboard mouse? ELSE lda <V.ULCase,u keyboard mouse? bita #KeyMse ENDC bne setmous2 clra fcb $8c setmous2 lda #$FF sta <G.KyMse,x ENDC clra puls pc,x restore and return * Write * * Entry: * A = character to write * Y = address of path descriptor * U = address of device memory area * * Exit: * CC = carry set on error * B = error code * Write ldb <V.ParmCnt,u are we in the process of getting parameters? lbne L0600 yes, go process sta <V.DevPar,u save off character cmpa #C$SPAC space or higher? bcc CoWrite yes, normal write cmpa #$1E 1E escape code? bcc L05EF yes, go process cmpa #$1B $1B escape code? beq L05F3 yes, go handle it cmpa #$05 $05 escape code? (cursor on/off) beq L05F3 yep, go handle it cmpa #C$BELL Bell? bne CoWrite no, control char jmp [>WGlobal+G.BelVec] for whom the bell tolls... CoWrite ldb #$03 write entry point in co-module CallCo lda <V.DevPar,u get character stored earlier L0593 ldx <D.CCMem get ptr to CC mem stu G.CurDvM,x save dev mem ptr for current device L0597 pshs a leax <G.CoTble,x point to co-module entry vectors lda <V.WinType,u get window type from device mem ldx a,x get vector to proper co-module puls a beq L05EB vector empty, exit with module not found leax b,x bsr L05C0 ldb <V.WinType,u beq L05B4 jsr ,x go execute co-module L05B0 pshs cc bra L05BB L05B4 jsr ,x L05B6 pshs cc clr >WGlobal+G.WIBusy L05BB clr >WGlobal+G.CrDvFl puls pc,cc L05C0 pshs x,b ldx <D.CCMem get ptr to CC mem clr G.WIBusy,x clear WindInt busy flag ldb <V.WinType,u get window type (0 = WindInt) bne L05CE branch if VDGInt incb else make B = 1 stb G.WIBusy,x and make WindInt busy L05CE clr G.CrDvFl,x clear 'we are current device' cmpu <G.CurDev,x bne L05D8 inc g000A,x L05D8 puls pc,x,b * U = ptr to CC memory L05DA pshs u,y,x ldu <G.CurDev,u get ptr to curr dev mem L05DF ldb #$0F ldx <D.CCMem get ptr to CC memory in X bsr L0597 puls pc,u,y,x restore regs and return L05E7 pshs u,y,x save regs bra L05DF L05EB comb ldb #E$MNF rts * $1E & $1F codes go here L05EF cmpa #$1E $1E code? beq Do1E branch if so * $1F codes fall through to here * Escape code handler : Initial code handled by CC3IO, any parameters past * $1B xx are handled by co-module later * NOTE: Notice that is does NOT update <DevPar,u to contain the param byte, * but leaves the initial <ESC> ($1b) code there. The co-module checks it * to see it as an ESC, and then checks for the first parameter byte for the * required action. L05F3 leax <CoWrite,pcr point to parameter vector entry point ldb #$01 get parameter count (need 1 to determine code) stx <V.ParmVct,u save vector stb <V.ParmCnt,u save # param bytes needed before exec'ing vect. Do1E clrb no error rts return * Processing parameters * A=parameter byte from SCF * B=# parameter bytes left (not including one in A) * U=device mem ptr L0600 ldx <V.NxtPrm,u get ptr of where to put next param byte sta ,x+ put it there stx <V.NxtPrm,u update pointer decb decrement parameter count stb <V.ParmCnt,u update it bne Do1E if still more to get, exit without error * B=0, flag to say we are not current device * We have all parameter bytes we need at this point. ldx <D.CCMem get ptr to CC mem bsr L05C0 stu G.CurDvM,x ldx <V.PrmStrt,u reset next param ptr to start stx <V.NxtPrm,u ldb <V.WinType,u is this device using WindInt? beq L0624 yes, special processing for WindInt jsr [<V.ParmVct,u] go execute parameter handler bra L05B0 L0624 jsr [<V.ParmVct,u] bra L05B6 * GetStat * * Entry: * A = function code * Y = address of path descriptor * U = address of device memory area * * Exit: * CC = carry set on error * B = error code * GetStat cmpa #SS.EOF beq SSEOF ldx PD.RGS,y cmpa #SS.ComSt beq GSComSt cmpa #SS.Joy beq GSJoy cmpa #SS.Mouse lbeq GSMouse cmpa #SS.Ready beq GSReady cmpa #SS.KySns beq GSKySns cmpa #SS.Montr beq GSMontr ldb #$06 carry over to co-module lbra L0593 * SS.ComSt - get baud/parity info GSComSt lda V.TYPE,u get device type clrb clear parity, etc. std R$Y,x save it in register stack rts return GSReady ldb <V.EndPtr,u get input buffer end pointer cmpb <V.InpPtr,u anything there? beq NotReady nope, exit with error bhi L0660 higher? addb #$80 nope, add 128 to count L0660 subb <V.InpPtr,u calculate number of characters there stb R$B,x save it in register stack SSEOF clrb clear errors rts return NotReady comb set carry ldb #E$NotRdy get error code rts return * Return special key status * X = pointer to caller's register stack GSKySns ldy <D.CCMem get ptr to CC mem clrb clear key code cmpu <G.CurDev,y are we the active device? bne L0678 branch if not ldb <G.KySns,y get key codes L0678 stb R$A,x save to caller reg clrb clear errors rts return * GetStat: SS.Montr (get Monitor type) * X = pointer to caller's register stack GSMontr ldb >WGlobal+G.MonTyp get monitor type * tfr b,a put in A clra std R$X,x save in caller's X rts return * GetStat: SS.JOY (get joystick X/Y/button values) * X = pointer to caller's register stack GSJoy clrb default to no errors tfr x,y transfer caller's registers to Y ldx <D.CCMem get ptr to CC mem cmpu <G.CurDev,x are we the current active device? beq GetJoy if so, go read joysticks clra else D = 0 std R$X,y X pos = 0 std R$Y,y Y pos = 0 sta R$A,y no buttons held down rts return * Get button status first GetJoy ldx >WGlobal+G.JoyEnt pshs u save driver static ldu <D.CCMem get ptr to CC mem leau >G.JoyMem,u point to subroutine module's static mem jsr J$JyBtn,x call entry point to get button * Joysticks button states returned in B puls u restore driver static lda R$X+1,y left or right? beq L06AB branch if right joystick lsrb shift over so same range as if right joystick L06AB andb #$05 preserve button bits lsrb button 1 down? (shifts button 2 to bit 2 too) bcc L06B2 no, go on orb #$01 turn on button 1 L06B2 stb R$A,y save button status to caller * * Now get actual joystick values (note: IRQs still off) * pshs y save ptr to caller's regs lda R$X+1,y get switch to indicate left or right joystick inca now 1 or 2 ldy #$0000 force low res?? pshs u save driver static mem ldu <D.CCMem get ptr to CC mem ldx >WGlobal+G.JoyEnt get address of joystick sub module leau >G.JoyMem,u get ptr to sub module's static mem jsr J$JyXY,x call routine in sub module to get joy X/Y * X = joystick X pos, Y = joystick Y pos puls u restore driver static mem pshs y save joystick Y ldy $02,s get ptr to caller's regs stx R$X,y save joystick X in caller's X ldd #63 subd ,s++ std R$Y,y save joystick Y in caller's Y clrb cleary carry puls pc,y return * GetStat: SS.Mouse (get mouse info) * X = pointer to caller's register stack GSMouse pshs u,y,x ldx <D.CCMem get ptr to CC mem cmpu <G.CurDev,x is caller in current window? beq L06FA branch if so ldy ,s get ptr to caller's regs ldb #Pt.Siz size of packet L06EC clr ,-s make room on stack decb bne L06EC leax ,s point X to temp mouse buffer on stack bsr MovMsPkt leas <Pt.Siz,s clean up stack puls pc,u,y,x and return * here the caller is in the current window L06FA tst <G.KyMse,x mouse keyboard active? bne L071A branch if so lda <G.MSmpRV,x ready to sample? bne L071A no, return packet pshs u,y,x bsr L073B read external mouse puls u,y,x lda <G.AutoMs,x get automouse flag anda <G.MseMv,x has mouse moved? beq L071A no, return packet lda #$03 update auto-follow mouse sub-function call lbsr L05E7 call co-module to update mouse clr <G.MseMv,x flag that the mouse hasn't moved L071A lda #$01 'special' co-mod function code: move mouse packet? lbsr L05E7 leax <G.Mouse,x move X to point to mouse packet ldy ,s get register stack pointer bsr MovMsPkt move packet to caller puls pc,u,y,x * Move mouse packet to process * Y = ptr to caller's register stack MovMsPkt ldu R$X,y get destination pointer ldy <D.Proc get process descriptor pointer ldb P$Task,y get destination task number clra get source task number ldy #Pt.Siz get length of packet os9 F$Move move it to the process rts return L0739 ldx <D.CCMem L073B leax <G.Mouse,x move X to mouse packet clra clear MSB of mouse resolution ldb <Pt.Res,x get resolution (0 = lores, 1 = hires) tfr d,y move mouse res to Y lda Pt.Actv,x get mouse side pshs u,y,x,b,a preserve regs ldx >WGlobal+G.JoyEnt get ptr to mouse sub module ldu <D.CCMem get mem pointer leau >G.JoyMem,u and point to mouse sub module statics jsr J$MsXY,x get data pshs y,x ldx $06,s get ptr to mouse packet in CC mem puls b,a get X value into D leay <Pt.AcX,x point X to mouse X/Y in mouse packet bsr L0764 puls b,a get Y value into D bsr L0764 puls pc,u,y,x,b,a * X = Address of G.Mouse in D.CCMem L0764 cmpd ,y++ compare mouse's current X to Pt.AcX beq L0770 branch if same std -$02,y else store new X into Pt.AcX lda #$01 sta <(G.MseMv-G.Mouse),x update mouse moved flag L0770 rts SSTone ldx >WGlobal+G.SndEnt get address of sound sub module jmp S$SetStt,x go execute routine in sub module * Animate Palette? This obviously isn't implemented yet SSAnPal ldx >WGlobal+G.SndEnt jmp S$Term,x * SetStat * * Entry: * A = function code * Y = address of path descriptor * U = address of device memory area * * Exit: * CC = carry set on error * B = error code * SetStat ldx PD.RGS,y cmpa #SS.ComSt lbeq SSComSt cmpa #SS.Montr lbeq SSMontr cmpa #SS.KySns lbeq SSKySns cmpa #SS.Tone beq SSTone cmpa #SS.AnPal beq SSAnPal cmpa #SS.SSig beq SSSig cmpa #SS.MsSig beq SSMsSig cmpa #SS.Relea beq SSRelea cmpa #SS.Mouse beq SSMouse cmpa #SS.GIP lbeq SSGIP cmpa #SS.Open bne L07B5 SSOpen ldx PD.DEV,y get device table entry stx V.PORT,u save it as port address L07B5 ldb #$09 call setstt entry point in co-module lbra L0593 go do it * SS.SSig - send signal on data ready SSSig pshs cc save interrupt status * The next line doesn't exist in the NitrOS version * clr <V.SSigID,u lda <V.InpPtr,u get input buffer pointer suba <V.EndPtr,u get how many chars are there pshs a save it temporarily bsr L07EC get current process ID tst ,s+ anything in buffer? bne L07F7 yes, go send the signal std <V.SSigID,u save process ID & signal puls pc,cc restore interrupts & return * SS.MsSig - send signal on mouse button SSMsSig pshs cc save interrupt status * The next line doesn't exist in the NitrOS version * clr <V.MSigID,u bsr L07EC get process ID ldx <D.CCMem get ptr to CC mem cmpu <G.CurDev,x are we active device? bne L07E7 no, save ID & signal tst >G.MsSig,x has button been down? bne L07F3 yes, go send the signal L07E7 std <V.MSigID,u save ID & signal code puls pc,cc restore interrupts & return L07EC orcc #IntMasks disable interrupts lda PD.CPR,y get curr proc # ldb R$X+1,x get user signal code rts return L07F3 clr >G.MsSig,x clear mouse button down flag L07F7 puls cc restore interrupts os9 F$Send send the signal rts return * SS.Relea - release a path from SS.SSig SSRelea lda PD.CPR,y get curr proc # cmpa <V.SSigID,u same as keyboard? bne L0807 branch if not clr <V.SSigID,u clear process ID L0807 cmpa <V.MSigID,u same as mouse? bne L0871 no, return clr <V.MSigID,u else clear process ID rts return * SS.Mouse - set mouse sample rate and button timeout * * Entry: * R$X = mouse sample rate and timeout * MSB = mouse sample rate * LSB = mouse button timeout * R$Y = mouse auto-follow feature * MSB = don't care * LSB = auto-follow ($00 = OFF, else = ON) * * NOTE: Default mouse params @ $28,u are $0078 * It modifies the static mem variables (for caller's window) first, and * then modifies global memory only if we are the current active device. SSMouse ldd R$X,x get sample rate & timeout from caller cmpa #$FF sample rate 256? beq L0819 yes, can't have it so go on sta <V.MSmpl,u save new timeout L0819 cmpb #$FF timeout 256? beq L0820 yes, can't have it so go on stb <V.MTime,u save new timeout L0820 ldb R$Y+1,x get auto-follow flag stb <V.MAutoF,u save it was MS.Side wrong RG ldy <D.CCMem get ptr to CC mem cmpu <G.CurDev,y are we current device? bne L083D no, exit without error stb <G.AutoMs,y save auto-follow flag for this dev ldd <V.MSmpl,u get sample rate/timeout sta <G.MSmpRV,y save it (reset value) sta <G.MSmpRt,y save it (current value) stb <G.Mouse+Pt.ToTm,y save timeout too L083D clrb exit without error rts * SS.GIP SSGIP ldy <D.CCMem get ptr to CC mem cmpu <G.CurDev,y current window? bne L0866 branch if not ldd R$Y,x get caller's Y (key repeat info) cmpd #$FFFF unchanged? beq L0853 yes, don't change current key info std <G.KyDly,y else save key delay and speed info L0853 ldd R$X,x get mouse info cmpa #$01 set for hi res adapter? bgt L088F branch to error if greater sta <G.Mouse+Pt.Res,y save new resolution value * B = mouse port (1 = right, 2 = left) tstb side above legal value? beq L088F no, exit with error cmpb #$02 side below legal value? bgt L088F no, exit with error stb <G.Mouse+Pt.Actv,y save new side L0866 clrb clear errors rts and return * SS.KySns - setstat??? SSKySns ldd R$X,x get monitor type requested beq L086E below legal value? ldb #$FF no, exit with error L086E stb <V.KySnsFlg,u save new sense mode L0871 clrb clear errors rts return * SS.Montr - change monitor type SSMontr ldd R$X,x get monitor type requested cmpd #$0002 below legal value? bhi L088F no, exit with error lda <D.VIDMD get current GIME video mode register anda #$EF get rid of monochrome bit bitb #$02 mono requested? beq L0885 no, keep checking ora #$10 switch to monochrome L0885 sta <D.VIDMD update video mode register stb >WGlobal+G.MonTyp save new monitor type inc <V.ScrChg,u flag a screen change clrb clear errors rts return * Illegal argument error handler L088F comb set carry for error ldb #E$IllArg get illegal argument error code rts return with it * SS.ComSt - set baud/parity params SSComSt ldd R$Y,x get requested window type eora V.TYPE,u same type as now? anda #$80 trying to flip from window to VDG? bne L088F yes, error lda R$Y,x no, get requested window type again bsr FindCoMod go make sure co-module for new type exists lbcc L07B5 carry it over to co-module rts return VDGInt fcs /VDGInt/ * * Link to proper co-module * Try VDGInt first * * Entry: A = window type (If bit 7 is set, it's a window, else VDG screen) * FindCoMod sta V.TYPE,u save new type bmi FindWind if hi-bit if A is set, we're a window pshs u,y,a ..else VDG lda #$02 get code for VDG type window sta <V.WinType,u save it leax <VDGInt,pcr point to VDGInt name bsr L08D4 link to it if it exists puls pc,u,y,a restore regs & return WindInt fcs /WindInt/ GrfInt fcs /GrfInt/ ++ *CC3GfxInt fcs /CC3GfxInt/ ++ * * Try WindInt * FindWind pshs u,y preserve regs clra set window type sta <V.WinType,u leax <WindInt,pcr point to WindInt name lda #$80 get driver type code bsr L08D4 try and link it *++ bcc ok leax <GrfInt,pcr point to GrfInt name lda #$80 bsr L08D4 *++ ok puls pc,u,y restore regs and return L08D2 clrb rts * * Check if co-module is in memory * L08D4 ldb <V.PrmStrt,u any parameter vector? bne L08D2 no, return pshs u save statics ldu <D.CCMem get ptr to CC mem bita <G.BCFFlg,u BCFFlg already linked? puls u restore statics bne L0900 yes, initialize co-module tsta Window type device? bpl L08E8 no, go on clra set co-module vector offset for window L08E8 pshs y,a preserve registers bsr L0905 try and link module bcc L08F0 we linked it, go on puls pc,y,a restore registers & return error L08F0 puls a restore vector offset ldx <D.CCMem get ptr to CC mem leax <G.CoTble,x point to vector offsets sty a,x store co-module entry vector puls y restore path descriptor pointer cmpa #$02 was it WindInt? bgt L08D2 no, return L0900 clrb lbra CallCo send it to co-module * * Link or load a co-module * L0905 ldd <D.Proc get current process descriptor pointer pshs u,x,b,a preserve it along with registers ldd <D.SysPrc get system process descriptor pointer std <D.Proc save it as current process lda #Systm+Objct get codes for link os9 F$Link link to it ldx $02,s get name pointer bcc L091B does module exist? ldu <D.SysPrc no, get system process descriptor pointer os9 F$Load load it L091B puls u,x,b,a restore regs std <D.Proc restore current process descriptor lbcs L05EB exit if error from load or link rts return emod eom equ * end