You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
MangoHud/src/nvctrl.cpp

160 lines
4.2 KiB
C++

#include <iostream>
#include <cstring>
#include <sstream>
#include <unordered_map>
#include <memory>
#include <functional>
#include <spdlog/spdlog.h>
#include "nvctrl.h"
#include "loaders/loader_nvctrl.h"
#include "loaders/loader_x11.h"
#include "string_utils.h"
#include "overlay.h"
typedef std::unordered_map<std::string, std::string> string_map;
static std::unique_ptr<Display, std::function<void(Display*)>> display;
struct nvctrlInfo nvctrl_info;
bool nvctrlSuccess = false;
static bool find_nv_x11(libnvctrl_loader& nvctrl, Display*& dpy)
{
char buf[8] {};
for (int i = 0; i < 16; i++) {
snprintf(buf, sizeof(buf), ":%d", i);
Display *d = g_x11->XOpenDisplay(buf);
if (d) {
if (nvctrl.XNVCTRLIsNvScreen(d, 0)) {
dpy = d;
spdlog::debug("XNVCtrl is using display {}", buf);
return true;
}
g_x11->XCloseDisplay(d);
}
}
return false;
}
bool checkXNVCtrl()
{
if (!g_x11->IsLoaded())
return false;
auto& nvctrl = get_libnvctrl_loader();
if (!nvctrl.IsLoaded()) {
spdlog::error("XNVCtrl loader failed to load");
return false;
}
Display *dpy;
nvctrlSuccess = find_nv_x11(nvctrl, dpy);
if (!nvctrlSuccess) {
spdlog::error("XNVCtrl didn't find the correct display");
return false;
}
auto local_x11 = g_x11;
display = { dpy,
[local_x11](Display *dpy) {
local_x11->XCloseDisplay(dpy);
}
};
// get device id at init
int64_t pci_id;
nvctrl.XNVCTRLQueryTargetAttribute64(display.get(),
NV_CTRL_TARGET_TYPE_GPU,
0,
0,
NV_CTRL_PCI_ID,
&pci_id);
deviceID = (pci_id & 0xFFFF);
return true;
}
static void parse_token(std::string token, string_map& options) {
std::string param, value;
size_t equal = token.find("=");
if (equal == std::string::npos)
return;
value = token.substr(equal+1);
param = token.substr(0, equal);
trim(param);
trim(value);
if (!param.empty())
options[param] = value;
}
char* get_attr_target_string(libnvctrl_loader& nvctrl, int attr, int target_type, int target_id) {
char* c = nullptr;
if (!nvctrl.XNVCTRLQueryTargetStringAttribute(display.get(), target_type, target_id, 0, attr, &c)) {
spdlog::error("Failed to query attribute '{}'", attr);
}
return c;
}
void getNvctrlInfo(){
string_map params;
std::string token;
auto& nvctrl = get_libnvctrl_loader();
if (!display)
return;
int enums[] = {
NV_CTRL_STRING_GPU_UTILIZATION,
NV_CTRL_STRING_GPU_CURRENT_CLOCK_FREQS,
0 // keep null
};
for (size_t i=0; enums[i]; i++) {
char* str = get_attr_target_string(nvctrl, enums[i], NV_CTRL_TARGET_TYPE_GPU, 0);
if (!str)
continue;
std::stringstream ss (str);
while (std::getline(ss, token, ',')) {
parse_token(token, params);
}
free(str);
}
if (!try_stoi(nvctrl_info.load, params["graphics"]))
nvctrl_info.load = 0;
if (!try_stoi(nvctrl_info.CoreClock, params["nvclock"]))
nvctrl_info.CoreClock = 0;
if (!try_stoi(nvctrl_info.MemClock, params["memclock"]))
nvctrl_info.MemClock = 0;
int64_t temp = 0;
nvctrl.XNVCTRLQueryTargetAttribute64(display.get(),
NV_CTRL_TARGET_TYPE_GPU,
0,
0,
NV_CTRL_GPU_CORE_TEMPERATURE,
&temp);
nvctrl_info.temp = temp;
int64_t memtotal = 0;
nvctrl.XNVCTRLQueryTargetAttribute64(display.get(),
NV_CTRL_TARGET_TYPE_GPU,
0,
0,
NV_CTRL_TOTAL_DEDICATED_GPU_MEMORY,
&memtotal);
nvctrl_info.memoryTotal = memtotal;
int64_t memused = 0;
nvctrl.XNVCTRLQueryTargetAttribute64(display.get(),
NV_CTRL_TARGET_TYPE_GPU,
0,
0,
NV_CTRL_USED_DEDICATED_GPU_MEMORY,
&memused);
nvctrl_info.memoryUsed = memused;
}