#!/usr/bin/env python3

import os, sys

device = "up5k"

# This script is designed to determine which DSPs have configuration bits
# not in their usual position, as in some cases DSP and IPConnect tiles have
# their config bits swapped

# Unfortunately, arbitrary configurations are not allowed by icecube, so
# we define a set that gives us maximum coverage (full coverage is not
# possible as one CBIT is never set)

allowed_configs = ["1110000010000001001110110", "0010000101000010111111111", "0001111000101100000000000"]

coverage = set()
for c in allowed_configs:
    for i in range(25):
        if c[i] == "1":
            coverage.add(i)

assert len(coverage) >= 24

def parse_exp(f):
    current_x = 0
    current_y = 0
    bits = set()
    for line in f:
        splitline = line.split(' ')
        if splitline[0].endswith("_tile"):
            current_x = int(splitline[1])
            current_y = int(splitline[2])
        elif splitline[0] == "IpConfig":
            if splitline[1][:5] == "CBIT_":
                bitidx = int(splitline[1][5:])
                bits.add((current_x, current_y, bitidx))    
    return bits
dsp_locs = [( 0, 5, 0), ( 0, 10, 0), ( 0, 15, 0), ( 0, 23, 0),
            (25, 5, 0), (25, 10, 0), (25, 15, 0), (25, 23, 0)]
            
dsp_data = {}

if not os.path.exists("./work_dsp_cbit"):
    os.mkdir("./work_dsp_cbit")

for loc in dsp_locs:
    x, y, z = loc
    missing_bits = set()
    new_bits = set()
    for config in allowed_configs:
        params = config[::-1]
        with open("./work_dsp_cbit/dsp_cbit.v","w") as f:
            print("""
            module top(input clk, input a, input b, input c, input d, output y);
            """, file=f)
            print("""
                SB_MAC16 #(
                    .C_REG(1'b%s),
                    .A_REG(1'b%s),
                    .B_REG(1'b%s),
                    .D_REG(1'b%s),
                    .TOP_8x8_MULT_REG(1'b%s),
                    .BOT_8x8_MULT_REG(1'b%s),
                    .PIPELINE_16x16_MULT_REG1(1'b%s),
                    .PIPELINE_16x16_MULT_REG2(1'b%s),
                    .TOPOUTPUT_SELECT(2'b%s),
                    .TOPADDSUB_LOWERINPUT(2'b%s),
                    .TOPADDSUB_UPPERINPUT(1'b%s),
                    .TOPADDSUB_CARRYSELECT(2'b%s),
                    .BOTOUTPUT_SELECT(2'b%s),
                    .BOTADDSUB_LOWERINPUT(2'b%s),
                    .BOTADDSUB_UPPERINPUT(1'b%s),
                    .BOTADDSUB_CARRYSELECT(2'b%s),
                    .MODE_8x8(1'b%s),
                    .A_SIGNED(1'b%s),
                    .B_SIGNED(1'b%s)
                ) dsp (
                    .CLK(clk),
                    .C(c),
                    .A(a),
                    .B(b),
                    .D(d),
                    .O(y)
                );"""
                % (
                params[0], params[1], params[2], params[3],
                params[4], params[5], params[6], params[7],
                params[8:10][::-1], params[10:12][::-1], params[12], params[13:15][::-1],
                params[15:17][::-1], params[17:19][::-1], params[19], params[20:22][::-1],
                params[22], params[23], params[24]), file=f)
            print("endmodule",file=f)
        with open("./work_dsp_cbit/dsp_cbit.pcf","w") as f:
            print("set_location dsp %d %d %d" % loc, file=f)
        retval = os.system("bash ../../icecube.sh -" + device + " ./work_dsp_cbit/dsp_cbit.v > ./work_dsp_cbit/icecube.log 2>&1")
        if retval != 0:
            sys.stderr.write('ERROR: icecube returned non-zero error code\n')
            sys.exit(1)
        retval = os.system("../../../icebox/icebox_explain.py ./work_dsp_cbit/dsp_cbit.asc > ./work_dsp_cbit/dsp_cbit.exp")
        if retval != 0:
            sys.stderr.write('ERROR: icebox_explain returned non-zero error code\n')
            sys.exit(1)
        bits = set()
        known = set()
        with open('./work_dsp_cbit/dsp_cbit.exp', 'r') as f:
            bits = parse_exp(f)
        for i in range(25):
            if params[i] == "1":
                exp_pos = (x, y + (i // 8), i % 8)
                if exp_pos not in bits:
                    missing_bits.add(exp_pos)
                else:
                    known.add(exp_pos)
        for bit in bits:
            if bit not in known:
                new_bits.add(bit)
    if len(missing_bits) > 0 or len(new_bits) > 0:
        print("DSP (%d, %d):" % (x, y))
        for bit in missing_bits:
            print("\tMissing (%d, %d, CBIT_%d)" % bit)
        for bit in new_bits:
            print("\tNew: (%d, %d, CBIT_%d)" % bit)
        dsp_data[loc] = (missing_bits, new_bits)
with open("dsp_cbits_%s.txt" % device, 'w') as f:
    for loc in dsp_data:
        x, y, z = loc
        missing_bits, new_bits = dsp_data[loc]
        print("DSP (%d, %d):" % (x, y), file=f)
        for bit in missing_bits:
            print("\tMissing (%d, %d, CBIT_%d)" % bit,file=f)
        for bit in new_bits:
            print("\tNew: (%d, %d, CBIT_%d)" % bit,file=f)