about summary refs log tree commit diff stats
path: root/102keyboard.subx
blob: 39788075881d091fe0aee458c31738dfe4f4d3f9 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
# check keyboard for a key
# return 0 on no keypress or unrecognized key
#
# We need to do this in machine code because Mu doesn't have global variables
# yet (for the keyboard buffer).

== code

# Most keys correspond to their ASCII/Unicode values.
# TODO: Support for international keyboards and multi-byte Unicode.
#
# However there are some exceptions that have no assigned place in Unicode
# (and with good reason):
#   0x80 = left arrow ←
#   0x81 = down arrow ↓
#   0x82 = up arrow ↑
#   0x83 = right arrow →
# These code points are not used by Unicode and their semantics are agreed to
# be context-sensitive: https://en.wikipedia.org/wiki/C0_and_C1_control_codes#C1_controls.
# Mu cannibalizes them in yet another non-standard way.
read-key:  # kbd: (addr keyboard) -> result/eax: byte
    # . prologue
    55/push-ebp
    89/<- %ebp 4/r32/esp
    # . save registers
    51/push-ecx
    # result = 0
    b8/copy-to-eax 0/imm32
    # ecx = keyboard
    8b/-> *(ebp+8) 1/r32/ecx
    81 7/subop/compare %ecx 0/imm32
    {
      75/jump-if-!= break/disp8
      # var buffer-byte-addr/ecx: (addr byte)
      8b/-> *Keyboard-buffer:read 1/r32/CL
      81 0/subop/add %ecx Keyboard-buffer:data/imm32
      # var next-key/eax: byte = *buffer-byte-addr
      8a/byte-> *ecx 0/r32/AL
      # if (next-key != 0) lock and remove from keyboard buffer
      3d/compare-eax-with 0/imm32
      {
        74/jump-if-= break/disp8
        fa/disable-interrupts
        c6 0/subop/copy-byte *ecx 0/imm8
        ff 0/subop/increment *Keyboard-buffer:read
        81 4/subop/and *Keyboard-buffer:read 0x0f/imm32
        fb/enable-interrupts
      }
      # return
      eb $read-key:end/disp8
    }
    # TODO: fake keyboard
$read-key:end:
    # . restore registers
    59/pop-to-ecx
    # . epilogue
    89/<- %esp 5/r32/ebp
    5d/pop-to-ebp
    c3/return
. # copy byte at *edx to CL # if (c == '\0') break 81 7/subop/compare 3/mod/direct 1/rm32/ecx . . . . . 0/imm32/null # compare ecx 74/jump-if-= $ascii-length:end/disp8 # ++s 42/increment-edx # ++result 40/increment-eax # loop eb/jump $ascii-length:loop/disp8 $ascii-length:end: # return eax c3/return == data # . . vim:nowrap:textwidth=0