1 /*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
4 This file is part of systemd.
6 Copyright 2010 Kay Sievers
8 systemd is free software; you can redistribute it and/or modify it
9 under the terms of the GNU Lesser General Public License as published by
10 the Free Software Foundation; either version 2.1 of the License, or
11 (at your option) any later version.
13 systemd is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 Lesser General Public License for more details.
18 You should have received a copy of the GNU Lesser General Public License
19 along with systemd; If not, see <http://www.gnu.org/licenses/>.
34 #include <sys/ioctl.h>
36 #include <linux/tiocl.h>
44 static bool is_vconsole(int fd) {
45 unsigned char data[1];
47 data[0] = TIOCL_GETFGCONSOLE;
48 return ioctl(fd, TIOCLINUX, data) >= 0;
51 static bool is_locale_utf8(void) {
54 if (!setlocale(LC_ALL, ""))
57 set = nl_langinfo(CODESET);
61 return streq(set, "UTF-8");
64 static int disable_utf8(int fd) {
67 if (ioctl(fd, KDSKBMODE, K_XLATE) < 0)
70 if (loop_write(fd, "\033%@", 3, false) < 0)
73 k = write_one_line_file("/sys/module/vt/parameters/default_utf8", "0");
78 log_warning("Failed to disable UTF-8: %s", strerror(-r));
83 static int enable_utf8(int fd) {
86 if (ioctl(fd, KDSKBMODE, K_UNICODE) < 0)
89 if (loop_write(fd, "\033%G", 3, false) < 0)
92 k = write_one_line_file("/sys/module/vt/parameters/default_utf8", "1");
97 log_warning("Failed to enable UTF-8: %s", strerror(-r));
102 static int load_keymap(const char *vc, const char *map, const char *map_toggle, bool utf8, pid_t *_pid) {
108 /* An empty map means kernel map */
113 args[i++] = KBD_LOADKEYS;
121 args[i++] = map_toggle;
126 log_error("Failed to fork: %m");
128 } else if (pid == 0) {
129 execv(args[0], (char **) args);
137 static int load_font(const char *vc, const char *font, const char *map, const char *unimap, pid_t *_pid) {
143 /* An empty font means kernel font */
148 args[i++] = KBD_SETFONT;
164 log_error("Failed to fork: %m");
166 } else if (pid == 0) {
167 execv(args[0], (char **) args);
175 int main(int argc, char **argv) {
177 char *vc_keymap = NULL;
178 char *vc_keymap_toggle = NULL;
179 char *vc_font = NULL;
180 char *vc_font_map = NULL;
181 char *vc_font_unimap = NULL;
183 char *vc_unicode = NULL;
185 #if defined(TARGET_MANDRIVA) || defined(TARGET_MAGEIA)
186 char *vc_keytable = NULL;
190 int r = EXIT_FAILURE;
191 pid_t font_pid = 0, keymap_pid = 0;
193 log_set_target(LOG_TARGET_AUTO);
194 log_parse_environment();
204 fd = open_terminal(vc, O_RDWR|O_CLOEXEC);
206 log_error("Failed to open %s: %m", vc);
210 if (!is_vconsole(fd)) {
211 log_error("Device %s is not a virtual console.", vc);
215 utf8 = is_locale_utf8();
217 vc_keymap = strdup("us");
220 log_error("Failed to allocate string.");
226 if (detect_container(NULL) <= 0) {
227 r = parse_env_file("/proc/cmdline", WHITESPACE,
228 "vconsole.keymap", &vc_keymap,
229 "vconsole.keymap.toggle", &vc_keymap_toggle,
230 "vconsole.font", &vc_font,
231 "vconsole.font.map", &vc_font_map,
232 "vconsole.font.unimap", &vc_font_unimap,
235 if (r < 0 && r != -ENOENT)
236 log_warning("Failed to read /proc/cmdline: %s", strerror(-r));
239 /* Hmm, nothing set on the kernel cmd line? Then let's
240 * try /etc/vconsole.conf */
242 r = parse_env_file("/etc/vconsole.conf", NEWLINE,
243 "KEYMAP", &vc_keymap,
244 "KEYMAP_TOGGLE", &vc_keymap_toggle,
246 "FONT_MAP", &vc_font_map,
247 "FONT_UNIMAP", &vc_font_unimap,
250 if (r < 0 && r != -ENOENT)
251 log_warning("Failed to read /etc/vconsole.conf: %s", strerror(-r));
255 #if defined(TARGET_FEDORA)
256 r = parse_env_file("/etc/sysconfig/i18n", NEWLINE,
258 "SYSFONTACM", &vc_font_map,
259 "UNIMAP", &vc_font_unimap,
261 if (r < 0 && r != -ENOENT)
262 log_warning("Failed to read /etc/sysconfig/i18n: %s", strerror(-r));
264 r = parse_env_file("/etc/sysconfig/keyboard", NEWLINE,
265 "KEYTABLE", &vc_keymap,
266 "KEYMAP", &vc_keymap,
268 if (r < 0 && r != -ENOENT)
269 log_warning("Failed to read /etc/sysconfig/keyboard: %s", strerror(-r));
271 if (access("/etc/sysconfig/console/default.kmap", F_OK) >= 0) {
274 t = strdup("/etc/sysconfig/console/default.kmap");
284 #elif defined(TARGET_SUSE)
285 r = parse_env_file("/etc/sysconfig/keyboard", NEWLINE,
286 "KEYTABLE", &vc_keymap,
288 if (r < 0 && r != -ENOENT)
289 log_warning("Failed to read /etc/sysconfig/keyboard: %s", strerror(-r));
291 r = parse_env_file("/etc/sysconfig/console", NEWLINE,
292 "CONSOLE_FONT", &vc_font,
293 "CONSOLE_SCREENMAP", &vc_font_map,
294 "CONSOLE_UNICODEMAP", &vc_font_unimap,
296 if (r < 0 && r != -ENOENT)
297 log_warning("Failed to read /etc/sysconfig/console: %s", strerror(-r));
299 #elif defined(TARGET_ARCH)
300 r = parse_env_file("/etc/rc.conf", NEWLINE,
301 "KEYMAP", &vc_keymap,
302 "CONSOLEFONT", &vc_font,
303 "CONSOLEMAP", &vc_font_map,
305 if (r < 0 && r != -ENOENT)
306 log_warning("Failed to read /etc/rc.conf: %s", strerror(-r));
308 #elif defined(TARGET_FRUGALWARE)
309 r = parse_env_file("/etc/sysconfig/keymap", NEWLINE,
310 "keymap", &vc_keymap,
312 if (r < 0 && r != -ENOENT)
313 log_warning("Failed to read /etc/sysconfig/keymap: %s", strerror(-r));
315 r = parse_env_file("/etc/sysconfig/font", NEWLINE,
318 if (r < 0 && r != -ENOENT)
319 log_warning("Failed to read /etc/sysconfig/font: %s", strerror(-r));
321 #elif defined(TARGET_ALTLINUX)
322 r = parse_env_file("/etc/sysconfig/keyboard", NEWLINE,
323 "KEYTABLE", &vc_keymap,
325 if (r < 0 && r != -ENOENT)
326 log_warning("Failed to read /etc/sysconfig/keyboard: %s", strerror(-r));
328 r = parse_env_file("/etc/sysconfig/consolefont", NEWLINE,
331 if (r < 0 && r != -ENOENT)
332 log_warning("Failed to read /etc/sysconfig/consolefont: %s", strerror(-r));
334 #elif defined(TARGET_GENTOO)
335 r = parse_env_file("/etc/rc.conf", NEWLINE,
336 "unicode", &vc_unicode,
338 if (r < 0 && r != -ENOENT)
339 log_warning("Failed to read /etc/rc.conf: %s", strerror(-r));
344 rc_unicode = parse_boolean(vc_unicode);
346 log_warning("Unknown value for /etc/rc.conf unicode=%s", vc_unicode);
348 if (rc_unicode && !utf8)
349 log_warning("/etc/rc.conf wants unicode, but current locale is not UTF-8 capable!");
350 else if (!rc_unicode && utf8) {
351 log_debug("/etc/rc.conf does not want unicode, leave it on in kernel but does not apply to vconsole.");
357 /* /etc/conf.d/consolefont comments and gentoo
358 * documentation mention uppercase, but the actual
359 * contents are lowercase. the existing
360 * /etc/init.d/consolefont tries both
362 r = parse_env_file("/etc/conf.d/consolefont", NEWLINE,
363 "CONSOLEFONT", &vc_font,
364 "consolefont", &vc_font,
365 "consoletranslation", &vc_font_map,
366 "CONSOLETRANSLATION", &vc_font_map,
367 "unicodemap", &vc_font_unimap,
368 "UNICODEMAP", &vc_font_unimap,
370 if (r < 0 && r != -ENOENT)
371 log_warning("Failed to read /etc/conf.d/consolefont: %s", strerror(-r));
373 r = parse_env_file("/etc/conf.d/keymaps", NEWLINE,
374 "keymap", &vc_keymap,
375 "KEYMAP", &vc_keymap,
377 if (r < 0 && r != -ENOENT)
378 log_warning("Failed to read /etc/conf.d/keymaps: %s", strerror(-r));
380 #elif defined(TARGET_MANDRIVA) || defined (TARGET_MAGEIA)
382 r = parse_env_file("/etc/sysconfig/i18n", NEWLINE,
384 "SYSFONTACM", &vc_font_map,
385 "UNIMAP", &vc_font_unimap,
387 if (r < 0 && r != -ENOENT)
388 log_warning("Failed to read /etc/sysconfig/i18n: %s", strerror(-r));
390 r = parse_env_file("/etc/sysconfig/keyboard", NEWLINE,
391 "KEYTABLE", &vc_keytable,
392 "KEYMAP", &vc_keymap,
393 "UNIKEYTABLE", &vc_keymap,
394 "GRP_TOGGLE", &vc_keymap_toggle,
396 if (r < 0 && r != -ENOENT)
397 log_warning("Failed to read /etc/sysconfig/keyboard: %s", strerror(-r));
402 if (endswith(vc_keytable, ".uni") || strstr(vc_keytable, ".uni."))
403 vc_keymap = strdup(vc_keytable);
406 s = strstr(vc_keytable, ".map");
408 vc_keytable[s-vc_keytable+1] = '\0';
409 vc_keymap = strappend(vc_keytable, ".uni");
412 vc_keymap = strdup(vc_keytable);
422 if (access("/etc/sysconfig/console/default.kmap", F_OK) >= 0) {
425 t = strdup("/etc/sysconfig/console/default.kmap");
445 if (load_keymap(vc, vc_keymap, vc_keymap_toggle, utf8, &keymap_pid) >= 0 &&
446 load_font(vc, vc_font, vc_font_map, vc_font_unimap, &font_pid) >= 0)
451 wait_for_terminate_and_warn(KBD_LOADKEYS, keymap_pid);
454 wait_for_terminate_and_warn(KBD_SETFONT, font_pid);
459 free(vc_font_unimap);
462 close_nointr_nofail(fd);