mirror of
https://github.com/enjoy-digital/litedram.git
synced 2025-01-04 09:52:25 -05:00
300 lines
10 KiB
Python
300 lines
10 KiB
Python
# This file is Copyright (c) 2018-2019 Florent Kermarrec <florent@enjoy-digital.fr>
|
|
# License: BSD
|
|
|
|
import unittest
|
|
import random
|
|
|
|
from migen import *
|
|
|
|
from litedram.common import *
|
|
from litedram.frontend.ecc import *
|
|
|
|
from litex.gen.sim import *
|
|
from litex.soc.cores.ecc import *
|
|
|
|
from test.common import *
|
|
|
|
|
|
def bits(value, width=32):
|
|
# convert int to a string representing binary value
|
|
# reverse it so that we can index bits easily with s[0] being LSB
|
|
return f"{value:0{width}b}"[::-1]
|
|
|
|
def frombits(bits):
|
|
# reverse of bits()
|
|
return int(bits[::-1], 2)
|
|
|
|
def bits_pp(value, width=32):
|
|
# pretty print binary value, with 0b, groupped by nibbles
|
|
if isinstance(value, str):
|
|
value = frombits(value)
|
|
return f"{value:#0{width}_b}"
|
|
|
|
|
|
def extract_ecc_data(data_width, codeword_width, codeword_bits):
|
|
extracted = ""
|
|
for i in range(8):
|
|
word = codeword_bits[codeword_width*i:codeword_width*(i+1)]
|
|
# remove parity bit
|
|
word = word[1:]
|
|
data_pos = compute_data_positions(codeword_width - 1) # -1 for parity
|
|
# extract data bits
|
|
word_ex = list(bits(0, 32))
|
|
for j, d in enumerate(data_pos):
|
|
word_ex[j] = word[d-1]
|
|
word_ex = "".join(word_ex)
|
|
extracted += word_ex
|
|
return extracted
|
|
|
|
|
|
class TestECC(unittest.TestCase):
|
|
def test_eccw_connected(self):
|
|
class DUT(Module):
|
|
def __init__(self):
|
|
self.submodules.eccw = LiteDRAMNativePortECCW(32*8, 39*8)
|
|
|
|
def main_generator(dut):
|
|
sink_data = seed_to_data(0, nbits=32*8)
|
|
|
|
yield dut.eccw.sink.data.eq(sink_data)
|
|
yield
|
|
source_data = (yield dut.eccw.source.data)
|
|
|
|
sink_data_bits = bits(sink_data, 32*8)
|
|
source_data_bits = bits(source_data, 39*8)
|
|
self.assertNotEqual(sink_data_bits, source_data_bits[:len(sink_data_bits)])
|
|
|
|
source_extracted = extract_ecc_data(32, 39, source_data_bits)
|
|
# assert each word separately for more readable assert messages
|
|
for i in range(8):
|
|
word = slice(32*i, 32*(i+1))
|
|
self.assertEqual(bits_pp(source_extracted[word]), bits_pp(sink_data_bits[word]),
|
|
msg=f"Mismatch at i = {i}")
|
|
|
|
dut = DUT()
|
|
run_simulation(dut, main_generator(dut))
|
|
|
|
def test_eccw_we_enabled(self):
|
|
# currently byte enable is not supported so it should be enabled all the time
|
|
class DUT(Module):
|
|
def __init__(self):
|
|
self.submodules.eccw = LiteDRAMNativePortECCW(32*8, 39*8)
|
|
|
|
def main_generator(dut):
|
|
yield
|
|
source_we = (yield dut.eccw.source.we)
|
|
|
|
self.assertEqual(bits_pp(source_we, 39//8), bits_pp(2**len(dut.eccw.source.we) - 1))
|
|
|
|
dut = DUT()
|
|
run_simulation(dut, main_generator(dut))
|
|
|
|
def test_eccr_connected(self):
|
|
class DUT(Module):
|
|
def __init__(self):
|
|
self.submodules.eccr = LiteDRAMNativePortECCR(32*8, 39*8)
|
|
|
|
def main_generator(dut):
|
|
sink_data = seed_to_data(0, nbits=(39*8 // 32 + 1) * 32)
|
|
|
|
yield dut.eccr.sink.data.eq(sink_data)
|
|
yield
|
|
source_data = (yield dut.eccr.source.data)
|
|
|
|
sink_data_bits = bits(sink_data, 39*8)
|
|
source_data_bits = bits(source_data, 32*8)
|
|
self.assertNotEqual(sink_data_bits[:len(source_data_bits)], source_data_bits)
|
|
|
|
sink_extracted = extract_ecc_data(32, 39, sink_data_bits)
|
|
self.assertEqual(bits_pp(sink_extracted), bits_pp(source_data_bits))
|
|
# assert each word separately for more readable assert messages
|
|
for i in range(8):
|
|
word = slice(32*i, 32*(i+1))
|
|
self.assertEqual(bits_pp(sink_extracted[word]), bits_pp(source_data_bits[word]),
|
|
msg=f"Mismatch at i = {i}")
|
|
|
|
dut = DUT()
|
|
run_simulation(dut, main_generator(dut))
|
|
|
|
def test_eccr_errors_connected_when_sink_valid(self):
|
|
class DUT(Module):
|
|
def __init__(self):
|
|
self.submodules.eccr = LiteDRAMNativePortECCR(32*8, 39*8)
|
|
|
|
def main_generator(dut):
|
|
yield dut.eccr.enable.eq(1)
|
|
yield dut.eccr.sink.data.eq(0b10) # wrong parity bit
|
|
yield
|
|
|
|
# decder sec/ded not connected when valid=0
|
|
self.assertEqual((yield dut.eccr.sec), 0)
|
|
self.assertEqual((yield dut.eccr.ded), 0)
|
|
|
|
yield dut.eccr.sink.valid.eq(1)
|
|
yield
|
|
self.assertEqual((yield dut.eccr.sec), 1)
|
|
self.assertEqual((yield dut.eccr.ded), 0)
|
|
|
|
dut = DUT()
|
|
run_simulation(dut, main_generator(dut))
|
|
|
|
def ecc_encode_decode_test(self, from_width, to_width, n, pre=None, post=None, **kwargs):
|
|
class DUT(Module):
|
|
def __init__(self):
|
|
self.port_from = LiteDRAMNativePort("both", 24, from_width)
|
|
self.port_to = LiteDRAMNativePort("both", 24, to_width)
|
|
self.submodules.ecc = LiteDRAMNativePortECC(self.port_from, self.port_to, **kwargs)
|
|
self.mem = DRAMMemory(to_width, n)
|
|
|
|
self.wdata = [seed_to_data(i, nbits=from_width) for i in range(n)]
|
|
self.rdata = []
|
|
|
|
def main_generator(dut):
|
|
if pre is not None:
|
|
yield from pre(dut)
|
|
|
|
port = dut.port_from
|
|
|
|
# write
|
|
for i in range(n):
|
|
yield port.cmd.valid.eq(1)
|
|
yield port.cmd.we.eq(1)
|
|
yield port.cmd.addr.eq(i)
|
|
yield
|
|
while (yield port.cmd.ready) == 0:
|
|
yield
|
|
yield port.cmd.valid.eq(0)
|
|
yield
|
|
yield port.wdata.valid.eq(1)
|
|
yield port.wdata.data.eq(dut.wdata[i])
|
|
yield
|
|
while (yield port.wdata.ready) == 0:
|
|
yield
|
|
yield port.wdata.valid.eq(0)
|
|
yield
|
|
|
|
# read
|
|
for i in range(n):
|
|
yield port.cmd.valid.eq(1)
|
|
yield port.cmd.we.eq(0)
|
|
yield port.cmd.addr.eq(i)
|
|
yield
|
|
while (yield port.cmd.ready) == 0:
|
|
yield
|
|
yield port.cmd.valid.eq(0)
|
|
yield
|
|
while (yield port.rdata.valid) == 0:
|
|
yield
|
|
dut.rdata.append((yield port.rdata.data))
|
|
yield port.rdata.ready.eq(1)
|
|
yield
|
|
yield port.rdata.ready.eq(0)
|
|
yield
|
|
|
|
if post is not None:
|
|
yield from post(dut)
|
|
|
|
dut = DUT()
|
|
generators = [
|
|
main_generator(dut),
|
|
dut.mem.write_handler(dut.port_to),
|
|
dut.mem.read_handler(dut.port_to),
|
|
]
|
|
run_simulation(dut, generators)
|
|
return dut
|
|
|
|
def test_ecc_32_7(self):
|
|
# 32 data bits + 6 code bits + parity bit
|
|
dut = self.ecc_encode_decode_test(32*8, 39*8, 2)
|
|
self.assertEqual(dut.wdata, dut.rdata)
|
|
|
|
def test_ecc_64_8(self):
|
|
# 64 data bits + 7 code bits + parity bit
|
|
dut = self.ecc_encode_decode_test(64*8, 72*8, 2)
|
|
self.assertEqual(dut.wdata, dut.rdata)
|
|
|
|
def test_ecc_sec_errors(self):
|
|
def pre(dut):
|
|
yield from dut.ecc.flip.write(0b00000100)
|
|
|
|
def post(dut):
|
|
dut.sec_errors = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors = (yield from dut.ecc.ded_errors.read())
|
|
|
|
dut = self.ecc_encode_decode_test(8*8, 13*8, 4, pre, post, with_error_injection=True)
|
|
self.assertEqual(dut.wdata, dut.rdata)
|
|
self.assertEqual(dut.sec_errors, 4)
|
|
self.assertEqual(dut.ded_errors, 0)
|
|
|
|
def test_ecc_ded_errors(self):
|
|
def pre(dut):
|
|
yield from dut.ecc.flip.write(0b00001100)
|
|
|
|
def post(dut):
|
|
dut.sec_errors = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors = (yield from dut.ecc.ded_errors.read())
|
|
|
|
dut = self.ecc_encode_decode_test(8*8, 13*8, 4, pre, post, with_error_injection=True)
|
|
self.assertNotEqual(dut.wdata, dut.rdata)
|
|
self.assertEqual(dut.sec_errors, 0)
|
|
self.assertEqual(dut.ded_errors, 4)
|
|
|
|
def test_ecc_decoder_disable(self):
|
|
def pre(dut):
|
|
yield from dut.ecc.flip.write(0b10101100)
|
|
yield from dut.ecc.enable.write(0)
|
|
|
|
def post(dut):
|
|
dut.sec_errors = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors = (yield from dut.ecc.ded_errors.read())
|
|
|
|
dut = self.ecc_encode_decode_test(8*8, 13*8, 4, pre, post, with_error_injection=True)
|
|
self.assertNotEqual(dut.wdata, dut.rdata)
|
|
self.assertEqual(dut.sec_errors, 0)
|
|
self.assertEqual(dut.ded_errors, 0)
|
|
|
|
def test_ecc_clear_sec_errors(self):
|
|
def pre(dut):
|
|
yield from dut.ecc.flip.write(0b00000100)
|
|
|
|
def post(dut):
|
|
dut.sec_errors = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors = (yield from dut.ecc.ded_errors.read())
|
|
|
|
yield from dut.ecc.clear.write(1)
|
|
yield
|
|
|
|
dut.sec_errors_c = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors_c = (yield from dut.ecc.ded_errors.read())
|
|
|
|
dut = self.ecc_encode_decode_test(8*8, 13*8, 4, pre, post, with_error_injection=True)
|
|
self.assertEqual(dut.wdata, dut.rdata)
|
|
self.assertNotEqual(dut.sec_errors, 0)
|
|
self.assertEqual(dut.ded_errors, 0)
|
|
self.assertEqual(dut.sec_errors_c, 0)
|
|
self.assertEqual(dut.ded_errors_c, 0)
|
|
|
|
def test_ecc_clear_ded_errors(self):
|
|
def pre(dut):
|
|
yield from dut.ecc.flip.write(0b10101100)
|
|
|
|
def post(dut):
|
|
dut.sec_errors = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors = (yield from dut.ecc.ded_errors.read())
|
|
|
|
yield from dut.ecc.clear.write(1)
|
|
yield
|
|
|
|
dut.sec_errors_c = (yield from dut.ecc.sec_errors.read())
|
|
dut.ded_errors_c = (yield from dut.ecc.ded_errors.read())
|
|
|
|
dut = self.ecc_encode_decode_test(8*8, 13*8, 4, pre, post, with_error_injection=True)
|
|
self.assertNotEqual(dut.wdata, dut.rdata)
|
|
self.assertEqual(dut.sec_errors, 0)
|
|
self.assertNotEqual(dut.ded_errors, 0)
|
|
self.assertEqual(dut.sec_errors_c, 0)
|
|
self.assertEqual(dut.ded_errors_c, 0)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|