1 # streams: data structure for operating on arrays in a stateful manner
3 # A stream looks like this:
4 # write: int # index at which writes go
5 # read: int # index that we've read until
6 # data: (array byte) # prefixed by size as usual
8 # some primitives for operating on streams:
9 # - clear-stream (clears everything but the data size)
10 # - rewind-stream (resets read pointer)
12 # We need to do this in machine code because streams need to be opaque types,
13 # and we don't yet support opaque types in Mu.
16 # instruction effective address register displacement immediate
17 # . op subop mod rm32 base index scale r32
18 # . 1-3 bytes 3 bits 2 bits 3 bits 3 bits 3 bits 2 bits 2 bits 0/1/2/4 bytes 0/1/2/4 bytes
20 clear-stream: # f: (addr stream byte)
23 89/copy 3/mod/direct 5/rm32/ebp . . . 4/r32/esp . . # copy esp to ebp
28 8b/copy 1/mod/*+disp8 5/rm32/ebp . . 0/r32/eax 8/disp8 . # copy *(ebp+8) to eax
29 # var count/ecx: int = f->size
30 8b/copy 1/mod/*+disp8 0/rm32/eax . . . 1/r32/ecx 8/disp8 . # copy *(eax+8) to ecx
31 # var max/ecx: (addr byte) = &f->data[f->size]
32 8d/copy-address 1/mod/*+disp8 4/rm32/sib 0/base/eax 1/index/ecx . 1/r32/ecx 0xc/disp8 . # copy eax+ecx+12 to ecx
34 c7 0/subop/copy 0/mod/direct 0/rm32/eax . . . . . 0/imm32 # copy to *eax
36 c7 0/subop/copy 1/mod/*+disp8 0/rm32/eax . . . . 4/disp8 0/imm32 # copy to *(eax+4)
37 # - clear all stream data
38 # - this isn't strictly necessary, and it can slow things down *a lot*, but better safe than sorry.
39 # var curr/eax: (addr byte) = f->data
40 81 0/subop/add 3/mod/direct 0/rm32/eax . . . . . 0xc/imm32 # add to eax
42 # if (curr >= max) break
43 39/compare 3/mod/direct 0/rm32/eax . . . 1/r32/ecx . . # compare eax with ecx
44 73/jump-if-addr>= $clear-stream:end/disp8
46 c6 0/subop/copy-byte 0/mod/direct 0/rm32/eax . . . . . 0/imm8 # copy byte to *eax
49 eb/jump $clear-stream:loop/disp8
55 89/copy 3/mod/direct 4/rm32/esp . . . 5/r32/ebp . . # copy ebp to esp
59 rewind-stream: # f: (addr stream byte)
62 89/copy 3/mod/direct 5/rm32/ebp . . . 4/r32/esp . . # copy esp to ebp
66 8b/copy 1/mod/*+disp8 5/rm32/ebp . . 0/r32/eax 8/disp8 . # copy *(ebp+8) to eax
68 c7 0/subop/copy 1/mod/*+disp8 0/rm32/eax . . . . 4/disp8 0/imm32 # copy to *(eax+4)
73 89/copy 3/mod/direct 4/rm32/esp . . . 5/r32/ebp . . # copy ebp to esp
77 # . . vim:nowrap:textwidth=0