minor, make xen an uncertainty and move a comment that was misplaced
[systembsd.git] / src / interfaces / hostnamed / hostnamed.c
CommitLineData
baa40e28 1/*
2 * Copyright (c) 2014 Ian Sutton <ian@kremlin.cc>
3 *
4 * Permission to use, copy, modify, and distribute this software for any
5 * purpose with or without fee is hereby granted, provided that the above
6 * copyright notice and this permission notice appear in all copies.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15 */
16
71e3eef1 17#include <unistd.h>
18#include <limits.h>
3eb1ef91 19#include <signal.h>
041ce20b 20#include <string.h>
ed99526e 21
043bd2ed 22#include <sys/param.h>
962fad6d 23#include <sys/sysctl.h>
041ce20b 24#include <sys/sensors.h>
25#include <sys/ioctl.h>
8504b5df 26#include <sys/utsname.h>
041ce20b 27
28#include <machine/apmvar.h>
043bd2ed 29
cd16a588 30#include <glib/gprintf.h>
ba9914b6 31#include <glib-unix.h>
7a9b162a 32/* #include <gtk/gtk.h> */
cd16a588 33
3c3794ac 34#include "hostnamed-gen.h"
3eb1ef91 35#include "hostnamed.h"
71e3eef1 36
7a9b162a 37/* format: {
38 * (1) string to be matched against runtime machine's sysctl output.
39 * can be either the exact string or a substring contained
40 * within sysctl strings. no "guesses" here, a match should
41 * reliably indicate the chassis/icon.
42 *
43 * (2) string describing chassis type divulged by (1).
44 * must be one of "desktop", "laptop", "server",
45 * "tablet", "handset", "vm", "container" or NULL
46 * if only icon string can be ascertained. "vm" refers
47 * to operating systems running on baremetal hypervisors
48 * (hardware virtualization, like XEN) while "container"
49 * refers to OSs running on shared hypervisors like
50 * virtualbox or VMware. consider the distinction carefully
51 * as common virtualization software like KVM may share
52 * characteristics of both "vm" and "container" types.
53 *
54 * (3) string specifying icon to use. follows XDG icon spec.
55 * see http://standards.freedesktop.org/icon-naming-spec/icon-naming-spec-latest.html
56 * for allowed strings.
57 *
58 * (4) chassis precedence bit. TRUE if (2) is defined and
59 * we're certain it is the proper string. FALSE in the
60 * circumstance (2) may be the correct string, unless
61 * a match with this bit set to TRUE overrides it.
62 * if (2) is NULL, this bit is inconsequential.
63 *
64 * (5) icon precedence bit. see previous definition.
65 * } */
66struct SYSCTL_LOOKUP_TABLE {
67 gchar *match_string;
68 gchar *chassis;
69 gchar *icon;
70 gboolean chassis_precedence;
71 gboolean icon_precedence;
72};
73
ed99526e 74GPtrArray *hostnamed_freeable;
9d2500b0 75Hostname1 *hostnamed_interf;
7ce16a35 76
3eb1ef91 77GMainLoop *hostnamed_loop;
78
79guint bus_descriptor;
80gboolean dbus_interface_exported; /* reliable because of gdbus operational guarantees */
81
7a9b162a 82gchar *CHASSIS, *ICON;
8504b5df 83gchar *OS_CPENAME;
84gchar *KERN_NAME, *KERN_RELEASE, *KERN_VERS;
7a9b162a 85
86/* TODO no specific vm or laptop icon in gnome
87 * NOTE paravirtualization on xen is only available for linuxes right now
88 * dmesg on linux systems reveals xen and virtualization method (HVM or PVM)
89 * but we will worry about those later */
2e337f46 90
91/* add any sysctl strings that suggest virtualization here */
7a9b162a 92const struct SYSCTL_LOOKUP_TABLE chassis_indicator_table[] =
93{
a194b156 94 { "QEMU Virtual CPU", "vm", NULL, FALSE, FALSE }, /* could be QEMU running in userspace or as part of KVM */
7c613054 95 { "KVM", "vm", "drive-multidisk", FALSE, FALSE },
df95f94a 96 { "SmartDC HVM", "vm", "drive-multidisk", TRUE, TRUE }, /* illumos-joyent kvm */
a194b156 97 { "VirtualBox", "vm", "drive-multidisk", TRUE, TRUE },
98 { "VMware, Inc.", "vm", "drive-multidisk", TRUE, TRUE },
99 { "VMware Virtual Platform", "vm", "drive-multidisk", TRUE, TRUE },
100 { "Parallels", "vm", "drive-multidisk", TRUE, TRUE }, /* need verification */
2e337f46 101 { "Xen", "vm", "drive-multidisk", FALSE, FALSE }
a194b156 102}; /* TODO: chroots, etc. are the actual "containers", add them */
355f60ae 103
041ce20b 104/* archs to check against when determining if machine is server */
105const gchar *server_archs[] = {
106 "hppa",
107 "sparc",
108 "sparc64"
109};
962fad6d 110
7a9b162a 111/* --- begin method/property/dbus signal code --- */
962fad6d 112
7ce16a35 113static gboolean
9d2500b0 114on_handle_set_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 115 GDBusMethodInvocation *invoc,
116 const gchar *greet,
117 gpointer data) {
118 return FALSE;
119}
120
121static gboolean
9d2500b0 122on_handle_set_static_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 123 GDBusMethodInvocation *invoc,
124 const gchar *greet,
125 gpointer data) {
126 return FALSE;
127}
128
129static gboolean
9d2500b0 130on_handle_set_pretty_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 131 GDBusMethodInvocation *invoc,
132 const gchar *greet,
133 gpointer data) {
134 return FALSE;
135}
136
137static gboolean
9d2500b0 138on_handle_set_chassis(Hostname1 *hn1_passed_interf,
7ce16a35 139 GDBusMethodInvocation *invoc,
140 const gchar *greet,
141 gpointer data) {
142 return FALSE;
143}
144
145static gboolean
9d2500b0 146on_handle_set_icon_name(Hostname1 *hn1_passed_interf,
7ce16a35 147 GDBusMethodInvocation *invoc,
148 const gchar *greet,
149 gpointer data) {
150 return FALSE;
151}
152
355f60ae 153/* note: all hostnamed/hostname1's properties are read-only,
154 * and do not need set_ functions, gdbus-codegen realized
155 * this from the XML and handled the to-be error of trying
156 * to set a read-only property's value
157 */
158
159const gchar *
160our_get_hostname() {
161
99ef55dd 162 gchar *hostname_buf, *ret;
163 size_t hostname_divider;
043bd2ed 164
8e18351c 165 hostname_buf = (gchar*) g_malloc0(MAXHOSTNAMELEN);
99ef55dd 166 ret = (gchar*) g_malloc0(MAXHOSTNAMELEN);
8e18351c 167
99ef55dd 168 g_ptr_array_add(hostnamed_freeable, hostname_buf);
169 g_ptr_array_add(hostnamed_freeable, ret);
af9f8b50 170
8e18351c 171 if(gethostname(hostname_buf, MAXHOSTNAMELEN) || g_strcmp0(hostname_buf, "") == 0)
172 return "localhost";
043bd2ed 173
99ef55dd 174 hostname_divider = strcspn(hostname_buf, ".");
8f3d38b1 175
99ef55dd 176 return strncpy(ret, hostname_buf, hostname_divider);
355f60ae 177}
178
179const gchar *
180our_get_static_hostname() {
181
82a6a0d8 182 const gchar *pretty_hostname;
183 const gchar *ret;
8e18351c 184
185 pretty_hostname = our_get_pretty_hostname();
186
187 if(g_strcmp0(pretty_hostname, "") == 0)
188 ret = our_get_hostname();
189
82a6a0d8 190 else if((ret = g_hostname_to_ascii(pretty_hostname))) {
8e18351c 191
82a6a0d8 192 g_ptr_array_add(hostnamed_freeable, (gpointer)ret);
8e18351c 193 return ret;
194 }
195
196 return ret;
355f60ae 197}
198
199const gchar *
200our_get_pretty_hostname() {
201
8e18351c 202 GKeyFile *config;
203 gchar *ret;
204
962fad6d 205 config = g_key_file_new();
206
8e18351c 207 if(g_key_file_load_from_file(config, "/etc/systemd_compat.conf", G_KEY_FILE_NONE, NULL)
82a6a0d8 208 && (ret = g_key_file_get_value(config, "hostnamed", "PrettyHostname", NULL))) { /* ret might need to be freed, docs dont specify but i am suspicious */
8e18351c 209
962fad6d 210 g_key_file_unref(config);
8e18351c 211 return ret;
212 }
213
214 if(config)
215 g_free(config);
216
217 return "";
355f60ae 218}
219
220const gchar *
221our_get_chassis() {
222
041ce20b 223 if(CHASSIS)
224 return CHASSIS;
962fad6d 225
041ce20b 226 return "desktop";
355f60ae 227}
228
229const gchar *
230our_get_icon_name() {
231
041ce20b 232 if(ICON)
233 return ICON;
234
235 return "";
355f60ae 236}
237
238const gchar *
239our_get_kernel_name() {
240
8504b5df 241 if(KERN_NAME)
242 return KERN_NAME;
243
244 return "";
355f60ae 245}
246
247const gchar *
248our_get_kernel_version() {
249
8504b5df 250 if(KERN_VERS)
251 return KERN_VERS;
252
253 return "";
355f60ae 254}
255
256const gchar *
257our_get_kernel_release() {
258
8504b5df 259 if(KERN_RELEASE)
260 return KERN_RELEASE;
261
262 return "";
355f60ae 263}
264
265const gchar *
266our_get_os_cpename() {
267
8504b5df 268 return "ONEDAY";
355f60ae 269}
270
271const gchar *
272our_get_os_pretty_name() {
273
8504b5df 274 return "OpenBSD";
355f60ae 275}
276
ff036f75 277/* --- end method/property/dbus signal code, begin bus/name handlers --- */
0df0018d 278
828caf9a 279static void hostnamed_on_bus_acquired(GDBusConnection *conn,
fbc0295f 280 const gchar *name,
281 gpointer user_data) {
11475670 282
3eb1ef91 283 g_printf("got bus/name, exporting %s's interface...\n", name);
11475670 284
99ef55dd 285 hostnamed_interf = hostname1_skeleton_new();
7ce16a35 286
355f60ae 287 /* attach function pointers to generated struct's method handlers */
7ce16a35 288 g_signal_connect(hostnamed_interf, "handle-set-hostname", G_CALLBACK(on_handle_set_hostname), NULL);
289 g_signal_connect(hostnamed_interf, "handle-set-static-hostname", G_CALLBACK(on_handle_set_static_hostname), NULL);
290 g_signal_connect(hostnamed_interf, "handle-set-pretty-hostname", G_CALLBACK(on_handle_set_pretty_hostname), NULL);
291 g_signal_connect(hostnamed_interf, "handle-set-chassis", G_CALLBACK(on_handle_set_chassis), NULL);
292 g_signal_connect(hostnamed_interf, "handle-set-icon-name", G_CALLBACK(on_handle_set_icon_name), NULL);
293
355f60ae 294 /* set our properties before export */
9d2500b0 295 hostname1_set_hostname(hostnamed_interf, our_get_hostname());
296 hostname1_set_static_hostname(hostnamed_interf, our_get_static_hostname());
297 hostname1_set_pretty_hostname(hostnamed_interf, our_get_pretty_hostname());
298 hostname1_set_chassis(hostnamed_interf, our_get_chassis());
299 hostname1_set_icon_name(hostnamed_interf, our_get_icon_name());
300 hostname1_set_kernel_name(hostnamed_interf, our_get_kernel_name());
301 hostname1_set_kernel_version(hostnamed_interf, our_get_kernel_version());
302 hostname1_set_kernel_release(hostnamed_interf, our_get_kernel_release());
303 hostname1_set_operating_system_cpename(hostnamed_interf, our_get_os_cpename());
304 hostname1_set_operating_system_pretty_name(hostnamed_interf, our_get_os_pretty_name());
355f60ae 305
7ce16a35 306 if(!g_dbus_interface_skeleton_export(G_DBUS_INTERFACE_SKELETON(hostnamed_interf),
9d2500b0 307 conn,
308 "/org/freedesktop/hostname1",
309 NULL)) {
7ce16a35 310
3eb1ef91 311 g_printf("failed to export %s's interface!\n", name); /* unusual edge case, TODO check errno */
99ef55dd 312 hostnamed_mem_clean();
7ce16a35 313
3eb1ef91 314 } else {
ea207ed3 315
99ef55dd 316 dbus_interface_exported = TRUE;
317 g_printf("exported %s's interface on the system bus...\n", name);
318 }
3eb1ef91 319}
3c3794ac 320
3eb1ef91 321static void hostnamed_on_name_acquired(GDBusConnection *conn,
99ef55dd 322 const gchar *name,
3eb1ef91 323 gpointer user_data) {
324
325 g_printf("success!\n");
3c3794ac 326}
327
828caf9a 328static void hostnamed_on_name_lost(GDBusConnection *conn,
9d2500b0 329 const gchar *name,
330 gpointer user_data) {
80043b36 331
99ef55dd 332 if(!conn) {
3eb1ef91 333
99ef55dd 334 g_printf("failed to connect to the system bus while trying to acquire name '%s': either dbus-daemon isn't running or we don't have permission to push names and/or their interfaces to it.\n", name);
335 hostnamed_mem_clean();
336 }
3eb1ef91 337
ca5a7f9f 338 g_printf("lost name %s, exiting...\n", name);
ed99526e 339
fbc0295f 340 hostnamed_mem_clean();
3eb1ef91 341}
342
343/* --- end bus/name handlers, begin misc unix functions --- */
344
345/* safe call to clean and then exit
ca5a7f9f 346 * this stops our GMainLoop safely before letting main() return */
3eb1ef91 347void hostnamed_mem_clean() {
348
99ef55dd 349 g_printf("exiting...\n");
3eb1ef91 350
99ef55dd 351 if(dbus_interface_exported)
352 g_dbus_interface_skeleton_unexport(G_DBUS_INTERFACE_SKELETON(hostnamed_interf));
3eb1ef91 353
99ef55dd 354 if(g_main_loop_is_running(hostnamed_loop))
355 g_main_loop_quit(hostnamed_loop);
ed99526e 356
ea207ed3 357}
358
3eb1ef91 359/* wrapper for glib's unix signal handling; called only once if terminatating signal is raised against us */
360gboolean unix_sig_terminate_handler(gpointer data) {
a3041e78 361
99ef55dd 362 g_printf("caught SIGINT/HUP/TERM, exiting\n");
3eb1ef91 363
99ef55dd 364 hostnamed_mem_clean();
365 return G_SOURCE_REMOVE;
3eb1ef91 366}
367
368void set_signal_handlers() {
369
99ef55dd 370 /* we don't care about its descriptor, we never need to unregister these */
371 g_unix_signal_add(SIGINT, unix_sig_terminate_handler, NULL);
372 g_unix_signal_add(SIGHUP, unix_sig_terminate_handler, NULL);
373 g_unix_signal_add(SIGTERM, unix_sig_terminate_handler, NULL);
5c2460ab 374
99ef55dd 375 /* TODO: the "only once" guarantee only counts towards specific signals.
376 * make sure calling a SIGINT and SIGHUP doesn't cause term_handler()
377 * to be called twice */
3eb1ef91 378}
379
380int main() {
8504b5df 381
382 hostnamed_freeable = g_ptr_array_new();
383
4babc29e 384 /* TODO: check for valid, writable config at init. if no, complain to `make install` */
8504b5df 385
386 CHASSIS = ICON = OS_CPENAME = 0;
387 KERN_NAME = KERN_RELEASE = KERN_VERS = 0;
4babc29e 388
99ef55dd 389 set_signal_handlers();
387173cb 390
8504b5df 391 if(!determine_chassis_and_icon() || !set_uname_properties())
7a9b162a 392 return 1;
393
394 hostnamed_loop = g_main_loop_new(NULL, TRUE);
ed99526e 395
99ef55dd 396 bus_descriptor = g_bus_own_name(G_BUS_TYPE_SYSTEM,
3eb1ef91 397 "org.freedesktop.hostname1",
398 G_BUS_NAME_OWNER_FLAGS_NONE,
399 hostnamed_on_bus_acquired,
400 hostnamed_on_name_acquired,
401 hostnamed_on_name_lost,
402 NULL,
403 NULL);
496f5d66 404
99ef55dd 405 g_main_loop_run(hostnamed_loop);
406 /* runs until single g_main_loop_quit() call is raised inside <interface>_mem_clean() */
407 g_main_loop_unref(hostnamed_loop);
3c3794ac 408
99ef55dd 409 /* guaranteed unownable */
410 g_bus_unown_name(bus_descriptor);
a3041e78 411
99ef55dd 412 /* at this point no operations can occur with our data, it is safe to free it + its container */
413 g_ptr_array_free(hostnamed_freeable, TRUE);
af9f8b50 414
99ef55dd 415 return 0;
71e3eef1 416}
417
8504b5df 418gboolean set_uname_properties() {
419
420 struct utsname un;
421
422 if(-1 == uname(&un))
423 return FALSE;
424
425 KERN_NAME = (gchar*)g_malloc0(sizeof(un.sysname));
426 g_ptr_array_add(hostnamed_freeable, KERN_NAME);
427 g_strlcpy(KERN_NAME, un.sysname, sizeof(un.sysname));
428
429 KERN_RELEASE = (gchar*)g_malloc0(sizeof(un.release));
430 g_ptr_array_add(hostnamed_freeable, KERN_RELEASE);
431 g_strlcpy(KERN_RELEASE, un.release, sizeof(un.release));
432
433 KERN_VERS = (gchar*)g_malloc0(sizeof(un.version));
434 g_ptr_array_add(hostnamed_freeable, KERN_VERS);
435 g_strlcpy(KERN_VERS, un.version, sizeof(un.version));
436
437 return TRUE;
438}
439
7a9b162a 440gboolean determine_chassis_and_icon() {
441
22b13250 442 const size_t bufsize = 4096;
443
041ce20b 444 char *hwproduct, *hwmodel, *hwvendor, *hwmachine;
445 size_t hwproduct_size, hwmodel_size, hwvendor_size, hwmachine_size;
446 int hwproduct_name[2], hwmodel_name[2], hwvendor_name[2], hwmachine_name[2];
962fad6d 447 unsigned int i;
041ce20b 448 gboolean UNSURE_CHASSIS_FLAG, UNSURE_ICON_FLAG;
449
22b13250 450 hwproduct_size = hwmodel_size = hwvendor_size = hwmachine_size = bufsize;
a112db4c 451 UNSURE_CHASSIS_FLAG = UNSURE_ICON_FLAG = FALSE;
1ed17375 452 i = 0;
962fad6d 453
22b13250 454 hwproduct = (char*)g_malloc0(4096);
455 hwmodel = (char*)g_malloc0(4096);
456 hwvendor = (char*)g_malloc0(4096);
457 hwmachine = (char*)g_malloc0(4096);
458
8504b5df 459 g_ptr_array_add(hostnamed_freeable, hwproduct);
460 g_ptr_array_add(hostnamed_freeable, hwmodel);
461 g_ptr_array_add(hostnamed_freeable, hwvendor);
462 g_ptr_array_add(hostnamed_freeable, hwmachine);
463
7a9b162a 464 hwproduct_name[0] = CTL_HW;
465 hwproduct_name[1] = HW_PRODUCT;
962fad6d 466
7a9b162a 467 hwmodel_name[0] = CTL_HW;
468 hwmodel_name[1] = HW_MODEL;
469
470 hwvendor_name[0] = CTL_HW;
471 hwvendor_name[1] = HW_VENDOR;
472
041ce20b 473 hwmachine_name[0] = CTL_HW;
474 hwmachine_name[1] = HW_MACHINE;
475
7a9b162a 476 /* pass NULL buffer to check size first, then pass hw to be filled according to freshly-set hw_size */
477 if(-1 == sysctl(hwproduct_name, 2, NULL, &hwproduct_size, NULL, 0) || -1 == sysctl(hwproduct_name, 2, hwproduct, &hwproduct_size, NULL, 0))
478 return FALSE;
479
480 if(-1 == sysctl(hwmodel_name, 2, NULL, &hwmodel_size, NULL, 0) || -1 == sysctl(hwmodel_name, 2, hwmodel, &hwmodel_size, NULL, 0))
481 return FALSE;
482
483 if(-1 == sysctl(hwvendor_name, 2, NULL, &hwvendor_size, NULL, 0) || -1 == sysctl(hwvendor_name, 2, hwvendor, &hwvendor_size, NULL, 0))
484 return FALSE;
485
041ce20b 486 if(-1 == sysctl(hwmachine_name, 2, NULL, &hwmachine_size, NULL, 0) || -1 == sysctl(hwmachine_name, 2, hwmachine, &hwmachine_size, NULL, 0))
487 return FALSE;
488
7a9b162a 489 /* TODO: test for laptop, if not, dmidecode for desktop vs. server
490 * probably move this code to vm test func and set a global after running it early, once */
491
041ce20b 492 for(; i < G_N_ELEMENTS(chassis_indicator_table); i++) {
493 if(strcasestr(hwproduct, chassis_indicator_table[i].match_string)
494 || strcasestr(hwmodel, chassis_indicator_table[i].match_string)
495 || strcasestr(hwvendor, chassis_indicator_table[i].match_string)) {
496
497 if(!UNSURE_CHASSIS_FLAG && chassis_indicator_table[i].chassis) {
498
499 UNSURE_CHASSIS_FLAG = chassis_indicator_table[i].chassis_precedence;
500 CHASSIS = chassis_indicator_table[i].chassis;
501 }
502
503 if(!UNSURE_ICON_FLAG && chassis_indicator_table[i].icon) {
504
505 UNSURE_ICON_FLAG = chassis_indicator_table[i].icon_precedence;
506 ICON = chassis_indicator_table[i].icon;
507 }
508 }
509 }
510
511 if(up_native_is_laptop()) {
512
513 if(!CHASSIS)
514 CHASSIS = "laptop";
515 if(!ICON)
516 ICON = "input-touchpad"; /* TODO pull an icon package that actually has the icons we're looking for */
517
518 } else if(is_server(hwmachine)) {
519
520 if(!CHASSIS)
521 CHASSIS = "server";
522 if(!ICON)
523 ICON = "uninterruptible-power-supply";
524
525 } else if(!CHASSIS || !ICON) {
962fad6d 526
041ce20b 527 if(!CHASSIS)
528 CHASSIS = "desktop";
529 if(!ICON)
530 ICON = "computer";
531 }
532
533 return (CHASSIS && ICON);
534}
535
536gboolean is_server(gchar *arch) {
537
538 unsigned int i;
7a9b162a 539
041ce20b 540 for(; i < G_N_ELEMENTS(server_archs); i++)
541 if(strcasestr(arch, server_archs[i]))
542 return TRUE;
543
544 return FALSE;
545}
546
547gboolean up_native_is_laptop() {
548
549 struct apm_power_info bstate;
550 struct sensordev acpiac;
551
552 if (up_native_get_sensordev("acpiac0", &acpiac))
553 return TRUE;
554
555 if (-1 == ioctl(up_apm_get_fd(), APM_IOC_GETPOWER, &bstate))
556 g_error("ioctl on apm fd failed : %s", g_strerror(errno));
557
558 return bstate.ac_state != APM_AC_UNKNOWN;
559}
560
561int up_apm_get_fd() {
562
563 static int apm_fd = 0;
564
565 if(apm_fd == 0) {
566
567 g_debug("apm_fd is not initialized yet, opening");
568
569 /* open /dev/apm */
570 if((apm_fd = open("/dev/apm", O_RDONLY)) == -1) {
571 if(errno != ENXIO && errno != ENOENT)
572 g_error("cannot open device file");
573 }
574 }
575
576 return apm_fd;
577}
578
579gboolean up_native_get_sensordev(const char * id, struct sensordev * snsrdev) {
580
581 int devn;
582 size_t sdlen = sizeof(struct sensordev);
583 int mib[] = {CTL_HW, HW_SENSORS, 0, 0 ,0};
584
585 for (devn = 0 ; ; devn++) {
586 mib[2] = devn;
587 if(sysctl(mib, 3, snsrdev, &sdlen, NULL, 0) == -1) {
588 if(errno == ENXIO)
589 continue;
590 if(errno == ENOENT)
591 break;
592 }
593
594 if (!strcmp(snsrdev->xname, id))
595 return TRUE;
596 }
597
598 return FALSE;
962fad6d 599}