Skip to content

Commit

Permalink
Refactor Windows DLL path handling
Browse files Browse the repository at this point in the history
  • Loading branch information
dalcinl committed Oct 20, 2023
1 parent 6de1d61 commit 5ead19c
Show file tree
Hide file tree
Showing 3 changed files with 99 additions and 32 deletions.
9 changes: 9 additions & 0 deletions .cibw/merge-wheels.py
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,10 @@ def wheel_tagline(tags: list[str]) -> str:
if libdir.exists():
libdir.rmdir()

mpipth = root_dir / "mpi.pth"
if mpipth.exists():
mpipth.unlink()

record = distinfo_dir / "RECORD"
record.unlink()

Expand Down Expand Up @@ -142,6 +146,11 @@ def wheel_tagline(tags: list[str]) -> str:
fh.write("# Register MPI ABI variants\n")
for variant in variant_registry:
fh.write(f"_mpiabi._register({variant!r})\n")
if tags[-1].startswith("win"):
fh.write(textwrap.dedent("""\
# Setup Windows DLL search path
_mpiabi._setup_windows_dll_path()
"""))

output_dir.mkdir(parents=True, exist_ok=True)
wheel_pack.pack(root_dir, output_dir, None)
Expand Down
106 changes: 86 additions & 20 deletions .cibw/mpi4py_mpiabi.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,38 +14,47 @@ def _verbose_info(message, verbosity=1):
print(f"# [{__spec__.parent}] {message}", file=sys.stderr)


def _site_prefixes():
prefixes = []
site = sys.modules.get("site")
if site is not None:
if sys.exec_prefix != sys.base_exec_prefix:
venv_base = sys.exec_prefix
prefixes.append(venv_base)
if site.ENABLE_USER_SITE:
user_base = os.path.abspath(site.USER_BASE)
prefixes.append(user_base)
if sys.base_exec_prefix in site.PREFIXES:
system_base = sys.base_exec_prefix
if sys.platform == "win32":
prefixes.append(system_base)
return prefixes


def _dlopen_rpath():
rpath = []

def add_rpath(*paths):
rpath.extend(paths)
def add_rpath(*directory):
path = os.path.join(*directory)
if path not in rpath:
rpath.append(path)

def add_rpath_prefix(prefix):
if sys.platform == "linux":
add_rpath(os.path.join(prefix, "lib"))
add_rpath(prefix, "lib")
if sys.platform == "win32":
add_rpath(os.path.join(prefix, "DLLs"))
add_rpath(os.path.join(prefix, "Library", "bin"))
add_rpath(prefix, "DLLs")
add_rpath(prefix, "Library", "bin")

site = sys.modules.get("site")
if site is not None and site.ENABLE_USER_SITE:
user_base = os.path.abspath(site.USER_BASE)
user_site = os.path.abspath(site.USER_SITE)
site_pkgs = os.path.commonpath((user_site, __file__))
if site_pkgs == user_site:
add_rpath_prefix(user_base)

if sys.exec_prefix != sys.base_exec_prefix:
add_rpath_prefix(sys.exec_prefix)
for prefix in _site_prefixes():
add_rpath_prefix(prefix)

add_rpath("")

if sys.platform == "darwin":
add_rpath(
"/usr/local/lib",
"/opt/homebrew/lib",
"/opt/local/lib",
)
add_rpath("/usr/local/lib")
add_rpath("/opt/homebrew/lib")
add_rpath("/opt/local/lib")

return rpath

Expand Down Expand Up @@ -226,3 +235,60 @@ def find_spec(cls, fullname, path, target=None):
def _install_finder():
if _Finder not in sys.meta_path:
sys.meta_path.append(_Finder)


def _setup_windows_dll_path(): # noqa: C901
i_mpi_root = os.environ.get("I_MPI_ROOT")
i_mpi_library_kind = (
os.environ.get("I_MPI_LIBRARY_KIND") or
os.environ.get("library_kind") or
"release"
)
i_mpi_ofi_library_internal = (
os.environ.get("I_MPI_OFI_LIBRARY_INTERNAL", "").lower()
not in ("0", "no", "off", "false", "disable")
)
msmpi_bin = os.environ.get("MSMPI_BIN")

