2024-02-16 01:15:09 +00:00
|
|
|
//go:build linux || windows
|
|
|
|
|
|
|
|
package gpu
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"log/slog"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-03-30 16:50:05 +00:00
|
|
|
"runtime"
|
2024-02-16 01:15:09 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Determine if the given ROCm lib directory is usable by checking for existence of some glob patterns
|
|
|
|
func rocmLibUsable(libDir string) bool {
|
|
|
|
slog.Debug("evaluating potential rocm lib dir " + libDir)
|
|
|
|
for _, g := range ROCmLibGlobs {
|
|
|
|
res, _ := filepath.Glob(filepath.Join(libDir, g))
|
|
|
|
if len(res) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetSupportedGFX(libDir string) ([]string, error) {
|
|
|
|
var ret []string
|
|
|
|
files, err := filepath.Glob(filepath.Join(libDir, "rocblas", "library", "TensileLibrary_lazy_gfx*.dat"))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, file := range files {
|
|
|
|
ret = append(ret, strings.TrimSuffix(strings.TrimPrefix(filepath.Base(file), "TensileLibrary_lazy_"), ".dat"))
|
|
|
|
}
|
|
|
|
return ret, nil
|
|
|
|
}
|
|
|
|
|
2024-03-30 16:50:05 +00:00
|
|
|
func rocmGetVisibleDevicesEnv(gpuInfo []GpuInfo) (string, string) {
|
|
|
|
ids := []string{}
|
|
|
|
for _, info := range gpuInfo {
|
|
|
|
if info.Library != "rocm" {
|
|
|
|
// TODO shouldn't happen if things are wired correctly...
|
|
|
|
slog.Debug("rocmGetVisibleDevicesEnv skipping over non-rocm device", "library", info.Library)
|
2024-02-16 01:15:09 +00:00
|
|
|
continue
|
|
|
|
}
|
2024-03-30 16:50:05 +00:00
|
|
|
ids = append(ids, info.ID)
|
2024-02-16 01:15:09 +00:00
|
|
|
}
|
2024-03-30 16:50:05 +00:00
|
|
|
return "HIP_VISIBLE_DEVICES", strings.Join(ids, ",")
|
|
|
|
}
|
2024-02-16 01:15:09 +00:00
|
|
|
|
2024-03-30 16:50:05 +00:00
|
|
|
func commonAMDValidateLibDir() (string, error) {
|
|
|
|
// We try to favor system paths first, so that we can wire up the subprocess to use
|
|
|
|
// the system version. Only use our bundled version if the system version doesn't work
|
|
|
|
// This gives users a more recovery options if versions have subtle problems at runtime
|
|
|
|
|
|
|
|
// Prefer explicit HIP env var
|
|
|
|
hipPath := os.Getenv("HIP_PATH")
|
|
|
|
if hipPath != "" {
|
|
|
|
hipLibDir := filepath.Join(hipPath, "bin")
|
|
|
|
if rocmLibUsable(hipLibDir) {
|
|
|
|
slog.Debug("detected ROCM via HIP_PATH=" + hipPath)
|
|
|
|
return hipLibDir, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan the LD_LIBRARY_PATH or PATH
|
|
|
|
pathEnv := "LD_LIBRARY_PATH"
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
pathEnv = "PATH"
|
|
|
|
}
|
|
|
|
|
|
|
|
paths := os.Getenv(pathEnv)
|
|
|
|
for _, path := range filepath.SplitList(paths) {
|
|
|
|
d, err := filepath.Abs(path)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if rocmLibUsable(d) {
|
|
|
|
return d, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Well known location(s)
|
2024-05-01 22:47:12 +00:00
|
|
|
for _, path := range RocmStandardLocations {
|
|
|
|
if rocmLibUsable(path) {
|
|
|
|
return path, nil
|
|
|
|
}
|
2024-03-30 16:50:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Installer payload location if we're running the installed binary
|
|
|
|
exe, err := os.Executable()
|
|
|
|
if err == nil {
|
|
|
|
rocmTargetDir := filepath.Join(filepath.Dir(exe), "rocm")
|
|
|
|
if rocmLibUsable(rocmTargetDir) {
|
|
|
|
slog.Debug("detected ROCM next to ollama executable " + rocmTargetDir)
|
|
|
|
return rocmTargetDir, nil
|
|
|
|
}
|
2024-02-16 01:15:09 +00:00
|
|
|
}
|
2024-03-30 16:50:05 +00:00
|
|
|
return "", fmt.Errorf("no suitable rocm found, falling back to CPU")
|
2024-02-16 01:15:09 +00:00
|
|
|
}
|