diff options
author | Clifford Wolf <clifford@clifford.at> | 2017-09-14 18:55:50 +0200 |
---|---|---|
committer | GitHub <noreply@github.com> | 2017-09-14 18:55:50 +0200 |
commit | 4749394e23350fdd67ca8c4dca31b827889193c9 (patch) | |
tree | b36b8fbd3ca7ec5f2499a71857690cf5cc4feedc | |
parent | 69613a26d3e3a5c6f0836a773717b03360df85ce (diff) | |
parent | c9c181fef21740b45653e23a4989d22530b543f3 (diff) | |
download | icestorm-4749394e23350fdd67ca8c4dca31b827889193c9.tar.gz icestorm-4749394e23350fdd67ca8c4dca31b827889193c9.tar.bz2 icestorm-4749394e23350fdd67ca8c4dca31b827889193c9.zip |
Merge pull request #97 from rlutz/hlc-fixes
Fixes to high-level configuration converters
-rw-r--r-- | icebox/Makefile | 11 | ||||
-rwxr-xr-x | icebox/icebox_asc2hlc.py | 1134 | ||||
-rwxr-xr-x | icebox/icebox_hlc2asc.py | 1063 | ||||
-rw-r--r-- | icebox/tc_logic_xpr.py | 44 | ||||
-rw-r--r-- | icebox/tc_rxlat_netnames.py | 71 | ||||
-rw-r--r-- | icebox/tc_xlat_netnames.py | 81 |
6 files changed, 2403 insertions, 1 deletions
diff --git a/icebox/Makefile b/icebox/Makefile index 430fb17..a6bd23b 100644 --- a/icebox/Makefile +++ b/icebox/Makefile @@ -18,6 +18,11 @@ chipdb-8k.txt: icebox.py iceboxdb.py icebox_chipdb.py python3 icebox_chipdb.py -8 > chipdb-8k.new mv chipdb-8k.new chipdb-8k.txt +check: all + python3 tc_xlat_netnames.py + python3 tc_rxlat_netnames.py + python3 tc_logic_xpr.py + clean: rm -f chipdb-1k.txt chipdb-8k.txt chipdb-384.txt chipdb-5k.txt rm -f icebox.pyc iceboxdb.pyc @@ -33,6 +38,8 @@ install: all cp icebox_chipdb.py $(DESTDIR)$(PREFIX)/bin/icebox_chipdb$(PY_EXE) cp icebox_diff.py $(DESTDIR)$(PREFIX)/bin/icebox_diff$(PY_EXE) cp icebox_explain.py $(DESTDIR)$(PREFIX)/bin/icebox_explain$(PY_EXE) + cp icebox_asc2hlc.py $(DESTDIR)$(PREFIX)/bin/icebox_asc2hlc$(PY_EXE) + cp icebox_hlc2asc.py $(DESTDIR)$(PREFIX)/bin/icebox_hlc2asc$(PY_EXE) cp icebox_colbuf.py $(DESTDIR)$(PREFIX)/bin/icebox_colbuf$(PY_EXE) cp icebox_html.py $(DESTDIR)$(PREFIX)/bin/icebox_html$(PY_EXE) cp icebox_maps.py $(DESTDIR)$(PREFIX)/bin/icebox_maps$(PY_EXE) @@ -45,6 +52,8 @@ uninstall: rm -f $(DESTDIR)$(PREFIX)/bin/icebox_chipdb$(PY_EXE) rm -f $(DESTDIR)$(PREFIX)/bin/icebox_diff$(PY_EXE) rm -f $(DESTDIR)$(PREFIX)/bin/icebox_explain$(PY_EXE) + rm -f $(DESTDIR)$(PREFIX)/bin/icebox_asc2hlc$(PY_EXE) + rm -f $(DESTDIR)$(PREFIX)/bin/icebox_hlc2asc$(PY_EXE) rm -f $(DESTDIR)$(PREFIX)/bin/icebox_colbuf$(PY_EXE) rm -f $(DESTDIR)$(PREFIX)/bin/icebox_html$(PY_EXE) rm -f $(DESTDIR)$(PREFIX)/bin/icebox_maps$(PY_EXE) @@ -55,4 +64,4 @@ uninstall: rm -f $(DESTDIR)$(PREFIX)/share/icebox/chipdb-8k.txt -rmdir $(DESTDIR)$(PREFIX)/share/icebox -.PHONY: all clean install uninstall +.PHONY: all check clean install uninstall diff --git a/icebox/icebox_asc2hlc.py b/icebox/icebox_asc2hlc.py new file mode 100755 index 0000000..4f0f6fa --- /dev/null +++ b/icebox/icebox_asc2hlc.py @@ -0,0 +1,1134 @@ +#!/usr/bin/env python3 +# Copyright (C) 2017 Roland Lutz +# +# Permission to use, copy, modify, and/or distribute this software for any +# purpose with or without fee is hereby granted, provided that the above +# copyright notice and this permission notice appear in all copies. +# +# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +import getopt, os, re, sys +import icebox + +GLB_NETWK_EXTERNAL_BLOCKS = [(13, 8, 1), (0, 8, 1), (7, 17, 0), (7, 0, 0), + (0, 9, 0), (13, 9, 0), (6, 0, 1), (6, 17, 1)] +GLB_NETWK_INTERNAL_TILES = [(7, 0), (7, 17), (13, 9), (0, 9), + (6, 17), (6, 0), (0, 8), (13, 8)] + + +## Get the global name of a net. +# +# \param x, y coordinates of the tile to which the net belongs +# \param fw, fh width and height of the tile fabric (excluding I/O tiles) +# \param net net name +# +# \return the global name of the net if it is a span wire, otherwise +# the unmodified net name +# +# There are 46624 span wires on the 1k (not counting dummies): +# +# span4_x[1..12]_g[1..20]_[0..11] +# span4_y[1..16]_g[1..16]_[0..11] +# span12_x[1..12]_g[1..28]_[0..1] +# span12_y[1..16]_g[1..24]_[0..1] +# +# span4_left_g[3..16]_[0..3] +# span4_right_g[5..18]_[0..3] +# span4_bottom_g[3..12]_[0..3] +# span4_top_g[5..14]_[0..3] +# +# span4_topleft[2,4,6,8]_[0..3] +# span4_bottomright[2,4,6,8]_[0..3] +# +# dummy_y[1..16]_g[0..3]_[0..11] +# +# "Dummy" nets are horizontal accesses to non-existing vertical span +# wires on the right edge which are listed by icebox but don't +# actually connect to anything outside the tile itself. + +def translate_netname(x, y, fw, fh, net): + def group_and_index(s, group_size): + n = int(s) + g = n // group_size + i = n % group_size + if g % 2 == 1: + i = i + 1 - (i % 2) * 2 + return g, i + + # logic and RAM tiles + + match = re.match(r'sp4_h_r_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + return 'span4_y%d_g%d_%d' % (y, x - g + 4, i) + match = re.match(r'sp4_h_l_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + return 'span4_y%d_g%d_%d' % (y, x - g + 3, i) + + match = re.match(r'sp4_v_b_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + return 'span4_x%d_g%d_%d' % (x, y + g, i) + match = re.match(r'sp4_v_t_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + return 'span4_x%d_g%d_%d' % (x, y + g + 1, i) + match = re.match(r'sp4_r_v_b_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + if x == fw: + # this net doesn't connect anywhere + return 'dummy_y%d_g%d_%d' % (y, g, i) + else: + return 'span4_x%d_g%d_%d' % (x + 1, y + g, i) + + match = re.match(r'sp12_h_r_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + return 'span12_y%d_g%d_%d' % (y, x - g + 12, i) + match = re.match(r'sp12_h_l_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + return 'span12_y%d_g%d_%d' % (y, x - g + 11, i) + + match = re.match(r'sp12_v_b_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + return 'span12_x%d_g%d_%d' % (x, y + g, i) + match = re.match(r'sp12_v_t_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + return 'span12_x%d_g%d_%d' % (x, y + g + 1, i) + + # I/O tiles + + match = re.match(r'span4_horz_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + if x == 0: + return 'span4_y%d_g%d_%d' % (y, x - g + 4, i) + else: + return 'span4_y%d_g%d_%d' % (y, x - g + 3, i) + + match = re.match(r'span4_vert_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 12) + if y == 0: + return 'span4_x%d_g%d_%d' % (x, y + g + 1, i) + else: + return 'span4_x%d_g%d_%d' % (x, y + g, i) + + match = re.match(r'span12_horz_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + if x == 0: + return 'span12_y%d_g%d_%d' % (y, x - g + 12, i) + else: + return 'span12_y%d_g%d_%d' % (y, x - g + 11, i) + + match = re.match(r'span12_vert_(\d+)$', net) + if match is not None: + g, i = group_and_index(match.group(1), 2) + if y == 0: + return 'span12_x%d_g%d_%d' % (x, y + g + 1, i) + else: + return 'span12_x%d_g%d_%d' % (x, y + g, i) + + # I/O tiles - peripheral wires + + match = re.match(r'span4_horz_r_(\d+)$', net) + if match is not None: + n = int(match.group(1)); g = n // 4; i = n % 4 + if y == 0: + if fw - x + g - 4 < 0: + return 'span4_bottomright%d_%d' % ((fw - x + 1 + g) * 2, i) + elif x - g + 1 < 0: + return 'span4_left_g%d_%d' % (-x + 1 + g, i) + else: + return 'span4_bottom_g%d_%d' % (x + 4 - g, i) + else: + if x - g - 1 < 0: + return 'span4_topleft%d_%d' % ((x + 4 - g) * 2, i) + elif x - g + 1 >= fw: + return 'span4_right_g%d_%d' % (fh + fw - x + 1 + g, i) + else: + return 'span4_top_g%d_%d' % (x + 4 - g, i) + + match = re.match(r'span4_horz_l_(\d+)$', net) + if match is not None: + n = int(match.group(1)); g = n // 4; i = n % 4 + if y == 0: + if x - g < 0: + return 'span4_left_g%d_%d' % (-x + 2 + g, i) + else: + return 'span4_bottom_g%d_%d' % (x + 3 - g, i) + else: + if x - g - 2 < 0: + return 'span4_topleft%d_%d' % ((x + 3 - g) * 2, i) + else: + return 'span4_top_g%d_%d' % (x + 3 - g, i) + + match = re.match(r'span4_vert_b_(\d+)$', net) + if match is not None: + n = int(match.group(1)); g = n // 4; i = n % 4 + if x == 0: + if y + g - 3 < 0: + return 'span4_bottom_g%d_%d' % (-y + 5 - g, i) + if fh - y - g < 0: + return 'span4_topleft%d_%d' % ((fh + 5 - y - g) * 2, i) + else: + return 'span4_left_g%d_%d' % (y + g, i) + else: + if y + g - 5 < 0: + return 'span4_bottomright%d_%d' % ((y + g) * 2, i) + elif y + g >= fh + 3: + return 'span4_top_g%d_%d' % (fw + fh + 5 - y - g, i) + else: + return 'span4_right_g%d_%d' % (y + g, i) + + match = re.match(r'span4_vert_t_(\d+)$', net) + if match is not None: + n = int(match.group(1)); g = n // 4; i = n % 4 + if x == 0: + if fh - y - g - 1 < 0: + return 'span4_topleft%d_%d' % ((fh + 4 - y - g) * 2, i) + else: + return 'span4_left_g%d_%d' % (y + g + 1, i) + else: + if y + g >= fh + 2: + return 'span4_top_g%d_%d' % (fw + fh + 4 - y - g, i) + else: + return 'span4_right_g%d_%d' % (y + g + 1, i) + + return net + +## Return the human-readable name of the \c fabout net of IO tile +## <tt>(x, y)</tt>. + +def lookup_fabout(x, y): + if (x, y) in GLB_NETWK_INTERNAL_TILES: + return 'glb_netwk_%d' % GLB_NETWK_INTERNAL_TILES.index((x, y)) + + return 'fabout' + + +## Remove an argument from a LUT string and an associated list of +## argument names. +# +# This is a helper function for \ref lut_to_logic_expression. +# +# \param lut string of 2^N `0' or `1' characters representing the +# logic of an Nx1 look-up table +# \param args list of N strings containing the human-readable names +# of the arguments +# \param i index of the argument to remove +# \param keep boolean value indicating which value of the removed +# argument is to be assumed in the resulting LUT +# +# \return a new pair <tt>(lut, args)</tt> with the argument removed + +def discard_argument(lut, args, i, keep): + assert len(lut) == 1 << len(args) + assert i >= 0 and i < len(args) + return ''.join(bit for j, bit in enumerate(lut) + if (j & (1 << i) != 0) == keep), \ + args[:i] + args[i + 1:] + +## Negate a tuple representation of a logic expression. +# +# This is a helper function for \ref lut_to_logic_expression. + +def negate_expr(expr): + if len(expr) == 2: + op, a = expr + assert op == 'not' + return a + if len(expr) != 3: + return 'not', expr + a, op, b = expr + if op == 'and': + return negate_expr(a), 'or', negate_expr(b) + if op == 'or': + return negate_expr(a), 'and', negate_expr(b) + assert op == 'xor' + if len(a) == 2 and a[0] == 'not': + return a[1], op, b + if len(b) == 2 and b[0] == 'not': + return a, op, b[1] + return negate_expr(a), op, b + +## Convert a tuple representation of a logic expression into a string. +# +# This is a helper function for \ref lut_to_logic_expression. +# +# \param expr the expression to convert +# \param parenthize whether a compound expression should be +# surrounded by parentheses + +def stringify(expr, parenthize): + if type(expr) == str: + return expr + assert type(expr) == tuple + + if len(expr) == 2: + op, a = expr + assert op == 'not' + assert type(a) == str + return "!" + a + + if len(expr) == 5: + a, op0, b, op1, c = expr + assert op0 == '?' and op1 == ':' + s = '%s ? %s : %s' % (stringify(a, False), stringify(b, False), + stringify(c, False)) + if parenthize: + return '(%s)' % s + return s + + assert len(expr) == 3 + + a, op, b = expr + l = [a, b] + i = 0 + while i < len(l): + if type(l[i]) == tuple and len(l[i]) == 3 and l[i][1] == op: + l = l[:i] + [l[i][0], l[i][2]] + l[i + 1:] + else: + i += 1 + + if op == 'and': + op = '&' + elif op == 'xor': + op = '^' + elif op == 'or': + op = '|' + + s = (' %s ' % op).join(stringify(x, True) for x in l) + if parenthize: + return '(%s)' % s + return s + +## Remove arguments which don't affect the result from a LUT string +## and an associated list of argument names. +# +# This is a helper function for \ref lut_to_logic_expression. +# +# \param lut string of 2^N `0' or `1' characters representing the +# logic of an Nx1 look-up table +# \param args list of N strings containing the human-readable names +# of the arguments +# +# \return a new pair <tt>(lut, args)</tt> with all unused arguments +# removed + +def discard_unused_arguments(lut, args): + assert len(lut) == 1 << len(args) + i = 0 + while i < len(args): + diff = False + for j in range(len(lut)): + if j & (1 << i) == 0 and lut[j] != lut[j | (1 << i)]: + diff = True + if not diff: + lut, args = discard_argument(lut, args, i, False) + else: + i += 1 + return lut, args + +## Convert a LUT string to a logic expression. +# +# \param lut string of 2^N `0' or `1' characters representing the +# logic of an Nx1 look-up table +# \param args list of N strings containing the human-readable names +# of the arguments +# +# \return a string containing a human-readable logic expression +# equivalent to the look-up table +# +# Example: lut_to_logic_expression('00010000', ['a', 'b', 'c']) -> 'a & b & !c' + +def lut_to_logic_expression(lut, args): + lut, args = discard_unused_arguments(lut, args) + + # filter out independent top-level arguments + toplevel_args = [] + i = 0 + while i < len(args) and len(args) >= 2: + ai_0 = set(bit for j, bit in enumerate(lut) if j & (1 << i) == 0) + ai_1 = set(bit for j, bit in enumerate(lut) if j & (1 << i) != 0) + assert len(ai_0) == 2 or len(ai_1) == 2 + + if len(ai_0) == 1: + # expression is constant if this argument is 0 + # e = (...) & arg or e = (...) | !arg + if tuple(ai_0)[0] == '0': + toplevel_args.append(('and', args[i])) + else: + toplevel_args.append(('or', ('not', args[i]))) + lut, args = discard_argument(lut, args, i, True) + i = 0 + continue + + if len(ai_1) == 1: + # expression is constant if this argument is 1 + # e = (...) & !arg or e = (...) | arg + if tuple(ai_1)[0] == '0': + toplevel_args.append(('and', ('not', args[i]))) + else: + toplevel_args.append(('or', args[i])) + lut, args = discard_argument(lut, args, i, False) + i = 0 + continue + + i += 1 + + i = 0 + while i < len(args) and len(args) >= 2: + is_xor = True + for j in range(len(lut)): + if j & (1 << i) == 0 and lut[j] == lut[j | (1 << i)]: + is_xor = False + break + + if is_xor: + toplevel_args.append(('xor', args[i])) + lut, args = discard_argument(lut, args, i, False) + continue + + i += 1 + + # detect simple top-level ternary conditions + i = 0 + while i < len(args) and len(args) >= 3: + j = i + 1 + while j < len(args): + ai_0_aj_0 = set(bit for k, bit in enumerate(lut) + if k & (1 << i) == 0 and k & (1 << j) == 0) + ai_0_aj_1 = set(bit for k, bit in enumerate(lut) + if k & (1 << i) == 0 and k & (1 << j) != 0) + ai_1_aj_0 = set(bit for k, bit in enumerate(lut) + if k & (1 << i) != 0 and k & (1 << j) == 0) + ai_1_aj_1 = set(bit for k, bit in enumerate(lut) + if k & (1 << i) != 0 and k & (1 << j) != 0) + assert len(ai_0_aj_0) == 2 or len(ai_0_aj_1) == 2 or \ + len(ai_1_aj_0) == 2 or len(ai_1_aj_1) == 2 + + if (len(ai_0_aj_0) == 2 or len(ai_0_aj_1) == 2) and \ + (len(ai_1_aj_0) == 2 or len(ai_1_aj_1) == 2) and \ + (len(ai_0_aj_0) == 2 or len(ai_1_aj_0) == 2) and \ + (len(ai_0_aj_1) == 2 or len(ai_1_aj_1) == 2): + j += 1 + continue + + ai_doesnt_matter_for_aj_0 = True + ai_doesnt_matter_for_aj_1 = True + aj_doesnt_matter_for_ai_0 = True + aj_doesnt_matter_for_ai_1 = True + + for k in range(len(lut)): + if k & (1 << i) != 0 or k & (1 << j) != 0: + continue + if lut[k] != lut[k | (1 << i)]: + ai_doesnt_matter_for_aj_0 = False + if lut[k | (1 << j)] != lut[k | (1 << i) | (1 << j)]: + ai_doesnt_matter_for_aj_1 = False + if lut[k] != lut[k | (1 << j)]: + aj_doesnt_matter_for_ai_0 = False + if lut[k | (1 << i)] != lut[k | (1 << i) | (1 << j)]: + aj_doesnt_matter_for_ai_1 = False + + if len(ai_0_aj_0) == 1 and len(ai_0_aj_1) == 1 and \ + aj_doesnt_matter_for_ai_1: + assert tuple(ai_0_aj_0)[0] != tuple(ai_0_aj_1)[0] + if tuple(ai_0_aj_0)[0] == '0': + toplevel_args.append((args[i], '?', ':', args[j])) + else: + toplevel_args.append((args[i], '?', ':', ('not', args[j]))) + lut, args = discard_argument(lut, args, i, True) + + # break loops + i = len(args) + j = len(args) + break + + if len(ai_1_aj_0) == 1 and len(ai_1_aj_1) == 1 and \ + aj_doesnt_matter_for_ai_0: + assert tuple(ai_1_aj_0)[0] != tuple(ai_1_aj_1)[0] + if tuple(ai_1_aj_0)[0] == '0': + toplevel_args.append((args[i], '?', args[j], ':')) + else: + toplevel_args.append((args[i], '?', ('not', args[j]), ':')) + lut, args = discard_argument(lut, args, i, False) + + # break loops + i = len(args) + j = len(args) + break + + if len(ai_0_aj_0) == 1 and len(ai_1_aj_0) == 1 and \ + ai_doesnt_matter_for_aj_1: + assert tuple(ai_0_aj_0)[0] != tuple(ai_1_aj_0)[0] + if tuple(ai_0_aj_0)[0] == '0': + toplevel_args.append((args[j], '?', ':', args[i])) + else: + toplevel_args.append((args[j], '?', ':', ('not', args[i]))) + lut, args = discard_argument(lut, args, j, True) + + # break loops + i = len(args) + j = len(args) + break + + if len(ai_0_aj_1) == 1 and len(ai_1_aj_1) == 1 and \ + ai_doesnt_matter_for_aj_0: + assert tuple(ai_0_aj_1)[0] != tuple(ai_1_aj_1)[0] + if tuple(ai_0_aj_1)[0] == '0': + toplevel_args.append((args[j], '?', args[i], ':')) + else: + toplevel_args.append((args[j], '?', ('not', args[i]), ':')) + lut, args = discard_argument(lut, args, j, False) + + # break loops + i = len(args) + j = len(args) + break + + j += 1 + i += 1 + + lut, args = discard_unused_arguments(lut, args) + + # group pairwise isolated arguments + i = 0 + while i < len(args): + j = i + 1 + while j < len(args): + ai_doesnt_matter_for_aj_0 = True + ai_doesnt_matter_for_aj_1 = True + aj_doesnt_matter_for_ai_0 = True + aj_doesnt_matter_for_ai_1 = True + both_dont_matter_if_equal = True + both_dont_matter_if_unequal = True + + for k in range(len(lut)): + if k & (1 << i) != 0 or k & (1 << j) != 0: + continue + if lut[k] != lut[k | (1 << i)]: + ai_doesnt_matter_for_aj_0 = False + if lut[k | (1 << j)] != lut[k | (1 << i) | (1 << j)]: + ai_doesnt_matter_for_aj_1 = False + if lut[k] != lut[k | (1 << j)]: + aj_doesnt_matter_for_ai_0 = False + if lut[k | (1 << i)] != lut[k | (1 << i) | (1 << j)]: + aj_doesnt_matter_for_ai_1 = False + if lut[k] != lut[k | (1 << i) | (1 << j)]: + both_dont_matter_if_equal = False + if lut[k | (1 << i)] != lut[k | (1 << j)]: + both_dont_matter_if_unequal = False + + # There are five possibilities of coupled arguments: one + # of the four combinations differs from the other three, + # or they are xor'ed + + if ai_doesnt_matter_for_aj_1 and \ + aj_doesnt_matter_for_ai_1 and \ + both_dont_matter_if_unequal: + # special case is ai=0 aj=0 + args = args[:i] + ((args[i], 'or', args[j]), ) + args[i + 1:] + lut, args = discard_argument(lut, args, j, False) + j = i + 1 + elif ai_doesnt_matter_for_aj_1 and \ + aj_doesnt_matter_for_ai_0 and \ + both_dont_matter_if_equal: + # special case is ai=1 aj=0 + args = args[:i] + ((args[i], 'and', negate_expr(args[j])), ) + \ + args[i + 1:] + lut, args = discard_argument(lut, args, j, False) + j = i + 1 + elif ai_doesnt_matter_for_aj_0 and \ + aj_doesnt_matter_for_ai_1 and \ + both_dont_matter_if_equal: + # special case is ai=0 aj=1 + args = args[:i] + ((args[i], 'or', negate_expr(args[j])), ) + \ + args[i + 1:] + lut, args = discard_argument(lut, args, j, True) + j = i + 1 + elif ai_doesnt_matter_for_aj_0 and \ + aj_doesnt_matter_for_ai_0 and \ + both_dont_matter_if_unequal: + # special case is ai=1 aj=1 + args = args[:i] + ((args[i], 'and', args[j]), ) + args[i + 1:] + lut, args = discard_argument(lut, args, j, True) + j = i + 1 + + elif both_dont_matter_if_equal and \ + both_dont_matter_if_unequal: + args = args[:i] + ((args[i], 'xor', args[j]), ) + args[i + 1:] + lut, args = discard_argument(lut, args, j, False) + j = i + 1 + else: + j += 1 + i += 1 + + # collect the result + + if not args: + # constant expression + assert len(lut) == 1 + return lut + + negate_result = lut.count('1') > lut.count('0') + if negate_result: + lut = ''.join('1' if bit == '0' else '0' for bit in lut) + + result = None + for i, bit in enumerate(lut): + if bit == '0': + continue + expr = None + for j, arg in enumerate(args): + if i & (1 << j) == 0: + arg = negate_expr(arg) + if expr is None: + expr = arg + else: + expr = (expr, 'and', arg) + if result is None: + result = expr + else: + result = (result, 'or', expr) + + if negate_result: + result = negate_expr(result) + + for toplevel_arg in reversed(toplevel_args): + if len(toplevel_arg) != 4: + result = tuple(reversed(toplevel_arg)) + (result, ) + elif toplevel_arg[2] == ':': + result = toplevel_arg[0:2] + (result, ) + toplevel_arg[2:4] + else: + assert toplevel_arg[3] == ':' + result = toplevel_arg + (result, ) + + return stringify(result, False) + + +class Fabric: + def __init__(self, ic): + self.ic = ic + self.tiles = {} + #self.colbuf = set() + + io_blocks = {} + ieren_blocks = {} + + for x0, y0, b0, x1, y1, b1 in self.ic.ieren_db(): + i = IOBlock() + assert (x0, y0, b0) not in io_blocks + io_blocks[x0, y0, b0] = i + assert (x1, y1, b1) not in ieren_blocks + ieren_blocks[x1, y1, b1] = i + + for xy in ic.io_tiles: + assert xy not in self.tiles + self.tiles[xy] = IOTile(self, xy, + (io_blocks.pop((xy[0], xy[1], 0), None), + io_blocks.pop((xy[0], xy[1], 1), None)), + (ieren_blocks.pop((xy[0], xy[1], 0), None), + ieren_blocks.pop((xy[0], xy[1], 1), None))) + assert not io_blocks + assert not ieren_blocks + + for xy in ic.logic_tiles: + assert xy not in self.tiles + self.tiles[xy] = LogicTile(self, xy) + + for xy in ic.ramb_tiles: + assert xy not in self.tiles + self.tiles[xy] = RAMBTile(self, xy) + + for xy in ic.ramt_tiles: + assert xy not in self.tiles + self.tiles[xy] = RAMTTile(self, xy) + + for x, y in self.tiles: + assert x >= 0 and x <= self.ic.max_x + assert y >= 0 and y <= self.ic.max_y + for x in range(self.ic.max_x + 1): + for y in range(self.ic.max_y + 1): + should_exist = (x > 0 and x < self.ic.max_x) or \ + (y > 0 and y < self.ic.max_y) + assert ((x, y) in self.tiles) == should_exist + + for xy in ic.ram_data: + assert type(self.tiles.get(xy, None)) == RAMBTile + + #colbuf_db = ic.colbuf_db() + #for x, y, i in self.colbuf: + # exists = False + # for src_x, src_y, dst_x, dst_y in colbuf_db: + # if src_x != x or src_y != y: + # continue + # assert (dst_x, dst_y) in self.tiles + # assert not self.tiles[dst_x, dst_y].colbuf[i] + # self.tiles[dst_x, dst_y].colbuf[i] = True + # exists = True + # assert exists + # + #for xy in self.tiles: + # for br in self.tiles[xy].buffer_and_routing: + # if br[0].startswith('glb_netwk_'): + # assert self.tiles[xy].colbuf[int(br[0][10:])] + + for bit in self.ic.extra_bits: + directive, arg = self.ic.lookup_extra_bit(bit) + assert directive == 'padin_glb_netwk' + x, y, n = GLB_NETWK_EXTERNAL_BLOCKS[int(arg)] + assert type(self.tiles.get((x, y), None)) == IOTile + block = self.tiles[x, y].io_blocks[n] + assert block is not None + block.padin_glb_netwk = True + + def printout(self, options): + print('device "%s" %d %d' % (self.ic.device, self.ic.max_x - 1, + self.ic.max_y - 1)) + + print('') + # internal_configuration_oscillator_frequency = low | medium | high + #print('coldboot = off') + print('warmboot = on') # IceStorm assumes this to be always on + + for xy in sorted(self.tiles.keys(), key = lambda xy: (xy[1], xy[0])): + self.tiles[xy].printout(options) + +class Tile: + def __init__(self, fabric, xy, data, is_logic_block): + self.fabric = fabric + self.ic = fabric.ic + self.xy = xy + self.data = data + + self.buffer_and_routing = set() + self.used_buffer_and_routing = set() + self.text = set() + self.bitinfo = list() + self.unknown_bits = False + + x, y = xy + db = self.ic.tile_db(x, y) + mapped_bits = set() + + # 'data' is a list of strings containing a series of zeroes and + # ones. 'bits' is a set of strings containing an entry + # "B<row>[<col>]" or "!B<row>[<col>]" for each bit. + + bits = set() + for k, line in enumerate(data): + for i in range(len(line)): + if line[i] == '1': + bits.add('B%d[%d]' % (k, i)) + else: + bits.add('!B%d[%d]' % (k, i)) + + for entry in db: + # LC bits don't have a useful entry in the database; skip them + # for now + if re.match(r'LC_', entry[1]): + continue + + # some nets have different names depending on the tile; filter + # out non-applicable net names + if entry[1] in ('routing', 'buffer') and ( + not self.ic.tile_has_net(x, y, entry[2]) or + not self.ic.tile_has_net(x, y, entry[3])): + continue + + # are all required bits set/unset? + match = True + for bit in entry[0]: + if not bit in bits: + match = False + if match: + for bit in entry[0]: + mapped_bits.add(bit) + + if entry[1:] == ['IoCtrl', 'IE_0']: + if match != (self.ic.device == '1k'): + self.ieren_blocks[0].enable_input = True + continue + if entry[1:] == ['IoCtrl', 'REN_0']: + if match: + self.ieren_blocks[0].disable_pull_up = True + continue + if entry[1:] == ['IoCtrl', 'IE_1']: + if match != (self.ic.device == '1k'): + self.ieren_blocks[1].enable_input = True + continue + if entry[1:] == ['IoCtrl', 'REN_1']: + if match: + self.ieren_blocks[1].disable_pull_up = True + continue + + if entry[1].startswith('IOB_') and entry[2].startswith('PINTYPE_'): + if match: + self.io_blocks[int(entry[1][4:])].pintype \ + |= 1 << int(entry[2][8:]) + continue + + if entry[1:] == ['RamConfig', 'PowerUp']: + if match != (self.ic.device == '1k'): + self.text.add('power_up') + continue + + if entry[1] == 'routing': + if match: + src = translate_netname(self.xy[0], self.xy[1], + self.ic.max_x - 1, + self.ic.max_y - 1, entry[2]) + dst = translate_netname(self.xy[0], self.xy[1], + self.ic.max_x - 1, + self.ic.max_y - 1, entry[3]) + if dst == 'fabout': + dst = lookup_fabout(*self.xy) + self.buffer_and_routing.add((src, '<->', dst)) + continue + if entry[1] == 'buffer': + if match: + src = translate_netname(self.xy[0], self.xy[1], + self.ic.max_x - 1, + self.ic.max_y - 1, entry[2]) + dst = translate_netname(self.xy[0], self.xy[1], + self.ic.max_x - 1, + self.ic.max_y - 1, entry[3]) + if dst == 'fabout': + dst = lookup_fabout(*self.xy) + self.buffer_and_routing.add((src, '->', dst)) + continue + + if entry[1] == 'ColBufCtrl': + assert entry[2].startswith('glb_netwk_') + #if match: + # fabric.colbuf.add(self.xy + (int(entry[2][10:]), )) + continue + + if match: + self.text.add(' '.join(entry[1:])) + + for prefix in ('local_', 'glb2local_'): + for fst in [fst for fst in self.buffer_and_routing + if fst[-1].startswith(prefix)]: + used = False + for snd in [snd for snd in self.buffer_and_routing + if snd[0] == fst[-1]]: + self.buffer_and_routing.remove(snd) + self.buffer_and_routing.add(fst[:-1] + snd) + used = True + if used: + self.buffer_and_routing.remove(fst) + + for k, line in enumerate(data): + self.bitinfo.append('') + extra_text = '' + for i in range(len(line)): + if 36 <= i <= 45 and is_logic_block: + self.bitinfo[-1] += '*' if line[i] == '1' else '-' + elif line[i] == '1' and 'B%d[%d]' % (k, i) not in mapped_bits: + self.unknown_bits = True + extra_text += ' B%d[%d]' % (k, i) + self.bitinfo[-1] += '?' + else: + self.bitinfo[-1] += '+' if line[i] == '1' else '-' + self.bitinfo[-1] += extra_text + + def get_hlc(self): + return sorted(set.union(self.text, + set(' '.join(t) + for t in set.difference( + self.buffer_and_routing, + self.used_buffer_and_routing)))) + + def printout(self, stmt, options): + text = self.get_hlc() + if text or self.unknown_bits or options.print_all: + if self.unknown_bits or options.print_map: + print() + if self.unknown_bits: + print("; Warning: No DB entries for some bits:") + for k, line in enumerate(self.bitinfo): + print("; %4s %s" % ('B%d' % k, line)) + print() + print("%s %d %d {" % (stmt, self.xy[0], self.xy[1])) + for line in text: + print(" " + line) + print("}") + +class LogicCell: + def __init__(self, tile, lcidx): + self.lut = ''.join(icebox.get_lutff_lut_bits(tile.data, lcidx)) + self.expr = lut_to_logic_expression( + self.lut, ('in_0', 'in_1', 'in_2', 'in_3')) + + self.options = [] + lutff_option_bits = ''.join(icebox.get_lutff_seq_bits(tile.data, lcidx)) + if lutff_option_bits[0] == '1': self.options.append('enable_carry') + if lutff_option_bits[1] == '1': self.options.append('enable_dff') + if lutff_option_bits[2] == '1': self.options.append('set_noreset') + if lutff_option_bits[3] == '1': self.options.append('async_setreset') + + self.buffer_and_routing0 = set() + self.buffer_and_routing1 = set() + for br in tuple(tile.buffer_and_routing): + if br[0] == 'lutff_%d/out' % lcidx: + self.buffer_and_routing1.add((br[0][8:], ) + br[1:]) + tile.used_buffer_and_routing.add(br) + elif br[-1].startswith('lutff_%d/' % lcidx): + self.buffer_and_routing0.add(br[:-1] + (br[-1][8:], )) + tile.used_buffer_and_routing.add(br) + + def get_hlc(self): + if self.lut == '0000000000000000' and not self.options: + t = [] + elif len(self.expr) > 64: + t = ['lut ' + self.lut] + else: + t = ['out = ' + self.expr] + return [' '.join(t) for t in sorted(self.buffer_and_routing0, + key = lambda x: x[-1])] + \ + t + self.options + \ + [' '.join(t) for t in sorted(self.buffer_and_routing1, + key = lambda x: x[-1])] + +class LogicTile(Tile): + def __init__(self, fabric, xy): + super().__init__(fabric, xy, fabric.ic.logic_tiles[xy], True) + self.cells = tuple(LogicCell(self, lcidx) for lcidx in range(8)) + + def get_hlc(self): + text = super().get_hlc() + + for i, cell in reversed(tuple(enumerate(self.cells))): + t = cell.get_hlc() + if t: + text = ['lutff_%d {' % i] + \ + [' %s' % s for s in t] + \ + ['}'] + \ + text + + return text + + def printout(self, options): + super().printout('logic_tile', options) + +class IOBlock: + def __init__(self): + # stored in the I/O tile where this block is located + self.pintype = 0 + + # stored in the I/O tile where this is an IE/REN block + self.enable_input = False + self.disable_pull_up = False + + # stored as an extra bit + self.padin_glb_netwk = False + +class IOTile(Tile): + def __init__(self, fabric, xy, io_blocks, ieren_blocks): + self.io_blocks = io_blocks + self.ieren_blocks = ieren_blocks + super().__init__(fabric, xy, fabric.ic.io_tiles[xy], False) + #self.cells = tuple(IOCell() for i in range(2)) + + for i, block in enumerate(io_blocks): + if block is None: + continue + block.buffer_and_routing0 = set() + block.buffer_and_routing1 = set() + for br in tuple(self.buffer_and_routing): + if br[0].startswith('io_%d/D_IN_' % i): + block.buffer_and_routing1.add((br[0][5:], ) + br[1:]) + self.used_buffer_and_routing.add(br) + elif br[-1].startswith('io_%d/' % i): + block.buffer_and_routing0.add(br[:-1] + (br[-1][5:], )) + self.used_buffer_and_routing.add(br) + + def get_hlc(self): + # if io_blocks[N] is None, this means there's no I/O pin there + + text = super().get_hlc() + for n in (1, 0): + block = self.io_blocks[n] + if block is None: + continue + + t = [] + input_pt = block.pintype & 3 + output_pt = block.pintype >> 2 & 15 + unknown_pt = block.pintype >> 6 + if input_pt != 0: + t.append('input_pin_type = %s' % ( + 'registered_pin', + 'simple_input_pin', + 'latched_registered_pin', + 'latched_pin')[input_pt]) + if output_pt != 0: + t.append('output_pin_type = %s' % ( + 'no_output', + '1', + '2', + '3', + 'DDR', + 'REGISTERED', + 'simple_output_pin', + 'REGISTERED_INVERTED', + 'DDR_ENABLE', + 'REGISTERED_ENABLE', + 'OUTPUT_TRISTATE', + 'REGISTERED_ENABLE_INVERTED', + 'DDR_ENABLE_REGISTERED', + 'REGISTERED_ENABLE_REGISTERED', + 'ENABLE_REGISTERED', + 'REGISTERED_ENABLE_REGISTERED_INVERTED')[output_pt]) + if unknown_pt != 0: + t.append('unknown_pin_type = %d' % unknown_pt) + if block.enable_input: + t.append('enable_input') + if block.disable_pull_up: + t.append('disable_pull_up') + + t += [' '.join(t) for t in sorted(block.buffer_and_routing0, + key = lambda x: x[-1])] + t += [' '.join(t) for t in sorted(block.buffer_and_routing1, + key = lambda x: x[0])] + if block.padin_glb_netwk: + t += ['GLOBAL_BUFFER_OUTPUT -> glb_netwk_%d' + % GLB_NETWK_EXTERNAL_BLOCKS.index(self.xy + (n, ))] + + if t: + text = ['io_%d {' % n] + \ + [' %s' % s for s in t] + \ + ['}'] + \ + text + + return text + + def printout(self, options): + super().printout('io_tile', options) + +class IOCell: + pass + +class RAMBTile(Tile): + def __init__(self, fabric, xy): + super().__init__(fabric, xy, fabric.ic.ramb_tiles[xy], False) + if xy in fabric.ic.ram_data: + self.data = fabric.ic.ram_data[xy] + else: + self.data = None + + def get_hlc(self): + text = super().get_hlc() + if self.data is not None: + text.append('') + text.append('data {') + for line in self.data: + text.append(' ' + line) + text.append('}') + return text + + def printout(self, options): + super().printout('ramb_tile', options) + +class RAMTTile(Tile): + def __init__(self, fabric, xy): + super().__init__(fabric, xy, fabric.ic.ramt_tiles[xy], False) + + def printout(self, options): + super().printout('ramt_tile', options) + + +class Options: + def __init__(self): + self.print_map = False + self.print_all = False + +def main(): + program_short_name = os.path.basename(sys.argv[0]) + options = Options() + + try: + opts, args = getopt.getopt(sys.argv[1:], 'mA', ['help', 'version']) + except getopt.GetoptError as e: + sys.stderr.write("%s: %s\n" % (program_short_name, e.msg)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + for opt, arg in opts: + if opt == '--help': + sys.stderr.write("""\ +Create a high-level representation from an ASCII bitstream. +Usage: %s [OPTION]... FILE + + -m print tile config bitmaps + -A don't skip uninteresting tiles + + --help display this help and exit + --version output version information and exit + +If you have a bug report, please file an issue on github: + https://github.com/rlutz/icestorm/issues +""" % sys.argv[0]) + sys.exit(0) + + if opt == '--version': + sys.stderr.write("""\ +icebox_asc2hlc - create a high-level representation from an ASCII bitstream +Copyright (C) 2017 Roland Lutz + +This program is free software: you can redistribute it and/or +modify it under the terms of the GNU General Public License as +published by the Free Software Foundation, either version 3 of +the License, or (at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. +""") + sys.exit(0) + + if opt == '-m': + options.print_map = True + elif opt == '-A': + options.print_all = True + + if not args: + sys.stderr.write("%s: missing argument\n" % (program_short_name)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + if len(args) != 1: + sys.stderr.write("%s: too many arguments\n" % (program_short_name)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + ic = icebox.iceconfig() + if args[0] == '-': + ic.read_file('/dev/stdin') + else: + ic.read_file(args[0]) + + fabric = Fabric(ic) + fabric.printout(options) + +if __name__ == '__main__': + main() diff --git a/icebox/icebox_hlc2asc.py b/icebox/icebox_hlc2asc.py new file mode 100755 index 0000000..0982942 --- /dev/null +++ b/icebox/icebox_hlc2asc.py @@ -0,0 +1,1063 @@ +#!/usr/bin/env python3 +# Copyright (C) 2017 Roland Lutz +# +# Permission to use, copy, modify, and/or distribute this software for any +# purpose with or without fee is hereby granted, provided that the above +# copyright notice and this permission notice appear in all copies. +# +# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +import getopt, os, re, sys +import icebox + +GLB_NETWK_EXTERNAL_BLOCKS = [(13, 8, 1), (0, 8, 1), (7, 17, 0), (7, 0, 0), + (0, 9, 0), (13, 9, 0), (6, 0, 1), (6, 17, 1)] +GLB_NETWK_INTERNAL_TILES = [(7, 0), (7, 17), (13, 9), (0, 9), + (6, 17), (6, 0), (0, 8), (13, 8)] + + +## Get the tile-local name of a net. +# +# \param x, y coordinates of the tile to which the net belongs +# \param fw, fh width and height of the tile fabric (excluding I/O tiles) +# \param net global net name +# +# \return the tile-local name of the net if it is a span wire, +# otherwise the unmodified net name + +def untranslate_netname(x, y, fw, fh, net): + def index(g, i, group_size): + if g % 2 == 1: + i = i + 1 - (i % 2) * 2 + return g * group_size + i + + match = re.match(r'span4_y(\d+)_g(\d+)_(\d+)$', net) + if match is not None: + my = int(match.group(1)) + mw = int(match.group(2)) + mi = int(match.group(3)) + assert my == y + assert mi >= 0 and mi < 12 + + mg = x - mw + 4 + assert mg >= 0 and mg <= 4 + + if x == 0: + return 'span4_horz_%d' % index(mg, mi, 12) + if x == fw + 1: + return 'span4_horz_%d' % index(mg - 1, mi, 12) + + if mg == 4: + return 'sp4_h_l_%d' % index(mg - 1, mi, 12) + else: + return 'sp4_h_r_%d' % index(mg, mi, 12) + + match = re.match(r'span4_x(\d+)_g(\d+)_(\d+)$', net) + if match is not None: + mx = int(match.group(1)) + mw = int(match.group(2)) + mi = int(match.group(3)) + assert mi >= 0 and mi < 12 + mg = mw - y + assert mg >= 0 and mg <= 4 + + if y == 0: + return 'span4_vert_%d' % index(mg - 1, mi, 12) + if y == fh + 1: + return 'span4_vert_%d' % index(mg, mi, 12) + + if mx == x + 1: + assert mg < 4 + return 'sp4_r_v_b_%d' % index(mg, mi, 12) + + assert mx == x + if mg == 4: + return 'sp4_v_t_%d' % index(mg - 1, mi, 12) + else: + return 'sp4_v_b_%d' % index(mg, mi, 12) + + match = re.match(r'dummy_y(\d+)_g(\d+)_(\d+)$', net) + if match is not None: + my = int(match.group(1)) + mw = int(match.group(2)) + mi = int(match.group(3)) + assert my == y + + mg = mw + assert mg >= 0 and mg < 4 + + return 'sp4_r_v_b_%d' % index(mg, mi, 12) + + match = re.match(r'span12_y(\d+)_g(\d+)_(\d+)$', net) + if match is not None: + my = int(match.group(1)) + mw = int(match.group(2)) + mi = int(match.group(3)) + assert my == y + assert mi >= 0 and mi < 2 + + mg = x - mw + 12 + assert mg >= 0 and mg <= 12 + + if x == 0: + return 'span12_horz_%d' % index(mg, mi, 2) + if x == fw + 1: + return 'span12_horz_%d' % index(mg - 1, mi, 2) + + if mg == 12: + return 'sp12_h_l_%d' % index(mg - 1, mi, 2) + else: + return 'sp12_h_r_%d' % index(mg, mi, 2) + + match = re.match(r'span12_x(\d+)_g(\d+)_(\d+)$', net) + if match is not None: + mx = int(match.group(1)) + mw = int(match.group(2)) + mi = int(match.group(3)) + assert mx == x + assert mi >= 0 and mi < 2 + + mg = mw - y + assert mg >= 0 and mg <= 12 + + if y == 0: + return 'span12_vert_%d' % index(mg - 1, mi, 2) + if y == fh + 1: + return 'span12_vert_%d' % index(mg, mi, 2) + + if mg == 12: + return 'sp12_v_t_%d' % index(mg - 1, mi, 2) + else: + return 'sp12_v_b_%d' % index(mg, mi, 2) + + match = re.match(r'span4_bottom_g(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mi >= 0 and mi < 4 + + if x == 0: + assert y != 0 + mg = -y + 5 - mw + assert y + mg - 3 < 0 + return 'span4_vert_b_%d' % (mg * 4 + mi) + else: + assert y == 0 + mg = x + 4 - mw + assert x - mg + 1 >= 0 + + if mg == 4: + return 'span4_horz_l_%d' % (mg * 4 + mi - 4) + else: + assert fw - x + mg - 4 >= 0 + return 'span4_horz_r_%d' % (mg * 4 + mi) + + match = re.match(r'span4_left_g(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mi >= 0 and mi < 4 + + if y == 0: + assert x != 0 + mg = mw + x - 1 + assert x - mg + 1 < 0 + + if mg == 4: + return 'span4_horz_l_%d' % (mg * 4 + mi - 4) + else: + assert fw - x + mg - 4 >= 0 + return 'span4_horz_r_%d' % (mg * 4 + mi) + + else: + assert x == 0 + mg = mw - y + assert fh - y - mg >= 0 + + if mg == 4: + return 'span4_vert_t_%d' % (mg * 4 + mi - 4) + else: + assert y + mg - 3 >= 0 + return 'span4_vert_b_%d' % (mg * 4 + mi) + + match = re.match(r'span4_right_g(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mi >= 0 and mi < 4 + + if y == fh + 1: + mg = mw - fh - fw + x - 1 + assert x - mg - 1 >= 0 + assert x - mg + 1 >= fw + return 'span4_horz_r_%d' % (mg * 4 + mi) + + assert x == fw + 1 + mg = mw - y + + if mg == 4: + assert y + mg - 1 < fh + 2 + return 'span4_vert_t_%d' % (mg * 4 + mi - 4) + else: + assert y + mg - 5 >= 0 + assert y + mg < fh + 3 + return 'span4_vert_b_%d' % (mg * 4 + mi) + + match = re.match(r'span4_top_g(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mi >= 0 and mi < 4 + + if x == fw + 1: + assert y != 0 + mg = fw + fh + 5 - y - mw + assert y + mg >= fh + 3 + + if mg == 4: + return 'span4_vert_t_%d' % (mg * 4 + mi - 4) + else: + assert y + mg - 5 >= 0 + return 'span4_vert_b_%d' % (mg * 4 + mi) + + assert y != 0 + mg = x + 4 - mw + assert x - mg - 1 >= 0 + + if mg == 4: + return 'span4_horz_l_%d' % (mg * 4 + mi - 4) + else: + assert x - mg + 1 < fw + return 'span4_horz_r_%d' % (mg * 4 + mi) + + match = re.match(r'span4_bottomright(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mw % 2 == 0 + assert mi >= 0 and mi < 4 + + if y == 0: + assert x != 0 + mg = mw // 2 - fw + x - 1 + assert fw - x + mg - 4 < 0 + return 'span4_horz_r_%d' % (mg * 4 + mi) + else: + assert x == fw + 1 + mg = mw // 2 - y + assert y + mg - 5 < 0 + return 'span4_vert_b_%d' % (mg * 4 + mi) + + match = re.match(r'span4_topleft(\d+)_(\d+)$', net) + if match is not None: + mw = int(match.group(1)) + mi = int(match.group(2)) + assert mw % 2 == 0 + assert mi >= 0 and mi < 4 + + if x == 0: + assert y != 0 + mg = fh + 5 - y - mw // 2 + assert fh - y - mg < 0 + + if mg == 4: + return 'span4_vert_t_%d' % (mg * 4 + mi - 4) + else: + return 'span4_vert_b_%d' % (mg * 4 + mi) + else: + assert y == fh + 1 + mg = x + 4 - mw // 2 + assert x - mg - 1 < 0 + + if mg == 4: + return 'span4_horz_l_%d' % (mg * 4 + mi - 4) + else: + return 'span4_horz_r_%d' % (mg * 4 + mi) + + return net + +## Check if the name of a destination net is the human-readable form +## of the \c fabout net of IO tile <tt>(x, y)</tt>. +# +# \return \c 'fabout' if it is the \c fabout net, otherwise the +# unchanged net name + +def revert_to_fabout(x, y, net): + if net.startswith('glb_netwk_'): + for i, xy in enumerate(GLB_NETWK_INTERNAL_TILES): + if net == 'glb_netwk_%d' % i and (x, y) == xy: + return 'fabout' + raise ParseError + + return net + + +EXPR_AND, EXPR_XOR, EXPR_OR, EXPR_TERN, EXPR_NOT, EXPR_ZERO, EXPR_ONE = range(7) + +## Evaluate a list representation of a logic expression for given +## input values. +# +# This is a helper function for \ref logic_expression_to_lut. +# +# \param expr list representation of a logic expression (see below) +# \param args list of boolean values representing the input values +# +# \result \c False or \c True, depending on the expression and arguments +# +# Expression | Result +# ---------------------------------------+---------------------------------- +# <tt>i</tt> | value of argument \a i +# <tt>(EXPR_AND, [expr, ...])</tt> | AND operation of all expressions +# <tt>(EXPR_XOR, [expr, ...])</tt> | XOR operation of all expressions +# <tt>(EXPR_OR, [expr, ...])</tt> | OR operation of all expressions +# <tt>(EXPR_TERN, ex_a, ex_b, ex_c)</tt> | result of \c ex_b if \c ex_a +# | evaluates to \c True, otherwise +# | result of \c ex_c +# <tt>(EXPR_NOT, expr)</tt> | negated result of \a expr +# <tt>(EXPR_ZERO, )</tt> | \c False +# <tt>(EXPR_ONE, )</tt> | \c True + +def evaluate(expr, args): + if type(expr) == int: + return args[expr] + + op = expr[0] + + if op == EXPR_AND: + assert len(expr) == 2 + for o in expr[1]: + if not evaluate(o, args): + return False + return True + + if op == EXPR_XOR: + assert len(expr) == 2 + result = False + for o in expr[1]: + if evaluate(o, args): + result = not result + return result + + if op == EXPR_OR: + assert len(expr) == 2 + for o in expr[1]: + if evaluate(o, args): + return True + return False + + if op == EXPR_TERN: + assert len(expr) == 4 + if evaluate(expr[1], args): + return evaluate(expr[2], args) + else: + return evaluate(expr[3], args) + + if op == EXPR_NOT: + assert len(expr) == 2 + return not evaluate(expr[1], args) + + if op == EXPR_ZERO: + assert len(expr) == 1 + return False + + if op == EXPR_ONE: + assert len(expr) == 1 + return True + + assert False # unknown operator + +## Convert a logic expression to a LUT string. +# +# \param lut string containing a human-readable logic expression +# \param args list of N strings containing the names of the arguments +# +# \return a string of 2^N `0' or `1' characters representing the logic +# of an Nx1 look-up table equivalent to the logic expression +# +# Example: logic_expression_to_lut('a & b & !c', ['a', 'b', 'c']) -> '00010000' + +def logic_expression_to_lut(s, args): + # make sure argument names are unique + assert len(set(args)) == len(args) + + stack = [[None, None, None], [[], None, None]] + stack[0][2] = l = []; stack[1][0].append((EXPR_OR, l)) + stack[0][1] = l = []; stack[0][2].append((EXPR_XOR, l)) + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + expect_expr = True + negate_count = 0 + + i = 0 + while i < len(s): + if s[i] == ' ': + pass + elif s[i] == '!': + assert expect_expr + negate_count += 1 + elif s[i] == '&': + assert not expect_expr + + expect_expr = True + negate_count = 0 + elif s[i] == '^': + assert not expect_expr + + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + + expect_expr = True + negate_count = 0 + elif s[i] == '|': + assert not expect_expr + + stack[0][1] = l = []; stack[0][2].append((EXPR_XOR, l)) + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + + expect_expr = True + negate_count = 0 + elif s[i] == '?': + assert not expect_expr + assert stack[1][0][-1][0] == EXPR_OR + stack[1][0][-1] = (EXPR_TERN, stack[1][0][-1], (EXPR_OR, []), + (EXPR_OR, [])) + stack[0][2] = l = stack[1][0][-1][2][1] + stack[0][1] = l = []; stack[0][2].append((EXPR_XOR, l)) + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + + expect_expr = True + negate_count = 0 + elif s[i] == ':': + assert not expect_expr + assert stack[1][0][-1][0] == EXPR_TERN + stack[0][2] = l = stack[1][0][-1][3][1] + stack[0][1] = l = []; stack[0][2].append((EXPR_XOR, l)) + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + + expect_expr = True + negate_count = 0 + elif s[i] == '(': + assert expect_expr + + stack.insert(0, [None, None, None]) + + stack[0][2] = l = [] + if negate_count % 2: + stack[1][0].append((EXPR_NOT, (EXPR_OR, l))) + else: + stack[1][0].append((EXPR_OR, l)) + + stack[0][1] = l = []; stack[0][2].append((EXPR_XOR, l)) + stack[0][0] = l = []; stack[0][1].append((EXPR_AND, l)) + + expect_expr = True + negate_count = 0 + + elif s[i] == ')': + assert not expect_expr + stack.pop(0) + + elif s[i] == '0': + assert expect_expr + if negate_count % 2: + stack[0][0].append((EXPR_ONE, )) + else: + stack[0][0].append((EXPR_ZERO, )) + + expect_expr = False + negate_count = None + + elif s[i] == '1': + assert expect_expr + if negate_count % 2: + stack[0][0].append((EXPR_ZERO, )) + else: + stack[0][0].append((EXPR_ONE, )) + + expect_expr = False + negate_count = None + + else: + assert expect_expr + + found = None + for j, arg in enumerate(args): + if s.startswith(arg, i): + found = j + i += len(arg) + break + assert found is not None + + if negate_count % 2: + stack[0][0].append((EXPR_NOT, found)) + else: + stack[0][0].append(found) + + expect_expr = False + negate_count = None + continue + + i += 1 + + assert len(stack) == 2 + return ''.join('1' if evaluate(stack[1][0][0], + tuple(i & (1 << j) != 0 + for j in range(len(args)))) else '0' + for i in range(1 << len(args))) + + +class ParseError(Exception): + pass + +def parse_bool(s): + if s == 'on': + return True + if s == 'off': + return False + raise ParseError + +class Main: + def __init__(self): + self.ic = None + self.device = None + #self.coldboot = None + self.warmboot = None + self.tiles = {} + + def read(self, fields): + if fields[0] == 'device' and len(fields) == 4 \ + and len(fields[1]) >= 2 and fields[1][0] == '"' \ + and fields[1][-1] == '"' \ + and self.ic is None and self.device is None: + self.device = fields[1][1:-1] + if self.device == '1k': + self.ic = icebox.iceconfig() + self.ic.setup_empty_1k() + elif self.device == '8k': + self.ic = icebox.iceconfig() + self.ic.setup_empty_8k() + elif self.device == '384': + self.ic = icebox.iceconfig() + self.ic.setup_empty_384() + else: + raise ParseError + #elif fields[0] == 'coldboot' and fields[1] == '=' \ + # and self.coldboot is None: + # # parsed but ignored (can't be represented in IceStorm .asc format) + # self.coldboot = parse_bool(fields[2]) + elif fields[0] == 'warmboot' and fields[1] == '=' \ + and self.warmboot is None: + # parsed but ignored (can't be represented in IceStorm .asc format) + self.warmboot = parse_bool(fields[2]) + else: + raise ParseError + + def new_block(self, fields): + if len(fields) != 3: + raise ParseError + x = int(fields[1]) + y = int(fields[2]) + if (x, y) in self.tiles: + return self.tiles[x, y] + if fields[0] == 'logic_tile': + if (x, y) not in self.ic.logic_tiles: + raise ParseError + tile = LogicTile(self.ic, x, y) + elif fields[0] == 'ramb_tile': + if (x, y) not in self.ic.ramb_tiles: + raise ParseError + tile = RAMBTile(self.ic, x, y) + elif fields[0] == 'ramt_tile': + if (x, y) not in self.ic.ramt_tiles: + raise ParseError + tile = RAMTTile(self.ic, x, y) + elif fields[0] == 'io_tile': + if (x, y) not in self.ic.io_tiles: + raise ParseError + tile = IOTile(self.ic, x, y) + else: + raise ParseError + self.tiles[x, y] = tile + return tile + + def writeout(self): + if self.ic is None: + raise ParseError + + # fix up IE/REN bits + unused_ieren = set() + + for x in range(1, self.ic.max_x): + unused_ieren.add((x, 0, 0)) + unused_ieren.add((x, 0, 1)) + unused_ieren.add((x, self.ic.max_y, 0)) + unused_ieren.add((x, self.ic.max_y, 1)) + + for y in range(1, self.ic.max_y): + unused_ieren.add((0, y, 0)) + unused_ieren.add((0, y, 1)) + unused_ieren.add((self.ic.max_x, y, 0)) + unused_ieren.add((self.ic.max_x, y, 1)) + + for x0, y0, b0, x1, y1, b1 in self.ic.ieren_db(): + if (x0, y0) in self.tiles: + io_tile = self.tiles[x0, y0] + else: + io_tile = IOTile(self.ic, x0, y0) + self.tiles[x0, y0] = io_tile + if io_tile.blocks[b0] is not None: + io_block = io_tile.blocks[b0] + else: + io_block = IOBlock(io_tile, b0) + io_tile.blocks[b0] = io_block + + if (x1, y1) in self.tiles: + ieren_tile = self.tiles[x1, y1] + else: + ieren_tile = IOTile(self.ic, x1, y1) + self.tiles[x1, y1] = ieren_tile + + if io_block.enable_input != (self.ic.device == '1k'): + ieren_tile.apply_directive('IoCtrl', 'IE_%d' % b1) + if io_block.disable_pull_up: + ieren_tile.apply_directive('IoCtrl', 'REN_%d' % b1) + + unused_ieren.remove((x1, y1, b1)) + + if self.ic.device == '1k': + for x1, y1, b1 in unused_ieren: + if (x1, y1) in self.tiles: + ieren_tile = self.tiles[x1, y1] + else: + ieren_tile = IOTile(self.ic, x1, y1) + self.tiles[x1, y1] = ieren_tile + ieren_tile.apply_directive('IoCtrl', 'IE_%d' % b1) + + # fix up RAMB power-up bits + + for x, y in self.ic.ramb_tiles: + if (x, y) in self.tiles: + tile = self.tiles[x, y] + else: + tile = RAMBTile(self.ic, x, y) + self.tiles[x, y] = tile + + if tile.power_up != (self.ic.device == '1k'): + tile.apply_directive('RamConfig', 'PowerUp') + + # enable column buffers + colbuf_db = self.ic.colbuf_db() + for x, y in list(self.tiles): + for src, dst in self.tiles[x, y].buffers + \ + self.tiles[x, y].routings: + if not src.startswith('glb_netwk_'): + continue + driving_xy = [(src_x, src_y) + for src_x, src_y, dst_x, dst_y in colbuf_db + if dst_x == x and dst_y == y] + assert len(driving_xy) == 1 + driving_xy, = driving_xy + + if driving_xy not in self.tiles: + if driving_xy in self.ic.logic_tiles: + tile = LogicTile(self.ic, *driving_xy) + elif driving_xy in self.ic.ramb_tiles: + tile = RAMBTile(self.ic, *driving_xy) + elif driving_xy in self.ic.ramt_tiles: + tile = RAMTTile(self.ic, *driving_xy) + elif driving_xy in self.ic.io_tiles: + tile = IOTile(self.ic, *driving_xy) + else: + assert False + self.tiles[driving_xy] = tile + + self.tiles[driving_xy].apply_directive('ColBufCtrl', src) + + self.ic.write_file('/dev/stdout') + +class Tile: + def __init__(self, ic, x, y): + self.ic = ic + self.x = x + self.y = y + self.data = ic.tile(x, y) + self.db = ic.tile_db(x, y) + + self.buffers = [] + self.routings = [] + self.bits_set = set() + self.bits_cleared = set() + + def apply_directive(self, *fields): + fields = list(fields) + bits, = [entry[0] for entry in self.db if entry[1:] == fields] + self.set_bits(bits) + + def set_bits(self, bits): + bits_set = set() + bits_clear = set() + + for bit in bits: + match = re.match(r'(!?)B(\d+)\[(\d+)\]$', bit) + if not match: + raise ValueError("invalid bit description: %s" % bit) + if match.group(1): + bits_clear.add((int(match.group(2)), int(match.group(3)))) + else: + bits_set.add((int(match.group(2)), int(match.group(3)))) + + if set.intersection(bits_set, bits_clear): + raise ValueError("trying to set/clear the same bit(s) at once") + + if set.intersection(bits_set, self.bits_cleared) or \ + set.intersection(bits_clear, self.bits_set): + raise ParseError("conflicting bits") + + self.bits_set.update(bits_set) + self.bits_cleared.update(bits_clear) + + for row, col in bits_set: + assert row < len(self.data) + assert col < len(self.data[row]) + self.data[row] = self.data[row][:col] + '1' + \ + self.data[row][col + 1:] + + def read(self, fields): + if len(fields) == 3 and fields[1] == '->': + src = untranslate_netname(self.x, self.y, + self.ic.max_x - 1, + self.ic.max_y - 1, fields[0]) + dst = untranslate_netname(self.x, self.y, + self.ic.max_x - 1, + self.ic.max_y - 1, fields[2]) + dst = revert_to_fabout(self.x, self.y, dst) + if (src, dst) not in self.buffers: + self.buffers.append((src, dst)) + self.apply_directive('buffer', src, dst) + elif len(fields) == 3 and fields[1] == '<->': + src = untranslate_netname(self.x, self.y, + self.ic.max_x - 1, + self.ic.max_y - 1, fields[0]) + dst = untranslate_netname(self.x, self.y, + self.ic.max_x - 1, + self.ic.max_y - 1, fields[2]) + dst = revert_to_fabout(self.x, self.y, dst) + if (src, dst) not in self.routings: + self.routings.append((src, dst)) + self.apply_directive('routing', src, dst) + elif len(fields) >= 5 and (fields[1] == '->' or fields[1] == '<->'): + self.read(fields[:3]) + self.read(fields[2:]) + else: + raise ParseError + + def new_block(self, fields): + raise ParseError + +class LogicTile(Tile): + def __init__(self, ic, x, y): + super().__init__(ic, x, y) + self.cells = [None, None, None, None, None, None, None, None] + self.neg_clk = False + self.carry_in_set = False # not in global bit list?! + + def read(self, fields): + if fields == ['NegClk'] and not self.neg_clk: + self.neg_clk = True + self.apply_directive('NegClk') + elif fields == ['CarryInSet'] and not self.carry_in_set: + self.carry_in_set = True + self.apply_directive('CarryInSet') + else: + super().read(fields) + + def new_block(self, fields): + for i in range(8): + if fields == ['lutff_%d' % i] and self.cells[i] is None: + self.cells[i] = LogicCell(self, i) + return self.cells[i] + raise ParseError + +class LogicCell: + def __init__(self, tile, index): + self.tile = tile + self.index = index + self.lut_bits = None + self.seq_bits = ['0'] * 4 + + def read(self, fields): + if fields[0] == 'lut' and len(fields) == 2 and self.lut_bits is None: + self.lut_bits = fields[1] + elif fields[0] == 'out' and len(fields) >= 3 and fields[1] == '=': + self.lut_bits = logic_expression_to_lut( + ' '.join(fields[2:]), ('in_0', 'in_1', 'in_2', 'in_3')) + elif fields == ['enable_carry']: + self.seq_bits[0] = '1' + elif fields == ['enable_dff']: + self.seq_bits[1] = '1' + elif fields == ['set_noreset']: + self.seq_bits[2] = '1' + elif fields == ['async_setreset']: + self.seq_bits[3] = '1' + elif len(fields) >= 3 and (fields[1] == '->' or fields[1] == '<->'): + prefix = 'lutff_%d/' % self.index + if fields[0] == 'out': + self.tile.read([prefix + fields[0]] + fields[1:]) + elif fields[-1].startswith('in_'): + self.tile.read(fields[:-1] + [prefix + fields[-1]]) + else: + raise ParseError + return + + bits = ''.join([ + self.lut_bits[15], self.lut_bits[12], + self.lut_bits[11], self.lut_bits[ 8], + self.lut_bits[ 0], self.lut_bits[ 3], + self.lut_bits[ 4], self.lut_bits[ 7], + self.seq_bits[ 0], self.seq_bits[ 1], + self.lut_bits[14], self.lut_bits[13], + self.lut_bits[10], self.lut_bits[ 9], + self.lut_bits[ 1], self.lut_bits[ 2], + self.lut_bits[ 5], self.lut_bits[ 6], + self.seq_bits[ 2], self.seq_bits[ 3] + ]) + self.tile.data[self.index * 2] = \ + self.tile.data[self.index * 2][:36] + bits[:10] + \ + self.tile.data[self.index * 2][46:] + self.tile.data[self.index * 2 + 1] = \ + self.tile.data[self.index * 2 + 1][:36] + bits[10:] + \ + self.tile.data[self.index * 2 + 1][46:] + + def new_block(self, fields): + raise ParseError + +class RAMData: + def __init__(self, data): + self.data = data + + def read(self, fields): + if len(fields) == 1: + self.data.append(fields[0]) + else: + raise ParseError + + def new_block(self, fields): + raise ParseError + +class RAMBTile(Tile): + def __init__(self, ic, x, y): + super().__init__(ic, x, y) + self.power_up = False + + def read(self, fields): + if fields == ['power_up'] and not self.power_up: + self.power_up = True + else: + super().read(fields) + + def new_block(self, fields): + if fields == ['data'] and (self.x, self.y) not in self.ic.ram_data: + self.ic.ram_data[self.x, self.y] = data = [] + return RAMData(data) + raise ParseError + +class RAMTTile(Tile): + def __init__(self, ic, x, y): + super().__init__(ic, x, y) + + def read(self, fields): + if fields == ['NegClk'] or fields[0] == 'RamConfig': + self.apply_directive(*fields) # TODO + else: + super().read(fields) + +class IOTile(Tile): + def __init__(self, ic, x, y): + super().__init__(ic, x, y) + self.blocks = [None, None] + + def read(self, fields): + if len(fields) == 2 and fields[0] == 'PLL': + self.apply_directive(*fields) # TODO + else: + super().read(fields) + + def new_block(self, fields): + if fields == ['io_0'] and self.blocks[0] is None: + self.blocks[0] = IOBlock(self, 0) + return self.blocks[0] + if fields == ['io_1'] and self.blocks[1] is None: + self.blocks[1] = IOBlock(self, 1) + return self.blocks[1] + raise ParseError + +class IOBlock: + def __init__(self, tile, index): + self.tile = tile + self.index = index + self.input_pin_type = None + self.output_pin_type = None + self.enable_input = False + self.disable_pull_up = False + + def read(self, fields): + if fields[0] == 'input_pin_type' and fields[1] == '=' \ + and len(fields) == 3 and self.input_pin_type is None: + self.input_pin_type = [ + 'registered_pin', + 'simple_input_pin', + 'latched_registered_pin', + 'latched_pin'].index(fields[2]) + for i in range(2): + if self.input_pin_type & 1 << i: + self.tile.apply_directive('IOB_%d' % self.index, + 'PINTYPE_%d' % i) + elif fields[0] == 'output_pin_type' and fields[1] == '=' \ + and len(fields) == 3 and self.output_pin_type is None: + self.output_pin_type = [ + 'no_output', + '1', + '2', + '3', + 'DDR', + 'REGISTERED', + 'simple_output_pin', + 'REGISTERED_INVERTED', + 'DDR_ENABLE', + 'REGISTERED_ENABLE', + 'OUTPUT_TRISTATE', + 'REGISTERED_ENABLE_INVERTED', + 'DDR_ENABLE_REGISTERED', + 'REGISTERED_ENABLE_REGISTERED', + 'ENABLE_REGISTERED', + 'REGISTERED_ENABLE_REGISTERED_INVERTED'].index(fields[2]) + for i in range(4): + if self.output_pin_type & 1 << i: + self.tile.apply_directive('IOB_%d' % self.index, + 'PINTYPE_%d' % (i + 2)) + elif fields == ['enable_input'] and not self.enable_input: + self.enable_input = True + elif fields == ['disable_pull_up'] and not self.disable_pull_up: + self.disable_pull_up = True + elif fields[0] == 'GLOBAL_BUFFER_OUTPUT' and fields[1] == '->' \ + and fields[2].startswith('glb_netwk_'): + if GLB_NETWK_EXTERNAL_BLOCKS[int(fields[2][10:])] \ + != (self.tile.x, self.tile.y, self.index): + raise ParseError + bit = [bit for bit in self.tile.ic.extra_bits_db() + if self.tile.ic.extra_bits_db()[bit] + == ("padin_glb_netwk", fields[2][10:])] + assert len(bit) == 1 + self.tile.ic.extra_bits.add(bit[0]) + elif len(fields) >= 3 and (fields[1] == '->' or fields[1] == '<->'): + prefix = 'io_%d/' % self.index + if fields[0] in ('D_IN_0', 'D_IN_1'): + self.tile.read([prefix + fields[0]] + fields[1:]) + elif fields[-1] in ('cen', + 'D_OUT_0', + 'D_OUT_1', + 'inclk', + #'LATCH_INPUT_VALUE', + 'outclk', + 'OUT_ENB'): + self.tile.read(fields[:-1] + [prefix + fields[-1]]) + else: + raise ParseError + else: + raise ParseError + + def new_block(self, fields): + raise ParseError + +def main1(path): + f = open(path, 'r') + stack = [Main()] + for i, line in enumerate(f): + fields = line.split('#')[0].split() + try: + if not fields: + pass # empty line + elif fields == ['}']: + stack.pop() + if not stack: + raise ParseError + elif fields[-1] == '{': + stack.append(stack[-1].new_block(fields[:-1])) + else: + stack[-1].read(fields) + except ParseError: + sys.stderr.write("Parse error in line %d:\n" % (i + 1)) + sys.stderr.write(line) + sys.exit(1) + if len(stack) != 1: + sys.stderr.write("Parse error: unexpected end of file") + sys.exit(1) + f.close() + + stack[0].writeout() + +def main(): + program_short_name = os.path.basename(sys.argv[0]) + + try: + opts, args = getopt.getopt(sys.argv[1:], '', ['help', 'version']) + except getopt.GetoptError as e: + sys.stderr.write("%s: %s\n" % (program_short_name, e.msg)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + for opt, arg in opts: + if opt == '--help': + sys.stderr.write("""\ +Create an ASCII bitstream from a high-level bitstream representation. +Usage: %s [OPTION]... FILE + + --help display this help and exit + --version output version information and exit + +If you have a bug report, please file an issue on github: + https://github.com/rlutz/icestorm/issues +""" % sys.argv[0]) + sys.exit(0) + + if opt == '--version': + sys.stderr.write("""\ +icebox_hlc2asc - create an ASCII bitstream from a high-level representation +Copyright (C) 2017 Roland Lutz + +This program is free software: you can redistribute it and/or +modify it under the terms of the GNU General Public License as +published by the Free Software Foundation, either version 3 of +the License, or (at your option) any later version. + +This program is distributed in the hope that it will be useful, +but WITHOUT ANY WARRANTY; without even the implied warranty of +MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +GNU General Public License for more details. +""") + sys.exit(0) + + if not args: + sys.stderr.write("%s: missing argument\n" % (program_short_name)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + if len(args) != 1: + sys.stderr.write("%s: too many arguments\n" % (program_short_name)) + sys.stderr.write("Try `%s --help' for more information.\n" + % sys.argv[0]) + sys.exit(1) + + if args[0] == '-': + main1('/dev/stdin') + else: + main1(args[0]) + +if __name__ == '__main__': + main() diff --git a/icebox/tc_logic_xpr.py b/icebox/tc_logic_xpr.py new file mode 100644 index 0000000..10f031c --- /dev/null +++ b/icebox/tc_logic_xpr.py @@ -0,0 +1,44 @@ +# Test case for `icebox_asc2hlc' and `icebox_hlc2asc': Does conversion +# from LUT strings to logic expressions and back work correctly? +# Copyright (C) 2017 Roland Lutz +# +# Permission to use, copy, modify, and/or distribute this software for any +# purpose with or without fee is hereby granted, provided that the above +# copyright notice and this permission notice appear in all copies. +# +# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +import sys +import icebox +from icebox_asc2hlc import lut_to_logic_expression +from icebox_hlc2asc import logic_expression_to_lut + +def main(): + sys.stderr.write("testing conversion from LUT strings " + "to logic expressions and back") + + for i in range(65536): + if i % 4096 == 0: + sys.stderr.write(".") + sys.stderr.flush() + + lut = bin(i)[2:].zfill(16) + s = lut_to_logic_expression(lut, ('a', 'b', 'c', 'd')) + l = logic_expression_to_lut(s, ('a', 'b', 'c', 'd')) + + if l != lut: + sys.stderr.write("\nERROR at LUT = %s\n" % lut) + sys.stderr.write("stringified = %s\n" % s) + sys.stderr.write("resulting LUT = %s\n" % l) + sys.exit(1) + + sys.stderr.write("\n") + +if __name__ == '__main__': + main() diff --git a/icebox/tc_rxlat_netnames.py b/icebox/tc_rxlat_netnames.py new file mode 100644 index 0000000..e8717ed --- /dev/null +++ b/icebox/tc_rxlat_netnames.py @@ -0,0 +1,71 @@ +# Test case for `icebox_hlc2asc': Does net name translation work correctly? +# Copyright (C) 2017 Roland Lutz +# +# Permission to use, copy, modify, and/or distribute this software for any +# purpose with or without fee is hereby granted, provided that the above +# copyright notice and this permission notice appear in all copies. +# +# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +import sys +import icebox +from icebox_asc2hlc import translate_netname +from icebox_hlc2asc import untranslate_netname + +def test_netname_translation(ic): + sys.stderr.write("testing backward netname translation " + "for the `%s' device...\n" % ic.device) + all_tiles = set() + for x in range(ic.max_x + 1): + for y in range(ic.max_y + 1): + if ic.tile(x, y) is not None: + all_tiles.add((x, y)) + + netnames = set() + failed = False + + for group in ic.group_segments(all_tiles, connect_gb = False): + is_span = set(net.startswith('sp') for x, y, net in group) + assert len(is_span) == 1 + if True not in is_span: + # only span nets are interesting here + continue + + netname = translate_netname(group[0][0], group[0][1], + ic.max_x - 1, ic.max_y - 1, group[0][2]) + if netname in netnames: + failed = True + print("duplicate netname: %s" % netname) + netnames.add(netname) + + for x, y, net in group: + s = untranslate_netname(x, y, ic.max_x - 1, ic.max_y - 1, netname) + if s != net: + failed = True + print("%-20s %s -> %s" % ("%d %d %s" % (x, y, net), netname, s)) + + if failed: + sys.stderr.write("ERROR\n") + sys.exit(1) + +def main(): + ic = icebox.iceconfig() + ic.setup_empty_384() + test_netname_translation(ic) + + ic = icebox.iceconfig() + ic.setup_empty_1k() + test_netname_translation(ic) + + ic = icebox.iceconfig() + ic.setup_empty_8k() + test_netname_translation(ic) + +if __name__ == '__main__': + main() diff --git a/icebox/tc_xlat_netnames.py b/icebox/tc_xlat_netnames.py new file mode 100644 index 0000000..39c89b1 --- /dev/null +++ b/icebox/tc_xlat_netnames.py @@ -0,0 +1,81 @@ +# Test case for `icebox_asc2hlc': Does net name translation work correctly? +# Copyright (C) 2017 Roland Lutz +# +# Permission to use, copy, modify, and/or distribute this software for any +# purpose with or without fee is hereby granted, provided that the above +# copyright notice and this permission notice appear in all copies. +# +# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES +# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF +# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR +# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES +# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN +# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF +# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + +import sys +import icebox +from icebox_asc2hlc import translate_netname + +def test_netname_translation(ic): + sys.stderr.write("testing forward netname translation " + "for the `%s' device...\n" % ic.device) + all_tiles = set() + for x in range(ic.max_x + 1): + for y in range(ic.max_y + 1): + if ic.tile(x, y) is not None: + all_tiles.add((x, y)) + + netnames = set() + failed = False + + for group in ic.group_segments(all_tiles, connect_gb = False): + is_span = set(net.startswith('sp') for x, y, net in group) + assert len(is_span) == 1 + if True not in is_span: + # only span nets are interesting here + continue + + s = set() + for seg in group: + s.add(translate_netname(seg[0], seg[1], + ic.max_x - 1, ic.max_y - 1, seg[2])) + if len(s) != 1: + failed = True + print("translated netnames don't match") + for seg in group: + print("%d %d %s" % seg, "->", + translate_netname(seg[0], seg[1], + ic.max_x - 1, ic.max_y - 1, seg[2])) + print() + + for dulpicate_netname in netnames.intersection(s): + failed = True + print("duplicate netname: %s" % dulpicate_netname) + for seg in group: + print("%d %d %s" % seg, "->", + translate_netname(seg[0], seg[1], + ic.max_x - 1, ic.max_y - 1, seg[2])) + print() + + netnames.update(s) + + if failed: + sys.stderr.write("ERROR\n") + sys.exit(1) + +def main(): + ic = icebox.iceconfig() + ic.setup_empty_384() + test_netname_translation(ic) + + ic = icebox.iceconfig() + ic.setup_empty_1k() + test_netname_translation(ic) + + ic = icebox.iceconfig() + ic.setup_empty_8k() + test_netname_translation(ic) + +if __name__ == '__main__': + main() |