// Raid.bcpl - Raid debugger for InterLisp-D
// Packages code added October 8, 1986 by Bill van Melle
// Last change March 20, 1985  10:30 AM by Bill van Melle
// Last change January 21, 1985  11:46 AM by Bill van Melle
// Last change May 21, 1984  3:49 PM by Bill van Melle
// Last change November 15, 1983  5:50 PM by Bill van Melle
// Last change March 30, 1983  11:27 AM by Bill van Melle
// Last change December 16, 1982  10:15 PM by Bill van Melle
// Last change April 15, 1982  4:23 PM by Bill van Melle
// Last change December 30, 1981  10:49 PM by Bill van Melle
// Last change September 27, 1981  9:36 PM by Bill van Melle
// Last change September 19, 1981  2:02 PM by Bill van Melle
// Last change August 4, 1981  12:41 PM by Beau Sheil

	get "Raid.decl"
	get "Streams.d"
	get "AltoDefs.d"

external [	// procedures defined
	RAID; uCodeCheck; RAIDCode; RaidReset; ReadChar

		// from RaidStack
	PrintFxtn; PrintBF; PrintAddrs; PrintBytes; GetFXP
		// from RaidPrint
	Lprint; LispStack; LispFrame; ShowStackBlocks
	Wb; Wn; Wo; PrintPtr; PrintStr; Confirm; Type; SpaceCheck
	CRLF 
		// from RaidProcs
	ShowRealCore; TeleRaid; ReadNum; ReadAtom; AtomNum
	DoYankDef; DoSetTopVal; FetchAtomComponent 
		// from Stack
	CtxtSwitch 
		// from VMemExtra
	PrintPageTable
		// from vmem
	LISPFINISH; WriteSwapBuf

		// from OS
	ShowDisplayStream; Endofs; Gets; Resets; Puts; Wc; Ws
        GetBitPos; SetBitPos; GetRmarg; GetLmarg; SetLmarg
        CallSwat; MyFrame; Min
		// misc
	ReadFlags; ReadRP; @BGetBasePtr; @BGetBase; @BPutBase; IGetBase
	@BGetBase32; @BPutBase32; MkSmallPos; SmallUnbox; EqNIL
	Iresume; UCase

		// statics used
	lvAbortFlag; keys; @LispKbd; dsp; @lvKT; @lvNIL; @RMSK; @lvVPtr
	uradix; uprintlevel; ulistlength
	@dlispDsp; @DisplayAddrHi; @dspArea
	EmulatorSpace; sysFontCharWidth
		// from RaidStack	
	lastFrame; linkUsed; raidStackFX
		// from RaidPrint	
	packagesOn
		// from VmemB	
	SwapBufVp

		// statics defined
	doRaid	// label used to abort typein
	insideRaid; typeDecoding; crCount; crMax
	TeleRaidSocket; rmargBitPos; stringLimit
	]

static [ insideRaid = false
	crCount = 1		// count for autohold after screenful
	crMax = 0
	typeDecoding = true	// => decode type names where possible
	rmargBitPos
	RaidFrame		// for RaidReset
	stringLimit = 200	// point at which to truncate strings
	]

manifest SubrArgsAddr = #210

structure String: [ length byte; char↑1,255 byte ]
 

let RAID(Mess1, Mess2, Flg; numargs na) =
   na eq 0 ? RAIDCode("Called from Swat", lvNIL) ,
      (na eq 1) % EqNIL(Mess2) ? RAIDCode("Called from Lisp:", Mess1) ,
	na eq 2 ? RAIDCode(Mess1, Mess2) ,
		    RAIDCode(Mess1, Mess2, not EqNIL(Flg))

and uCodeCheck(code) = RAIDCode("Called from uCode", code)

