Files
clang-p2996/clang/tools/amdgpu-arch/AMDGPUArchByHSA.cpp
Yaxun (Sam) Liu 661d91a0fd [clang] Make amdgpu-arch tool work on Windows
Currently amdgpu-arch tool detects AMD GPU by dynamically
loading HSA runtime shared library and using HSA API's,
which is not available on Windows.

This patch makes it work on Windows by dynamically loading
HIP runtime dll and using HIP API's.

Reviewed by: Matt Arsenault, Joseph Huber, Johannes Doerfert

Differential Revision: https://reviews.llvm.org/D153725
2023-07-08 00:01:02 -04:00

123 lines
3.7 KiB
C++

//===- AMDGPUArchLinux.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 AMDGPU installed in system
// using HSA on Linux. This tool is used by AMDGPU OpenMP and HIP driver.
//
//===----------------------------------------------------------------------===//
#include "clang/Basic/Version.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/DynamicLibrary.h"
#include "llvm/Support/Error.h"
#include "llvm/Support/raw_ostream.h"
#include <memory>
#include <string>
#include <vector>
using namespace llvm;
typedef enum {
HSA_STATUS_SUCCESS = 0x0,
} hsa_status_t;
typedef enum {
HSA_DEVICE_TYPE_CPU = 0,
HSA_DEVICE_TYPE_GPU = 1,
} hsa_device_type_t;
typedef enum {
HSA_AGENT_INFO_NAME = 0,
HSA_AGENT_INFO_DEVICE = 17,
} hsa_agent_info_t;
typedef struct hsa_agent_s {
uint64_t handle;
} hsa_agent_t;
hsa_status_t (*hsa_init)();
hsa_status_t (*hsa_shut_down)();
hsa_status_t (*hsa_agent_get_info)(hsa_agent_t, hsa_agent_info_t, void *);
hsa_status_t (*hsa_iterate_agents)(hsa_status_t (*)(hsa_agent_t, void *),
void *);
constexpr const char *DynamicHSAPath = "libhsa-runtime64.so";
llvm::Error loadHSA() {
std::string ErrMsg;
auto DynlibHandle = std::make_unique<llvm::sys::DynamicLibrary>(
llvm::sys::DynamicLibrary::getPermanentLibrary(DynamicHSAPath, &ErrMsg));
if (!DynlibHandle->isValid()) {
return llvm::createStringError(llvm::inconvertibleErrorCode(),
"Failed to 'dlopen' %s", DynamicHSAPath);
}
#define DYNAMIC_INIT(SYMBOL) \
{ \
void *SymbolPtr = DynlibHandle->getAddressOfSymbol(#SYMBOL); \
if (!SymbolPtr) \
return llvm::createStringError(llvm::inconvertibleErrorCode(), \
"Failed to 'dlsym' " #SYMBOL); \
SYMBOL = reinterpret_cast<decltype(SYMBOL)>(SymbolPtr); \
}
DYNAMIC_INIT(hsa_init);
DYNAMIC_INIT(hsa_shut_down);
DYNAMIC_INIT(hsa_agent_get_info);
DYNAMIC_INIT(hsa_iterate_agents);
#undef DYNAMIC_INIT
return llvm::Error::success();
}
static hsa_status_t iterateAgentsCallback(hsa_agent_t Agent, void *Data) {
hsa_device_type_t DeviceType;
hsa_status_t Status =
hsa_agent_get_info(Agent, HSA_AGENT_INFO_DEVICE, &DeviceType);
// continue only if device type if GPU
if (Status != HSA_STATUS_SUCCESS || DeviceType != HSA_DEVICE_TYPE_GPU) {
return Status;
}
std::vector<std::string> *GPUs =
static_cast<std::vector<std::string> *>(Data);
char GPUName[64];
Status = hsa_agent_get_info(Agent, HSA_AGENT_INFO_NAME, GPUName);
if (Status != HSA_STATUS_SUCCESS) {
return Status;
}
GPUs->push_back(GPUName);
return HSA_STATUS_SUCCESS;
}
int printGPUsByHSA() {
// Attempt to load the HSA runtime.
if (llvm::Error Err = loadHSA()) {
logAllUnhandledErrors(std::move(Err), llvm::errs());
return 1;
}
hsa_status_t Status = hsa_init();
if (Status != HSA_STATUS_SUCCESS) {
return 1;
}
std::vector<std::string> GPUs;
Status = hsa_iterate_agents(iterateAgentsCallback, &GPUs);
if (Status != HSA_STATUS_SUCCESS) {
return 1;
}
for (const auto &GPU : GPUs)
llvm::outs() << GPU << '\n';
if (GPUs.size() < 1)
return 1;
hsa_shut_down();
return 0;
}