mirror of https://github.com/openXC7/prjxray.git
190 lines
5.7 KiB
Python
Executable File
190 lines
5.7 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
#
|
|
# Copyright (C) 2017-2020 The Project X-Ray Authors.
|
|
#
|
|
# Use of this source code is governed by a ISC-style
|
|
# license that can be found in the LICENSE file or at
|
|
# https://opensource.org/licenses/ISC
|
|
#
|
|
# SPDX-License-Identifier: ISC
|
|
# TODO: need better coverage for different tile types
|
|
|
|
from io import StringIO
|
|
import os
|
|
import os.path
|
|
import re
|
|
import unittest
|
|
import tempfile
|
|
|
|
import prjxray
|
|
import utils.fasm2frames as fasm2frames
|
|
|
|
from textx.exceptions import TextXSyntaxError
|
|
|
|
|
|
def frm2bits(txt):
|
|
'''
|
|
Convert output .frm file text to set of (frame addr, word #, bit index) tuples
|
|
'''
|
|
bits_out = set()
|
|
for l in txt.split('\n'):
|
|
l = l.strip()
|
|
if not l:
|
|
continue
|
|
# 0x00020500 0x00000000,0x00000000,0x00000000,...
|
|
addr, words = l.split(' ')
|
|
addr = int(addr, 0)
|
|
words = words.split(',')
|
|
assert (101 == len(words))
|
|
for wordi, word in enumerate(words):
|
|
word = int(word, 0)
|
|
for biti in range(32):
|
|
val = word & (1 << biti)
|
|
if val:
|
|
bits_out.add((addr, wordi, biti))
|
|
return bits_out
|
|
|
|
|
|
def bitread2bits(txt):
|
|
'''
|
|
Convert .bits text file (ie bitread output) to set of (frame addr, word #, bit index) tuples
|
|
'''
|
|
bits_ref = set()
|
|
for l in txt.split('\n'):
|
|
l = l.strip()
|
|
if not l:
|
|
continue
|
|
# bit_0002050b_004_14
|
|
m = re.match(r'bit_(.{8})_(.{3})_(.{2})', l)
|
|
addr = int(m.group(1), 16)
|
|
word = int(m.group(2), 10)
|
|
bit = int(m.group(3), 10)
|
|
bits_ref.add((addr, word, bit))
|
|
return bits_ref
|
|
|
|
|
|
class TestStringMethods(unittest.TestCase):
|
|
def filename_test_data(self, fname):
|
|
return os.path.join(os.path.dirname(__file__), 'test_data', fname)
|
|
|
|
def get_test_data(self, fname):
|
|
with open(self.filename_test_data(fname)) as f:
|
|
return f.read()
|
|
|
|
def fasm2frames(self, fin_data, **kw):
|
|
with tempfile.NamedTemporaryFile(suffix='.fasm') as fin:
|
|
fin.write(fin_data.encode('utf-8'))
|
|
fin.flush()
|
|
|
|
fout = StringIO()
|
|
fasm2frames.run(
|
|
self.filename_test_data('db'), "xc7", fin.name, fout, **kw)
|
|
|
|
return fout.getvalue()
|
|
|
|
def test_lut(self):
|
|
'''Simple smoke test on just the LUTs'''
|
|
self.fasm2frames(self.get_test_data('lut.fasm'))
|
|
|
|
def bitread_frm_equals(self, frm_fn, bitread_fn):
|
|
fout = self.fasm2frames(self.get_test_data(frm_fn))
|
|
|
|
# Build a list of output used bits
|
|
bits_out = frm2bits(fout)
|
|
|
|
# Build a list of reference used bits
|
|
bits_ref = bitread2bits(self.get_test_data(bitread_fn))
|
|
|
|
# Now check for equivilence vs reference design
|
|
self.assertEqual(len(bits_ref), len(bits_out))
|
|
self.assertEqual(bits_ref, bits_out)
|
|
|
|
def test_lut_int(self):
|
|
self.bitread_frm_equals('lut_int.fasm', 'lut_int/design.bits')
|
|
|
|
def test_ff_int(self):
|
|
self.bitread_frm_equals('ff_int.fasm', 'ff_int/design.bits')
|
|
|
|
@unittest.skip
|
|
def test_ff_int_op1(self):
|
|
'''Omitted key set to '''
|
|
self.bitread_frm_equals('ff_int_op1.fasm', 'ff_int/design.bits')
|
|
|
|
# Same check as above, but isolated test case
|
|
def test_opkey_01_default(self):
|
|
'''Optional key with binary omitted value should produce valid result'''
|
|
self.fasm2frames("CLBLM_L_X10Y102.SLICEM_X0.SRUSEDMUX")
|
|
|
|
@unittest.skip
|
|
def test_opkey_01_1(self):
|
|
self.fasm2frames("CLBLM_L_X10Y102.SLICEM_X0.SRUSEDMUX 1")
|
|
|
|
@unittest.skip
|
|
def test_opkey_enum(self):
|
|
'''Optional key with enumerated value should produce syntax error'''
|
|
try:
|
|
# CLBLM_L.SLICEM_X0.AMUXFF.O6 !30_06 !30_07 !30_08 30_11
|
|
self.fasm2frames("CLBLM_L_X10Y102.SLICEM_X0.AFFMUX.O6")
|
|
self.fail("Expected syntax error")
|
|
except TextXSyntaxError:
|
|
pass
|
|
|
|
def test_ff_int_0s(self):
|
|
'''Explicit 0 entries'''
|
|
self.bitread_frm_equals('ff_int_0s.fasm', 'ff_int/design.bits')
|
|
|
|
def test_badkey(self):
|
|
'''Bad key should throw syntax error'''
|
|
try:
|
|
self.fasm2frames("CLBLM_L_X10Y102.SLICEM_X0.SRUSEDMUX 2")
|
|
self.fail("Expected syntax error")
|
|
except TextXSyntaxError:
|
|
pass
|
|
|
|
@unittest.skip
|
|
def test_dupkey(self):
|
|
'''Duplicate key should throw syntax error'''
|
|
try:
|
|
self.fasm2frames(
|
|
"""\
|
|
CLBLM_L_X10Y102.SLICEM_X0.SRUSEDMUX 0
|
|
CLBLM_L_X10Y102.SLICEM_X0.SRUSEDMUX 1
|
|
""")
|
|
self.fail("Expected syntax error")
|
|
except TextXSyntaxError:
|
|
pass
|
|
|
|
@unittest.skip
|
|
def test_sparse(self):
|
|
'''Verify sparse equivalent to normal encoding'''
|
|
frm_fn = 'lut_int.fasm'
|
|
|
|
fout_sparse_txt = self.fasm2frames(
|
|
self.get_test_data(frm_fn), sparse=True)
|
|
bits_sparse = frm2bits(fout_sparse_txt)
|
|
|
|
fout_full_txt = self.fasm2frames(
|
|
self.get_test_data(frm_fn), sparse=False)
|
|
bits_full = frm2bits(fout_full_txt)
|
|
|
|
# Now check for equivilence vs reference design
|
|
self.assertEquals(len(bits_sparse), len(bits_full))
|
|
self.assertEquals(bits_sparse, bits_full)
|
|
|
|
# Verify the full ROI is way bigger description
|
|
# It will still be decent size though since even sparse occupies all columns in that area
|
|
self.assertGreaterEqual(len(fout_full_txt), len(fout_sparse_txt) * 4)
|
|
|
|
def test_stepdown_1(self):
|
|
self.bitread_frm_equals(
|
|
'iob/liob_stepdown.fasm', 'iob/liob_stepdown.bits')
|
|
|
|
def test_stepdown_2(self):
|
|
self.bitread_frm_equals(
|
|
'iob/riob_stepdown.fasm', 'iob/riob_stepdown.bits')
|
|
|
|
|
|
if __name__ == '__main__':
|
|
unittest.main()
|