and RAIDCode(st, param, isaddr; numargs na) = valof
  [
   RaidFrame = MyFrame()
   let wasInsideRaid = insideRaid
   insideRaid = true
   if crMax eq 0
      then crMax = Min(60, dspArea rshift 8)
   if dsp eq 0
      then CallSwat("Raid: No dsp", st)
   if DisplayAddrHi
      then ShowDisplayStream(dsp, DSalone)
   rmargBitPos = GetRmarg (dsp)
   Ws("*NRaid: ")
   PrintErrorMsg (st, param, (na gr 2) & isaddr)
   [ if ((not @kbdAd) % (not @(kbdAd+1)) %
	((not @(kbdAd+2)) & #173677) % ((not @(kbdAd+3)) & #177567)) eq 0
	then break ] repeat
				// wait until all keys (but ctrl/shift) up
   Resets(keys)				// clear any type-ahead
//   @lvAbortFlag = 0		// allow shift-swat
   raidStackFX = GetFXP()
   lastFrame = 0
   linkUsed = 0

   packagesOn = not EqNIL(FetchAtomComponent (TOPVALspace, AtomNum("**PACKAGE**")))

   // This ATROCITY is due to Bcpl's refusal to allow strings as table entries
   // The effect of this series of bindings is to create a table of character
   // code, prompt message pairs on the stack
   let c1, av1  = #1,  "Ascii bytes"			// ↑A
   let c2,  v2  = #2,  "Show bytes"			// ↑B
   let c4,  v4  = #4,  "Return to top level"		// ↑D
   let c6,  v6  = #6,  "Show basic frame"		// ↑F
   let c7,  v7  = #7,  "Show stack blocks"		// ↑G
   let c12, v12 = #12, "Next frame"			// LF
   let c13, v13 = #13, "Kill Lisp"			// ↑K
   let c14, v14 = #14, "Lisp Stack from frame"		// ↑L
   let c16, v16 = #16, "Return NIL"			// ↑N
   let c17, v17 = #17, "Atom number for atom "		// ↑O
//   let c20, v20 = #20, "Turn microcode PC tracing "	// ↑P
   let c23, v23 = #23, "Call Swat"			// ↑S
   let c24, v24 = #24, "Return T"			// ↑T
   let c25, v25 = #25, "Show Lisp user screen"		// ↑U
   let c26, v26 = #26, "Set top value of atom "		// ↑V
   let c30, v30 = #30, "Show frame extension"		// ↑X
   let c31, v31 = #31, "Yank definition"		// ↑Y
   let cc,  vc  = $,,  "Word from 2 bytes "
   let ce,  ve  = $.,  "2 bytes from word "
   let cf,  vf  = $;,  "Page# from virtual address "
   let cp,  vp  = $+,  "Add 2 octal numbers "
   let ca,  va  = $←,  "Set word"
   let cd,  vd  = $<,  "Set cell"
   let cb,  vb  = $↑,  "Previous frame"
   let cq,  vq  = $?,  "Show help"
   let cA,  vA  = $A,  "Atom top level value"
   let cB,  vB  = $B,  "Show virtual addrs"
   let cC,  vC  = $C,  "Coremap"
   let cD,  vD  = $D,  "Atom definition"
   let cE,  vE  = $E,  "Error msg"
   let cF,  vF  = $F,  "Show frame number "
   let cJ,  vJ  = $J,  "Set Raid list length"
   let cL,  vL  = $L,  "Lisp stack"
//   let cM,  vM  = $M,  "Memory map check"
   let cN,  vN  = $N,  "Set Raid list depth"
   let cO,  vO  = $O,  "Show emulator addrs"
   let cP,  vP  = $P,  "Property list of atom "
   let cQ,  vQ  = $Q,  "Set Raid screen size"
   let cR,  vR  = $R,  "switch to teleRaid"
   let cS,  vS  = $S,  "Show stack addrs"
   let cT,  vT  = $T,  "Type decoding "
   let cU,  vU  = $U,  "Set Raid radix"
   let cV,  vV  = $V,  "Show Lisp object"
   let cW,  vW  = $W,  "Walk stack blocks"
   let cY,  vY  = $Y,  "Show page table"
   let cZ,  vZ  = $Z,  "Show Vmem flags"
   let cend = 0				// 0 marks end of command table
   let V = nil
   [
doRaid: Ws("*N@"); crCount = 1
   switchon GetCom(lv c1) into
      [
      case 0:				//Noop
	loop
      case 1:				//↑A{onum, onum, onum}
	StartLoc(""); PrintBytes(true)
	loop
      case 2:				//↑B{onum, onum, onum}
	StartLoc(""); PrintBytes(false)
	loop
      case 4:				//↑D{} call \RAIDEXITFN
	if Confirm() then
	   [ 
	   V = 0; break
	   ]
	CRLF()
	loop
      case 6:				//↑F{onum}
	AtLoc(" stack"); PrintBF(ReadNum(8))
	loop
      case 7:				//↑G{onum}
	StartLoc(" stack"); ShowStackBlocks(ReadNum(8), true)
	loop
      case #13:				//↑K{} to kill
	Ws(" Note: (LOGOUT T) is much safer!*N  Type OK to confirm: ")
	if UCase(ReadChar()) eq $O & UCase(ReadChar()) eq $K
	   then LISPFINISH()
	Ws(" xxx")
	loop
      case #14:				//↑L{from fx; $A or $C}
	AtLoc(" stack"); Ws ("/ context# ")
	LispStack(ReadNum(8))
	lastFrame = 0
	loop
      case $*N:				//do nothing
	loop
      case #16:				//↑N{} return NIL
	V = lvNIL
	break
      case #17:				//↑O{string} get atom number
	[ let a = ReadAtom(); Ws("is "); Wo(a) ]
	CRLF()
	loop
//      case #20:				//↑P{} uPC tracing
//	if Confirm(uPCTraceAddr ? "off", "on") then uPCTracing(true)
//	loop
      case #23:				//↑S{} gets Swat
	if Confirm() then CallSwat("Raid")
	loop
      case #24:				//↑T{} return T
	V = lvKT
	break
      case #25:				//↑U{} show Lisp display
	if Confirm() then
	 [ unless DisplayAddrHi do [ Ws(" No Lisp display to show"); loop ]
	   ShowDisplayStream(dlispDsp, DSalone)	// show lisp display
	   until Gets(keys) do loop		// wait for a keystroke
	   ShowDisplayStream(dsp, DSalone)	// restore Raid dsp
	 ]
	loop
      case #26:				//↑V{atom, value}
	DoSetTopVal()
	loop
      case #30:				//↑X{onum}
	AtLoc(" stack")
	PrintFxtn(ReadNum(8))
	loop
      case #31:				//↑Y{atom,atom}
	DoYankDef()
	loop
      case $,:				//,(onum}: 2 bytes -> word
	Wo(ReadNum(8) lshift 8 + ReadNum(8))
	CRLF()
	loop
      case $.:				//.(onum}: word -> 2 bytes
	[ let a = ReadNum(8)
	  Wo(a rshift 8); Wc($*S); Wo(a & RMSK)
	]
	CRLF()
	loop
      case $;:				//;(onum}: Va -> VP
	Wo(ReadNum(8) lshift 8 + ReadNum(8) rshift 8)
	CRLF()
	loop
      case $+:				//+(onum}
	Wo(ReadNum(8) + ReadNum(8))
	CRLF()
	loop
      case $←:				//←(onum, onum, onum}
	AtLoc("")
	[ let v0, v1 = ReadNum(8), ReadNum(8)
	  BPutBase(v0, v1, GetNewNum(BGetBase(v0, v1), 8)) ]
	CRLF()
	loop
      case $<:				//<(onum, onum, onum, onum}
	AtLoc("")
	[ let v0, v1 = ReadNum(8), ReadNum(8)
	  Ws(" currently ")
	  Wo(BGetBase(v0, v1)); Wc($*S); Wo(BGetBase(v0, v1+1))
	  Ws(" to ")
	  let n0, n1 = ReadNum(8), ReadNum(8)
	  if Confirm()
	     then [ BPutBase(v0, v1, n0); BPutBase(v0, v1+1, n1) ]
	]	  
	CRLF()
	loop
      case $?:				//help
	ShowHelp(lv c1)
	loop
      case $A:				//A{onum}
	Ws(" for ")
	Lprint(FetchAtomComponent(TOPVALspace,ReadAtom()))
	loop
      case $B:				//B{onum, onum, onum}
	StartLoc(""); PrintAddrs(ReadNum(8))
	loop
      case $C:				//C{}
	CRLF()
	ShowRealCore()
	loop
      case $D:				//D{onum}
	Ws(" for ")
	[ let a = ReadAtom()
	PrintAddrs(DEFspace+(a rshift 15), a lshift 1, 2)
	]
	loop
      case $E:				//E
	Ws(" was: "); PrintErrorMsg(st, param, (na gr 2) & isaddr)
	loop
      case $F:				//F{dnum, $A or $C}
	LispFrame()
	loop
      case #12:				//LF (next frame)
	LispFrame(lastFrame+1, linkUsed)
	loop
      case $↑:				//↑ (previous frame)
	LispFrame(lastFrame-1, linkUsed)
	loop
      case $J:				//J{dnum}
	ulistlength = GetNewNum(ulistlength, 10)
	loop
      case $L:				//L{$A or $C}
	LispStack()
	loop
      case $N:				//N{dnum}
	uprintlevel = GetNewNum(uprintlevel, 10)
	loop
      case $O:				//O{onum, onum}
	StartLoc(" Alto")
	PrintAddrs(EmulatorSpace)
	loop
      case $P:				//P{onum}
	Lprint(FetchAtomComponent(PLISTspace,ReadAtom()))
	loop
      case $Q:				//Q{dnum}
	crMax = GetNewNum(crMax, 10)
	loop
      case $R:				//R call teleRaid
	if Confirm() then
	   [ 
	   V = -1; break
	   ]
	CRLF()
	loop
      case $S:				//S{onum, onum}
	StartLoc(" stack")
	PrintAddrs(STACKspace)
	loop
      case $T:				//T type decode switch
	if Confirm(typeDecoding ? "off", "on")
	   then typeDecoding = not typeDecoding
	loop
      case $U:				//U{dnum}
	[ let N = GetNewNum(uradix, 10)
	  test (N ge 2)&(N le 10)
	    ifso  uradix = N
	    ifnot Ws("Invalid, uradix not set") ]
	loop
      case $V:				//V{onum, onum}
	AtLoc("")
	[ let v0, v1 = ReadNum(8), ReadNum(8)
	  Lprint(lv v0) ]
	loop
      case $W:				//W
	StartLoc(" stack")
	ShowStackBlocks(ReadNum(8), false)
	loop
      case $Y:				//Y 
	if Confirm()
	   then PrintPageTable()
	loop
      case $Z:				//Z{onum, onum}
	Ws(" for virtual pages from ")
	[ let s = ReadNum(8); Ws(" to ")
	  let f = ReadNum(8)
	  Ws("*N    VP   Flags RealP*N")
	  for i = s to f
	   do [ Wo(i); Ws("  ")
		Wo(ReadFlags(i))
		Wo(ReadRP(i))
		CRLF() ]
	]
	loop
      default: Ws("??"); Resets(keys)
	loop
      ]
    ] repeat

//  Exit sequence. Just return, unless ↑D, in which case reset Lisp

    insideRaid = wasInsideRaid
    CRLF()
    Resets(keys)
    [ if ((not @kbdAd) % (not @(kbdAd+1)) %
		    (not @(kbdAd+2)) % ((not @(kbdAd+3)) & #177577)) eq 0
		    then break	// Wait until everything but shiftlock up
	] repeat
    if DisplayAddrHi
      then ShowDisplayStream(dlispDsp, DSalone)

    WriteSwapBuf(); SwapBufVp = 0
    if (V ne 0) & (V ne -1) then resultis V

//  ↑D exit (V=0) used to worry about flushing the Bcpl stack. But reentry
//  from Lisp does this automatically since the Chord change.

    test V eq 0 
	ifso [ 
	     // disable Lisp kbd
		@displayInterrupt = @displayInterrupt & (not LispKeyMask)
	     V = ResetFXP
	     ]
	ifnot V = TeleRaidFXP
    CtxtSwitch(V)
    resultis Iresume(MkSmallPos(V))
  ]

and PrintErrorMsg (str1, param, isaddr) be
  [
   test str1 ne SubrArgsAddr
      ifso Ws(str1)
     ifnot 		// Gross hack: Raid's first arg from Lisp
	test Type(str1) eq STRINGPTRTYPE
	   ifso PrintStr(str1, true)
	  ifnot Lprint (str1, true)
   test (GetBitPos(dsp) gr (rmargBitPos rshift 1)) & LongType(param)
      ifso Ws("*N     ")	// if far to right, start new line
     ifnot Wc($*S)
   test isaddr
	ifso 		// param is explicitly an addr, don't interpret
		[ PrintPtr (param>>VA.vahi, param>>VA.valo)
		  CRLF()
		]
	ifnot test Type(param) eq STRINGPTRTYPE
		   ifso [ PrintStr(param, true); CRLF() ]
		  ifnot Lprint (param)
  ]
 
and LongType(obj) = valof		// true if obj might be long
  [ 
    let typ = Type(obj)
    resultis (typ le ATOMTYPE) % (typ eq STRINGPTRTYPE)
  ]
 
and GetCom(CT, s) = valof			// CT => command table
  [ 
    let c = UCase(ReadChar())
    [ if c eq CT!0
	then  [ Ws(" - ")			// found it
		Ws(CT!1)
		break ]
      CT=CT+2
    ] repeatwhile CT!0		// 0 entry ends the table
    resultis c
  ]
 
and GetNewNum(old, rad) = valof
  [ 
    Ws(" currently "); Wn(old, rad); Ws(" to ")
    resultis ReadNum(rad)
  ]
 
and AtLoc(s) be					// prints common msg
  [ Ws(" at"); Ws(s); Ws(" location ") ]

and StartLoc(s) be					// prints common msg
  [ Ws(" starting at"); Ws(s); Ws(" location ") ]

and ReadChar() = valof	// read character and echo it, abort on DEL
  [
    let c = Gets(keys)
    if c eq DEL then RaidReset(" XXX")
    PrintComChar (c)
    resultis c
  ]

and PrintComChar (ch) be
  [
    test (ch ge $*S) % (ch eq $*N)
	ifso Wc(ch)
	ifnot test ch eq #12
		ifso Ws ("LF")
		ifnot [ Wc($↑); Wc (ch%#100) ]
  ]

and RaidReset(errmsg; numargs na) be	// retto Raid command loop
 [
  if na gr 0 then Ws(errmsg)
  Wc($*N)
  SetLmarg(dsp, 8)	// reset margin
  MyFrame()!0 = RaidFrame
  RaidFrame!1 = doRaid - 1		// to restart command loop
 ]

and ShowHelp(CT) be			// CT => start of command table
  [ 
   Ws("*N*NRAID commands*N*N")
   [ PrintComChar(CT!0); Ws(" - "); Ws(CT!1)
     test GetBitPos(dsp) gr 300
	ifso CRLF()
	ifnot SetBitPos(dsp, 300)
     CT=CT+2
   ] repeatwhile CT!0		// 0 entry ends the table
   CRLF()
  ]