-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathlocation.go
592 lines (536 loc) · 9.9 KB
/
location.go
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
package zog
import (
"fmt"
"strings"
)
type Loc8 interface {
Read8(z *Zog) (byte, error)
Write8(z *Zog, n byte) error
String() string
}
type Loc16 interface {
Read16(z *Zog) (uint16, error)
Write16(z *Zog, nn uint16) error
String() string
}
// (HL), (BC), (SP) could refer to a byte addr or a word addr
type Loc interface {
Read8(z *Zog) (byte, error)
Write8(z *Zog, n byte) error
Read16(z *Zog) (uint16, error)
Write16(z *Zog, nn uint16) error
String() string
}
type Registers struct {
A, F byte
B, C byte
D, E byte
H, L byte
IXH, IXL byte
IYH, IYL byte
I, R byte
// Alternate register set
A_PRIME, F_PRIME byte
B_PRIME, C_PRIME byte
D_PRIME, E_PRIME byte
H_PRIME, L_PRIME byte
SP uint16
PC uint16
}
func (r Registers) Summary() string {
return fmt.Sprintf("AF %04X BC %04X DE %04X HL %04X SP %04X IX %04X IY %04X",
r.Read16(AF),
r.Read16(BC),
r.Read16(DE),
r.Read16(HL),
r.Read16(SP),
r.Read16(IX),
r.Read16(IY))
}
func (r Registers) String() string {
return fmt.Sprintf("AF %04X BC %04X DE %04X HL %04X AF' %04X BC' %04X DE' %04X HL' %04X SP %04X PC %04X IX %04X IY %04X",
r.Read16(AF),
r.Read16(BC),
r.Read16(DE),
r.Read16(HL),
r.Read16(AF_PRIME),
r.Read16(BC_PRIME),
r.Read16(DE_PRIME),
r.Read16(HL_PRIME),
r.Read16(SP),
r.Read16(PC),
r.Read16(IX),
r.Read16(IY))
}
type R8 int
const (
A R8 = iota
F
B
C
D
E
H
L
IXL
IXH
IYL
IYH
I
R
)
func (r *Registers) Read8(l R8) byte {
switch l {
case A:
return r.A
case F:
return r.F
case B:
return r.B
case C:
return r.C
case D:
return r.D
case E:
return r.E
case H:
return r.H
case L:
return r.L
case IXL:
return r.IXL
case IXH:
return r.IXH
case IYL:
return r.IYL
case IYH:
return r.IYH
case I:
return r.I
case R:
return r.R
default:
panic(fmt.Sprintf("Unknown R8: %d", l))
}
}
func (r *Registers) Write8(l R8, n byte) {
switch l {
case A:
r.A = n
case F:
r.F = n
case B:
r.B = n
case C:
r.C = n
case D:
r.D = n
case E:
r.E = n
case H:
r.H = n
case L:
r.L = n
case IXL:
r.IXL = n
case IXH:
r.IXH = n
case IYL:
r.IYL = n
case IYH:
r.IYH = n
case I:
r.I = n
case R:
r.R = n
default:
panic(fmt.Sprintf("Unknown R8: %d", l))
}
}
type r8name struct {
r R8
name string
}
var R8Names []r8name = []r8name{
{A, "A"},
{F, "F"},
{B, "B"},
{C, "C"},
{D, "D"},
{E, "E"},
{H, "H"},
{L, "L"},
{IXL, "IXL"},
{IXH, "IXH"},
{IYL, "IYL"},
{IYH, "IYH"},
{I, "I"},
{R, "R"},
}
func LookupR8Name(name string) R8 {
name = strings.ToUpper(name)
for _, r8name := range R8Names {
if r8name.name == name {
return r8name.r
}
}
panic(fmt.Errorf("Unrecognised R8 name : %s", name))
}
func (r R8) String() string {
for _, r8name := range R8Names {
if r8name.r == r {
return r8name.name
}
}
panic(fmt.Errorf("Unrecognised R8 : %d", int(r)))
}
func (r R8) Read8(z *Zog) (byte, error) {
n := z.reg.Read8(r)
//fmt.Printf("Z: %02X <- %s\n", n, r)
return n, nil
}
func (r R8) Write8(z *Zog, n byte) error {
z.reg.Write8(r, n)
//fmt.Printf("Z: %s <- %02X\n", r, n)
return nil
}
type R16 int
const (
AF R16 = iota
BC
DE
HL
IX
IY
SP
PC
AF_PRIME
BC_PRIME
DE_PRIME
HL_PRIME
)
func (r *Registers) Read16(l R16) uint16 {
var lo, hi byte
switch l {
case AF:
hi = r.A
lo = r.F
case BC:
hi = r.B
lo = r.C
case DE:
hi = r.D
lo = r.E
case HL:
hi = r.H
lo = r.L
case IX:
hi = r.IXH
lo = r.IXL
case IY:
hi = r.IYH
lo = r.IYL
case SP:
return r.SP
case PC:
return r.PC
case AF_PRIME:
hi = r.A_PRIME
lo = r.F_PRIME
case BC_PRIME:
hi = r.B_PRIME
lo = r.C_PRIME
case DE_PRIME:
hi = r.D_PRIME
lo = r.E_PRIME
case HL_PRIME:
hi = r.H_PRIME
lo = r.L_PRIME
default:
panic(fmt.Sprintf("Unknown R16: %d", l))
}
return (uint16(hi) << 8) | uint16(lo)
}
func (r *Registers) Write16(l R16, nn uint16) {
hi := byte(nn >> 8)
lo := byte(nn)
switch l {
case AF:
r.A = hi
r.F = lo
case BC:
r.B = hi
r.C = lo
case DE:
r.D = hi
r.E = lo
case HL:
r.H = hi
r.L = lo
case IX:
r.IXH = hi
r.IXL = lo
case IY:
r.IYH = hi
r.IYL = lo
case SP:
r.SP = nn
case PC:
r.PC = nn
case AF_PRIME:
r.A_PRIME = hi
r.F_PRIME = lo
case BC_PRIME:
r.B_PRIME = hi
r.C_PRIME = lo
case DE_PRIME:
r.D_PRIME = hi
r.E_PRIME = lo
case HL_PRIME:
r.H_PRIME = hi
r.L_PRIME = lo
default:
panic(fmt.Sprintf("Unknown R16: %d", l))
}
}
type r16name struct {
r R16
name string
}
var R16Names []r16name = []r16name{
{AF, "AF"},
{AF_PRIME, "AF'"},
{BC, "BC"},
{DE, "DE"},
{HL, "HL"},
{IX, "IX"},
{IY, "IY"},
{SP, "SP"},
}
func LookupR16Name(name string) R16 {
name = strings.ToUpper(name)
for _, r16name := range R16Names {
if r16name.name == name {
return r16name.r
}
}
panic(fmt.Errorf("Unrecognised R16 name : %s", name))
}
func (r R16) String() string {
for _, r16name := range R16Names {
if r16name.r == r {
return r16name.name
}
}
panic(fmt.Errorf("Unrecognised R16 : %d", int(r)))
}
func (r R16) Read16(z *Zog) (uint16, error) {
nn := z.reg.Read16(r)
//fmt.Printf("Z: %04X <- %s\n", nn, r)
return nn, nil
}
func (r R16) Write16(z *Zog, nn uint16) error {
z.reg.Write16(r, nn)
//fmt.Printf("Z: %s <- %04X\n", r, nn)
return nil
}
type Contents struct {
addr Loc16
}
func (c Contents) String() string {
return fmt.Sprintf("(%s)", c.addr)
}
func (c Contents) Read8(z *Zog) (byte, error) {
addr, err := c.addr.Read16(z)
if err != nil {
return 0, fmt.Errorf("Can't get contents of [%s]: %s", c.addr, err)
}
n, err := z.Mem.Peek(addr)
if err != nil {
return 0, fmt.Errorf("Can't read contents of [%s]: %s", c, err)
}
//fmt.Printf("Z: %02X <- %s\n", n, c)
return n, nil
}
func (c Contents) Write8(z *Zog, n byte) error {
addr, err := c.addr.Read16(z)
if err != nil {
return fmt.Errorf("Can't get contents of [%s]: %s", c.addr, err)
}
err = z.Mem.Poke(addr, n)
if err != nil {
return fmt.Errorf("Can't write contents of [%s]: %s", c, err)
}
//fmt.Printf("Z: %s <- %02X\n", c, n)
return nil
}
func (c Contents) Read16(z *Zog) (uint16, error) {
addr, err := c.addr.Read16(z)
if err != nil {
return 0, fmt.Errorf("Can't get contents of [%s]: %s", c.addr, err)
}
nn, err := z.Mem.Peek16(addr)
if err != nil {
return 0, fmt.Errorf("Can't read contents of [%s]: %s", c, err)
}
//fmt.Printf("Z: %04X <- %s\n", nn, c)
return nn, nil
}
func (c Contents) Write16(z *Zog, nn uint16) error {
addr, err := c.addr.Read16(z)
if err != nil {
return fmt.Errorf("Can't get contents of [%s]: %s", c.addr, err)
}
err = z.Mem.Poke16(addr, nn)
if err != nil {
return fmt.Errorf("Can't write contents of [%s]: %s", c, err)
}
//fmt.Printf("Z: %s <- %04X\n", c, nn)
return nil
}
type IndexedContents struct {
addr Loc16
d Disp
}
func (ic IndexedContents) String() string {
return fmt.Sprintf("(%s%+d)", ic.addr, int8(ic.d))
}
func (ic IndexedContents) Read8(z *Zog) (byte, error) {
addr, err := ic.addr.Read16(z)
if err != nil {
return 0, fmt.Errorf("Can't get contents of [%s]: %s", ic.addr, err)
}
addr += uint16(ic.d)
n, err := z.Mem.Peek(addr)
if err != nil {
return 0, fmt.Errorf("Can't read contents of [%s]: %s", ic, err)
}
//fmt.Printf("Z: %02X <- %s\n", n, ic)
return n, nil
}
func (ic IndexedContents) Write8(z *Zog, n byte) error {
addr, err := ic.addr.Read16(z)
if err != nil {
return fmt.Errorf("Can't get contents of [%s]: %s", ic.addr, err)
}
addr += uint16(ic.d)
err = z.Mem.Poke(addr, n)
if err != nil {
return fmt.Errorf("Can't write contents of [%s]: %s", ic, err)
}
//fmt.Printf("Z: %s <- %02X\n", ic, n)
return nil
}
func (ic IndexedContents) Read16(z *Zog) (uint16, error) {
// TODO: debug
var nn uint16
//fmt.Printf("Z: %04X <- %s\n", nn, ic.addr)
return nn, nil
}
func (ic IndexedContents) Write16(z *Zog, nn uint16) error {
// TODO: debug
//fmt.Printf("Z: %s <- %04X\n", ic.addr, nn)
return nil
}
type Label struct {
// Default to the zero value because we need to encode
// this location *before* it is resolved with an address
// so that we can calculate instruction length.
// See assemble/ResolveAddr
Imm16
name string
}
func (l Label) String() string {
addrStr := l.Imm16.String()
return fmt.Sprintf("%s (%s)", l.name, addrStr)
}
type Imm16 uint16
func (nn Imm16) String() string {
return fmt.Sprintf("0x%04X", uint16(nn))
}
func (nn Imm16) Read16(z *Zog) (uint16, error) {
return uint16(nn), nil
}
func (nn Imm16) Write16(z *Zog, n uint16) error {
return fmt.Errorf("Attempt to write [%02X] to immediate 16bit value [%04X]", n, nn)
}
type Imm8 byte
func (n Imm8) String() string {
return fmt.Sprintf("0x%02X", byte(n))
}
func (n Imm8) Read8(z *Zog) (byte, error) {
return byte(n), nil
}
func (n Imm8) Write8(z *Zog, tmp byte) error {
return fmt.Errorf("Attempt to write [%02X] to immediate 8bit value [%04X]", n, tmp)
}
type Conditional interface {
String() string
IsTrue(z *Zog) bool
}
type FlagTest int
const (
FT_Z FlagTest = iota
FT_C
FT_PO
FT_PE
FT_P
FT_M
)
func (ft FlagTest) String() string {
switch ft {
case FT_Z:
return "Z"
case FT_C:
return "C"
case FT_PO:
return "PO"
case FT_PE:
return "PE"
case FT_P:
return "P"
case FT_M:
return "M"
default:
panic(fmt.Sprintf("Unknown flag test [%d]", int(ft)))
}
}
func (ft FlagTest) IsTrue(z *Zog) bool {
switch ft {
case FT_Z:
return z.GetFlag(F_Z)
case FT_C:
return z.GetFlag(F_C)
case FT_PO:
return !z.GetFlag(F_PV)
case FT_PE:
return z.GetFlag(F_PV)
case FT_P:
return !z.GetFlag(F_S)
case FT_M:
return z.GetFlag(F_S)
default:
panic(fmt.Sprintf("Unknown flag test [%d]", int(ft)))
}
}
type TrueLogicConstant struct{}
var True TrueLogicConstant
func (l TrueLogicConstant) String() string {
panic("Attempt to render 'true' as string")
}
func (l TrueLogicConstant) IsTrue(z *Zog) bool {
return true
}
type Not struct {
ft FlagTest
}
func (n Not) String() string {
return fmt.Sprintf("N%s", n.ft)
}
func (n Not) IsTrue(z *Zog) bool {
return !n.ft.IsTrue(z)
}
type Disp int8
func (d Disp) String() string {
return fmt.Sprintf("%d", d)
}