type environment { screen: (handle screen) program: (handle program) cursor-word: (handle word) nrows: int ncols: int code-separator-col: int } fn initialize-environment _env: (addr environment) { var env/esi: (addr environment) <- copy _env var program-ah/eax: (addr handle program) <- get env, program allocate program-ah var program/eax: (addr program) <- lookup *program-ah var cursor-word-ah/ecx: (addr handle word) <- get env, cursor-word initialize-program program, cursor-word-ah # initialize screen var screen-ah/eax: (addr handle screen) <- get env, screen var _screen/eax: (addr screen) <- lookup *screen-ah var screen/edi: (addr screen) <- copy _screen var nrows/eax: int <- copy 0 var ncols/ecx: int <- copy 0 nrows, ncols <- screen-size screen var dest/edx: (addr int) <- get env, nrows copy-to *dest, nrows dest <- get env, ncols copy-to *dest, ncols var repl-col/ecx: int <- copy ncols repl-col <- shift-right 1 dest <- get env, code-separator-col copy-to *dest, repl-col } fn draw-screen _env: (addr environment) { var env/esi: (addr environment) <- copy _env var screen-ah/eax: (addr handle screen) <- get env, screen var _screen/eax: (addr screen) <- lookup *screen-ah var screen/edi: (addr screen) <- copy _screen var dest/edx: (addr int) <- get env, code-separator-col var tmp/eax: int <- copy *dest clear-canvas env tmp <- add 2 # repl-margin-left move-cursor screen, 3, tmp # input-row } fn initialize-environment-with-fake-screen _self: (addr environment), nrows: int, ncols: int { var self/esi: (addr environment) <- copy _self var screen-ah/eax: (addr handle screen) <- get self, screen allocate screen-ah var screen-addr/eax: (addr screen) <- lookup *screen-ah initialize-screen screen-addr, nrows, ncols initialize-environment self } fn process _self: (addr environment), key: grapheme { $process:body: { var self/esi: (addr environment) <- copy _self compare key, 0x445b1b # left-arrow { break-if-!= var cursor-word-ah/edi: (addr handle word) <- get self, cursor-word var _cursor-word/eax: (addr word) <- lookup *cursor-word-ah var cursor-word/ecx: (addr word) <- copy _cursor-word # if not at start, move left within current word var at-start?/eax: boolean <- cursor-at-start? cursor-word compare at-start?, 0 # false { break-if-= cursor-left cursor-word break $process:body } # otherwise, move to end of prev word var prev-word-ah/esi: (addr handle word) <- get cursor-word, prev var prev-word/eax: (addr word) <- lookup *prev-word-ah { compare prev-word, 0 break-if-= copy-object prev-word-ah, cursor-word-ah cursor-to-end prev-word } break $process:body } compare key, 0x435b1b # right-arrow { break-if-!= var cursor-word-ah/edi: (addr handle word) <- get self, cursor-word var _cursor-word/eax: (addr word) <- lookup *cursor-word-ah var cursor-word/ecx: (addr word) <- copy _cursor-word # if not at end, move right within current word var at-end?/eax: boolean <- cursor-at-end? cursor-word compare at-end?, 0 # false { break-if-= cursor-right cursor-word break $process:body } # otherwise, move to start of next word var next-word-ah/esi: (addr handle word) <- get cursor-word, next var next-word/eax: (addr word) <- lookup *next-word-ah { compare next-word, 0 break-if-= copy-object next-word-ah, cursor-word-ah cursor-to-start next-word } break $process:body } compare key, 0x7f # del (backspace on Macs) { break-if-!= var cursor-word-ah/edi: (addr handle word) <- get self, cursor-word var _cursor-word/eax: (addr word) <- lookup *cursor-word-ah var cursor-word/ecx: (addr word) <- copy _cursor-word # if not at start of some word, delete grapheme before cursor within current word var at-start?/eax: boolean <- cursor-at-start? cursor-word compare at-start?, 0 # false { break-if-= delete-before-cursor cursor-word break $process:body } # otherwise delete current word and move to end of prev word var prev-word-ah/esi: (addr handle word) <- get cursor-word, prev var prev-word/eax: (addr word) <- lookup *prev-word-ah { compare prev-word, 0 break-if-= copy-object prev-word-ah, cursor-word-ah cursor-to-end prev-word delete-next prev-word } break $process:body } compare key, 0x20 # space { break-if-!= # insert new word var cursor-word-ah/edx: (addr handle word) <- get self, cursor-word append-word cursor-word-ah var cursor-word/eax: (addr word) <- lookup *cursor-word-ah var next-word-ah/ecx: (addr handle word) <- get cursor-word, next copy-object next-word-ah, cursor-word-ah break $process:body } compare key, 0xa # enter { break-if-!= # toggle display of subsidiary stack var cursor-word-ah/edx: (addr handle word) <- get self, cursor-word var cursor-word/eax: (addr word) <- lookup *cursor-word-ah var display-subsidiary-stack?/eax: (addr boolean) <- get cursor-word, display-subsidiary-stack? var tmp/ecx: int <- copy 1 tmp <- subtract *display-subsidiary-stack? copy-to *display-subsidiary-stack?, tmp break $process:body } # otherwise insert key within current word var g/edx: grapheme <- copy key var print?/eax: boolean <- real-grapheme? key { compare print?, 0 # false break-if-= var cursor-word-ah/eax: (addr handle word) <- get self, cursor-word var cursor-word/eax: (addr word) <- lookup *cursor-word-ah add-grapheme-to-word cursor-word, g break $process:body } # silently ignore other hotkeys } } fn evaluate-environment _env: (addr environment), stack: (addr value-stack) { var env/esi: (addr environment) <- copy _env # program var program-ah/eax: (addr handle program) <- get env, program var _program/eax: (addr program) <- lookup *program-ah var program/esi: (addr program) <- copy _program # defs var defs/edx: (addr handle function) <- get program, defs # line var sandbox-ah/esi: (addr handle sandbox) <- get program, sandboxes var sandbox/eax: (addr sandbox) <- lookup *sandbox-ah var line-ah/eax: (addr handle line) <- get sandbox, data var _line/eax: (addr line) <- lookup *line-ah var line/esi: (addr line) <- copy _line evaluate defs, 0, line, 0, stack } fn render _env: (addr environment) { var env/esi: (addr environment) <- copy _env clear-canvas env var screen-ah/edi: (addr handle screen) <- get env, screen var _screen/eax: (addr screen) <- lookup *screen-ah var screen/edi: (addr screen) <- copy _screen var _repl-col/ecx: (addr int) <- get env, code-separator-col var repl-col/ecx: int <- copy *_repl-col repl-col <- add 2 # repl-margin-left # cursor-word var cursor-word-ah/ebx: (addr handle word) <- get env, cursor-word var _cursor-word/eax: (addr word) <- lookup *cursor-word-ah var cursor-word/ebx: (addr word) <- copy _cursor-word # program var program-ah/eax: (addr handle program) <- get env, program var _program/eax: (addr program) <- lookup *program-ah var program/esi: (addr program) <- copy _program # defs var defs/edx: (addr handle function) <- get program, defs # line var sandbox-ah/esi: (addr handle sandbox) <- get program, sandboxes var sandbox/eax: (addr sandbox) <- lookup *sandbox-ah var line-ah/eax: (addr handle line) <- get sandbox, data var _line/eax: (addr line) <- lookup *line-ah var line/esi: (addr line) <- copy _line # cursor-col var cursor-col: int var cursor-col-a/eax: (addr int) <- address cursor-col # var dummy/ecx: int <- render-line screen, defs, 0, line, 3, repl-col, cursor-word, cursor-col-a # input-row=3 move-cursor screen, 3, cursor-col # input-row } fn render-line screen: (addr screen), defs: (addr handle function), bindings: (addr table), _line: (addr line), top-row: int, left-col: int, cursor-word: (addr word), cursor-col-a: (addr int) -> right-col/ecx: int { # curr-word var line/esi: (addr line) <- copy _line var first-word-ah/eax: (addr handle word) <- get line, data var curr-word/eax: (addr word) <- lookup *first-word-ah # loop-carried dependency var curr-col/ecx: int <- copy left-col # { compare curr-word, 0 break-if-= # if necessary, first render columns for subsidiary stack $render-line:subsidiary: { { var display-subsidiary-stack?/eax: (addr boolean) <- get curr-word, display-subsidiary-stack? compare *display-subsidiary-stack?, 0 # false break-if-= $render-line:subsidiary } # does function exist? var callee/edi: (addr function) <- copy 0 { var curr-stream-storage: (stream byte 0x10) var curr-stream/esi: (addr stream byte) <- address curr-stream-storage emit-word curr-word, curr-stream var callee-h: (handle function) var callee-ah/eax: (addr handle function) <- address callee-h find-function defs, curr-stream, callee-ah var _callee/eax: (addr function) <- lookup *callee-ah callee <- copy _callee compare callee, 0 break-if-= $render-line:subsidiary } move-cursor screen, top-row, curr-col print-word screen, curr-word { var word-len/eax: int <- word-length curr-word curr-col <- add word-len curr-col <- add 2 add-to top-row, 1 } # obtain stack at call site var stack-storage: value-stack var stack/edx: (addr value-stack) <- address stack-storage initialize-value-stack stack, 0x10 { var prev-word-ah/eax: (addr handle word) <- get curr-word, prev var prev-word/eax: (addr word) <- lookup *prev-word-ah compare prev-word, 0 break-if-= evaluate defs, bindings, line, prev-word, stack } # construct new bindings var callee-bindings-storage: table var callee-bindings/esi: (addr table) <- address callee-bindings-storage initialize-table callee-bindings, 0x10 bind-args callee, stack, callee-bindings # obtain body var callee-body-ah/eax: (addr handle line) <- get callee, body var callee-body/eax: (addr line) <- lookup *callee-body-ah # - render subsidiary stack curr-col <- render-line screen, defs, callee-bindings, callee-body, top-row, curr-col, cursor-word, cursor-col-a # move-cursor screen, top-row, curr-col print-code-point screen, 0x21d7 # ⇗ # curr-col <- add 2 subtract-from top-row, 1 } # now render main column curr-col <- render-column screen, defs, bindings, line, curr-word, top-row, curr-col, cursor-word, cursor-col-a var next-word-ah/edx: (addr handle word) <- get curr-word, next curr-word <- lookup *next-word-ah loop } right-col <- copy curr-col } # Render: # - starting at top-row, left-col: final-word # - starting somewhere below at left-col: the stack result from interpreting first-world to final-word (inclusive) # unless final-word is truly the final word, in which case it might be incomplete # # Outputs: # - Return the farthest column written. # - If final-word is same as cursor-word, do some additional computation to set # cursor-col-a. fn render-column screen: (addr screen), defs: (addr handle function), bindings: (addr table), scratch: (addr line), final-word: (addr word), top-row: int, left-col: int, cursor-word: (addr word), cursor-col-a: (addr int) -> right-col/ecx: int { var max-width/ecx: int <- copy 0 { # indent stack var indented-col/ebx: int <- copy left-col indented-col <- add 1 # margin-right - 2 for padding spaces # compute stack var stack: value-stack var stack-addr/edi: (addr value-stack) <- address stack initialize-value-stack stack-addr, 0x10 # max-words evaluate defs, bindings, scratch, final-word, stack-addr # render stack var curr-row/edx: int <- copy top-row curr-row <- add 3 # stack-margin-top var _max-width/eax: int <- value-stack-max-width stack-addr var max-width/esi: int <- copy _max-width var i/eax: int <- value-stack-length stack-addr { compare i, 0 break-if-<= move-cursor screen, curr-row, indented-col { var val/eax: int <- pop-int-from-value-stack stack-addr render-integer screen, val, max-width var size/eax: int <- decimal-size val compare size, max-width break-if-<= max-width <- copy size } curr-row <- increment i <- decrement loop } } # render word, initialize result reset-formatting screen move-cursor screen, top-row, left-col print-word screen, final-word { var size/eax: int <- word-length final-word compare size, max-width break-if-<= max-width <- copy size } # update cursor { var f/eax: (addr word) <- copy final-word compare f, cursor-word break-if-!= var cursor-index/eax: int <- cursor-index cursor-word cursor-index <- add left-col var dest/edi: (addr int) <- copy cursor-col-a copy-to *dest, cursor-index } # post-process right-col right-col <- copy max-width right-col <- add left-col right-col <- add 3 # margin-right } # synaesthesia fn render-integer screen: (addr screen), val: int, max-width: int { var bg/eax: int <- hash-color val var fg/ecx: int <- copy 7 { compare bg, 2 break-if-!= fg <- copy 0 } { compare bg, 3 break-if-!= fg <- copy 0 } { compare bg, 6 break-if-!= fg <- copy 0 } start-color screen, fg, bg print-grapheme screen, 0x20 # space print-int32-decimal-right-justified screen, val, max-width print-grapheme screen, 0x20 # space } fn hash-color val: int -> result/eax: int { result <- try-modulo val, 7 # assumes that 7 is always the background color } fn clear-canvas _env: (addr environment) { var env/esi: (addr environment) <- copy _env var screen-ah/edi: (addr handle screen) <- get env, screen var _screen/eax: (addr screen) <- lookup *screen-ah var screen/edi: (addr screen) <- copy _screen clear-screen screen var nrows/eax: (addr int) <- get env, nrows var _repl-col/ecx: (addr int) <- get env, code-separator-col var repl-col/ecx: int <- copy *_repl-col draw-vertical-line screen, 1, *nrows, repl-col move-cursor screen, 3, 2 print-string screen, "x 2* = x 2 *" move-cursor screen, 4, 2 print-string screen, "x 1+ = x 1 +" move-cursor screen, 5, 2 print-string screen, "x 2+ = x 1+ 1+" } fn real-grapheme? g: grapheme -> result/eax: boolean { $real-grapheme?:body: { # if g == newline return true compare g, 0xa { break-if-!= result <- copy 1 # true break $real-grapheme?:body } # if g == tab return true compare g, 9 { break-if-!= result <- copy 1 # true break $real-grapheme?:body } # if g < 32 return false compare g, 0x20 { break-if->= result <- copy 0 # false break $real-grapheme?:body } # if g <= 255 return true compare g, 0xff { break-if-> result <- copy 1 # true break $real-grapheme?:body } # if (g&0xff == Esc) it's an escape sequence and-with g, 0xff compare g, 0x1b # Esc { break-if-!= result <- copy 0 # false break $real-grapheme?:body } # otherwise return true result <- copy 1 # true } }