scripts/liberty: Rework how attribute types are discovered.

* Add functions for producing different types of values into liberty
   output (plus doctests for them).
 * Respect the type given by a `define()` statement.

Signed-off-by: Tim 'mithro' Ansell <me@mith.ro>
This commit is contained in:
Tim 'mithro' Ansell 2020-07-09 21:16:04 -07:00
parent c9d0c8131f
commit 296ef2066e
1 changed files with 240 additions and 15 deletions

View File

@ -343,7 +343,8 @@ def generate(library_dir, lib, corner, ocorner_type, icorner_type, cells):
if ocorner_type != TimingType.ccsnoise: if ocorner_type != TimingType.ccsnoise:
remove_ccsnoise_from_library(common_data, "library") remove_ccsnoise_from_library(common_data, "library")
output = liberty_dict("library", lib+"__"+corner, common_data) attribute_types = {}
output = liberty_dict("library", lib+"__"+corner, common_data, attribute_types)
assert output[-1] == '}', output assert output[-1] == '}', output
top_write(output[:-1]) top_write(output[:-1])
@ -360,7 +361,13 @@ def generate(library_dir, lib, corner, ocorner_type, icorner_type, cells):
remove_ccsnoise_from_cell(cell_data, cell_with_size) remove_ccsnoise_from_cell(cell_data, cell_with_size)
top_write(['']) top_write([''])
top_write(liberty_dict("cell", "%s__%s" % (lib, cell_with_size), cell_data, [cell_with_size])) top_write(liberty_dict(
"cell",
"%s__%s" % (lib, cell_with_size),
cell_data,
[cell_with_size],
attribute_types,
))
top_write(['']) top_write([''])
top_write(['}']) top_write(['}'])
@ -602,6 +609,135 @@ def is_liberty_list(k):
return k in ('variable', 'index', 'values') return k in ('variable', 'index', 'values')
def liberty_guess(o):
"""
>>> liberty_guess('hello') # doctest: +ELLIPSIS
<function liberty_str at ...>
>>> liberty_guess(1.0) # doctest: +ELLIPSIS
<function liberty_float at ...>
>>> liberty_guess(1) # doctest: +ELLIPSIS
<function liberty_float at ...>
>>> liberty_guess(None)
Traceback (most recent call last):
...
ValueError: None has unguessable type: <class 'NoneType'>
"""
if isinstance(o, str):
return liberty_str
elif isinstance(o, (float,int)):
return liberty_float
else:
raise ValueError("%r has unguessable type: %s" % (o, type(o)))
def liberty_bool(b):
"""
>>> liberty_bool(True)
'true'
>>> liberty_bool(False)
'false'
>>> liberty_bool(1.0)
'true'
>>> liberty_bool(1.5)
Traceback (most recent call last):
...
ValueError: 1.5 is not a bool
>>> liberty_bool(0.0)
'false'
>>> liberty_bool(0)
'false'
>>> liberty_bool(1)
'true'
>>> liberty_bool("error")
Traceback (most recent call last):
...
ValueError: 'error' is not a bool
"""
try:
b2 = bool(b)
except ValueError:
b2 = None
if b2 != b:
raise ValueError("%r is not a bool" % b)
return {True: 'true', False: 'false'}[b]
def liberty_str(s):
"""
>>> liberty_str("hello")
'"hello"'
>>> liberty_str('he"llo')
Traceback (most recent call last):
...
ValueError: '"' is not allow in the string: 'he"llo'
>>> liberty_str(1.0)
'"1.0000000000"'
>>> liberty_str(1)
'"1.0000000000"'
>>> liberty_str([])
Traceback (most recent call last):
...
ValueError: [] is not a string
>>> liberty_str(True)
Traceback (most recent call last):
...
ValueError: True is not a string
"""
try:
if isinstance(s, (int, float)):
s = liberty_float(s)
except ValueError:
pass
if not isinstance(s, str):
raise ValueError("%r is not a string" % s)
if '"' in s:
raise ValueError("'\"' is not allow in the string: %r" % s)
return '"'+s+'"'
def liberty_int(f):
"""
>>> liberty_int(1.0)
1
>>> liberty_int(1.5)
Traceback (most recent call last):
...
ValueError: 1.5 is not an int
>>> liberty_int("error")
Traceback (most recent call last):
...
ValueError: 'error' is not an int
"""
try:
f2 = int(f)
except ValueError as e:
f2 = None
if f2 is None or f2 != f:
raise ValueError("%r is not an int" % f)
return int(f)
def liberty_float(f): def liberty_float(f):
""" """
@ -617,7 +753,42 @@ def liberty_float(f):
>>> liberty_float(1) >>> liberty_float(1)
'1.0000000000' '1.0000000000'
>>> liberty_float(True)
Traceback (most recent call last):
...
ValueError: True is not a float
>>> liberty_float(False)
Traceback (most recent call last):
...
ValueError: False is not a float
>>> liberty_float(0)
'0.0000000000'
>>> liberty_float(None)
Traceback (most recent call last):
...
ValueError: None is not a float
>>> liberty_float('hello')
Traceback (most recent call last):
...
ValueError: 'hello' is not a float
""" """
try:
f2 = float(f)
except (ValueError, TypeError):
f2 = None
if isinstance(f, bool):
f2 = None
if f is None or f2 != f:
raise ValueError("%r is not a float" % f)
WIDTH = len(str(0.0083333333)) WIDTH = len(str(0.0083333333))
s = json.dumps(f) s = json.dumps(f)
@ -639,6 +810,14 @@ def liberty_float(f):
return s return s
LIBERTY_ATTRIBUTE_TYPES = {
'boolean': liberty_bool,
'string': liberty_str,
'int': liberty_int,
'float': liberty_float,
}
INDENT=" " INDENT=" "
@ -738,8 +917,42 @@ def liberty_list(k, v, i=tuple()):
return o return o
def liberty_dict(dtype, dvalue, data, indent=tuple()): def liberty_dict(dtype, dvalue, data, indent=tuple(), attribute_types=None):
"""
>>> def g(a, b, c):
... return {"group_name": a, "attribute_name":b, "attribute_type": c}
>>> d = {'float': 1.0, "str": "str"}
>>> print('\\n'.join(liberty_dict("library", "test", d)))
library ("test") {
float : 1.0000000000;
str : "str";
}
>>> d['define'] = [g("cell", "float", "string")]
>>> print('\\n'.join(liberty_dict("library", "test", d)))
library ("test") {
define(float,cell,string);
float : 1.0000000000;
str : "str";
}
>>> d['define'] = [g("library", "float", "string")]
>>> print('\\n'.join(liberty_dict("library", "test", d)))
library ("test") {
define(float,library,string);
float : "1.0000000000";
str : "str";
}
"""
assert isinstance(data, dict), (dtype, dvalue, data) assert isinstance(data, dict), (dtype, dvalue, data)
if attribute_types is None:
attribute_types = {}
assert isinstance(attribute_types, dict), (dtype, dvalue, attribute_types)
o = [] o = []
if dvalue: if dvalue:
@ -785,24 +998,35 @@ def liberty_dict(dtype, dvalue, data, indent=tuple()):
# Output all the attributes # Output all the attributes
if dtype not in attribute_types:
dtype_attribute_types = {}
attribute_types[dtype] = dtype_attribute_types
dtype_attribute_types = attribute_types[dtype]
for _, ktype, _, kvalue, k, v in di: for _, ktype, _, kvalue, k, v in di:
indent_n = list(indent)+[k] indent_n = list(indent)+[k]
if ktype == 'define': if ktype == 'define':
for d in sorted(data['define'], key=lambda d: d['group_name']+'.'+d['attribute_name']): for d in sorted(data['define'], key=lambda d: d['group_name']+'.'+d['attribute_name']):
aname = d['attribute_name']
gname = d['group_name']
atype = d['attribute_type']
o.append('%sdefine(%s,%s,%s);' % ( o.append('%sdefine(%s,%s,%s);' % (
INDENT*len(indent_n), INDENT*len(indent_n), aname, gname, atype))
d['attribute_name'],
d['group_name'], assert atype in LIBERTY_ATTRIBUTE_TYPES, (atype, d)
d['attribute_type']), if gname not in attribute_types:
) attribute_types[gname] = {}
attribute_types[gname][aname] = LIBERTY_ATTRIBUTE_TYPES[atype]
elif ktype == "comp_attribute": elif ktype == "comp_attribute":
o.extend(liberty_composite(kvalue, v, indent_n)) o.extend(liberty_composite(kvalue, v, indent_n))
elif isinstance(v, dict): elif isinstance(v, dict):
assert isinstance(v, dict), (dtype, dvalue, k, v) assert isinstance(v, dict), (dtype, dvalue, k, v)
o.extend(liberty_dict(ktype, kvalue, v, indent_n)) o.extend(liberty_dict(ktype, kvalue, v, indent_n, attribute_types))
elif isinstance(v, list): elif isinstance(v, list):
assert len(v) > 0, (dtype, dvalue, k, v) assert len(v) > 0, (dtype, dvalue, k, v)
@ -811,7 +1035,7 @@ def liberty_dict(dtype, dvalue, data, indent=tuple()):
return o.items() return o.items()
for l in sorted(v, key=sk): for l in sorted(v, key=sk):
o.extend(liberty_dict(ktype, kvalue, l, indent_n)) o.extend(liberty_dict(ktype, kvalue, l, indent_n, attribute_types))
elif is_liberty_list(ktype): elif is_liberty_list(ktype):
o.extend(liberty_list(ktype, v, indent_n)) o.extend(liberty_list(ktype, v, indent_n))
@ -824,11 +1048,12 @@ def liberty_dict(dtype, dvalue, data, indent=tuple()):
raise ValueError("Unknown %s: %r\n%s" % (k, v, indent_n)) raise ValueError("Unknown %s: %r\n%s" % (k, v, indent_n))
else: else:
if isinstance(v, str): if ktype in dtype_attribute_types:
v = '"%s"' % v liberty_out = dtype_attribute_types[ktype]
elif isinstance(v, (float,int)): else:
v = liberty_float(v) liberty_out = liberty_guess(v)
o.append("%s%s : %s;" % (INDENT*len(indent_n), k, v)) ov = liberty_out(v)
o.append("%s%s : %s;" % (INDENT*len(indent_n), k, ov))
o.append("%s}" % (INDENT*len(indent))) o.append("%s}" % (INDENT*len(indent)))
return o return o