2018-12-10 20:22:40 +01:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
'''
|
|
|
|
|
Check:
|
|
|
|
|
-Individual files are valid
|
|
|
|
|
-No overlap between any tile
|
2018-12-10 23:42:53 +01:00
|
|
|
|
|
|
|
|
TODO:
|
|
|
|
|
Can we use prjxray?
|
|
|
|
|
Relies on 074, which is too far into the process
|
2018-12-10 20:22:40 +01:00
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
from prjxray import util
|
2018-12-10 23:42:53 +01:00
|
|
|
from prjxray import db as prjxraydb
|
|
|
|
|
import os
|
2018-12-10 20:22:40 +01:00
|
|
|
import parsedb
|
|
|
|
|
#from prjxray import db as prjxraydb
|
|
|
|
|
import glob
|
|
|
|
|
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
def make_tile_mask(db_root, tile_name, tilej, strict=False, verbose=False):
|
|
|
|
|
'''
|
|
|
|
|
Return dict
|
|
|
|
|
key: (address, bit index)
|
|
|
|
|
val: sample description of where it came from (there may be multiple, only one)
|
|
|
|
|
'''
|
|
|
|
|
|
|
|
|
|
# FIXME: fix mask files https://github.com/SymbiFlow/prjxray/issues/301
|
|
|
|
|
# in the meantime build them on the fly
|
|
|
|
|
# We may want this to build them anyway
|
|
|
|
|
|
|
|
|
|
ret = dict()
|
|
|
|
|
for absaddr, bitaddr, tag in util.gen_tile_bits(
|
|
|
|
|
db_root, tilej, strict=strict, verbose=verbose):
|
|
|
|
|
name = "%s.%s" % (tile_name, tag)
|
|
|
|
|
ret.setdefault((absaddr, bitaddr), name)
|
|
|
|
|
return ret
|
|
|
|
|
|
|
|
|
|
|
2018-12-10 20:22:40 +01:00
|
|
|
def parsedb_all(db_root, verbose=False):
|
|
|
|
|
'''Verify .db files are individually valid'''
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
files = 0
|
|
|
|
|
for bit_fn in glob.glob('%s/segbits_*.db' % db_root):
|
|
|
|
|
verbose and print("Checking %s" % bit_fn)
|
2018-12-10 20:22:40 +01:00
|
|
|
parsedb.run(bit_fn, fnout=None, strict=True, verbose=verbose)
|
2018-12-10 23:42:53 +01:00
|
|
|
files += 1
|
|
|
|
|
print("segbits_*.db: %d okay" % files)
|
2018-12-10 20:22:40 +01:00
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
files = 0
|
|
|
|
|
for bit_fn in glob.glob('%s/mask_*.db' % db_root):
|
|
|
|
|
verbose and print("Checking %s" % bit_fn)
|
2018-12-10 20:22:40 +01:00
|
|
|
parsedb.run(bit_fn, fnout=None, strict=True, verbose=verbose)
|
2018-12-10 23:42:53 +01:00
|
|
|
files += 1
|
|
|
|
|
print("mask_*.db: %d okay" % files)
|
2018-12-10 20:22:40 +01:00
|
|
|
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
def check_tile_overlap(db, db_root, strict=False, verbose=False):
|
2018-12-10 20:22:40 +01:00
|
|
|
'''
|
2018-12-10 23:42:53 +01:00
|
|
|
Verifies that no two tiles use the same bit
|
|
|
|
|
|
2018-12-10 20:22:40 +01:00
|
|
|
Assume .db files are individually valid
|
|
|
|
|
Create a mask for all the bits the tile type uses
|
|
|
|
|
For each tile, create bitmasks over the entire bitstream for current part
|
|
|
|
|
Throw an exception if two tiles share an address
|
|
|
|
|
'''
|
2018-12-10 23:42:53 +01:00
|
|
|
mall = dict()
|
2018-12-10 20:22:40 +01:00
|
|
|
|
|
|
|
|
tiles_checked = 0
|
2018-12-10 23:42:53 +01:00
|
|
|
for tile_name, tilej in db.tilegrid.items():
|
|
|
|
|
#for tile_name, tilej in [("CLBLL_L_X12Y138", db.tilegrid["CLBLL_L_X12Y138"])]:
|
|
|
|
|
mtile = make_tile_mask(
|
|
|
|
|
db_root, tile_name, tilej, strict=strict, verbose=verbose)
|
|
|
|
|
verbose and print(
|
|
|
|
|
"Checking %s, type %s, bits: %s" %
|
|
|
|
|
(tile_name, tilej["type"], len(mtile)))
|
|
|
|
|
if len(mtile) == 0:
|
2018-12-10 20:22:40 +01:00
|
|
|
continue
|
2018-12-10 23:42:53 +01:00
|
|
|
|
|
|
|
|
collisions = set(mall.keys()).intersection(set(mtile.keys()))
|
|
|
|
|
if collisions:
|
|
|
|
|
print("ERROR: %s collisions" % len(collisions))
|
|
|
|
|
for ck in sorted(collisions):
|
|
|
|
|
addr, bitaddr = ck
|
|
|
|
|
print(
|
|
|
|
|
" %08X_%04X: had %s, got %s" %
|
|
|
|
|
(addr, bitaddr, mall[ck], mtile[ck]))
|
|
|
|
|
raise ValueError("%s collisions" % len(collisions))
|
|
|
|
|
mall.update(mtile)
|
2018-12-10 20:22:40 +01:00
|
|
|
tiles_checked += 1
|
2018-12-10 23:42:53 +01:00
|
|
|
print("Checked %s tiles, %s bits" % (tiles_checked, len(mall)))
|
2018-12-10 20:22:40 +01:00
|
|
|
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
def run(db_root, strict=False, verbose=False):
|
2018-12-10 20:22:40 +01:00
|
|
|
# Start by running a basic check on db files
|
2018-12-10 23:42:53 +01:00
|
|
|
print("Checking individual .db...")
|
2018-12-10 20:22:40 +01:00
|
|
|
parsedb_all(db_root, verbose=verbose)
|
|
|
|
|
|
|
|
|
|
# Now load and verify tile consistency
|
|
|
|
|
db = prjxraydb.Database(db_root)
|
|
|
|
|
db._read_tilegrid()
|
2018-12-10 23:42:53 +01:00
|
|
|
'''
|
|
|
|
|
these don't load properly without .json files
|
|
|
|
|
See: https://github.com/SymbiFlow/prjxray/issues/303
|
2018-12-10 20:22:40 +01:00
|
|
|
db._read_tile_types()
|
|
|
|
|
print(db.tile_types.keys())
|
|
|
|
|
'''
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
verbose and print("")
|
|
|
|
|
|
|
|
|
|
print("Checking aggregate dir...")
|
|
|
|
|
check_tile_overlap(db, db_root, strict=strict, verbose=verbose)
|
2018-12-10 20:22:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
|
import argparse
|
|
|
|
|
|
|
|
|
|
parser = argparse.ArgumentParser(
|
|
|
|
|
description="Parse a db repository, checking for consistency")
|
|
|
|
|
|
|
|
|
|
util.db_root_arg(parser)
|
2018-12-10 23:42:53 +01:00
|
|
|
parser.add_argument('--strict', action='store_true', help='')
|
2018-12-10 20:22:40 +01:00
|
|
|
parser.add_argument('--verbose', action='store_true', help='')
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
2018-12-10 23:42:53 +01:00
|
|
|
run(args.db_root, strict=args.strict, verbose=args.verbose)
|
2018-12-10 20:22:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
main()
|