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