Skip to content

Commit

Permalink
[shortfin] Add dev_me.py script to standardize my development process. (
Browse files Browse the repository at this point in the history
  • Loading branch information
stellaraccident authored Oct 26, 2024
1 parent 6f3f8c7 commit 53820cd
Show file tree
Hide file tree
Showing 2 changed files with 255 additions and 13 deletions.
229 changes: 229 additions & 0 deletions shortfin/dev_me.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,229 @@
#!/usr/bin/env python3
# Copyright 2024 Advanced Micro Devices, Inc.
#
# Licensed under the Apache License v2.0 with LLVM Exceptions. See
# https://llvm.org/LICENSE.txt for license information. SPDX-License-Identifier:
# Apache-2.0 WITH LLVM-exception

# dev_me.py
#
# This is an opinionated development environment setup procedure aimed at
# making core contributors on the same golden path. It is not the only way
# to develop this project.
#
# First time build usage:
# rm -Rf build # Start with a fresh build dir
# python dev_me.py [--cmake=/path/to/cmake] [--clang=/path/to/clang] \
# [--iree=/path/to/iree] [--asan] [--build-type=Debug] \
# [--no-tracing]
#
# Subsequent build:
# ./dev_me.py
#
# This will perform an editable install into the used python with both
# default and tracing packages installed. After the initial build, ninja
# can be invoked directly under build/cmake/default or build/cmake/tracy.
# This can be done automatically just by running dev_me.py in a tree with
# an existing build directory.
#
# By default, if there is an iree source dir adjacent to this parent repository,
# that will be used (so you can just directly edit IREE runtime code and build.
# Otherwise, the shortfin build will download a pinned IREE source tree.

import argparse
import os
from packaging.version import Version
from pathlib import Path
import re
import subprocess
import shutil
import sys
import sysconfig


CMAKE_REQUIRED_VERSION = Version("3.29")
PYTHON_REQUIRED_VERSION = Version("3.12")
CLANG_REQUIRED_VERSION = Version("16")


class EnvInfo:
def __init__(self, args):
self.this_dir = Path(__file__).resolve().parent
self.python_exe = sys.executable
self.python_version = Version(".".join(str(v) for v in sys.version_info[1:2]))
self.debug = bool(sysconfig.get_config_var("Py_DEBUG"))
self.asan = "-fsanitize=address" in sysconfig.get_config_var("PY_LDFLAGS")
self.gil_disabled = bool(sysconfig.get_config_var("Py_GIL_DISABLED"))
self.cmake_exe, self.cmake_version = self.find_cmake(args)
self.ninja_exe = shutil.which("ninja")
self.clang_exe, self.clang_version = self.find_clang(args)
self.iree_dir = self.find_iree(args)

self.configured_dirs = []
self.add_configured(self.this_dir / "build" / "cmake" / "default")
self.add_configured(self.this_dir / "build" / "cmake" / "tracy")

def add_configured(self, path: Path):
probe = path / "CMakeCache.txt"
if probe.resolve().exists():
self.configured_dirs.append(path)

def find_cmake(self, args):
paths = []
if args.cmake:
paths.append(str(args.cmake))
else:
default_cmake = shutil.which("cmake")
if default_cmake:
paths.append(default_cmake)
for cmake_path in paths:
try:
cmake_output = subprocess.check_output(
[cmake_path, "--version"]
).decode()
except:
continue
if m := re.search("cmake version (.+)", cmake_output):
return cmake_path, Version(m.group(1))
return None, None

def find_clang(self, args):
if args.clang:
clang_exe = args.clang
else:
clang_exe = shutil.which("clang")
if not clang_exe:
return None, None
try:
clang_output = subprocess.check_output(
[clang_exe, "--version"]
).decode()
except:
return None, None
if m := re.search(r"clang version ([0-9\.]+)", clang_output):
return clang_exe, Version(m.group(1))
return None, None

def find_iree(self, args):
iree_dir = args.iree
if not iree_dir:
# See if a sibling iree directory exists.
iree_dir = self.this_dir.parent.parent / "iree"
if (iree_dir / "CMakeLists.txt").exists():
return str(iree_dir)
if not iree_dir.exists():
print(f"ERROR: --iree={iree_dir} directory does not exist")
sys.exit(1)
return str(iree_dir)

def check_prereqs(self, args):
if self.cmake_version is None or self.cmake_version < CMAKE_REQUIRED_VERSION:
print(
f"ERROR: cmake not found or of an insufficient version: {self.cmake_exe}"
)
print(f" Required: {CMAKE_REQUIRED_VERSION}, Found: {self.cmake_version}")
print(f" Configure explicitly with --cmake=")
sys.exit(1)
if self.python_version < PYTHON_REQUIRED_VERSION:
print(f"ERROR: python version too old: {self.python_exe}")
print(
f" Required: {PYTHON_REQUIRED_VERSION}, Found: {self.python_version}"
)
sys.exit(1)
if self.clang_exe and self.clang_version < CLANG_REQUIRED_VERSION:
print(f"ERROR: clang version too old: {self.clang_exe}")
print(f" REQUIRED: {CLANG_REQUIRED_VERSION}, Found {self.clang_version}")
elif not self.clang_exe:
print(f"WARNING: Building the project with clang is highly recommended")
print(f" (pass --clang= to select clang)")

if args.asan and not self.asan:
print(
f"ERROR: An ASAN build was requested but python was not built with ASAN support"
)
sys.exit(1)

def __repr__(self):
report = [
f"python: {self.python_exe}",
f"debug: {self.debug}",
f"asan: {self.asan}",
f"gil_disabled: {self.gil_disabled}",
f"cmake: {self.cmake_exe} ({self.cmake_version})",
f"ninja: {self.ninja_exe}",
f"clang: {self.clang_exe} ({self.clang_version})",
f"iree: {self.iree_dir}",
]
return "\n".join(report)


def main(argv: list[str]):
parser = argparse.ArgumentParser(
prog="shortfin dev", description="Shortfin dev setup helper"
)
parser.add_argument("--cmake", type=Path, help="CMake path")
parser.add_argument("--clang", type=Path, help="Clang path")
parser.add_argument("--iree", type=Path, help="Path to IREE source checkout")
parser.add_argument("--asan", action="store_true", help="Build with ASAN support")
parser.add_argument(
"--no-tracing", action="store_true", help="Disable IREE tracing build"
)
parser.add_argument(
"--build-type", default="Debug", help="CMake build type (default Debug)"
)
args = parser.parse_args(argv)
env_info = EnvInfo(args)

if env_info.configured_dirs:
print("First time configure...")
build_mode(env_info)
else:
configure_mode(env_info, args)


def configure_mode(env_info: EnvInfo, args):
print("Environment info:")
print(env_info)
env_info.check_prereqs(args)

env_vars = {
"SHORTFIN_DEV_MODE": "ON",
"SHORTFIN_CMAKE_BUILD_TYPE": args.build_type,
"SHORTFIN_ENABLE_ASAN": "ON" if args.asan else "OFF",
"SHORTFIN_CMAKE": env_info.cmake_exe,
}
if env_info.iree_dir:
env_vars["SHORTFIN_IREE_SOURCE_DIR"] = env_info.iree_dir
if env_info.clang_exe:
env_vars["CC"] = env_info.clang_exe
env_vars["CXX"] = f"{env_info.clang_exe}++"
env_vars["CMAKE_LINKER_TYPE"] = "LLD"
env_vars["SHORTFIN_ENABLE_TRACING"] = "OFF" if args.no_tracing else "ON"

print("Executing setup:")
setup_args = [
env_info.python_exe,
"-m",
"pip",
"install",
"--no-build-isolation",
"-v",
"-e",
str(env_info.this_dir),
]
print(f"{' '.join('='.join(kv) for kv in env_vars.items())} \\")
print(f" {' '.join(setup_args)}")
actual_env_vars = dict(os.environ)
actual_env_vars.update(env_vars)
subprocess.check_call(setup_args, cwd=env_info.this_dir, env=actual_env_vars)
print("You are now DEV'd!")


def build_mode(env_info: EnvInfo):
print("Building")
for build_dir in env_info.configured_dirs:
subprocess.check_call([env_info.cmake_exe, "--build", str(build_dir)])


if __name__ == "__main__":
main(sys.argv[1:])
39 changes: 26 additions & 13 deletions shortfin/setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
import shutil
import subprocess
import sys
import traceback
from distutils.command.build import build as _build
from distutils.core import setup, Extension
from pathlib import Path
Expand Down Expand Up @@ -38,8 +39,12 @@ def get_env_cmake_option(name: str, default_value: bool = False) -> str:
return f"-D{name}={svalue}"


def add_env_cmake_setting(args, env_name: str, cmake_name=None) -> str:
def add_env_cmake_setting(
args, env_name: str, cmake_name=None, default_value=None
) -> str:
svalue = os.getenv(env_name)
if svalue is None and default_value is not None:
svalue = default_value
if svalue is not None:
if not cmake_name:
cmake_name = env_name
Expand All @@ -62,6 +67,7 @@ def combine_dicts(*ds):
CPP_PREBUILT_BINARY_DIR = "@libshortfin_BINARY_DIR@"

SETUPPY_DIR = os.path.realpath(os.path.dirname(__file__))
CMAKE_EXE = os.getenv("SHORTFIN_CMAKE", "cmake")


def is_cpp_prebuilt():
Expand Down Expand Up @@ -223,13 +229,11 @@ def build_cmake_configuration(CMAKE_BUILD_DIR: Path, extra_cmake_args=[]):
] + extra_cmake_args

