Compare commits
No commits in common. "master" and "v2.1" have entirely different histories.
@ -3,7 +3,7 @@
|
|||||||
This program allows you to read temperature sensors off of supported EVGA 30-series iCX3 video cards, as well as control the fans individually.
|
This program allows you to read temperature sensors off of supported EVGA 30-series iCX3 video cards, as well as control the fans individually.
|
||||||
|
|
||||||
## Prerequisites
|
## Prerequisites
|
||||||
A supported EVGA 30-series card with iCX3. This includes:
|
A supported EVGA 30-series iCX3 card. I have not done extensive testing but belive this is every model of their:
|
||||||
* RTX 3060 Ti
|
* RTX 3060 Ti
|
||||||
* RTX 3070
|
* RTX 3070
|
||||||
* RTX 3070 Ti
|
* RTX 3070 Ti
|
||||||
@ -14,8 +14,6 @@ A supported EVGA 30-series card with iCX3. This includes:
|
|||||||
|
|
||||||
The number of fans supported depends, of course, on your particular model.
|
The number of fans supported depends, of course, on your particular model.
|
||||||
|
|
||||||
You must have the `i2c-dev` kernel module loaded with `modprobe i2c-dev`
|
|
||||||
|
|
||||||
Access to the `/dev/i2c` device files, which means either:
|
Access to the `/dev/i2c` device files, which means either:
|
||||||
* Run as root, or
|
* Run as root, or
|
||||||
* Install udev rules to allow user access. If you have the OpenRGB udev rules installed to control the LEDs you already have this set up.
|
* Install udev rules to allow user access. If you have the OpenRGB udev rules installed to control the LEDs you already have this set up.
|
||||||
@ -41,7 +39,6 @@ Note that when controlling fans directly through iCX3 they will fall offline fro
|
|||||||
|
|
||||||
```text
|
```text
|
||||||
Available options:
|
Available options:
|
||||||
--i2c N : Only probe I2C bus N instead of all (may help with stuttering or freezing when probing I2C devices)
|
|
||||||
--gpu N : Control only GPU N instead of all supported cards
|
--gpu N : Control only GPU N instead of all supported cards
|
||||||
--fan SPEED : Set all fans at once to SPEED (see below)
|
--fan SPEED : Set all fans at once to SPEED (see below)
|
||||||
--fanN SPEED : Set fan N (0-3) to SPEED
|
--fanN SPEED : Set fan N (0-3) to SPEED
|
||||||
@ -55,7 +52,6 @@ Available options:
|
|||||||
--watch N : Keep printing output every N seconds
|
--watch N : Keep printing output every N seconds
|
||||||
--overwrite : Overwrite previously displayed info with --watch and --compact instead of continuously logging new lines
|
--overwrite : Overwrite previously displayed info with --watch and --compact instead of continuously logging new lines
|
||||||
--color : Print headers in color in --compact mode for better readability
|
--color : Print headers in color in --compact mode for better readability
|
||||||
--no-reasons : Do not query NVML for clocks reasons (can cause stuttering)
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Examples:
|
### Examples:
|
||||||
|
11
evga-card.c
11
evga-card.c
@ -8,7 +8,7 @@
|
|||||||
|
|
||||||
/* Search all i2c device files for ones are on a PCI device of a supported GPU,
|
/* Search all i2c device files for ones are on a PCI device of a supported GPU,
|
||||||
and respond with the correct iCX3 version information */
|
and respond with the correct iCX3 version information */
|
||||||
int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus, int i2c_bus)
|
int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus)
|
||||||
{
|
{
|
||||||
char i2c_devices_path[NAME_MAX];
|
char i2c_devices_path[NAME_MAX];
|
||||||
char device_path[NAME_MAX];
|
char device_path[NAME_MAX];
|
||||||
@ -20,7 +20,6 @@ int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus, int i2c_bus)
|
|||||||
struct dirent *ent;
|
struct dirent *ent;
|
||||||
|
|
||||||
int num_gpus = 0;
|
int num_gpus = 0;
|
||||||
int current_i2c_bus = -1;
|
|
||||||
unsigned short pci_vendor, pci_device, pci_subsystem_vendor, pci_subsystem_device = 0;
|
unsigned short pci_vendor, pci_device, pci_subsystem_vendor, pci_subsystem_device = 0;
|
||||||
|
|
||||||
/* Start looking for I2C adapters in /sys/bus/i2c/devices/ */
|
/* Start looking for I2C adapters in /sys/bus/i2c/devices/ */
|
||||||
@ -38,14 +37,6 @@ int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus, int i2c_bus)
|
|||||||
if(strncmp(ent->d_name, "i2c-", 4) != 0)
|
if(strncmp(ent->d_name, "i2c-", 4) != 0)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
/* Only probe the specific device given (if provided) */
|
|
||||||
if (i2c_bus >= 0) {
|
|
||||||
sscanf(ent->d_name, "i2c-%i", ¤t_i2c_bus);
|
|
||||||
if (current_i2c_bus != i2c_bus)
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
strcpy(device_path, i2c_devices_path);
|
strcpy(device_path, i2c_devices_path);
|
||||||
strcat(device_path, ent->d_name);
|
strcat(device_path, ent->d_name);
|
||||||
|
|
||||||
|
@ -91,9 +91,6 @@ struct card_info {
|
|||||||
int i2c_fd; /* File descriptor for the i2c device file, for re-use */
|
int i2c_fd; /* File descriptor for the i2c device file, for re-use */
|
||||||
int product_id; /* EVGA internal product ID, as reported by the iCX3 controller */
|
int product_id; /* EVGA internal product ID, as reported by the iCX3 controller */
|
||||||
unsigned int bar0; /* Address of the card's PCI base address register */
|
unsigned int bar0; /* Address of the card's PCI base address register */
|
||||||
void *nvml_device; /* Pointer to nvmlDevice_t for use in NVML calls */
|
|
||||||
void *vram_addr; /* Memory mapping for GDDR6 temps */
|
|
||||||
void *hotspot_addr; /* Memory mapping for hotspot temperature */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct gpu_pci_info {
|
struct gpu_pci_info {
|
||||||
@ -165,7 +162,7 @@ static struct gpu_pci_info evga_pci_ids[] =
|
|||||||
{"EVGA GeForce RTX 3090 Ti FTW3 Ultra Gaming" , NVIDIA_VEN, NVIDIA_RTX3090TI_DEV, EVGA_SUB_VEN, EVGA_RTX3090TI_FTW3_ULTRA_GAMING_SUB_DEV }
|
{"EVGA GeForce RTX 3090 Ti FTW3 Ultra Gaming" , NVIDIA_VEN, NVIDIA_RTX3090TI_DEV, EVGA_SUB_VEN, EVGA_RTX3090TI_FTW3_ULTRA_GAMING_SUB_DEV }
|
||||||
};
|
};
|
||||||
|
|
||||||
int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus, int i2c_bus);
|
int find_evga_gpu_i2cs(struct card_info *infos, int max_gpus);
|
||||||
unsigned short read_pci_id(char *device_path, char *field);
|
unsigned short read_pci_id(char *device_path, char *field);
|
||||||
char *read_nvidia_pci_address(char *device_path);
|
char *read_nvidia_pci_address(char *device_path);
|
||||||
|
|
||||||
|
47
evga-icx.c
47
evga-icx.c
@ -24,8 +24,7 @@ char *header_start = "";
|
|||||||
char *header_end = "";
|
char *header_end = "";
|
||||||
|
|
||||||
static const char helpstring[] = "Available options:\n"
|
static const char helpstring[] = "Available options:\n"
|
||||||
"--i2c N : Only probe I2C bus N instead of all (may help with stuttering or freezing when probing I2C devices)\n"
|
"--gpu N : Control only GPU N instead of all supported cards\n"
|
||||||
"--gpu N : Control only GPU N instead of all supported cards\n"
|
|
||||||
"--fan SPEED : Set all fans at once to SPEED (see below)\n"
|
"--fan SPEED : Set all fans at once to SPEED (see below)\n"
|
||||||
"--fanN SPEED : Set fan N (0-3) to SPEED\n"
|
"--fanN SPEED : Set fan N (0-3) to SPEED\n"
|
||||||
" SPEED may be one of the following:\n"
|
" SPEED may be one of the following:\n"
|
||||||
@ -37,10 +36,9 @@ static const char helpstring[] = "Available options:\n"
|
|||||||
"--compact : Print sensor reading in a compact one-line per card format\n"
|
"--compact : Print sensor reading in a compact one-line per card format\n"
|
||||||
"--watch N : Keep printing output every N seconds\n"
|
"--watch N : Keep printing output every N seconds\n"
|
||||||
"--overwrite : Overwrite previously displayed info with --watch and --compact instead of continuously logging\n"
|
"--overwrite : Overwrite previously displayed info with --watch and --compact instead of continuously logging\n"
|
||||||
"--color : Print headers in color in --compact mode for better readability\n"
|
"--color : Print headers in color in --compact mode for better readability\n";
|
||||||
"--no-reasons : Do not query NVML for clocks reasons (can cause stuttering)\n";
|
|
||||||
|
|
||||||
void print_gpu_info(int gpu_num, struct card_info gpus[], int compact, int no_reasons);
|
void print_gpu_info(int gpu_num, struct card_info gpus[], int compact);
|
||||||
|
|
||||||
int main (int argc, char **argv)
|
int main (int argc, char **argv)
|
||||||
{
|
{
|
||||||
@ -49,23 +47,13 @@ int main (int argc, char **argv)
|
|||||||
int print_info = 0;
|
int print_info = 0;
|
||||||
int compact = 0;
|
int compact = 0;
|
||||||
int gpu_num = -1; /* Card to control */
|
int gpu_num = -1; /* Card to control */
|
||||||
int i2c_bus = -1;
|
|
||||||
int overwrite = 0;
|
int overwrite = 0;
|
||||||
int no_reasons = 0;
|
|
||||||
unsigned int watch = 0;
|
unsigned int watch = 0;
|
||||||
char *fan_speed[ICX3_MAX_FANS] = {NULL};
|
char *fan_speed[ICX3_MAX_FANS] = {NULL};
|
||||||
|
|
||||||
/* Input parsing */
|
/* Input parsing */
|
||||||
for (int i = 1; i < argc; i++){
|
for (int i = 1; i < argc; i++){
|
||||||
if (strcmp(argv[i], "--i2c") == 0) {
|
if (strcmp(argv[i], "--gpu") == 0) {
|
||||||
i++;
|
|
||||||
if (i < argc) {
|
|
||||||
i2c_bus = atoi(argv[i]);
|
|
||||||
} else {
|
|
||||||
printf(helpstring);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} else if (strcmp(argv[i], "--gpu") == 0) {
|
|
||||||
i++;
|
i++;
|
||||||
if (i < argc) {
|
if (i < argc) {
|
||||||
gpu_num = atoi(argv[i]);
|
gpu_num = atoi(argv[i]);
|
||||||
@ -112,8 +100,6 @@ int main (int argc, char **argv)
|
|||||||
} else if (strcmp(argv[i], "--color") == 0) {
|
} else if (strcmp(argv[i], "--color") == 0) {
|
||||||
header_start = HEADER_COLOR_START;
|
header_start = HEADER_COLOR_START;
|
||||||
header_end = HEADER_COLOR_END;
|
header_end = HEADER_COLOR_END;
|
||||||
} else if (strcmp(argv[i], "--no-reasons") == 0) {
|
|
||||||
no_reasons = 1;
|
|
||||||
} else {
|
} else {
|
||||||
printf(helpstring);
|
printf(helpstring);
|
||||||
return 0;
|
return 0;
|
||||||
@ -133,7 +119,7 @@ int main (int argc, char **argv)
|
|||||||
if (overwrite && !compact)
|
if (overwrite && !compact)
|
||||||
overwrite = 0;
|
overwrite = 0;
|
||||||
|
|
||||||
gpu_count = find_evga_gpu_i2cs(gpus, MAX_GPUS, i2c_bus);
|
gpu_count = find_evga_gpu_i2cs(gpus, MAX_GPUS);
|
||||||
|
|
||||||
if (gpu_count == -1) {
|
if (gpu_count == -1) {
|
||||||
printf("Error scanning I2C devices\n");
|
printf("Error scanning I2C devices\n");
|
||||||
@ -167,8 +153,6 @@ int main (int argc, char **argv)
|
|||||||
/* NVML init */
|
/* NVML init */
|
||||||
#ifdef USE_NVML
|
#ifdef USE_NVML
|
||||||
init_nvml();
|
init_nvml();
|
||||||
for (int i = 0; i < gpu_count; i++)
|
|
||||||
get_nvml_handle(&gpus[i]);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* PCI init for VRAM/hotspot temps */
|
/* PCI init for VRAM/hotspot temps */
|
||||||
@ -180,18 +164,17 @@ int main (int argc, char **argv)
|
|||||||
/* print sensor info */
|
/* print sensor info */
|
||||||
if (print_info) {
|
if (print_info) {
|
||||||
do {
|
do {
|
||||||
if (overwrite)
|
printf("\x1b[K"); /* Clear current console line (really just for overwrite mode) */
|
||||||
printf("\x1b[K"); /* Clear current console line */
|
|
||||||
|
|
||||||
if (gpu_num == -1) {
|
if (gpu_num == -1) {
|
||||||
/* No GPU specified on command line, loop over all supported GPUs */
|
/* No GPU specified on command line, loop over all supported GPUs */
|
||||||
for (int i = 0; i < gpu_count; i++){
|
for (int i = 0; i < gpu_count; i++){
|
||||||
if (i > 0)
|
if (i > 0)
|
||||||
printf("\n");
|
printf("\n");
|
||||||
print_gpu_info(i, &gpus[i], compact, no_reasons);
|
print_gpu_info(i, &gpus[i], compact);
|
||||||
}
|
}
|
||||||
} else if (gpu_num <= gpu_count - 1) {
|
} else if (gpu_num <= gpu_count - 1) {
|
||||||
print_gpu_info(gpu_num, &gpus[gpu_num], compact, no_reasons);
|
print_gpu_info(gpu_num, &gpus[gpu_num], compact);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!overwrite)
|
if (!overwrite)
|
||||||
@ -213,7 +196,7 @@ int main (int argc, char **argv)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
void print_gpu_info(int gpu_num, struct card_info *gpu, int compact, int no_reasons) {
|
void print_gpu_info(int gpu_num, struct card_info *gpu, int compact) {
|
||||||
if (compact) {
|
if (compact) {
|
||||||
/* One line per GPU */
|
/* One line per GPU */
|
||||||
printf("%s#%d FAN%s", header_start, gpu_num, header_end);
|
printf("%s#%d FAN%s", header_start, gpu_num, header_end);
|
||||||
@ -245,10 +228,8 @@ void print_gpu_info(int gpu_num, struct card_info *gpu, int compact, int no_reas
|
|||||||
#ifdef USE_NVML
|
#ifdef USE_NVML
|
||||||
printf("%s MEM %s", header_start, header_end);
|
printf("%s MEM %s", header_start, header_end);
|
||||||
printf("%3d%%", get_nvml_mem_util(gpu));
|
printf("%3d%%", get_nvml_mem_util(gpu));
|
||||||
if (!no_reasons) {
|
printf("%s CLK %s", header_start, header_end);
|
||||||
printf("%s CLK %s", header_start, header_end);
|
print_nvml_clock_reason(1, gpu);
|
||||||
print_nvml_clock_reason(1, gpu);
|
|
||||||
}
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
@ -278,10 +259,8 @@ void print_gpu_info(int gpu_num, struct card_info *gpu, int compact, int no_reas
|
|||||||
|
|
||||||
#ifdef USE_NVML
|
#ifdef USE_NVML
|
||||||
printf("Mem util: %d%%\n", get_nvml_mem_util(gpu));
|
printf("Mem util: %d%%\n", get_nvml_mem_util(gpu));
|
||||||
if (!no_reasons) {
|
printf("Clock reasons: ");
|
||||||
printf("Clock reasons: ");
|
print_nvml_clock_reason(0, gpu);
|
||||||
print_nvml_clock_reason(0, gpu);
|
|
||||||
}
|
|
||||||
printf("\n");
|
printf("\n");
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
95
gddr6.c
95
gddr6.c
@ -36,69 +36,70 @@ void init_gddr6(struct card_info *card)
|
|||||||
card->bar0 = (pci_dev->base_addr[0] & 0xFFFFFFFF);
|
card->bar0 = (pci_dev->base_addr[0] & 0xFFFFFFFF);
|
||||||
|
|
||||||
pci_cleanup(pacc);
|
pci_cleanup(pacc);
|
||||||
|
|
||||||
/* Open our memory mappings */
|
|
||||||
card->vram_addr = NULL;
|
|
||||||
card->hotspot_addr = NULL;
|
|
||||||
|
|
||||||
int fd;
|
|
||||||
if ((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1)
|
|
||||||
{
|
|
||||||
printf("Can't read memory for VRAM and Hotspot temperatures. If you are root, enable kernel parameter iomem=relaxed\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned int phys_addr, base_offset;
|
|
||||||
void *map_base;
|
|
||||||
|
|
||||||
for (int i = 0; i < sizeof(device_offset_info) / sizeof(struct device_offset); i++) {
|
|
||||||
if (card->pci_device_id == device_offset_info[i].device_id){
|
|
||||||
/* Map for VRAM */
|
|
||||||
phys_addr = (card->bar0 + device_offset_info[i].vram_offset);
|
|
||||||
base_offset = phys_addr & ~(PG_SZ-1);
|
|
||||||
map_base = mmap(0, PG_SZ, PROT_READ, MAP_SHARED, fd, base_offset);
|
|
||||||
if(map_base == (void *) -1)
|
|
||||||
printf("Can't map memory for VRAM temperature. If you are root, enable kernel parameter iomem=relaxed\n");
|
|
||||||
else
|
|
||||||
card->vram_addr = (void *) map_base + (phys_addr - base_offset);
|
|
||||||
|
|
||||||
|
|
||||||
/* Map for hotspot */
|
|
||||||
phys_addr = (card->bar0 + device_offset_info[i].hotspot_offset);
|
|
||||||
base_offset = phys_addr & ~(PG_SZ-1);
|
|
||||||
map_base = mmap(0, PG_SZ, PROT_READ, MAP_SHARED, fd, base_offset);
|
|
||||||
if(map_base == (void *) -1)
|
|
||||||
printf("Can't map memory for Hotspot temperature. If you are root, enable kernel parameter iomem=relaxed\n");
|
|
||||||
else
|
|
||||||
card->hotspot_addr = (void *) map_base + (phys_addr - base_offset);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
close(fd);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float get_vram_temp(struct card_info *card)
|
float get_vram_temp(struct card_info *card)
|
||||||
{
|
{
|
||||||
|
int fd;
|
||||||
float temp = 0.0;
|
float temp = 0.0;
|
||||||
|
|
||||||
if(card->vram_addr == NULL)
|
if ((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1)
|
||||||
return 0.0;
|
{
|
||||||
|
printf("Can't read memory. If you are root, enable kernel parameter iomem=relaxed\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int read_result = *((unsigned int *) card->vram_addr);
|
for (int i = 0; i < sizeof(device_offset_info) / sizeof(struct device_offset); i++) {
|
||||||
temp = ((read_result & 0x00000fff) / 0x20);
|
if (card->pci_device_id == device_offset_info[i].device_id){
|
||||||
|
unsigned int phys_addr = (card->bar0 + device_offset_info[i].vram_offset);
|
||||||
|
unsigned int base_offset = phys_addr & ~(PG_SZ-1);
|
||||||
|
void *map_base = mmap(0, PG_SZ, PROT_READ, MAP_SHARED, fd, base_offset);
|
||||||
|
if(map_base == (void *) -1)
|
||||||
|
{
|
||||||
|
if (fd != -1)
|
||||||
|
close(fd);
|
||||||
|
printf("Can't read memory for VRAM temperature. If you are root, enable kernel parameter iomem=relaxed\n");
|
||||||
|
}
|
||||||
|
void *virt_addr = (char *) map_base + (phys_addr - base_offset);
|
||||||
|
int read_result = *((unsigned int *) virt_addr);
|
||||||
|
temp = ((read_result & 0x00000fff) / 0x20);
|
||||||
|
munmap(map_base, PG_SZ);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
close(fd);
|
||||||
return temp;
|
return temp;
|
||||||
}
|
}
|
||||||
|
|
||||||
float get_hotspot_temp(struct card_info *card)
|
float get_hotspot_temp(struct card_info *card)
|
||||||
{
|
{
|
||||||
|
int fd;
|
||||||
float temp = 0.0;
|
float temp = 0.0;
|
||||||
|
|
||||||
if(card->hotspot_addr == NULL)
|
if ((fd = open("/dev/mem", O_RDWR | O_SYNC)) == -1)
|
||||||
return 0.0;
|
{
|
||||||
|
printf("Can't read memory. If you are root, enable kernel parameter iomem=relaxed\n");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
int read_result = *((unsigned int *) card->hotspot_addr);
|
for (int i = 0; i < sizeof(device_offset_info) / sizeof(struct device_offset); i++) {
|
||||||
temp = (read_result >> 8) & 0xff;
|
if (card->pci_device_id == device_offset_info[i].device_id){
|
||||||
|
unsigned int phys_addr = (card->bar0 + device_offset_info[i].hotspot_offset);
|
||||||
|
unsigned int base_offset = phys_addr & ~(PG_SZ-1);
|
||||||
|
void *map_base = mmap(0, PG_SZ, PROT_READ, MAP_SHARED, fd, base_offset);
|
||||||
|
if(map_base == (void *) -1)
|
||||||
|
{
|
||||||
|
if (fd != -1)
|
||||||
|
close(fd);
|
||||||
|
printf("Can't read memory for hotspot. If you are root, enable kernel parameter iomem=relaxed\n");
|
||||||
|
}
|
||||||
|
void *virt_addr = (char *) map_base + (phys_addr - base_offset);
|
||||||
|
int read_result = *((unsigned int *) virt_addr);
|
||||||
|
temp = (read_result >> 8) & 0xff;
|
||||||
|
munmap(map_base, PG_SZ);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
close(fd);
|
||||||
return temp;
|
return temp;
|
||||||
}
|
}
|
||||||
|
2
icx3.c
2
icx3.c
@ -237,6 +237,6 @@ void set_fan(int fan, char *setting, struct card_info *card)
|
|||||||
if (fan_readback.fanmode != fan_control.fanmode ||
|
if (fan_readback.fanmode != fan_control.fanmode ||
|
||||||
fan_readback.rpm_offset != fan_control.rpm_offset ||
|
fan_readback.rpm_offset != fan_control.rpm_offset ||
|
||||||
fan_readback.duty != fan_control.duty)
|
fan_readback.duty != fan_control.duty)
|
||||||
printf("Error setting fan %d on %s\n", fan, card->i2c_dev_path);
|
printf("Error setting fan %d on %s\n", fan, card->i2c_fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -10,14 +10,15 @@ void init_nvml()
|
|||||||
printf("Could not init NVML: %s\n", nvmlErrorString(result));
|
printf("Could not init NVML: %s\n", nvmlErrorString(result));
|
||||||
}
|
}
|
||||||
|
|
||||||
void get_nvml_handle(struct card_info *card)
|
int get_nvml_handle(nvmlDevice_t *device, struct card_info *card)
|
||||||
{
|
{
|
||||||
nvmlReturn_t result;
|
nvmlReturn_t result;
|
||||||
result = nvmlDeviceGetHandleByPciBusId_v2(card->pci_id, card->nvml_device);
|
result = nvmlDeviceGetHandleByPciBusId_v2(card->pci_id, device);
|
||||||
if (result != NVML_SUCCESS) {
|
if (result != NVML_SUCCESS) {
|
||||||
printf("Failed to get NVML device handle for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
printf("Failed to get device handle for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
||||||
card->nvml_device = NULL;
|
return 0;
|
||||||
}
|
}
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void print_nvml_clock_reason(int compact, struct card_info *card)
|
void print_nvml_clock_reason(int compact, struct card_info *card)
|
||||||
@ -51,11 +52,12 @@ void print_nvml_clock_reason(int compact, struct card_info *card)
|
|||||||
|
|
||||||
unsigned int get_nvml_temp(struct card_info *card)
|
unsigned int get_nvml_temp(struct card_info *card)
|
||||||
{
|
{
|
||||||
if (card->nvml_device == NULL)
|
nvmlDevice_t nvml_device;
|
||||||
|
if (!get_nvml_handle(&nvml_device, card))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
unsigned int temp;
|
unsigned int temp;
|
||||||
nvmlReturn_t result = nvmlDeviceGetTemperature(*(nvmlDevice_t*)(card->nvml_device), NVML_TEMPERATURE_GPU, &temp);
|
nvmlReturn_t result = nvmlDeviceGetTemperature(nvml_device, NVML_TEMPERATURE_GPU, &temp);
|
||||||
if (result != NVML_SUCCESS) {
|
if (result != NVML_SUCCESS) {
|
||||||
printf("Failed to get temperature for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
printf("Failed to get temperature for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
||||||
return 0;
|
return 0;
|
||||||
@ -65,11 +67,12 @@ unsigned int get_nvml_temp(struct card_info *card)
|
|||||||
|
|
||||||
unsigned long long get_nvml_clock_reasons(struct card_info *card)
|
unsigned long long get_nvml_clock_reasons(struct card_info *card)
|
||||||
{
|
{
|
||||||
if (card->nvml_device == NULL)
|
nvmlDevice_t nvml_device;
|
||||||
|
if (!get_nvml_handle(&nvml_device, card))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
unsigned long long reasons;
|
unsigned long long reasons;
|
||||||
nvmlReturn_t result = nvmlDeviceGetCurrentClocksEventReasons(*(nvmlDevice_t*)(card->nvml_device), &reasons) ;
|
nvmlReturn_t result = nvmlDeviceGetCurrentClocksEventReasons(nvml_device, &reasons) ;
|
||||||
if (result != NVML_SUCCESS) {
|
if (result != NVML_SUCCESS) {
|
||||||
printf("Failed to get clock reasons for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
printf("Failed to get clock reasons for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
||||||
return 0;
|
return 0;
|
||||||
@ -80,11 +83,12 @@ unsigned long long get_nvml_clock_reasons(struct card_info *card)
|
|||||||
|
|
||||||
unsigned int get_nvml_mem_util(struct card_info *card)
|
unsigned int get_nvml_mem_util(struct card_info *card)
|
||||||
{
|
{
|
||||||
if (card->nvml_device == NULL)
|
nvmlDevice_t nvml_device;
|
||||||
|
if (!get_nvml_handle(&nvml_device, card))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
nvmlUtilization_t util;
|
nvmlUtilization_t util;
|
||||||
nvmlReturn_t result = nvmlDeviceGetUtilizationRates(*(nvmlDevice_t*)(card->nvml_device), &util);
|
nvmlReturn_t result = nvmlDeviceGetUtilizationRates(nvml_device, &util);
|
||||||
if (result != NVML_SUCCESS) {
|
if (result != NVML_SUCCESS) {
|
||||||
printf("Failed to get clock reasons for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
printf("Failed to get clock reasons for card at %s: %s\n", card->pci_id, nvmlErrorString(result));
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -22,7 +22,7 @@ static struct clock_reason clock_reason_names[] =
|
|||||||
};
|
};
|
||||||
|
|
||||||
void init_nvml();
|
void init_nvml();
|
||||||
void get_nvml_handle(struct card_info *card);
|
int get_nvml_handle(nvmlDevice_t *device, struct card_info *card);
|
||||||
void print_nvml_clock_reason(int compact, struct card_info *card);
|
void print_nvml_clock_reason(int compact, struct card_info *card);
|
||||||
unsigned int get_nvml_temp(struct card_info *card);
|
unsigned int get_nvml_temp(struct card_info *card);
|
||||||
unsigned long long get_nvml_clock_reasons(struct card_info *card);
|
unsigned long long get_nvml_clock_reasons(struct card_info *card);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user