Skip to content

Commit

Permalink
or_issue.py (RE: #424) (#428)
Browse files Browse the repository at this point in the history
* or_issue.py (RE: #424)

* Reflect changes based on @maliberty's review

* 66 -> os.EX_NOINPUT
  • Loading branch information
donn committed Jul 7, 2021
1 parent 6055672 commit 5731e51
Show file tree
Hide file tree
Showing 2 changed files with 233 additions and 1 deletion.
232 changes: 232 additions & 0 deletions or_issue.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,232 @@
#!/usr/bin/env python3

# Copyright 2021 Efabless Corporation
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

import os
import re
import sys
import shutil
import pathlib
import argparse
from collections import deque
from os.path import join, abspath, dirname, basename, isdir, relpath

openlane_path = abspath(dirname(__file__))

parser = argparse.ArgumentParser(description="""
This script creates a reproducible, self-contained package of files to demonstrate
OpenROAD behavior in a vaccum, suitable for filing issues.
Requires UNIX-like operating system. Final package printed to /dev/stdout. Don't forget to chomp/rstrip.
Usage example: IMAGE_NAME=efabless/openlane:v0.15 python3 or_issue.py -s ./scripts/openroad/or_pdn.tcl ./designs/inverter/runs/openlane_test
""")
parser.add_argument('--or-script', '-s', required=True, help='Name of the OpenROAD script causing the failure: i.e. or_antenna_check.tcl, or_pdn.tcl, etc.')
parser.add_argument('--pdk-root', required=(os.getenv("PDK_ROOT") is None), default=os.getenv("PDK_ROOT"), help='Path to the PDK root (Required if environment variable PDK_ROOT is not set.)')
parser.add_argument('-c', '--compression', default="gzip", help='Comma,delimited list of compression techniques to use after tar. Use "None" to disable compression altogether. Default is "gzip", which will create a .tar.gzip file. Valid technologies: gzip/gz, xzip/xz, bzip2/bz2')
parser.add_argument('run_path', help='Path to the run folder.')
args = parser.parse_args()

script_path = abspath(args.or_script)
run_path = abspath(args.run_path)
pdk_root = abspath(args.pdk_root)
compression = args.compression
or_scripts_path = join(openlane_path, "scripts", "openroad")

if not script_path.startswith(or_scripts_path):
print(f"⚠ The OpenROAD script {script_path} does not appear to be in {or_scripts_path}.", file=sys.stderr)
print("This script's behavior may be undefined.", file=sys.stderr)

run_name = basename(run_path)
script_basename = basename(args.or_script)[:-4]

script_path_containerized = script_path.replace(openlane_path, "/openLANE_flow")
run_path_containerized = run_path.replace(openlane_path, "/openLANE_flow")


# Phase 1: Read All Environment Variables
# pdk_config = join(args.pdk_root, "sky130A", "libs.tech", "openlane", "config.tcl")
print(f"Parsing config file(s)...", file=sys.stderr)
run_config = join(run_path, "config.tcl")

env = {}


def read_env(config_path: str, from_path: str, input_env={}) -> dict:
rx = r"\s*set\s*::env\((.+?)\)\s*(.+)"
env = input_env.copy()
string_data = ""
try:
string_data = open(config_path).read()
except FileNotFoundError:
print(f"❌ File {config_path} not found. The {from_path} path may have been specified incorrectly.", file=sys.stderr)
exit(os.EX_NOINPUT)

# Process \ at ends of lines, remove semicolons
entries = string_data.split("\n")
i = 0
while i < len(entries):
if not entries[i].endswith("\\"):
if entries[i].endswith(";"):
entries[i] = entries[i][:-1]
i += 1
continue
entries[i] = entries[i][:-1] + entries[i + 1]
del entries[i + 1]

for entry in entries:
match = re.match(rx, entry)
if match is None:
continue
name = match[1]; value = match[2]
# remove double quotes
if value.startswith('"') and value.endswith('"'):
value = value[1:-1]
# print(value)
env[name] = value

return env

env = read_env(run_config, "Run Path") # , read_env(pdk_config, "PDK Root"))

# Phase 2: Set up destination folder
destination_folder = abspath(join(".", "_build", f"{run_name}_{script_basename}_packaged"))
print(f"Setting up {destination_folder}...", file=sys.stderr)

def mkdirp(path):
return pathlib.Path(path).mkdir(parents=True, exist_ok=True)

try:
shutil.rmtree(destination_folder)
except FileNotFoundError:
pass

mkdirp(destination_folder)

# Phase 3: Process TCL Scripts To Find Full List Of Files
tcls_to_process = deque([ script_path ])
def shift(deque):
try:
return deque.popleft()
except:
return None

envs_used = ["OR_SCRIPT"]
env["OR_SCRIPT"] = script_path_containerized

current = shift(tcls_to_process)
while current is not None:
script = open(current).read()

for key, value in env.items():
key_accessor = f"$::env({key})"
if not key_accessor in script:
continue
envs_used.append(key)
if value.endswith(".tcl"):
tcls_to_process.append(value)

current = shift(tcls_to_process)

# Phase 4: Copy The Files
final_env_pairs = []

pdk_path = join(destination_folder, "pdk")
openlane_misc_path = join(destination_folder, "openlane")

def copy(frm, to):
parents = dirname(to)
mkdirp(parents)
try:
if isdir(frm):
shutil.copytree(frm, to)
else:
shutil.copyfile(frm, to)
except:
print(f"ℹ Couldn't copy {frm}, skipping...", file=sys.stderr)

for key in envs_used:
value = env[key]
if value.startswith(run_path_containerized):
relative = relpath(value, run_path_containerized)
final_value = join(".", relative)
final_path = join(destination_folder, final_value)
from_path = value.replace(run_path_containerized, run_path)
copy(from_path, final_path)
final_env_pairs.append((key, final_value))
elif value.startswith(pdk_root):
relative = relpath(value, pdk_root)
final_value = join("pdk", relative)
final_path = join(destination_folder, final_value)
copy(value, final_path)
final_env_pairs.append((key, final_value))
elif value.startswith("/openLANE_flow"):
relative = relpath(value, "/openLANE_flow")
final_value = join("openlane", relative)
final_path = join(destination_folder, final_value)
from_path = value.replace("/openLANE_flow", openlane_path)
copy(from_path, final_path)
final_env_pairs.append((key, final_value))
else:
final_env_pairs.append((key, value))

# Phase 5: Create Run Files
run_ol = join(destination_folder, "run_ol")
with open(run_ol, "w") as f:
env_list = "\\\n ".join([f"-e {key}='{value}'" for key, value in final_env_pairs])
f.write(f"""
#!/bin/sh
dir=$(cd -P -- "$(dirname -- "$0")" && pwd -P)
cd $dir;
docker run --rm\\
-tiv `pwd`:`pwd` -w `pwd`\\
{env_list}\\
{os.getenv("IMAGE_NAME") or "efabless/openlane:current"} openroad \\$::env\\(OR_SCRIPT\\)
""")
os.chmod(run_ol, 0o755)

run_raw = join(destination_folder, "run")
with open(run_raw, "w") as f:
env_list = "\n".join([f"export {key}='{value}';" for key, value in final_env_pairs])
f.write(f"""
#!/bin/sh
dir=$(cd -P -- "$(dirname -- "$0")" && pwd -P)
cd $dir;
{env_list}
openroad $OR_SCRIPT
""")
os.chmod(run_raw, 0o755)

# Phase 6: Tarball and output
last_output = destination_folder
if compression != "None":
technology_dict = {
"xz": ("xz", "xz"),
"xzip": ("xz", "xz"),
"gz": ("gzip", "gz"),
"gzip": ("gzip", "gz"),
"bz2": ("bzip2", "bz2"),
"bzip2": ("bzip2", "bz2"),
}
technologies = compression.split(',')

ext = ".".join([technology_dict[k][1] for k in technologies])
last_output = f"{destination_folder}.tar.{ext}"

pipes = " ".join([f"| {technology_dict[k][0]}" for k in technologies])
os.system(f"tar -cvC {destination_folder} . {pipes} > {last_output}")

print("✔ Done.", file=sys.stderr)
print(last_output)
2 changes: 1 addition & 1 deletion scripts/openroad/or_replace.tcl
Original file line number Diff line number Diff line change
Expand Up @@ -58,7 +58,7 @@ if { $::env(PL_BASIC_PLACEMENT) } {

if { $::env(PL_TIME_DRIVEN) } {
read_lib $::env(LIB_SYNTH)
read_sdc $::env(SCRIPTS_DIR)/base.sdc
read_sdc $::env(BASE_SDC_FILE)
read_verilog $::env(yosys_result_file_tag).v
} else {
set_replace_disable_timing_driven_mode_cmd
Expand Down

0 comments on commit 5731e51

Please sign in to comment.