if DEV_MODE:
cmake_args.extend(
[
"-DCMAKE_C_COMPILER=clang",
"-DCMAKE_CXX_COMPILER=clang++",
"-DCMAKE_LINKER_TYPE=LLD",
]
)
if not os.getenv("CC"):
cmake_args.append("-DCMAKE_C_COMPILER=clang")
if not os.getenv("CXX"):
cmake_args.append("-DCMAKE_CXX_COMPILER=clang++")
add_env_cmake_setting(cmake_args, "CMAKE_LINKER_TYPE", default_value="LLD")

add_env_cmake_setting(cmake_args, "SHORTFIN_IREE_SOURCE_DIR")
add_env_cmake_setting(cmake_args, "SHORTFIN_ENABLE_ASAN")
Expand All @@ -238,12 +242,13 @@ def build_cmake_configuration(CMAKE_BUILD_DIR: Path, extra_cmake_args=[]):
cmake_cache_file = os.path.join(CMAKE_BUILD_DIR, "CMakeCache.txt")
if not os.path.exists(cmake_cache_file):
print(f"Configuring with: {cmake_args}")
subprocess.check_call(["cmake", SOURCE_DIR] + cmake_args, cwd=CMAKE_BUILD_DIR)
subprocess.check_call([CMAKE_EXE, SOURCE_DIR] + cmake_args, cwd=CMAKE_BUILD_DIR)
print(f"CMake configure complete.")
else:
print(f"Not re-configing (already configured)")

# Build.
subprocess.check_call(["cmake", "--build", "."], cwd=CMAKE_BUILD_DIR)
subprocess.check_call([CMAKE_EXE, "--build", "."], cwd=CMAKE_BUILD_DIR)
print("Build complete.")

# Optionally run CTests.
Expand All @@ -264,9 +269,17 @@ def run(self):
if is_cpp_prebuilt():
return

self.build_default_configuration()
if ENABLE_TRACY:
self.build_tracy_configuration()
try:
self.build_default_configuration()
if ENABLE_TRACY:
self.build_tracy_configuration()
except subprocess.CalledProcessError as e:
print("Native build failed:")
traceback.print_exc()
# This is not great, but setuptools *swallows* exceptions from here
# and mis-reports them as deprecation warnings! This is fairly
# fatal, so just kill it.
sys.exit(1)

def build_default_configuration(self):
print(" *********************************")
Expand Down

0 comments on commit 53820cd

Please sign in to comment.