Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
| 2 | /* |
| 3 | * Utility routines. |
| 4 | * |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 5 | * Copyright (C) many different people. If you wrote this, please |
| 6 | * acknowledge your work. |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 7 | * |
Bernhard Reutner-Fischer | b1629b1 | 2006-05-19 19:29:19 +0000 | [diff] [blame] | 8 | * Licensed under GPLv2 or later, see file LICENSE in this tarball for details. |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 9 | */ |
| 10 | |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 11 | #include "libbb.h" |
| 12 | |
Eric Andersen | c7bda1c | 2004-03-15 08:29:22 +0000 | [diff] [blame] | 13 | /* From <linux/kd.h> */ |
Rob Landley | bc68cd1 | 2006-03-10 19:22:06 +0000 | [diff] [blame] | 14 | enum { KDGKBTYPE = 0x4B33 }; /* get keyboard type */ |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 15 | |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 16 | |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 17 | static int open_a_console(const char *fnam) |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 18 | { |
| 19 | int fd; |
| 20 | |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 21 | /* try read-write */ |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 22 | fd = open(fnam, O_RDWR); |
| 23 | |
| 24 | /* if failed, try read-only */ |
| 25 | if (fd < 0 && errno == EACCES) |
| 26 | fd = open(fnam, O_RDONLY); |
| 27 | |
| 28 | /* if failed, try write-only */ |
| 29 | if (fd < 0 && errno == EACCES) |
| 30 | fd = open(fnam, O_WRONLY); |
| 31 | |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 32 | return fd; |
| 33 | } |
| 34 | |
| 35 | /* |
| 36 | * Get an fd for use with kbd/console ioctls. |
| 37 | * We try several things because opening /dev/console will fail |
| 38 | * if someone else used X (which does a chown on /dev/console). |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 39 | */ |
| 40 | |
Eric Andersen | c38678d | 2002-09-16 06:22:25 +0000 | [diff] [blame] | 41 | int get_console_fd(void) |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 42 | { |
Denis Vlasenko | ec27feb | 2007-02-17 15:52:02 +0000 | [diff] [blame] | 43 | static const char *const console_names[] = { |
| 44 | DEV_CONSOLE, CURRENT_VC, CURRENT_TTY |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 45 | }; |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 46 | |
Denis Vlasenko | ec27feb | 2007-02-17 15:52:02 +0000 | [diff] [blame] | 47 | int fd; |
| 48 | |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 49 | for (fd = 2; fd >= 0; fd--) { |
| 50 | int fd4name; |
Denis Vlasenko | 51742f4 | 2007-04-12 00:32:05 +0000 | [diff] [blame] | 51 | int choice_fd; |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 52 | char arg; |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 53 | |
Denis Vlasenko | ec27feb | 2007-02-17 15:52:02 +0000 | [diff] [blame] | 54 | fd4name = open_a_console(console_names[fd]); |
Denis Vlasenko | 06af216 | 2007-02-03 17:28:39 +0000 | [diff] [blame] | 55 | chk_std: |
Denis Vlasenko | 51742f4 | 2007-04-12 00:32:05 +0000 | [diff] [blame] | 56 | choice_fd = (fd4name >= 0 ? fd4name : fd); |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 57 | |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 58 | arg = 0; |
Denis Vlasenko | 51742f4 | 2007-04-12 00:32:05 +0000 | [diff] [blame] | 59 | if (ioctl(choice_fd, KDGKBTYPE, &arg) == 0) |
| 60 | return choice_fd; |
| 61 | if (fd4name >= 0) { |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 62 | close(fd4name); |
| 63 | fd4name = -1; |
| 64 | goto chk_std; |
| 65 | } |
| 66 | } |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 67 | |
Denis Vlasenko | aa7a888 | 2007-10-20 00:17:34 +0000 | [diff] [blame] | 68 | bb_error_msg("can't open console"); |
Glenn L McGrath | 005f83a | 2003-09-01 08:53:32 +0000 | [diff] [blame] | 69 | return fd; /* total failure */ |
Eric Andersen | aad1a88 | 2001-03-16 22:47:14 +0000 | [diff] [blame] | 70 | } |
Bernhard Reutner-Fischer | ae4342c | 2008-05-19 08:18:50 +0000 | [diff] [blame] | 71 | |
| 72 | /* From <linux/vt.h> */ |
| 73 | enum { |
| 74 | VT_ACTIVATE = 0x5606, /* make vt active */ |
| 75 | VT_WAITACTIVE = 0x5607 /* wait for vt active */ |
| 76 | }; |
| 77 | |
| 78 | void console_make_active(int fd, const int vt_num) |
| 79 | { |
| 80 | xioctl(fd, VT_ACTIVATE, (void *)(ptrdiff_t)vt_num); |
| 81 | xioctl(fd, VT_WAITACTIVE, (void *)(ptrdiff_t)vt_num); |
| 82 | } |