
Summary: These two tools do the same thing, we should unify them into a single tool. We create symlinks for backward compatiblity and provide a way to get the old vendor specific behavior with `--amdgpu-only` and `--nvptx-only`.
78 lines
2.7 KiB
C++
78 lines
2.7 KiB
C++
//===- AMDGPUArchByKFD.cpp - list AMDGPU installed ------*- C++ -*---------===//
|
|
//
|
|
// Part of the LLVM Project, 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file implements a tool for detecting name of AMD GPUs installed in
|
|
// system using the Linux sysfs interface for the AMD KFD driver. This file does
|
|
// not respect ROCR_VISIBLE_DEVICES like the ROCm environment would.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
#include "llvm/Support/LineIterator.h"
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include <memory>
|
|
|
|
using namespace llvm;
|
|
|
|
constexpr static const char *KFD_SYSFS_NODE_PATH =
|
|
"/sys/devices/virtual/kfd/kfd/topology/nodes";
|
|
|
|
// See the ROCm implementation for how this is handled.
|
|
// https://github.com/ROCm/ROCT-Thunk-Interface/blob/master/src/libhsakmt.h#L126
|
|
constexpr static long getMajor(long Ver) { return (Ver / 10000) % 100; }
|
|
constexpr static long getMinor(long Ver) { return (Ver / 100) % 100; }
|
|
constexpr static long getStep(long Ver) { return Ver % 100; }
|
|
|
|
int printGPUsByKFD() {
|
|
SmallVector<std::pair<long, long>> Devices;
|
|
std::error_code EC;
|
|
for (sys::fs::directory_iterator Begin(KFD_SYSFS_NODE_PATH, EC), End;
|
|
Begin != End; Begin.increment(EC)) {
|
|
if (EC)
|
|
return 1;
|
|
|
|
long Node = 0;
|
|
if (sys::path::stem(Begin->path()).consumeInteger(10, Node))
|
|
return 1;
|
|
|
|
SmallString<0> Path(Begin->path());
|
|
sys::path::append(Path, "properties");
|
|
|
|
ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
|
|
MemoryBuffer::getFileOrSTDIN(Path);
|
|
if (std::error_code EC = BufferOrErr.getError())
|
|
return 1;
|
|
|
|
long GFXVersion = 0;
|
|
for (line_iterator Lines(**BufferOrErr, false); !Lines.is_at_end();
|
|
++Lines) {
|
|
StringRef Line(*Lines);
|
|
if (Line.consume_front("gfx_target_version")) {
|
|
if (Line.drop_while([](char C) { return std::isspace(C); })
|
|
.consumeInteger(10, GFXVersion))
|
|
return 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If this is zero the node is a CPU.
|
|
if (GFXVersion == 0)
|
|
continue;
|
|
Devices.emplace_back(Node, GFXVersion);
|
|
}
|
|
|
|
// Sort the devices by their node to make sure it prints in order.
|
|
llvm::sort(Devices, [](auto &L, auto &R) { return L.first < R.first; });
|
|
for (const auto &[Node, GFXVersion] : Devices)
|
|
std::fprintf(stdout, "gfx%ld%ld%lx\n", getMajor(GFXVersion),
|
|
getMinor(GFXVersion), getStep(GFXVersion));
|
|
|
|
return 0;
|
|
}
|