xref: /llvm-project/clang/tools/amdgpu-arch/AMDGPUArchByKFD.cpp (revision 55fad5e980efbcc4a581057c558fedaadf91e877)
1129a1a27SJoseph Huber //===- AMDGPUArchByKFD.cpp - list AMDGPU installed ------*- C++ -*---------===//
2129a1a27SJoseph Huber //
3129a1a27SJoseph Huber // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4129a1a27SJoseph Huber // See https://llvm.org/LICENSE.txt for license information.
5129a1a27SJoseph Huber // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6129a1a27SJoseph Huber //
7129a1a27SJoseph Huber //===----------------------------------------------------------------------===//
8129a1a27SJoseph Huber //
9129a1a27SJoseph Huber // This file implements a tool for detecting name of AMD GPUs installed in
10129a1a27SJoseph Huber // system using the Linux sysfs interface for the AMD KFD driver. This file does
11129a1a27SJoseph Huber // not respect ROCR_VISIBLE_DEVICES like the ROCm environment would.
12129a1a27SJoseph Huber //
13129a1a27SJoseph Huber //===----------------------------------------------------------------------===//
14129a1a27SJoseph Huber 
15129a1a27SJoseph Huber #include "llvm/Support/FileSystem.h"
16129a1a27SJoseph Huber #include "llvm/Support/LineIterator.h"
17129a1a27SJoseph Huber #include "llvm/Support/MemoryBuffer.h"
18129a1a27SJoseph Huber #include "llvm/Support/Path.h"
19129a1a27SJoseph Huber #include <memory>
20129a1a27SJoseph Huber 
21129a1a27SJoseph Huber using namespace llvm;
22129a1a27SJoseph Huber 
23129a1a27SJoseph Huber constexpr static const char *KFD_SYSFS_NODE_PATH =
24129a1a27SJoseph Huber     "/sys/devices/virtual/kfd/kfd/topology/nodes";
25129a1a27SJoseph Huber 
26129a1a27SJoseph Huber // See the ROCm implementation for how this is handled.
27129a1a27SJoseph Huber // https://github.com/ROCm/ROCT-Thunk-Interface/blob/master/src/libhsakmt.h#L126
28129a1a27SJoseph Huber constexpr static long getMajor(long Ver) { return (Ver / 10000) % 100; }
29129a1a27SJoseph Huber constexpr static long getMinor(long Ver) { return (Ver / 100) % 100; }
30129a1a27SJoseph Huber constexpr static long getStep(long Ver) { return Ver % 100; }
31129a1a27SJoseph Huber 
32129a1a27SJoseph Huber int printGPUsByKFD() {
33129a1a27SJoseph Huber   SmallVector<std::pair<long, long>> Devices;
34129a1a27SJoseph Huber   std::error_code EC;
35129a1a27SJoseph Huber   for (sys::fs::directory_iterator Begin(KFD_SYSFS_NODE_PATH, EC), End;
36129a1a27SJoseph Huber        Begin != End; Begin.increment(EC)) {
37129a1a27SJoseph Huber     if (EC)
38129a1a27SJoseph Huber       return 1;
39129a1a27SJoseph Huber 
40129a1a27SJoseph Huber     long Node = 0;
41129a1a27SJoseph Huber     if (sys::path::stem(Begin->path()).consumeInteger(10, Node))
42129a1a27SJoseph Huber       return 1;
43129a1a27SJoseph Huber 
44129a1a27SJoseph Huber     SmallString<0> Path(Begin->path());
45129a1a27SJoseph Huber     sys::path::append(Path, "properties");
46129a1a27SJoseph Huber 
47129a1a27SJoseph Huber     ErrorOr<std::unique_ptr<MemoryBuffer>> BufferOrErr =
48129a1a27SJoseph Huber         MemoryBuffer::getFileOrSTDIN(Path);
49129a1a27SJoseph Huber     if (std::error_code EC = BufferOrErr.getError())
50129a1a27SJoseph Huber       return 1;
51129a1a27SJoseph Huber 
52129a1a27SJoseph Huber     long GFXVersion = 0;
53129a1a27SJoseph Huber     for (line_iterator Lines(**BufferOrErr, false); !Lines.is_at_end();
54129a1a27SJoseph Huber          ++Lines) {
55129a1a27SJoseph Huber       StringRef Line(*Lines);
56129a1a27SJoseph Huber       if (Line.consume_front("gfx_target_version")) {
57*55fad5e9SJoseph Huber         if (Line.drop_while([](char C) { return std::isspace(C); })
58*55fad5e9SJoseph Huber                 .consumeInteger(10, GFXVersion))
59129a1a27SJoseph Huber           return 1;
60129a1a27SJoseph Huber         break;
61129a1a27SJoseph Huber       }
62129a1a27SJoseph Huber     }
63129a1a27SJoseph Huber 
64129a1a27SJoseph Huber     // If this is zero the node is a CPU.
65129a1a27SJoseph Huber     if (GFXVersion == 0)
66129a1a27SJoseph Huber       continue;
67129a1a27SJoseph Huber     Devices.emplace_back(Node, GFXVersion);
68129a1a27SJoseph Huber   }
69129a1a27SJoseph Huber 
70129a1a27SJoseph Huber   // Sort the devices by their node to make sure it prints in order.
71129a1a27SJoseph Huber   llvm::sort(Devices, [](auto &L, auto &R) { return L.first < R.first; });
72129a1a27SJoseph Huber   for (const auto &[Node, GFXVersion] : Devices)
73129a1a27SJoseph Huber     std::fprintf(stdout, "gfx%ld%ld%lx\n", getMajor(GFXVersion),
74129a1a27SJoseph Huber                  getMinor(GFXVersion), getStep(GFXVersion));
75129a1a27SJoseph Huber 
76129a1a27SJoseph Huber   return 0;
77129a1a27SJoseph Huber }
78