Add unit tests, refactoring

This commit is contained in:
Loic GUEGAN 2018-09-02 16:31:49 +02:00
parent cec2994481
commit a110465a42
11 changed files with 70 additions and 48 deletions

View file

@ -32,14 +32,29 @@ class Microprogram:
""" """
Read data into memory Read data into memory
""" """
data=self.c["RAM"].read() little_endian=self.c["RAM"].read()
self.c["MDR"]=data ##### Build little endian version of MDR ####
big_endian=(little_endian&0xFF)<<24
big_endian=big_endian|(((little_endian>>8)&0xFF)<<16)
big_endian=big_endian|(((little_endian>>16)&0xFF)<<8)
big_endian=big_endian|((little_endian>>24)&0xFF)
##############################################
self.c["MDR"]=big_endian
def wr(self): def wr(self):
""" """
Write data into memory Write data into memory
""" """
self.c["RAM"].write() ##### Build little endian version of MDR ####
little_endian=(self.c["MDR"]&0xFF)<<24
little_endian=little_endian|(((self.c["MDR"]>>8)&0xFF)<<16)
little_endian=little_endian|(((self.c["MDR"]>>16)&0xFF)<<8)
little_endian=little_endian|((self.c["MDR"]>>24)&0xFF)
##############################################
big_endian=self.c["MDR"] # Backup MDR before change it to little endian
self.c["MDR"]=little_endian # Load little endian value
self.c["RAM"].write() # Write little endian value into memory
self.c["MDR"]=big_endian # Restore big endian
def exec(self): # TODO: Implement opcode def exec(self): # TODO: Implement opcode
""" """
@ -187,16 +202,3 @@ class Microprogram:
self.fetch();self.c["PC"]+=1 # Needed because memory access take 1 cycle in simulation self.fetch();self.c["PC"]+=1 # Needed because memory access take 1 cycle in simulation
self.c["PC"]=self.c["PC"]+1 self.c["PC"]=self.c["PC"]+1
def dump(self):
"""
Print RAM, stack and registers
"""
print("-------------- RAM --------------")
self.c["RAM"].dump()
print("------------- Stack -------------")
self.c["RAM"].dumpRange(self.c["LV"]*4,self.c["SP"]*4,4) # Convert address to 32bits value
print("----------- Registers -----------")
for key,value in self.c.items():
if key!="RAM":
print("{}={}".format(key,value))
print("---------------------------------")

View file

@ -37,11 +37,11 @@ class Ram:
addr=self.c["MAR"]*4 # Don't forget MAR address 32bits block of memory addr=self.c["MAR"]*4 # Don't forget MAR address 32bits block of memory
if addr>self.lastAddr: if addr>self.lastAddr:
raise ValueError("You get out of the ram by trying to set a value at address {}, max address is {}".format(addr,self.lastAddr)) raise ValueError("You get out of the ram by trying to set a value at address {}, max address is {}".format(addr,self.lastAddr))
#### Little endian #### #### Split bytes and write ####
self.data[addr]=self.c["MDR"] & 0xFF self.data[addr+3]=self.c["MDR"] & 0xFF
self.data[addr+1]=self.c["MDR"] & 0xFF00 self.data[addr+2]=self.c["MDR"]>>8 & 0xFF
self.data[addr+2]=self.c["MDR"] & 0xFF0000 self.data[addr+1]=self.c["MDR"]>>16 & 0xFF
self.data[addr+3]=self.c["MDR"] & 0xFF000000 self.data[addr]=self.c["MDR"]>>24 & 0xFF
def read(self): def read(self):
@ -51,8 +51,8 @@ class Ram:
addr=self.c["MAR"]*4 # Don't forget MAR address 32bits block of memory addr=self.c["MAR"]*4 # Don't forget MAR address 32bits block of memory
value=None value=None
try: try:
#### Little endian #### #### Combine bytes ####
value=(self.data[addr+3]<<24)|(self.data[addr+2]<<16)|(self.data[addr+1]<<8)|self.data[addr] value=self.data[addr]<<24|(self.data[addr+1]<<16)|(self.data[addr+2]<<8)|(self.data[addr+3])
except: except:
if addr>self.lastAddr: if addr>self.lastAddr:
raise ValueError("You get out of the ram by trying to get value at address {}, max address is {}".format(addr,self.lastAddr)) raise ValueError("You get out of the ram by trying to get value at address {}, max address is {}".format(addr,self.lastAddr))
@ -77,18 +77,7 @@ class Ram:
def dump(self): def dump(self):
""" """
Simple dump helper Fetch RAM data (usefull for unit tests)
""" """
for key,value in self.data.items(): return(self.data)
#print("{}:{}".format(key,bin(value)[2:]))
print("{}:{}".format(key,value))
def dumpRange(self,start,end,step):
"""
Another dump helper
"""
for i in range(start,end+1,step):
try:
print("{}:{}".format(i,self.data[i]))
except:
print("{}:0".format(i))

View file

@ -12,4 +12,7 @@ c["RAM"]=RAM # Add ram to components
mic=Microprogram(c) # Create micro program mic=Microprogram(c) # Create micro program
mic.run() # Run the micro program mic.run() # Run the micro program
mic.dump() # Dump ram
mic.rd()
print(bin(c["MDR"]))
print(RAM.dump())

2
MicSim/ram.txt Normal file
View file

@ -0,0 +1,2 @@
BIPUSH
9

0
MicSim/test/__init__.py Normal file
View file

34
MicSim/test/test_ram.py Normal file
View file

@ -0,0 +1,34 @@
from components.ram import Ram
import unittest
from random import randint
class RamTest(unittest.TestCase):
def setUp(self):
"""
Init test
"""
self.caretaker=dict({"MDR":0,"MAR":0,"MBR":0,"PC":0})
def test_write(self):
"""
Test write method
"""
toWrite=randint(0,2**32) # Pick a random number to write
self.caretaker["MDR"]=toWrite
ram=Ram(self.caretaker,10000)
ram.write() # Write a random number at address 0
data=ram.dump() # Dump ram
##### Test if everything is written using big endian model #####
self.assertEqual((toWrite>>24)&0xFF,data[self.caretaker["MAR"]])
self.assertEqual((toWrite>>16)&0xFF,data[self.caretaker["MAR"]+1])
self.assertEqual((toWrite>>8)&0xFF,data[self.caretaker["MAR"]+2])
self.assertEqual(toWrite&0xFF,data[self.caretaker["MAR"]+3])
if __name__ == "__main__":
unittest.main()

View file

@ -9,7 +9,8 @@ It is simple, you have to:
3. Enjoy ! 3. Enjoy !
### I want to know more about it.... ### I want to know more about it....
All the architecture components are located in _components_ folder: Source code is located in MicSim folder. All the components used for the Mic-1 architecture are
located in _MicSim/components_ folder:
- **ijvm.py** Contains standard IJVM constant - **ijvm.py** Contains standard IJVM constant
- **microprogram.py** Contains IJVM implementation that use Mic-1 architecture - **microprogram.py** Contains IJVM implementation that use Mic-1 architecture
- **caretaker.py** Hold all the Mic-1 architecture components (registers, ram etc..) - **caretaker.py** Hold all the Mic-1 architecture components (registers, ram etc..)

View file

@ -1,9 +0,0 @@
BIPUSH
-4
IFLT
0
4
BIPUSH
6
BIPUSH
7