style
[psensor.git] / src / lib / nvidia.c
1 /*
2  * Copyright (C) 2010-2012 jeanfi@gmail.com
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301 USA
18  */
19 #include <locale.h>
20 #include <libintl.h>
21 #define _(str) gettext(str)
22
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 #include <X11/Xlib.h>
28
29 #include <NVCtrl/NVCtrl.h>
30 #include <NVCtrl/NVCtrlLib.h>
31
32 #include "psensor.h"
33
34 Display *display;
35
36 /*
37   Returns the temperature (Celcius) of a NVidia GPU.
38 */
39 static int get_temp(struct psensor *sensor)
40 {
41         int temp;
42         Bool res;
43
44         res = XNVCTRLQueryTargetAttribute(display,
45                                           NV_CTRL_TARGET_TYPE_GPU,
46                                           sensor->nvidia_id,
47                                           0,
48                                           NV_CTRL_GPU_CORE_TEMPERATURE,
49                                           &temp);
50
51         if (res == True)
52                 return temp;
53
54         fprintf(stderr, _("ERROR: failed to retrieve nvidia temperature\n"));
55         return 0;
56 }
57
58 static struct psensor *create_sensor(int id, int values_len)
59 {
60         char name[200];
61         char *sid;
62         struct psensor *s;
63
64         sprintf(name, "GPU%d", id);
65
66         sid = malloc(strlen("nvidia") + 1 + strlen(name) + 1);
67         sprintf(sid, "nvidia %s", name);
68
69         s = psensor_create(sid,
70                            strdup(name),
71                            SENSOR_TYPE_NVIDIA_TEMP,
72                            values_len);
73
74         s->nvidia_id = id;
75
76         return s;
77 }
78
79 /*
80   Opens connection to X server and returns the number
81   of NVidia GPUs.
82
83   Return 0 if no NVidia gpus or cannot get information.
84 */
85 static int init()
86 {
87         int evt, err, n;
88
89         display = XOpenDisplay(NULL);
90
91         if (!display) {
92                 fprintf(stderr,
93                         _("ERROR: Cannot open connection to X Server\n"));
94                 return 0;
95         }
96
97         if (XNVCTRLQueryExtension(display, &evt, &err) &&
98             XNVCTRLQueryTargetCount(display, NV_CTRL_TARGET_TYPE_GPU, &n))
99                 return n;
100
101         fprintf(stderr, _("ERROR: Cannot retrieve NVidia information\n"));
102
103         return 0;
104 }
105
106 void nvidia_psensor_list_update(struct psensor **sensors)
107 {
108         struct psensor **ss, *s;
109
110         ss = sensors;
111         while (*ss) {
112                 s = *ss;
113
114                 if (s->type == SENSOR_TYPE_NVIDIA_TEMP)
115                         psensor_set_current_value(s, get_temp(s));
116
117                 ss++;
118         }
119 }
120
121 struct psensor **nvidia_psensor_list_add(struct psensor **sensors,
122                                          int values_len)
123 {
124         int i, n;
125         struct psensor **tmp, **ss, *s;
126
127         n = init();
128
129         ss = sensors;
130         for (i = 0; i < n; i++) {
131                 s = create_sensor(i, values_len);
132
133                 tmp = psensor_list_add(ss, s);
134
135                 if (ss != tmp)
136                         free(ss);
137
138                 ss = tmp;
139         }
140
141         return ss;
142 }
143
144 void nvidia_cleanup()
145 {
146         if (display) {
147                 XCloseDisplay(display);
148                 display = NULL;
149         }
150 }