dllpath = []

def add_dllpath(*directory, dll=""):
path = os.path.join(*directory)
if path not in dllpath:
filename = os.path.join(path, f"{dll}.dll")
if os.path.isfile(filename):
dllpath.append(path)

def add_dllpath_impi(*rootdir):
if i_mpi_ofi_library_internal:
add_dllpath(*rootdir, "libfabric", "bin", dll="libfabric")
add_dllpath(*rootdir, "bin", "libfabric", dll="libfabric")
if i_mpi_library_kind:
add_dllpath(*rootdir, "bin", i_mpi_library_kind, dll="impi")
add_dllpath(*rootdir, "bin", dll="impi")

def add_dllpath_msmpi(*bindir):
add_dllpath(*bindir, dll="msmpi")

def add_dllpath_prefix(prefix):
add_dllpath_impi(prefix, "Library")
add_dllpath_msmpi(prefix, "Library", "bin")

for prefix in _site_prefixes():
add_dllpath_prefix(prefix)

if i_mpi_root:
add_dllpath_impi(i_mpi_root)

if msmpi_bin:
add_dllpath_msmpi(msmpi_bin)

ospath = os.environ["PATH"].split(os.path.pathsep)
for entry in dllpath:
if entry not in ospath:
ospath.append(entry)
os.environ["PATH"] = os.path.pathsep.join(ospath)

if hasattr(os, "add_dll_directory"):
for entry in dllpath:
os.add_dll_directory(entry)
16 changes: 4 additions & 12 deletions .github/workflows/cd-wheel.yml
Original file line number Diff line number Diff line change
Expand Up @@ -332,17 +332,12 @@ jobs:
Darwin-*)
;;
*NT*-impi_rt)
I_MPI_ROOT=$(cygpath -w "$CONDA_PREFIX/Library")
echo "I_MPI_ROOT=$I_MPI_ROOT" >> $GITHUB_ENV
echo "$I_MPI_ROOT\\bin" >> $GITHUB_PATH
echo "$I_MPI_ROOT\\bin\\libfabric" >> $GITHUB_PATH
#I_MPI_ROOT=$(cygpath -w "$CONDA_PREFIX/Library")
#echo "I_MPI_ROOT=$I_MPI_ROOT" >> $GITHUB_ENV
;;
*NT*-msmpi)
MSMPI_ROOT=$(cygpath -w "$CONDA_PREFIX/Library")
echo "MSMPI_BIN=$MSMPI_ROOT\\bin" >> $GITHUB_ENV
echo "MSMPI_INC=$MSMPI_ROOT\\include" >> $GITHUB_ENV
echo "MSMPI_LIB64=$MSMPI_ROOT\\lib" >> $GITHUB_ENV
echo "$MSMPI_BIN" >> $GITHUB_PATH
#MSMPI_ROOT=$(cygpath -w "$CONDA_PREFIX/Library")
#echo "MSMPI_BIN=$MSMPI_ROOT\\bin" >> $GITHUB_ENV
;;
esac
Expand Down Expand Up @@ -413,7 +408,6 @@ jobs:
mpiexec: ${{steps.user.outputs.mpiexec }}
env:
PYTHONUSERBASE: ${{ runner.temp }}/user
I_MPI_FABRICS: shm

- if: ${{ matrix.mpi == 'impi' }}
id: venv
Expand Down Expand Up @@ -442,8 +436,6 @@ jobs:
with:
python: ${{ steps.venv.outputs.python }}
mpiexec: ${{steps.venv.outputs.mpiexec }}
env:
I_MPI_FABRICS: shm

- uses: mpi4py/setup-mpi@v1
with:
Expand Down

0 comments on commit 5ead19c

Please sign in to comment.