-
Notifications
You must be signed in to change notification settings - Fork 383
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* or_issue.py (RE: #424) * Reflect changes based on @maliberty's review * 66 -> os.EX_NOINPUT
- Loading branch information
Showing
2 changed files
with
233 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters