|  | #!/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 | 
|  | """ | 
|  | This script loads the pin dump of the desired BEL from Vivado and groups pins into ports. | 
|  |  | 
|  | Ports that are not connected to the PL are not included. These ports are usually test and | 
|  | debug related ports, which are not useful from a P&R perspective. | 
|  |  | 
|  | Ports are then written to a JSON file. | 
|  | """ | 
|  | import argparse | 
|  | import csv | 
|  | import json | 
|  | import re | 
|  |  | 
|  | from collections import defaultdict | 
|  |  | 
|  | from prjxray.util import OpenSafeFile | 
|  |  | 
|  |  | 
|  | def main(): | 
|  |  | 
|  | BUS_REGEX = re.compile("(.*[A-Z_])([0-9]+)$") | 
|  |  | 
|  | # Parse arguments | 
|  | parser = argparse.ArgumentParser( | 
|  | description=__doc__, | 
|  | formatter_class=argparse.RawDescriptionHelpFormatter) | 
|  |  | 
|  | parser.add_argument("csv", type=str, help="BEL pin dump file") | 
|  | parser.add_argument( | 
|  | "json", | 
|  | type=str, | 
|  | help="Output JSON file with BEL pins grouped into ports") | 
|  |  | 
|  | parser.add_argument( | 
|  | "--special-pins", | 
|  | default="", | 
|  | type=str, | 
|  | required=False, | 
|  | help="Some pins cannot be decoded with the regex") | 
|  |  | 
|  | args = parser.parse_args() | 
|  |  | 
|  | # Load pin dump | 
|  | with OpenSafeFile(args.csv, "r") as fp: | 
|  | pin_dump = list(csv.DictReader(fp)) | 
|  |  | 
|  | # Group pins into ports | 
|  | ports = defaultdict(lambda: {"direction": None, "width": 0}) | 
|  |  | 
|  | special_pins = args.special_pins.split(",") | 
|  |  | 
|  | for pin in list(pin_dump): | 
|  | pin_name = pin["name"] | 
|  |  | 
|  | name = None | 
|  | if pin_name in special_pins: | 
|  | # Full match | 
|  | name = pin_name | 
|  | else: | 
|  | # Partial match | 
|  | for special_pin in special_pins: | 
|  | if pin_name.startswith(special_pin): | 
|  | name = special_pin | 
|  | break | 
|  |  | 
|  | if name is None: | 
|  | match = BUS_REGEX.match(pin_name) | 
|  | if match: | 
|  | name = match.group(1) | 
|  | else: | 
|  | name = pin_name | 
|  |  | 
|  | # Get direction | 
|  | is_input = int(pin["is_input"]) | 
|  | is_output = int(pin["is_output"]) | 
|  | is_clock = int(pin["is_clock"]) | 
|  |  | 
|  | if is_input: | 
|  | direction = "input" | 
|  | if is_clock: | 
|  | direction = "clock" | 
|  | elif is_output: | 
|  | direction = "output" | 
|  | else: | 
|  | assert False, pin | 
|  |  | 
|  | # Add to port | 
|  | port = ports[name] | 
|  |  | 
|  | if port["direction"] is None: | 
|  | port["direction"] = direction | 
|  | else: | 
|  | assert port["direction"] == direction, (port, direction, name) | 
|  |  | 
|  | port["width"] += 1 | 
|  |  | 
|  | # Write pin ports to a JSON file | 
|  | with OpenSafeFile(args.json, "w") as fp: | 
|  | json.dump(ports, fp, indent=1, sort_keys=True) | 
|  |  | 
|  |  | 
|  | if __name__ == "__main__": | 
|  | main() |