-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy path0asm.asm
811 lines (747 loc) · 23.7 KB
/
0asm.asm
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
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
;;; # 0asm
;;;
;;; 0asm is a 16-bit x86 assembler written in 16-bit x86 assembly which fits
;;; in under 512 bytes of x86 machine code! It runs on
;;; [bootOS](https://github.com/nanochess/bootOS/).
;;;
;;; ## Assembling
;;;
;;; You can assemble and run 0asm using nasm and QEMU (respectively):
;;;
;;; make run
;;;
;;; This also copies the input file `test.asm` into the disk under the filename
;;; `0`. One can then run the assembler in bootOS:
;;;
;;; $0asm
;;;
;;; and then run the output program under `@`:
;;;
;;; $@
;;; Hello, world! $
;;;
;;; Further examples are available under the `examples/` directory.
;;;
;;; ## Anticipated FAQ
;;;
;;; ### What does it support?
;;;
;;; 0asm supports several common instructions, labels (absolute and relative
;;; relocations), and octal literals (but only octal literals). It also
;;; supports all 16-bit x86 general purpose registers, but does not support
;;; segment registers.
;;;
;;; Many jump instructions are supported (call, jmp, jb, jnb, jz, jnz, jbe,
;;; jnbe), along with several string instructions (stosb, stosw, lodsb, lodsw,
;;; movsw, cbw, scasw), several stack instructions (push, pop, ret), a few
;;; fundamental arithmetic instructions in both register/register and
;;; register/immediate form (add, or, adc, and, xor, cmp, mov), and some misc
;;; instructions (int, stc, inc, dec, the pseudo-instruction db).
;;;
;;; Errors are not always handled, but the assembler does usually exit cleanly
;;; instead of producing garbage.
;;;
;;; ### How does it work?
;;;
;;; 0asm is a simple 2-pass assembler. The first pass outputs the machine
;;; code for instructions which do not require any relocation; it also collects
;;; labels into a "symbol table", and the addresses of immediates which need
;;; to be fixed into a "fixup table". The first pass also outputs the machine
;;; code for instructions which do not require any relocation.
;;;
;;; ### Is it self-hosting?
;;;
;;; Unfortunately not yet, for several reasons.
;;;
;;; 1. The underlying bootOS only supports 512 byte files.
;;; 2. The instruction encoding produced is not optimal, so it will not fit
;;; in 512 bytes once assembled.
;;; 3. Shift and memory-addressing instructions are not yet supported.
;;;
;;; These problems are not insurmountable, although it seems difficult.
;;; We could easily move the goalpost by typing the entire program using db,
;;; but of course that would be no fun.
;;;
;;; ## Development
;;;
;;; The code is decently commented. Many of the files in this repo are
;;; generated from `0asm.asm`, including the README. You can also debug by
;;; running `make debug` (to start QEMU in one pane) and `make gdbdebug`
;;; (to start GDB along with some helpful default scripts).
;;;
;;; ## License
;;;
;;; Copyright (c) 2019 Keyhan Vakil
;;;
;;; This program is free software: you can redistribute it and/or modify
;;; it under the terms of the GNU General Public License as published by
;;; the Free Software Foundation, either version 3 of the License, or
;;; (at your option) any later version.
;;;
;;; This program is distributed in the hope that it will be useful,
;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;;; GNU General Public License for more details.
;;;
;;; You should have received a copy of the GNU General Public License
;;; along with this program. If not, see <https://www.gnu.org/licenses/>.
;; Vocabulary
;; hash: a formula which maps identifiers to a 16-bit value (for shorter, but
;; lossy, string comparisons). See the hash procedure for its computation.
;; short table: an associative array of (key, value) pairs. Keys are typically
;; outputs of the hash function, while values are any 16-bit values.
;; long table: a short table with 16-bit values. Long tables are ONLY used for
;; the label fixup, otherwise we use short tables because they are more size
;; efficient.
;; clobbered: indicates that a register is changed by a procedure in some
;; (typically unspecified) fashion.
;; consumed: as we read characters from our input buffer, we "consume"
;; them. We generally keep the invariant that al is the next character in
;; the buffer, and si points to the rest of the buffer. Note that consuming
;; is different than clobbering, as consuming means that we maintain this
;; invariant!
;; ";!": this comment symbol indicates that some code or functionality is
;; easily added, but was removed so that the program fits in 512 bytes.
;; As more features are implemented, these comments are removed.
cpu 186
bits 16
fixup_table: equ 0x4000
symbol_table: equ 0x5000
register_table: equ 0x6000
input_start: equ 0x6b00
output_start: equ 0x6d00
int_exit: equ 0x20
int_read_file: equ 0x23
int_save_file: equ 0x24
; Macro which hashes the given string, and stores the result in the hash_result
; variable. This MUST be kept in sync with the implementation of the runtime
; hash function below.
%macro hash_s 1.nolist
%assign hash_result 0x23
%strlen %%len %1
%assign %%i 1
%rep %%len
%substr %%ch %1 %%i
%assign hash_result (31 * hash_result + %%ch)
%assign hash_result (hash_result & 0xffff)
%assign %%i (%%i+1)
%endrep
%endmacro
%macro dw_hash 1.nolist
hash_s %1
dw hash_result
%endmacro
global _start
_start:
;; Zero out fixup, symbol table, and the stack. Many of the functions below
;; (like lookup and add_table) assume that tables are zero-terminated, and
;; this initialization ensures that this is the case.
initialize_tables:
; This zeros out 0x4000 to 0x6f00.
; 0x4000 is the address of the fixup table.
; 0x5000 is the address of the symbol table.
; 0x6f00 is the start of the stack.
mov di,fixup_table
; We set only ch here, which saves us a byte. This means that the lower
; cl might be garbage, but this is OK -- the maximum value is 0x2fff,
; which would have us write up to 0x7000, right below bootOS. The minimum
; value is 0x2f00, which still zeros out everything we need.
;
; In any case, error-checking is definitely not the priority of this
; assembler...
mov ch,0x2f
xor ax,ax
rep stosb
;; Unlike other tables, the register table is generated at runtime, since the
;; lookup values are sequential, we can save bytes by only storing the keys.
initialize_register_table:
; It's shorter to do this manually than to use add_table.
mov si,partial_register_table
mov di,register_table
; ax is zero from above.
.initialize_register_table_loop:
; Copy over partial register table key.
movsw
; Store the index of this register.
stosb
inc ax
; If al == 16, then we've finished copying partial_register_table.
; From initialize_tables above, we know that this table was zero.
; It is important that we don't overwrite this final zero entry, since
; we are using it as a sentinel value.
cmp al,0x10
jne .initialize_register_table_loop
.initialize_register_table_end:
;; Initialize si to point to the input buffer and di to point to the output
;; buffer. Read in the input file into the input buffer.
init:
; Start stack pointer right below input buffer. This leaves space for
; 0x200 input bytes, which is the size of files in bootOS. The output
; buffer will start at 0x6d00.
mov sp,input_start
mov di,sp
; Read in the file.
mov bx,infile
int int_read_file
jc done_error
; All registers except sp could be clobbered at this point.
mov si,sp
; Give space for 0x200 output bytes, until 0x6f00.
mov di,output_start
;; Keep reading instructions.
read_instruction_loop:
; This lets us use "ret" when we successfully parse an instruction, which
; ends up saving many bytes over "jmp read_instruction", because we need
; to jump back so often.
call read_instruction
jmp read_instruction_loop
;; Read and assemble a single instruction.
read_instruction:
lodsb
;; This block will be executed when we are finished parsing the entire file.
done:
; The input file is NUL terminated. If this is not a NUL, then we need to
; keep parsing the file.
and al,al
jnz done_end
.fixup_labels:
; Set lookup to "long mode" by modifying lodsb -> lodsw.
; This allows us to lookup into long tables like the symbol table.
inc byte [lookup_lodsb]
mov si,fixup_table
.fixup_labels_loop:
; Load key from fixup table.
lodsw
and ax,ax
jz .fixup_labels_done
; Lookup key in symbol table.
mov bp,si
mov si,symbol_table
; Basically mov cx,ax, but shorter and ax is getting clobbered anyway.
xchg ax,cx
call lookup
; Error: could not find a label for this address we need to fix.
jnc done_error
; Currently there is a "fixup hint" (*i) stored at the address of the
; immediate we are fixing (i), and the label address is d. This code sets
; *i = *i + d - i. This allows us to support both relative and absolute
; relocations, by choosing the fixup hint appropriately. (See jmp_and_call
; as well as parse_2x.parse_2x_label for relative and absolute relocations
; respectively.)
; Store the label address.
push ax
; Get the location to fixup.
lodsw
mov di,ax
; *i -= i
sub [di],ax
pop ax
; *i += d
add [di],ax
jmp .fixup_labels_loop
.fixup_labels_done:
; Original start of output buffer.
mov di,output_start
; Output to file.
mov bx,outfile
int int_save_file
done_error:
int int_exit
done_end:
; Skip space or space-ish character
cmp al,' '
jbe read_instruction
;; Parse comment by consuming until we see a newline.
read_comment:
cmp al,';'
jnz .read_comment_end
.read_comment_loop:
lodsb
cmp al,`\n`
jnz .read_comment_loop
ret
.read_comment_end:
call hash
; At this point, al equals the first non-identifier character,
; and si points to right after al.
;
; Some examples follow. The position of al is marked a, the position
; of si is marked s. $ is used to indicate a newline.
;
; mov ax,cx$
; as
;
; hello:$
; as
;
; ret$
; as
;; If the next character in the buffer is a colon, this adds the label to our
;; symbol table.
add_to_label:
cmp al,':'
jne add_to_label_end
; SELF-MODIFYING.
symbol_table_addr:
mov ax,symbol_table
; Load address of the immediate in the above instruction
mov bx,symbol_table_addr+1
; Inlining the tail call to add_table here saves us a two bytes of a jmp.
;; Store (cx, di) at ax, and increment bx to point to the new end.
;; Inputs:
;; ax is the address of the end of the table.
;; bx is the address of the immediate in the instruction that loaded ax.
;; (Note: this code is self-modifying.)
;; cx is the key to write.
;; di is the value to write.
;; Outputs:
;; ax is clobbered to the initial value of di.
;; *bx is incremented by 4 to point to the new end.
;; cx is clobbered.
;; bx, di are preserved.
add_table:
; Set ax = old cx, di = old ax, cx = old di.
xchg ax,di
xchg ax,cx
; Store old di at old ax.
stosw
; Store old cx at old ax+2.
; This xchg could be mov ax,cx, but it's shorter this way.
xchg ax,cx
stosw
; Update the end of the table.
mov [bx],di
mov di,ax
ret
add_to_label_end:
; db
parse_db:
hash_s 'db'
cmp cx,hash_result
je store_odigit_byte
.parse_db_end:
; int
parse_int:
hash_s 'int'
cmp cx,hash_result
jne parse_int_end
mov al,0xcd
stosb
store_odigit_byte:
call odigit
; This "jump chain" allows us to jump to a single statement (done_error), from
; multiple places, without requiring long jumps (signed displacement greater
; than one byte). Each individual jump in the chain is within 128 bytes of the
; previous jump.
done_error_chain_0:
jnc done_error
stosb
ret
parse_int_end:
; All calls to lookup expect a saved value of si in bp, which is restored
; on success.
mov bp,si
;; Deals with jmp and call instructions, encoded as:
;; - opcode (1 or 2 bytes) - from the table.
;; - immediate (2 bytes) - needs to be relocated by linker.
jmp_and_call:
mov si,jmp_and_call_table
call lookup
jnc jmp_and_call_end
.jmp_and_call_match:
cmp al,0x90
jnb .single_byte
; If this is greater than 0x90, it's an extended instruction.
; The first byte should be 0xf, and the next byte is obtained from
; the table.
mov byte [di],0xf
inc di
.single_byte:
stosb
; Hash the label and add it to the fixup table.
call add_fixup
; Add a fixup hint of -2. (This effectively creates a relative relocation,
; see done.fixup_labels for details.)
mov ax,0xfffe
stosw
ret
add_fixup:
call hash_pre
; SELF-MODIFYING.
fixup_table_addr:
mov ax,fixup_table
; Load address of the immediate in the above instruction
mov bx,fixup_table_addr+1
jmp add_table
jmp_and_call_end:
;; Deals with single-byte no argument instructions, encoded as just the opcode.
parse_10:
; Note we don't need to set si or bp here.
; If the lookup above fails, then si points to one after jmp_and_call_table
; (i.e., table10), and bp is still correctly saved.
call lookup
jc stosb_ret
.parse_10_end:
;; Deals with single-byte single-register instructions, encoded as just the
;; opcode plus a register number.
parse_11:
call lookup
jnc parse_11_end
push ax
call accept_register
done_error_chain_1:
jnc done_error_chain_0
pop dx
add ax,dx
stosb_ret:
stosb
ret
parse_11_end:
;; Deals with instructions which take two arguments: either a
;; register-register, or a register-immediate. This also handles the case of
;; register-label (which is encoded as an absolute relocation of a
;; register-immediate).
;;
;; This is the trickiest case, consult an Intel manual for details.
parse_2x:
call lookup
done_error_chain_2:
jnc done_error_chain_1
; Save the opcode.
push ax
; The first argument MUST be a register, regardless.
call accept_register
; (Note accept_register clobbers the comma in ax.)
done_error_chain_3:
jnc done_error_chain_2
; Restore opcode in dx.
pop dx
; Check if this is a 16-bit register or an 8-bit register.
; bl will be 1 if we should use a 16-bit immediate, and 0 if we should use
; an 8-bit immediate. It also encodes if this is operating on an 8-bit or
; 16-bit register. Technically x86 makes a distinction between these two,
; but we treat them the same (leading to longer but still valid instruction
; encodings).
; Sets carry flag if al < 0x8.
cmp al,0x8
sbb bx,bx
inc bx
and al,0x7
; Set LSB of opcode byte correctly.
add dl,bl
; Try to get another register.
push ax
push si
call accept_register
; (Note accept_register clobbers the newline in ax.)
; If it's not a register, it must be an immediate or a label.
jnc .parse_2x_immediate
; In this case, we are parsing a register-register instruction.
; We store this as:
; - opcode (1 byte)
; - Mod R/M byte (1 byte)
; The parse was successful, so we need to clean the old si from the stack.
; We don't use cx from now on, so pop into it.
pop cx
; Compute the Mod R/M byte:
; 11 src[0:3] dst[0:3]
; where src and dst are the register numbers.
; Right now ax=src, and dst is on the top of stack.
; Shifting up by three here can overflow the second MSB, but it's
; overwritten below anyway.
shl al,0x3
; Note that we always set the top two bits, as we only support register
; addressing here.
or al,0xc0
; Get the destination register off the stack.
pop cx
or cl,al
; Mod R/M byte
mov dh,cl
; Opcode byte in dl
; Basically mov ax,dx, but dx doesn't matter.
xchg ax,dx
stosw
ret
.parse_2x_immediate:
; Backtrack to point to the immediate / label.
pop si
; Check if this is a MOV instruction.
cmp dl,0x88
; Opcode byte for all group 1 immediate instructions
mov al,0x80
je .parse_group1_immediate
; Opcode byte for MOV with immediate
mov al,0xc6
; In this case it's a MOV instruction, we kill the old top bits of the
; opcode. This effectively allows us to reuse the opcode byte for Group 1
; instructions, and makes the MOV instruction a "raw" encoding which uses
; an opcode byte of 0. It also maintains whether this instruction is
; supposed to operate on 8-bit or 16-bit registers.
mov dl,bl
.parse_group1_immediate:
; In the immediate case, the opcode byte of the register-register
; instruction is USUALLY reusable as the Mod R/M byte. (The exception
; is the case when we have a MOV instruction.) First, we can remove the
; LSB for the opcode byte (which indicates if the instruction operates
; on 8-bit or 16-bit registers). The middle three bits of this opcode
; byte select the correct operation for all Group 1 instructions.
add al,bl
stosb
; Construct the Mod R/M byte using the old register.
pop ax
add al,dl
; Clear off the last bit of the opcode.
sub al,bl
; Set the top two bits of Mod R/M to indicate registers.
or al,0xc0
stosb
; Parse off the octal number.
push si
call odigit
; If it's not an octal number, it must be a register.
jnc .parse_2x_label
; The parse was successful, so we need to clean the old si from the stack.
; We don't use cx from now on, so pop into it.
pop cx
.parse_2x_append_immediate:
; Blindly store both bytes of the immediate.
stosw
; If bx = 0, we want to decrement di, so that we effectively store only
; one byte of the immediate.
; If bx = 1, then we don't want to change di.
; Because we have bx available, this is shorter than the similar code in
; jmp_and_call.jmp_and_call_match.
add di,bx
dec di
ret
.parse_2x_label:
; Backtrack.
pop si
call add_fixup
; We want to add a value to the relocation such that, when di is added by
; the fixup, it will correspond to the absolute address of the loaded
; label. This is the start of the output buffer (output_start), minus the
; actual location the binary will start (0x7c00).
add ah,0xf
stosw
ret
;; Accept a register pointed to by the buffer.
;;
;; Inputs:
;; al is the first character in the buffer.
;; si is a buffer pointing to the input after al.
;; Outputs:
;; ax is DESTROYED. It now indicates the register number and does not point
;; to the buffer, therefore ignoring the character after this register.
;; bx is clobbered.
;; bp is clobbered.
;; si is consumed.
accept_register:
call hash_pre
mov bp,si
mov si,register_table
; FALLTHROUGH to lookup (saves us two bytes)
;; Lookup the key in the given SHORT table, returning the value.
;;
;; Inputs:
;; cx is the key to lookup.
;; si points to the first key of the table.
;; bp is the old value of si.
;; Outputs:
;; al is the returned value, or 0 on failure.
;; si is modified.
;; if the lookup is successful, it is set to bp.
;; if the lookup fails, it points one word after the end of the table.
;; carry flag is set iff the lookup was successful.
lookup:
lodsw
and ax,ax
jz lookup_not_found
lookup_keep_going:
cmp ax,cx
lookup_lodsb:
; SELF-MODIFYING.
; This can be changed to a lodsw to make lookup use long tables.
lodsb
jne lookup
lookup_done:
stc
mov si,bp
lookup_not_found:
ret
;; Compute a hash for table indexing. Stops at the first non-identifier
;; character after al, consuming all characters and incrementing si. This MUST
;; be kept in sync with the hash_s macro above.
;;
;; Inputs:
;; al is the first character in the buffer.
;; si is a buffer pointing to the input after al.
;; Outputs:
;; al is consumed.
;; si is consumed.
;; cx is the returned hash value.
hash_pre:
lodsb
hash:
mov cx,0x23
.hash_loop:
; Using cbw here lets us use ax as al, which we need (because we want to
; use the full 16-bit for the hash to reduce collisions).
cbw
; cx = 31 * cx + (next character)
sub ax,cx
shl cx,0x5
add cx,ax
lodsb
; Stop if we see a non-identifier character.
cmp al,'@'
jg .hash_loop
ret
;; Convert ASCII octal.
;; The representation must start with "0o".
;; Inputs:
;; al is the first character in the buffer.
;; si is a buffer pointing to the input after al.
;; Outputs:
;; ax is the returned value. It is clobbered if the parse fails.
;; cx is clobbered.
;; si is consumed.
;; the carry flag is set if the parse is successful.
odigit:
lodsb
cmp al,'0'
jne .odigit_bad
;! Error checking on o removed for byte count.
lodsb
xor cx,cx
.odigit_loop:
lodsb
; Convert by subtracting off ASCII 0.
sub al,'0'
; Note if al < '0', then CF is set to 1. This indicates we are now dealing
; with a whitespace character, and the parse was successful.
;! We don't check that the value is not too large.
jb .odigit_good
shl cx,0x3
; Note that the bottom three bits of cl are zero and al is at most 7,
; so adding only the lower bytes here is fine; we never overflow cl.
add cl,al
jmp .odigit_loop
.odigit_bad:
; Clear the carry flag to show parse was bad.
clc
.odigit_good:
; Basically mov ax,cx, but cx is getting clobbered.
xchg ax,cx
ret
;; THESE TABLES MUST APPEAR IN THIS ORDER. ;;
;; Lookup table for jump and call instructions (anything requiring relative
;; relocation with 16-bit addresses). Keys are the hashes, and values are the
;; opcode.
jmp_and_call_table:
dw_hash 'call'
db 0xe8
dw_hash 'jmp'
db 0xe9
dw_hash 'jb'
db 0x82
dw_hash 'jnb'
db 0x83
dw_hash 'jz'
db 0x84
dw_hash 'jnz'
db 0x85
dw_hash 'jbe'
db 0x86
dw_hash 'jnbe'
db 0x87
; NOT FOUND
dw 0x0
;; Lookup table for instructions which take up one byte and have no arguments.
;; Keys here are the hashes, values are simply the opcode.
table10:
dw_hash 'cbw'
db 0x98
dw_hash 'movsw'
db 0xa4
dw_hash 'stosb'
db 0xaa
dw_hash 'stosw'
db 0xab
dw_hash 'lodsb'
db 0xac
dw_hash 'lodsw'
db 0xad
dw_hash 'scasw'
db 0xaf
dw_hash 'ret'
db 0xc3
dw_hash 'stc'
db 0xf9
; NOT FOUND
dw 0x0
;; Lookup table for instructions with one opcode byte and one argument.
;; Keys here are the hashes. Values are the opcode which will be added to
;; (note that because we start numbering 16-bit registers at 0x8, most of
;; the values here are shifted down by 0x8).
table11:
dw_hash 'push'
db 0x48
dw_hash 'pop'
db 0x50
dw_hash 'inc'
db 0x38
dw_hash 'dec'
; Nasty trick here. We overlap outfile and infile with opcode bytes.
; This makes our input filename "0" and our output filename "@", and
; saves us three bytes.
outfile:
db 0x40
; NOT FOUND
dw 0x0
;; Lookup table for instructions which take two arguments. Keys here are the
;; hashes. Values are the opcode byte for non-immediate (register/register)
;; forms.
table2x:
dw_hash 'add'
db 0x00
dw_hash 'or'
db 0x08
dw_hash 'adc'
db 0x10
dw_hash 'and'
db 0x20
dw_hash 'cmp'
db 0x38
dw_hash 'mov'
db 0x88
dw_hash 'xor'
infile:
db 0x30
; NOT FOUND
dw 0x0
;; The "partial" register table contains only keys -- no values. The actual
;; register table is initialized using this and initialize_register_table.
partial_register_table:
dw_hash 'al'
dw_hash 'cl'
dw_hash 'dl'
dw_hash 'bl'
dw_hash 'ah'
dw_hash 'ch'
dw_hash 'dh'
dw_hash 'bh'
dw_hash 'ax'
dw_hash 'cx'
dw_hash 'dx'
dw_hash 'bx'
dw_hash 'sp'
dw_hash 'bp'
dw_hash 'si'
dw_hash 'di'
; NOT FOUND not required.
bytes_left_message:
%assign bytes_left 0x200-($-$$)
%warning Have bytes_left bytes left.
times bytes_left db 0x90