File size: 2,838 Bytes
2a0bc63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
import platform
from ctypes import CDLL
from typing import List, Optional
from pathlib import Path

from .logger import logger


def find_library(path: Optional[str] = None, gpu: bool = False) -> str:
    lib_directory = Path(__file__).parent.resolve() / "lib"

    if path:
        subdirs = [d.name for d in lib_directory.iterdir() if d.is_dir()]
        if path not in subdirs:
            return path

    system = platform.system()
    metal = gpu and system == "Darwin"
    cuda = gpu and not metal
    if not path:
        if (lib_directory / "local").is_dir():
            path = "local"
        elif cuda:
            path = "cuda"
        elif metal:
            path = "metal"
        elif platform.processor() == "arm":
            # Apple silicon doesn't support AVX/AVX2.
            path = "basic" if system == "Darwin" else ""
        else:
            from cpuinfo import get_cpu_info

            try:
                flags = get_cpu_info()["flags"]
            except:
                logger.warning(
                    "Unable to detect CPU features. "
                    "Please report at https://github.com/marella/ctransformers/issues"
                )
                flags = []

            if "avx2" in flags:
                path = "avx2"
            elif "avx" in flags and "f16c" in flags:
                path = "avx"
            else:
                path = "basic"

    name = "ctransformers"
    if system == "Linux":
        name = f"lib{name}.so"
    elif system == "Windows":
        name = f"{name}.dll"
    elif system == "Darwin":
        name = f"lib{name}.dylib"
    else:
        name = ""

    path = lib_directory / path / name
    if not path.is_file():
        if cuda:
            env = "CT_CUBLAS=1 "
        elif metal:
            env = "CT_METAL=1 "
        else:
            env = ""
        raise OSError(
            "Precompiled binaries are not available for the current platform. "
            "Please reinstall from source using:\n\n"
            "  pip uninstall ctransformers --yes\n"
            f"  {env}pip install ctransformers --no-binary ctransformers\n\n"
        )
    return str(path)


def load_cuda() -> bool:
    try:
        import nvidia
    except ImportError:
        return False
    path = Path(nvidia.__path__[0])
    system = platform.system()
    if system == "Windows":
        libs = [
            path / "cuda_runtime" / "bin" / "cudart64_12.dll",
            path / "cublas" / "bin" / "cublas64_12.dll",
        ]
    else:
        libs = [
            path / "cuda_runtime" / "lib" / "libcudart.so.12",
            path / "cublas" / "lib" / "libcublas.so.12",
        ]
    for lib in libs:
        if not lib.is_file():
            return False
    libs = [str(lib.resolve()) for lib in libs]
    for lib in libs:
        CDLL(lib)
    return True