minor, initialize counter to zero. i might be a bit tired.
[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>
26
27#include <machine/apmvar.h>
043bd2ed 28
cd16a588 29#include <glib/gprintf.h>
ba9914b6 30#include <glib-unix.h>
7a9b162a 31/* #include <gtk/gtk.h> */
cd16a588 32
3c3794ac 33#include "hostnamed-gen.h"
3eb1ef91 34#include "hostnamed.h"
71e3eef1 35
7a9b162a 36/* add any sysctl strings that suggest virtualization here */
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;
83
84/* TODO no specific vm or laptop icon in gnome
85 * NOTE paravirtualization on xen is only available for linuxes right now
86 * dmesg on linux systems reveals xen and virtualization method (HVM or PVM)
87 * but we will worry about those later */
88const struct SYSCTL_LOOKUP_TABLE chassis_indicator_table[] =
89{
041ce20b 90 { "QEMU Virtual CPU", "container", NULL, FALSE, FALSE }, /* could be QEMU running in userspace or as part of KVM */
7c613054 91 { "KVM", "vm", "drive-multidisk", FALSE, FALSE },
7a9b162a 92 { "SmartDC HVM", "vm", "drive-multidisk", TRUE, TRUE }, /* oracle solaris kvm */
93 { "VirtualBox", "container", "drive-optical", TRUE, TRUE },
94 { "VMware, Inc.", "container", "drive-optical", TRUE, TRUE },
95 { "VMware Virtual Platform", "container", "drive-optical", TRUE, TRUE },
96 { "Parallels", "container", "drive-optical", TRUE, TRUE } /* need verification */
97};
355f60ae 98
041ce20b 99/* archs to check against when determining if machine is server */
100const gchar *server_archs[] = {
101 "hppa",
102 "sparc",
103 "sparc64"
104};
962fad6d 105
7a9b162a 106/* --- begin method/property/dbus signal code --- */
962fad6d 107
7ce16a35 108static gboolean
9d2500b0 109on_handle_set_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 110 GDBusMethodInvocation *invoc,
111 const gchar *greet,
112 gpointer data) {
113 return FALSE;
114}
115
116static gboolean
9d2500b0 117on_handle_set_static_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 118 GDBusMethodInvocation *invoc,
119 const gchar *greet,
120 gpointer data) {
121 return FALSE;
122}
123
124static gboolean
9d2500b0 125on_handle_set_pretty_hostname(Hostname1 *hn1_passed_interf,
7ce16a35 126 GDBusMethodInvocation *invoc,
127 const gchar *greet,
128 gpointer data) {
129 return FALSE;
130}
131
132static gboolean
9d2500b0 133on_handle_set_chassis(Hostname1 *hn1_passed_interf,
7ce16a35 134 GDBusMethodInvocation *invoc,
135 const gchar *greet,
136 gpointer data) {
137 return FALSE;
138}
139
140static gboolean
9d2500b0 141on_handle_set_icon_name(Hostname1 *hn1_passed_interf,
7ce16a35 142 GDBusMethodInvocation *invoc,
143 const gchar *greet,
144 gpointer data) {
145 return FALSE;
146}
147
355f60ae 148/* note: all hostnamed/hostname1's properties are read-only,
149 * and do not need set_ functions, gdbus-codegen realized
150 * this from the XML and handled the to-be error of trying
151 * to set a read-only property's value
152 */
153
154const gchar *
155our_get_hostname() {
156
99ef55dd 157 gchar *hostname_buf, *ret;
158 size_t hostname_divider;
043bd2ed 159
8e18351c 160 hostname_buf = (gchar*) g_malloc0(MAXHOSTNAMELEN);
99ef55dd 161 ret = (gchar*) g_malloc0(MAXHOSTNAMELEN);
8e18351c 162
99ef55dd 163 g_ptr_array_add(hostnamed_freeable, hostname_buf);
164 g_ptr_array_add(hostnamed_freeable, ret);
af9f8b50 165
8e18351c 166 if(gethostname(hostname_buf, MAXHOSTNAMELEN) || g_strcmp0(hostname_buf, "") == 0)
167 return "localhost";
043bd2ed 168
99ef55dd 169 hostname_divider = strcspn(hostname_buf, ".");
8f3d38b1 170
99ef55dd 171 return strncpy(ret, hostname_buf, hostname_divider);
355f60ae 172}
173
174const gchar *
175our_get_static_hostname() {
176
82a6a0d8 177 const gchar *pretty_hostname;
178 const gchar *ret;
8e18351c 179
180 pretty_hostname = our_get_pretty_hostname();
181
182 if(g_strcmp0(pretty_hostname, "") == 0)
183 ret = our_get_hostname();
184
82a6a0d8 185 else if((ret = g_hostname_to_ascii(pretty_hostname))) {
8e18351c 186
82a6a0d8 187 g_ptr_array_add(hostnamed_freeable, (gpointer)ret);
8e18351c 188 return ret;
189 }
190
191 return ret;
355f60ae 192}
193
194const gchar *
195our_get_pretty_hostname() {
196
8e18351c 197 GKeyFile *config;
198 gchar *ret;
199
962fad6d 200 config = g_key_file_new();
201
8e18351c 202 if(g_key_file_load_from_file(config, "/etc/systemd_compat.conf", G_KEY_FILE_NONE, NULL)
82a6a0d8 203 && (ret = g_key_file_get_value(config, "hostnamed", "PrettyHostname", NULL))) { /* ret might need to be freed, docs dont specify but i am suspicious */
8e18351c 204
962fad6d 205 g_key_file_unref(config);
8e18351c 206 return ret;
207 }
208
209 if(config)
210 g_free(config);
211
212 return "";
355f60ae 213}
214
215const gchar *
216our_get_chassis() {
217
041ce20b 218 if(CHASSIS)
219 return CHASSIS;
962fad6d 220
041ce20b 221 return "desktop";
355f60ae 222}
223
224const gchar *
225our_get_icon_name() {
226
041ce20b 227 if(ICON)
228 return ICON;
229
230 return "";
355f60ae 231}
232
233const gchar *
234our_get_kernel_name() {
235
236 return "TODO";
237}
238
239const gchar *
240our_get_kernel_version() {
241
242 return "TODO";
243}
244
245const gchar *
246our_get_kernel_release() {
247
248 return "TODO";
249}
250
251const gchar *
252our_get_os_cpename() {
253
254 return "TODO";
255}
256
257const gchar *
258our_get_os_pretty_name() {
259
260 return "TODO";
261}
262
ff036f75 263/* --- end method/property/dbus signal code, begin bus/name handlers --- */
0df0018d 264
828caf9a 265static void hostnamed_on_bus_acquired(GDBusConnection *conn,
fbc0295f 266 const gchar *name,
267 gpointer user_data) {
11475670 268
3eb1ef91 269 g_printf("got bus/name, exporting %s's interface...\n", name);
11475670 270
99ef55dd 271 hostnamed_interf = hostname1_skeleton_new();
7ce16a35 272
355f60ae 273 /* attach function pointers to generated struct's method handlers */
7ce16a35 274 g_signal_connect(hostnamed_interf, "handle-set-hostname", G_CALLBACK(on_handle_set_hostname), NULL);
275 g_signal_connect(hostnamed_interf, "handle-set-static-hostname", G_CALLBACK(on_handle_set_static_hostname), NULL);
276 g_signal_connect(hostnamed_interf, "handle-set-pretty-hostname", G_CALLBACK(on_handle_set_pretty_hostname), NULL);
277 g_signal_connect(hostnamed_interf, "handle-set-chassis", G_CALLBACK(on_handle_set_chassis), NULL);
278 g_signal_connect(hostnamed_interf, "handle-set-icon-name", G_CALLBACK(on_handle_set_icon_name), NULL);
279
355f60ae 280 /* set our properties before export */
9d2500b0 281 hostname1_set_hostname(hostnamed_interf, our_get_hostname());
282 hostname1_set_static_hostname(hostnamed_interf, our_get_static_hostname());
283 hostname1_set_pretty_hostname(hostnamed_interf, our_get_pretty_hostname());
284 hostname1_set_chassis(hostnamed_interf, our_get_chassis());
285 hostname1_set_icon_name(hostnamed_interf, our_get_icon_name());
286 hostname1_set_kernel_name(hostnamed_interf, our_get_kernel_name());
287 hostname1_set_kernel_version(hostnamed_interf, our_get_kernel_version());
288 hostname1_set_kernel_release(hostnamed_interf, our_get_kernel_release());
289 hostname1_set_operating_system_cpename(hostnamed_interf, our_get_os_cpename());
290 hostname1_set_operating_system_pretty_name(hostnamed_interf, our_get_os_pretty_name());
355f60ae 291
7ce16a35 292 if(!g_dbus_interface_skeleton_export(G_DBUS_INTERFACE_SKELETON(hostnamed_interf),
9d2500b0 293 conn,
294 "/org/freedesktop/hostname1",
295 NULL)) {
7ce16a35 296
3eb1ef91 297 g_printf("failed to export %s's interface!\n", name); /* unusual edge case, TODO check errno */
99ef55dd 298 hostnamed_mem_clean();
7ce16a35 299
3eb1ef91 300 } else {
ea207ed3 301
99ef55dd 302 dbus_interface_exported = TRUE;
303 g_printf("exported %s's interface on the system bus...\n", name);
304 }
3eb1ef91 305}
3c3794ac 306
3eb1ef91 307static void hostnamed_on_name_acquired(GDBusConnection *conn,
99ef55dd 308 const gchar *name,
3eb1ef91 309 gpointer user_data) {
310
311 g_printf("success!\n");
3c3794ac 312}
313
828caf9a 314static void hostnamed_on_name_lost(GDBusConnection *conn,
9d2500b0 315 const gchar *name,
316 gpointer user_data) {
80043b36 317
99ef55dd 318 if(!conn) {
3eb1ef91 319
99ef55dd 320 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);
321 hostnamed_mem_clean();
322 }
3eb1ef91 323
ca5a7f9f 324 g_printf("lost name %s, exiting...\n", name);
ed99526e 325
fbc0295f 326 hostnamed_mem_clean();
3eb1ef91 327}
328
329/* --- end bus/name handlers, begin misc unix functions --- */
330
331/* safe call to clean and then exit
ca5a7f9f 332 * this stops our GMainLoop safely before letting main() return */
3eb1ef91 333void hostnamed_mem_clean() {
334
99ef55dd 335 g_printf("exiting...\n");
3eb1ef91 336
99ef55dd 337 if(dbus_interface_exported)
338 g_dbus_interface_skeleton_unexport(G_DBUS_INTERFACE_SKELETON(hostnamed_interf));
3eb1ef91 339
99ef55dd 340 if(g_main_loop_is_running(hostnamed_loop))
341 g_main_loop_quit(hostnamed_loop);
ed99526e 342
ea207ed3 343}
344
3eb1ef91 345/* wrapper for glib's unix signal handling; called only once if terminatating signal is raised against us */
346gboolean unix_sig_terminate_handler(gpointer data) {
a3041e78 347
99ef55dd 348 g_printf("caught SIGINT/HUP/TERM, exiting\n");
3eb1ef91 349
99ef55dd 350 hostnamed_mem_clean();
351 return G_SOURCE_REMOVE;
3eb1ef91 352}
353
354void set_signal_handlers() {
355
99ef55dd 356 /* we don't care about its descriptor, we never need to unregister these */
357 g_unix_signal_add(SIGINT, unix_sig_terminate_handler, NULL);
358 g_unix_signal_add(SIGHUP, unix_sig_terminate_handler, NULL);
359 g_unix_signal_add(SIGTERM, unix_sig_terminate_handler, NULL);
5c2460ab 360
99ef55dd 361 /* TODO: the "only once" guarantee only counts towards specific signals.
362 * make sure calling a SIGINT and SIGHUP doesn't cause term_handler()
363 * to be called twice */
3eb1ef91 364}
365
366int main() {
4babc29e 367
368 /* TODO: check for valid, writable config at init. if no, complain to `make install` */
369
99ef55dd 370 set_signal_handlers();
387173cb 371
041ce20b 372 if(!determine_chassis_and_icon())
7a9b162a 373 return 1;
374
375 hostnamed_loop = g_main_loop_new(NULL, TRUE);
99ef55dd 376 hostnamed_freeable = g_ptr_array_new();
ed99526e 377
99ef55dd 378 bus_descriptor = g_bus_own_name(G_BUS_TYPE_SYSTEM,
3eb1ef91 379 "org.freedesktop.hostname1",
380 G_BUS_NAME_OWNER_FLAGS_NONE,
381 hostnamed_on_bus_acquired,
382 hostnamed_on_name_acquired,
383 hostnamed_on_name_lost,
384 NULL,
385 NULL);
496f5d66 386
99ef55dd 387 g_main_loop_run(hostnamed_loop);
388 /* runs until single g_main_loop_quit() call is raised inside <interface>_mem_clean() */
389 g_main_loop_unref(hostnamed_loop);
3c3794ac 390
99ef55dd 391 /* guaranteed unownable */
392 g_bus_unown_name(bus_descriptor);
a3041e78 393
99ef55dd 394 /* at this point no operations can occur with our data, it is safe to free it + its container */
395 g_ptr_array_free(hostnamed_freeable, TRUE);
af9f8b50 396
99ef55dd 397 return 0;
71e3eef1 398}
399
7a9b162a 400gboolean determine_chassis_and_icon() {
401
22b13250 402 const size_t bufsize = 4096;
403
041ce20b 404 char *hwproduct, *hwmodel, *hwvendor, *hwmachine;
405 size_t hwproduct_size, hwmodel_size, hwvendor_size, hwmachine_size;
406 int hwproduct_name[2], hwmodel_name[2], hwvendor_name[2], hwmachine_name[2];
962fad6d 407 unsigned int i;
041ce20b 408 gboolean UNSURE_CHASSIS_FLAG, UNSURE_ICON_FLAG;
409
22b13250 410 hwproduct_size = hwmodel_size = hwvendor_size = hwmachine_size = bufsize;
a112db4c 411 UNSURE_CHASSIS_FLAG = UNSURE_ICON_FLAG = FALSE;
1ed17375 412 i = 0;
962fad6d 413
22b13250 414 hwproduct = (char*)g_malloc0(4096);
415 hwmodel = (char*)g_malloc0(4096);
416 hwvendor = (char*)g_malloc0(4096);
417 hwmachine = (char*)g_malloc0(4096);
418
7a9b162a 419 hwproduct_name[0] = CTL_HW;
420 hwproduct_name[1] = HW_PRODUCT;
962fad6d 421
7a9b162a 422 hwmodel_name[0] = CTL_HW;
423 hwmodel_name[1] = HW_MODEL;
424
425 hwvendor_name[0] = CTL_HW;
426 hwvendor_name[1] = HW_VENDOR;
427
041ce20b 428 hwmachine_name[0] = CTL_HW;
429 hwmachine_name[1] = HW_MACHINE;
430
7a9b162a 431 /* pass NULL buffer to check size first, then pass hw to be filled according to freshly-set hw_size */
432 if(-1 == sysctl(hwproduct_name, 2, NULL, &hwproduct_size, NULL, 0) || -1 == sysctl(hwproduct_name, 2, hwproduct, &hwproduct_size, NULL, 0))
433 return FALSE;
434
435 if(-1 == sysctl(hwmodel_name, 2, NULL, &hwmodel_size, NULL, 0) || -1 == sysctl(hwmodel_name, 2, hwmodel, &hwmodel_size, NULL, 0))
436 return FALSE;
437
438 if(-1 == sysctl(hwvendor_name, 2, NULL, &hwvendor_size, NULL, 0) || -1 == sysctl(hwvendor_name, 2, hwvendor, &hwvendor_size, NULL, 0))
439 return FALSE;
440
041ce20b 441 if(-1 == sysctl(hwmachine_name, 2, NULL, &hwmachine_size, NULL, 0) || -1 == sysctl(hwmachine_name, 2, hwmachine, &hwmachine_size, NULL, 0))
442 return FALSE;
443
7a9b162a 444 /* TODO: test for laptop, if not, dmidecode for desktop vs. server
445 * probably move this code to vm test func and set a global after running it early, once */
446
041ce20b 447 for(; i < G_N_ELEMENTS(chassis_indicator_table); i++) {
448 if(strcasestr(hwproduct, chassis_indicator_table[i].match_string)
449 || strcasestr(hwmodel, chassis_indicator_table[i].match_string)
450 || strcasestr(hwvendor, chassis_indicator_table[i].match_string)) {
451
452 if(!UNSURE_CHASSIS_FLAG && chassis_indicator_table[i].chassis) {
453
454 UNSURE_CHASSIS_FLAG = chassis_indicator_table[i].chassis_precedence;
455 CHASSIS = chassis_indicator_table[i].chassis;
456 }
457
458 if(!UNSURE_ICON_FLAG && chassis_indicator_table[i].icon) {
459
460 UNSURE_ICON_FLAG = chassis_indicator_table[i].icon_precedence;
461 ICON = chassis_indicator_table[i].icon;
462 }
463 }
464 }
465
466 if(up_native_is_laptop()) {
467
468 if(!CHASSIS)
469 CHASSIS = "laptop";
470 if(!ICON)
471 ICON = "input-touchpad"; /* TODO pull an icon package that actually has the icons we're looking for */
472
473 } else if(is_server(hwmachine)) {
474
475 if(!CHASSIS)
476 CHASSIS = "server";
477 if(!ICON)
478 ICON = "uninterruptible-power-supply";
479
480 } else if(!CHASSIS || !ICON) {
962fad6d 481
041ce20b 482 if(!CHASSIS)
483 CHASSIS = "desktop";
484 if(!ICON)
485 ICON = "computer";
486 }
487
488 return (CHASSIS && ICON);
489}
490
491gboolean is_server(gchar *arch) {
492
493 unsigned int i;
7a9b162a 494
041ce20b 495 for(; i < G_N_ELEMENTS(server_archs); i++)
496 if(strcasestr(arch, server_archs[i]))
497 return TRUE;
498
499 return FALSE;
500}
501
502gboolean up_native_is_laptop() {
503
504 struct apm_power_info bstate;
505 struct sensordev acpiac;
506
507 if (up_native_get_sensordev("acpiac0", &acpiac))
508 return TRUE;
509
510 if (-1 == ioctl(up_apm_get_fd(), APM_IOC_GETPOWER, &bstate))
511 g_error("ioctl on apm fd failed : %s", g_strerror(errno));
512
513 return bstate.ac_state != APM_AC_UNKNOWN;
514}
515
516int up_apm_get_fd() {
517
518 static int apm_fd = 0;
519
520 if(apm_fd == 0) {
521
522 g_debug("apm_fd is not initialized yet, opening");
523
524 /* open /dev/apm */
525 if((apm_fd = open("/dev/apm", O_RDONLY)) == -1) {
526 if(errno != ENXIO && errno != ENOENT)
527 g_error("cannot open device file");
528 }
529 }
530
531 return apm_fd;
532}
533
534gboolean up_native_get_sensordev(const char * id, struct sensordev * snsrdev) {
535
536 int devn;
537 size_t sdlen = sizeof(struct sensordev);
538 int mib[] = {CTL_HW, HW_SENSORS, 0, 0 ,0};
539
540 for (devn = 0 ; ; devn++) {
541 mib[2] = devn;
542 if(sysctl(mib, 3, snsrdev, &sdlen, NULL, 0) == -1) {
543 if(errno == ENXIO)
544 continue;
545 if(errno == ENOENT)
546 break;
547 }
548
549 if (!strcmp(snsrdev->xname, id))
550 return TRUE;
551 }
552
553 return FALSE;
962fad6d 554}