-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathregvm.py
125 lines (103 loc) · 3.12 KB
/
regvm.py
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
class REGVM:
def __init__(self):
self.pc = 0
self.registers = {
'A': 0,
'B': 0,
'C': 0,
'D': 0
}
self.flags = {
'Z': 0, # zero
'N': 0 # negative
}
self.memory = []
def load_program(self, program):
self.memory = program
def fetch(self):
if self.pc < len(self.memory):
instruction = self.memory[self.pc]
self.pc += 1
return instruction
else:
return None
def decode_and_execute(self, instruction):
if not instruction:
return False
parts = instruction.split()
opcode = parts[0]
args = parts[1:]
#print("op", opcode, ", args", args)
if opcode == 'MOV':
fst = args[0]
snd = args[1]
self.registers[fst] = int(snd)
elif opcode == 'ADD':
fst = args[0]
snd = args[1]
if snd.isalpha():
self.registers[fst] += self.registers[snd]
else:
self.registers[fst] += int(snd)
self.update_flags(self.registers[fst])
elif opcode == 'SUB':
fst = args[0]
snd = args[1]
if snd.isalpha():
self.registers[fst] -= self.registers[snd]
else:
self.registers[fst] -= int(snd)
self.update_flags(self.registers[fst])
elif opcode == 'MUL':
fst = args[0]
snd = args[1]
if snd.isalpha():
self.registers[fst] *= self.registers[snd]
else:
self.registers[fst] *= int(snd)
self.update_flags(self.registers[fst])
elif opcode == 'CMP':
fst = args[0]
snd = args[1]
if self.registers[fst] == int(snd):
self.flags['Z'] = 1
else:
self.flags['Z'] = 0
elif opcode == 'JMP':
fst = args[0]
self.pc = int(fst)
elif opcode == 'JZ':
fst = args[0]
if self.flags['Z']:
self.pc = int(fst)
elif opcode == 'PRINT':
fst = args[0]
print(f"Register {fst}: {self.registers[fst]}")
else:
raise ValueError(f"Unknown opcode: {opcode}")
return True
def update_flags(self, value):
self.flags['Z'] = 1 if value == 0 else 0
self.flags['N'] = 1 if value < 0 else 0
def run(self):
while True:
instruction = self.fetch()
if not instruction or not self.decode_and_execute(instruction):
break
# factorial
factorial = [
# start:
"MOV A 1", # init A with 1
"MOV B 5", # init B with 5
# loop:
"CMP B 0", # compare B with 0
"JZ 7", # if B is 0, jump to end
"MUL A B", # multiply A by B and store in A
"SUB B 1", # subtract 1 from B
"JMP 2", # jump to loop
# end:
"PRINT A", # Print the result in register A
]
vm = REGVM()
vm.load_program(factorial)
vm.run()