123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418 |
- import contextlib
- import io
- import os
- import re
- import subprocess
- from typing import List, Set
- import warnings
- from pathlib import Path
- from packaging.version import parse, Version
- import setuptools
- import torch
- import torch.utils.cpp_extension as torch_cpp_ext
- from torch.utils.cpp_extension import (
- BuildExtension, CUDAExtension, CUDA_HOME, ROCM_HOME)
- ROOT_DIR = os.path.dirname(__file__)
- MAIN_CUDA_VERSION = "12.1"
- # Supported NVIDIA GPU architectures.
- NVIDIA_SUPPORTED_ARCHS = {
- "6.1", "7.0", "7.5", "8.0", "8.6", "8.9", "9.0"
- }
- ROCM_SUPPORTED_ARCHS = {
- "gfx90a", "gfx908", "gfx906", "gfx1030", "gfx1100"
- }
- def _is_hip() -> bool:
- return torch.version.hip is not None
- def _is_cuda() -> bool:
- return torch.version.cuda is not None
- # Compiler flags.
- CXX_FLAGS = ["-g", "-O2", "-std=c++17"]
- # TODO: Should we use -O3?
- NVCC_FLAGS = ["-O2", "-std=c++17"]
- if _is_hip():
- if ROCM_HOME is None:
- raise RuntimeError(
- "Cannot find ROCM_HOME. ROCm must be available to build the "
- "package.")
- NVCC_FLAGS += ["-DUSE_ROCM"]
- if _is_cuda() and CUDA_HOME is None:
- raise RuntimeError(
- "Cannot find CUDA_HOME. CUDA must be available to build the package.")
- ABI = 1 if torch._C._GLIBCXX_USE_CXX11_ABI else 0
- CXX_FLAGS += [f"-D_GLIBCXX_USE_CXX11_ABI={ABI}"]
- NVCC_FLAGS += [f"-D_GLIBCXX_USE_CXX11_ABI={ABI}"]
- def get_amdgpu_offload_arch():
- command = "/opt/rocm/llvm/bin/amdgpu-offload-arch"
- try:
- output = subprocess.check_output([command])
- return output.decode('utf-8').strip()
- except subprocess.CalledProcessError as e:
- error_message = f"Error: {e}"
- raise RuntimeError(error_message) from e
- except FileNotFoundError as e:
- # If the command is not found, print an error message
- error_message = f"The command {command} was not found."
- raise RuntimeError(error_message) from e
- return None
- def get_hipcc_rocm_version():
- # Run the hipcc --version command
- result = subprocess.run(['hipcc', '--version'],
- stdout=subprocess.PIPE,
- stderr=subprocess.STDOUT,
- text=True)
- # Check if the command was executed successfully
- if result.returncode != 0:
- print("Error running 'hipcc --version'")
- return None
- # Extract the version using a regular expression
- match = re.search(r'HIP version: (\S+)', result.stdout)
- if match:
- # Return the version string
- return match.group(1)
- else:
- print("Could not find HIP version in the output")
- return None
- def glob(pattern: str):
- root = Path(__name__).parent
- return [str(p) for p in root.glob(pattern)]
- def get_nvcc_cuda_version(cuda_dir: str) -> Version:
- """Get the CUDA version from nvcc.
- Adapted from https://github.com/NVIDIA/apex/blob/8b7a1ff183741dd8f9b87e7bafd04cfde99cea28/setup.py
- """
- nvcc_output = subprocess.check_output([cuda_dir + "/bin/nvcc", "-V"],
- universal_newlines=True)
- output = nvcc_output.split()
- release_idx = output.index("release") + 1
- nvcc_cuda_version = parse(output[release_idx].split(",")[0])
- return nvcc_cuda_version
- def get_torch_arch_list() -> Set[str]:
- # TORCH_CUDA_ARCH_LIST can have one or more architectures,
- # e.g. "8.0" or "7.5,8.0,8.6+PTX". Here, the "8.6+PTX" option asks the
- # compiler to additionally include PTX code that can be runtime-compiled
- # and executed on the 8.6 or newer architectures. While the PTX code will
- # not give the best performance on the newer architectures, it provides
- # forward compatibility.
- env_arch_list = os.environ.get("TORCH_CUDA_ARCH_LIST", None)
- if env_arch_list is None:
- return set()
- # List are separated by ; or space.
- torch_arch_list = set(env_arch_list.replace(" ", ";").split(";"))
- if not torch_arch_list:
- return set()
- # Filter out the invalid architectures and print a warning.
- valid_archs = NVIDIA_SUPPORTED_ARCHS.union(
- {s + "+PTX"
- for s in NVIDIA_SUPPORTED_ARCHS})
- arch_list = torch_arch_list.intersection(valid_archs)
- # If none of the specified architectures are valid, raise an error.
- if not arch_list:
- raise RuntimeError(
- "None of the CUDA/ROCM architectures in `TORCH_CUDA_ARCH_LIST` "
- f"env variable ({env_arch_list}) is supported. "
- f"Supported CUDA architectures are: {valid_archs}.")
- invalid_arch_list = torch_arch_list - valid_archs
- if invalid_arch_list:
- warnings.warn(
- f"Unsupported CUDA/ROCM architectures ({invalid_arch_list}) are "
- "excluded from the `TORCH_CUDA_ARCH_LIST` env variable "
- f"({env_arch_list}). Supported CUDA/ROCM architectures are: "
- f"{valid_archs}.",
- stacklevel=2)
- return arch_list
- # First, check the TORCH_CUDA_ARCH_LIST environment variable.
- compute_capabilities = get_torch_arch_list()
- if _is_cuda() and not compute_capabilities:
- # If TORCH_CUDA_ARCH_LIST is not defined or empty, target all available
- # GPUs on the current machine.
- device_count = torch.cuda.device_count()
- for i in range(device_count):
- major, minor = torch.cuda.get_device_capability(i)
- if major < 6:
- raise RuntimeError(
- "GPUs with compute capability below 6.0 are not supported.")
- compute_capabilities.add(f"{major}.{minor}")
- ext_modules = []
- if _is_cuda():
- nvcc_cuda_version = get_nvcc_cuda_version(CUDA_HOME)
- if not compute_capabilities:
- # If no GPU is specified nor available, add all supported architectures
- # based on the NVCC CUDA version.
- compute_capabilities = NVIDIA_SUPPORTED_ARCHS.copy()
- if nvcc_cuda_version < Version("11.1"):
- compute_capabilities.remove("8.6")
- if nvcc_cuda_version < Version("11.8"):
- compute_capabilities.remove("8.9")
- compute_capabilities.remove("9.0")
- # Validate the NVCC CUDA version.
- if nvcc_cuda_version < Version("11.0"):
- raise RuntimeError(
- "CUDA 11.0 or higher is required to build the package.")
- if (nvcc_cuda_version < Version("11.1")
- and any(cc.startswith("8.6") for cc in compute_capabilities)):
- raise RuntimeError(
- "CUDA 11.1 or higher is required for compute capability 8.6.")
- if nvcc_cuda_version < Version("11.8"):
- if any(cc.startswith("8.9") for cc in compute_capabilities):
- # CUDA 11.8 is required to generate the code targeting compute capability 8.9.
- # However, GPUs with compute capability 8.9 can also run the code generated by
- # the previous versions of CUDA 11 and targeting compute capability 8.0.
- # Therefore, if CUDA 11.8 is not available, we target compute capability 8.0
- # instead of 8.9.
- warnings.warn(
- "CUDA 11.8 or higher is required for compute capability 8.9. "
- "Targeting compute capability 8.0 instead.",
- stacklevel=2)
- compute_capabilities = set(cc for cc in compute_capabilities
- if not cc.startswith("8.9"))
- compute_capabilities.add("8.0+PTX")
- if any(cc.startswith("9.0") for cc in compute_capabilities):
- raise RuntimeError(
- "CUDA 11.8 or higher is required for compute capability 9.0.")
- NVCC_FLAGS_PUNICA = NVCC_FLAGS.copy()
- # Add target compute capabilities to NVCC flags.
- for capability in compute_capabilities:
- num = capability[0] + capability[2]
- NVCC_FLAGS += ["-gencode", f"arch=compute_{num},code=sm_{num}"]
- if capability.endswith("+PTX"):
- NVCC_FLAGS += [
- "-gencode", f"arch=compute_{num},code=compute_{num}"
- ]
- if int(capability[0]) >= 8:
- NVCC_FLAGS_PUNICA += [
- "-gencode", f"arch=compute_{num},code=sm_{num}"
- ]
- if capability.endswith("+PTX"):
- NVCC_FLAGS_PUNICA += [
- "-gencode", f"arch=compute_{num},code=compute_{num}"
- ]
- # Use NVCC threads to parallelize the build.
- if nvcc_cuda_version >= Version("11.2"):
- nvcc_threads = int(os.getenv("NVCC_THREADS", 8))
- num_threads = min(os.cpu_count(), nvcc_threads)
- NVCC_FLAGS += ["--threads", str(num_threads)]
-
- if nvcc_cuda_version >= Version("11.8"):
- NVCC_FLAGS += ["-DENABLE_FP8_E5M2"]
- # changes for punica kernels
- NVCC_FLAGS += torch_cpp_ext.COMMON_NVCC_FLAGS
- REMOVE_NVCC_FLAGS = [
- '-D__CUDA_NO_HALF_OPERATORS__',
- '-D__CUDA_NO_HALF_CONVERSIONS__',
- '-D__CUDA_NO_BFLOAT16_CONVERSIONS__',
- '-D__CUDA_NO_HALF2_OPERATORS__',
- ]
- for flag in REMOVE_NVCC_FLAGS:
- with contextlib.suppress(ValueError):
- torch_cpp_ext.COMMON_NVCC_FLAGS.remove(flag)
- install_punica = bool(int(os.getenv("APHRODITE_INSTALL_PUNICA_KERNELS", "1")))
- device_count = torch.cuda.device_count()
- for i in range(device_count):
- major, minor = torch.cuda.get_device_capability(i)
- if major < 8:
- install_punica = False
- break
- if install_punica:
- ext_modules.append(
- CUDAExtension(
- name="aphrodite._punica_C",
- sources=["kernels/punica/punica_ops.cc"] +
- glob("kernels/punica/bgmv/*.cu"),
- extra_compile_args={
- "cxx": CXX_FLAGS,
- "nvcc": NVCC_FLAGS_PUNICA,
- },
- ))
-
- install_hadamard = bool(int(os.getenv("APHRODITE_INSTALL_HADAMARD_KERNELS", "1")))
- device_count = torch.cuda.device_count()
- for i in range(device_count):
- major, minor = torch.cuda.get_device_capability(i)
- if major < 7:
- install_hadamard = False
- break
- if install_hadamard:
- ext_modules.append(
- CUDAExtension(
- name="aphrodite._hadamard_C",
- sources=["kernels/hadamard/fast_hadamard_transform.cpp",
- "kernels/hadamard/fast_hadamard_transform_cuda.cu"],
- extra_compile_args={
- "cxx": CXX_FLAGS,
- "nvcc": NVCC_FLAGS,
- },
- ))
- elif _is_hip():
- amd_arch = get_amdgpu_offload_arch()
- if amd_arch not in ROCM_SUPPORTED_ARCHS:
- raise RuntimeError(
- f"Only the following arch is supported: {ROCM_SUPPORTED_ARCHS}"
- f"amdgpu_arch_found: {amd_arch}")
- aphrodite_extension_sources = [
- "kernels/cache_kernels.cu",
- "kernels/attention/attention_kernels.cu",
- "kernels/pos_encoding_kernels.cu",
- "kernels/activation_kernels.cu",
- "kernels/layernorm_kernels.cu",
- "kernels/quantization/squeezellm/quant_cuda_kernel.cu",
- "kernels/quantization/gguf/gguf_kernel.cu",
- "kernels/quantization/gptq/q_gemm.cu",
- "kernels/cuda_utils_kernels.cu",
- "kernels/moe/align_block_size_kernel.cu",
- "kernels/pybind.cpp",
- ]
- if _is_cuda():
- aphrodite_extension_sources.append("kernels/quantization/awq/gemm_kernels.cu")
- aphrodite_extension_sources.append("kernels/quantization/quip/origin_order.cu")
- aphrodite_extension_sources.append("kernels/quantization/marlin/marlin_cuda_kernel.cu")
- aphrodite_extension_sources.append("kernels/all_reduce/custom_all_reduce.cu")
- aphrodite_extension = CUDAExtension(
- name="aphrodite._C",
- sources=aphrodite_extension_sources,
- extra_compile_args={
- "cxx": CXX_FLAGS,
- "nvcc": NVCC_FLAGS,
- },
- libraries=["cuda", "conda/envs/aphrodite-runtime/lib",
- "conda/envs/aphrodite-runtime/lib/stubs"] if _is_cuda() else [],
- library_dirs=["conda/envs/aphrodite-runtime/lib",
- "conda/envs/aphrodite-runtime/lib/stubs"] if _is_cuda() else [],
- )
- ext_modules.append(aphrodite_extension)
- def get_path(*filepath) -> str:
- return os.path.join(ROOT_DIR, *filepath)
- def find_version(filepath: str) -> str:
- """Extract version information from the given filepath.
- Adapted from https://github.com/ray-project/ray/blob/0b190ee1160eeca9796bc091e07eaebf4c85b511/python/setup.py
- """
- with open(filepath) as fp:
- version_match = re.search(r"^__version__ = ['\"]([^'\"]*)['\"]",
- fp.read(), re.M)
- if version_match:
- return version_match.group(1)
- raise RuntimeError("Unable to find version string.")
- def get_aphrodite_version() -> str:
- version = find_version(get_path("aphrodite-engine", "__init__.py"))
-
- if _is_hip():
- # get the HIP version
- hipcc_version = get_hipcc_rocm_version()
- if hipcc_version != MAIN_CUDA_VERSION:
- rocm_version_str = hipcc_version.replace(".", "")[:3]
- version += f"+rocm{rocm_version_str}"
- else:
- cuda_version = str(nvcc_cuda_version)
- # Split the version into numerical and suffix parts
- version_parts = version.split('-')
- version_num = version_parts[0]
- version_suffix = version_parts[1] if len(version_parts) > 1 else ''
-
- if cuda_version != MAIN_CUDA_VERSION:
- cuda_version_str = cuda_version.replace(".", "")[:3]
- version_num += f"+cu{cuda_version_str}"
-
- # Reassemble the version string with the suffix, if any
- version = version_num + ('-' + version_suffix if version_suffix else '')
-
- return version
- def read_readme() -> str:
- """Read the README file if present."""
- p = get_path("README.md")
- if os.path.isfile(p):
- return io.open(get_path("README.md"), "r", encoding="utf-8").read()
- else:
- return ""
- def get_requirements() -> List[str]:
- """Get Python package dependencies from requirements.txt."""
- if _is_hip():
- with open(get_path("requirements-rocm.txt")) as f:
- requirements = f.read().strip().split("\n")
- else:
- with open(get_path("requirements.txt")) as f:
- requirements = f.read().strip().split("\n")
- return requirements
- setuptools.setup(
- name="aphrodite-engine",
- version=find_version(get_path("aphrodite", "__init__.py")),
- author="PygmalionAI",
- license="AGPL 3.0",
- description="The inference engine for PygmalionAI models",
- long_description=read_readme(),
- long_description_content_type="text/markdown",
- url="https://github.com/PygmalionAI/aphrodite-engine",
- project_urls={
- "Homepage": "https://pygmalion.chat",
- "Documentation": "https://docs.pygmalion.chat",
- "GitHub": "https://github.com/PygmalionAI",
- "Huggingface": "https://huggingface.co/PygmalionAI",
- },
- classifiers=[
- "Programming Language :: Python :: 3.8",
- "Programming Language :: Python :: 3.9",
- "Programming Language :: Python :: 3.10",
- "Programming Language :: Python :: 3.11",
- "License :: OSI Approved :: GNU Affero General Public License v3 or later (AGPLv3+)",
- "Topic :: Scientific/Engineering :: Artificial Intelligence",
- ],
- packages=setuptools.find_packages(exclude=("kernels", "examples",
- "tests")),
- python_requires=">=3.8",
- install_requires=get_requirements(),
- ext_modules=ext_modules,
- cmdclass={"build_ext": BuildExtension},
- package_data={"aphrodite-engine": ["aphrodite/endpoints/kobold/klite.embd",
- "aphrodite/modeling/layers/quantization/hadamard.safetensors",
- "py.typed"]},
- include_package_data=True,
- )
|