mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-08 14:13:53 +00:00
vt:tackle kbd_table
Keyboard struct lifetime is easy, but the locking is not and is completely ignored by the existing code. Tackle this one head on - Make the kbd_table private so we can run down all direct users - Hoick the relevant ioctl handlers into the keyboard layer - Lock them with the keyboard lock so they don't change mid keypress - Add helpers for things like console stop/start so we isolate the poking around properly - Tweak the braille console so it still builds There are a couple of FIXME locking cases left for ioctls that are so hideous they should be addressed in a later patch. After this patch the kbd_table is private and all the keyboard jiggery pokery is in one place. This update fixes speakup and also a memory leak in the original. Signed-off-by: Alan Cox <alan@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
This commit is contained in:
parent
0fb8379dab
commit
079c9534a9
@ -244,16 +244,13 @@ static int keyboard_notifier_call(struct notifier_block *blk,
|
|||||||
|
|
||||||
switch (val) {
|
switch (val) {
|
||||||
case KVAL(K_CAPS):
|
case KVAL(K_CAPS):
|
||||||
on_off = vc_kbd_led(kbd_table + fg_console,
|
on_off = vt_get_leds(fg_console, VC_CAPSLOCK);
|
||||||
VC_CAPSLOCK);
|
|
||||||
break;
|
break;
|
||||||
case KVAL(K_NUM):
|
case KVAL(K_NUM):
|
||||||
on_off = vc_kbd_led(kbd_table + fg_console,
|
on_off = vt_get_leds(fg_console, VC_NUMLOCK);
|
||||||
VC_NUMLOCK);
|
|
||||||
break;
|
break;
|
||||||
case KVAL(K_HOLD):
|
case KVAL(K_HOLD):
|
||||||
on_off = vc_kbd_led(kbd_table + fg_console,
|
on_off = vt_get_leds(fg_console, VC_SCROLLOCK);
|
||||||
VC_SCROLLOCK);
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (on_off == 1)
|
if (on_off == 1)
|
||||||
|
@ -1731,15 +1731,15 @@ static void do_handle_spec(struct vc_data *vc, u_char value, char up_flag)
|
|||||||
switch (value) {
|
switch (value) {
|
||||||
case KVAL(K_CAPS):
|
case KVAL(K_CAPS):
|
||||||
label = msg_get(MSG_KEYNAME_CAPSLOCK);
|
label = msg_get(MSG_KEYNAME_CAPSLOCK);
|
||||||
on_off = (vc_kbd_led(kbd_table + vc->vc_num, VC_CAPSLOCK));
|
on_off = vt_get_leds(fg_console, VC_CAPSLOCK);
|
||||||
break;
|
break;
|
||||||
case KVAL(K_NUM):
|
case KVAL(K_NUM):
|
||||||
label = msg_get(MSG_KEYNAME_NUMLOCK);
|
label = msg_get(MSG_KEYNAME_NUMLOCK);
|
||||||
on_off = (vc_kbd_led(kbd_table + vc->vc_num, VC_NUMLOCK));
|
on_off = vt_get_leds(fg_console, VC_NUMLOCK);
|
||||||
break;
|
break;
|
||||||
case KVAL(K_HOLD):
|
case KVAL(K_HOLD):
|
||||||
label = msg_get(MSG_KEYNAME_SCROLLLOCK);
|
label = msg_get(MSG_KEYNAME_SCROLLLOCK);
|
||||||
on_off = (vc_kbd_led(kbd_table + vc->vc_num, VC_SCROLLOCK));
|
on_off = vt_get_leds(fg_console, VC_SCROLLOCK);
|
||||||
if (speakup_console[vc->vc_num])
|
if (speakup_console[vc->vc_num])
|
||||||
speakup_console[vc->vc_num]->tty_stopped = on_off;
|
speakup_console[vc->vc_num]->tty_stopped = on_off;
|
||||||
break;
|
break;
|
||||||
@ -2020,7 +2020,7 @@ speakup_key(struct vc_data *vc, int shift_state, int keycode, u_short keysym,
|
|||||||
if (type >= 0xf0)
|
if (type >= 0xf0)
|
||||||
type -= 0xf0;
|
type -= 0xf0;
|
||||||
if (type == KT_PAD
|
if (type == KT_PAD
|
||||||
&& (vc_kbd_led(kbd_table + fg_console, VC_NUMLOCK))) {
|
&& (vt_get_leds(fg_console, VC_NUMLOCK))) {
|
||||||
if (up_flag) {
|
if (up_flag) {
|
||||||
spk_keydown = 0;
|
spk_keydown = 0;
|
||||||
goto out;
|
goto out;
|
||||||
|
@ -110,11 +110,9 @@ static struct sysrq_key_op sysrq_SAK_op = {
|
|||||||
#ifdef CONFIG_VT
|
#ifdef CONFIG_VT
|
||||||
static void sysrq_handle_unraw(int key)
|
static void sysrq_handle_unraw(int key)
|
||||||
{
|
{
|
||||||
struct kbd_struct *kbd = &kbd_table[fg_console];
|
vt_reset_unicode(fg_console);
|
||||||
|
|
||||||
if (kbd)
|
|
||||||
kbd->kbdmode = default_utf8 ? VC_UNICODE : VC_XLATE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct sysrq_key_op sysrq_unraw_op = {
|
static struct sysrq_key_op sysrq_unraw_op = {
|
||||||
.handler = sysrq_handle_unraw,
|
.handler = sysrq_handle_unraw,
|
||||||
.help_msg = "unRaw",
|
.help_msg = "unRaw",
|
||||||
|
@ -68,8 +68,6 @@ extern void ctrl_alt_del(void);
|
|||||||
|
|
||||||
#define KBD_DEFLOCK 0
|
#define KBD_DEFLOCK 0
|
||||||
|
|
||||||
void compute_shiftstate(void);
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Handler Tables.
|
* Handler Tables.
|
||||||
*/
|
*/
|
||||||
@ -100,35 +98,29 @@ static fn_handler_fn *fn_handler[] = { FN_HANDLERS };
|
|||||||
* Variables exported for vt_ioctl.c
|
* Variables exported for vt_ioctl.c
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/* maximum values each key_handler can handle */
|
|
||||||
const int max_vals[] = {
|
|
||||||
255, ARRAY_SIZE(func_table) - 1, ARRAY_SIZE(fn_handler) - 1, NR_PAD - 1,
|
|
||||||
NR_DEAD - 1, 255, 3, NR_SHIFT - 1, 255, NR_ASCII - 1, NR_LOCK - 1,
|
|
||||||
255, NR_LOCK - 1, 255, NR_BRL - 1
|
|
||||||
};
|
|
||||||
|
|
||||||
const int NR_TYPES = ARRAY_SIZE(max_vals);
|
|
||||||
|
|
||||||
struct kbd_struct kbd_table[MAX_NR_CONSOLES];
|
|
||||||
EXPORT_SYMBOL_GPL(kbd_table);
|
|
||||||
static struct kbd_struct *kbd = kbd_table;
|
|
||||||
|
|
||||||
struct vt_spawn_console vt_spawn_con = {
|
struct vt_spawn_console vt_spawn_con = {
|
||||||
.lock = __SPIN_LOCK_UNLOCKED(vt_spawn_con.lock),
|
.lock = __SPIN_LOCK_UNLOCKED(vt_spawn_con.lock),
|
||||||
.pid = NULL,
|
.pid = NULL,
|
||||||
.sig = 0,
|
.sig = 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
|
||||||
* Variables exported for vt.c
|
|
||||||
*/
|
|
||||||
|
|
||||||
int shift_state = 0;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Internal Data.
|
* Internal Data.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
static struct kbd_struct kbd_table[MAX_NR_CONSOLES];
|
||||||
|
static struct kbd_struct *kbd = kbd_table;
|
||||||
|
|
||||||
|
/* maximum values each key_handler can handle */
|
||||||
|
static const int max_vals[] = {
|
||||||
|
255, ARRAY_SIZE(func_table) - 1, ARRAY_SIZE(fn_handler) - 1, NR_PAD - 1,
|
||||||
|
NR_DEAD - 1, 255, 3, NR_SHIFT - 1, 255, NR_ASCII - 1, NR_LOCK - 1,
|
||||||
|
255, NR_LOCK - 1, 255, NR_BRL - 1
|
||||||
|
};
|
||||||
|
|
||||||
|
static const int NR_TYPES = ARRAY_SIZE(max_vals);
|
||||||
|
|
||||||
static struct input_handler kbd_handler;
|
static struct input_handler kbd_handler;
|
||||||
static DEFINE_SPINLOCK(kbd_event_lock);
|
static DEFINE_SPINLOCK(kbd_event_lock);
|
||||||
static unsigned long key_down[BITS_TO_LONGS(KEY_CNT)]; /* keyboard key bitmap */
|
static unsigned long key_down[BITS_TO_LONGS(KEY_CNT)]; /* keyboard key bitmap */
|
||||||
@ -138,6 +130,8 @@ static int npadch = -1; /* -1 or number assembled on pad */
|
|||||||
static unsigned int diacr;
|
static unsigned int diacr;
|
||||||
static char rep; /* flag telling character repeat */
|
static char rep; /* flag telling character repeat */
|
||||||
|
|
||||||
|
static int shift_state = 0;
|
||||||
|
|
||||||
static unsigned char ledstate = 0xff; /* undefined */
|
static unsigned char ledstate = 0xff; /* undefined */
|
||||||
static unsigned char ledioctl;
|
static unsigned char ledioctl;
|
||||||
|
|
||||||
@ -188,7 +182,7 @@ static int getkeycode_helper(struct input_handle *handle, void *data)
|
|||||||
return d->error == 0; /* stop as soon as we successfully get one */
|
return d->error == 0; /* stop as soon as we successfully get one */
|
||||||
}
|
}
|
||||||
|
|
||||||
int getkeycode(unsigned int scancode)
|
static int getkeycode(unsigned int scancode)
|
||||||
{
|
{
|
||||||
struct getset_keycode_data d = {
|
struct getset_keycode_data d = {
|
||||||
.ke = {
|
.ke = {
|
||||||
@ -215,7 +209,7 @@ static int setkeycode_helper(struct input_handle *handle, void *data)
|
|||||||
return d->error == 0; /* stop as soon as we successfully set one */
|
return d->error == 0; /* stop as soon as we successfully set one */
|
||||||
}
|
}
|
||||||
|
|
||||||
int setkeycode(unsigned int scancode, unsigned int keycode)
|
static int setkeycode(unsigned int scancode, unsigned int keycode)
|
||||||
{
|
{
|
||||||
struct getset_keycode_data d = {
|
struct getset_keycode_data d = {
|
||||||
.ke = {
|
.ke = {
|
||||||
@ -383,9 +377,11 @@ static void to_utf8(struct vc_data *vc, uint c)
|
|||||||
/*
|
/*
|
||||||
* Called after returning from RAW mode or when changing consoles - recompute
|
* Called after returning from RAW mode or when changing consoles - recompute
|
||||||
* shift_down[] and shift_state from key_down[] maybe called when keymap is
|
* shift_down[] and shift_state from key_down[] maybe called when keymap is
|
||||||
* undefined, so that shiftkey release is seen
|
* undefined, so that shiftkey release is seen. The caller must hold the
|
||||||
|
* kbd_event_lock.
|
||||||
*/
|
*/
|
||||||
void compute_shiftstate(void)
|
|
||||||
|
static void do_compute_shiftstate(void)
|
||||||
{
|
{
|
||||||
unsigned int i, j, k, sym, val;
|
unsigned int i, j, k, sym, val;
|
||||||
|
|
||||||
@ -418,6 +414,15 @@ void compute_shiftstate(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* We still have to export this method to vt.c */
|
||||||
|
void compute_shiftstate(void)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
do_compute_shiftstate();
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We have a combining character DIACR here, followed by the character CH.
|
* We have a combining character DIACR here, followed by the character CH.
|
||||||
* If the combination occurs in the table, return the corresponding value.
|
* If the combination occurs in the table, return the corresponding value.
|
||||||
@ -637,7 +642,7 @@ static void fn_SAK(struct vc_data *vc)
|
|||||||
|
|
||||||
static void fn_null(struct vc_data *vc)
|
static void fn_null(struct vc_data *vc)
|
||||||
{
|
{
|
||||||
compute_shiftstate();
|
do_compute_shiftstate();
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -990,6 +995,8 @@ unsigned char getledstate(void)
|
|||||||
|
|
||||||
void setledstate(struct kbd_struct *kbd, unsigned int led)
|
void setledstate(struct kbd_struct *kbd, unsigned int led)
|
||||||
{
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
if (!(led & ~7)) {
|
if (!(led & ~7)) {
|
||||||
ledioctl = led;
|
ledioctl = led;
|
||||||
kbd->ledmode = LED_SHOW_IOCTL;
|
kbd->ledmode = LED_SHOW_IOCTL;
|
||||||
@ -997,6 +1004,7 @@ void setledstate(struct kbd_struct *kbd, unsigned int led)
|
|||||||
kbd->ledmode = LED_SHOW_FLAGS;
|
kbd->ledmode = LED_SHOW_FLAGS;
|
||||||
|
|
||||||
set_leds();
|
set_leds();
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline unsigned char getleds(void)
|
static inline unsigned char getleds(void)
|
||||||
@ -1036,6 +1044,75 @@ static int kbd_update_leds_helper(struct input_handle *handle, void *data)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_get_leds - helper for braille console
|
||||||
|
* @console: console to read
|
||||||
|
* @flag: flag we want to check
|
||||||
|
*
|
||||||
|
* Check the status of a keyboard led flag and report it back
|
||||||
|
*/
|
||||||
|
int vt_get_leds(int console, int flag)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
ret = vc_kbd_led(kbd, flag);
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(vt_get_leds);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_set_led_state - set LED state of a console
|
||||||
|
* @console: console to set
|
||||||
|
* @leds: LED bits
|
||||||
|
*
|
||||||
|
* Set the LEDs on a console. This is a wrapper for the VT layer
|
||||||
|
* so that we can keep kbd knowledge internal
|
||||||
|
*/
|
||||||
|
void vt_set_led_state(int console, int leds)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
setledstate(kbd, leds);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_kbd_con_start - Keyboard side of console start
|
||||||
|
* @console: console
|
||||||
|
*
|
||||||
|
* Handle console start. This is a wrapper for the VT layer
|
||||||
|
* so that we can keep kbd knowledge internal
|
||||||
|
*/
|
||||||
|
void vt_kbd_con_start(int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
clr_vc_kbd_led(kbd, VC_SCROLLOCK);
|
||||||
|
set_leds();
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_kbd_con_stop - Keyboard side of console stop
|
||||||
|
* @console: console
|
||||||
|
*
|
||||||
|
* Handle console stop. This is a wrapper for the VT layer
|
||||||
|
* so that we can keep kbd knowledge internal
|
||||||
|
*/
|
||||||
|
void vt_kbd_con_stop(int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
set_vc_kbd_led(kbd, VC_SCROLLOCK);
|
||||||
|
set_leds();
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This is the tasklet that updates LED state on all keyboards
|
* This is the tasklet that updates LED state on all keyboards
|
||||||
* attached to the box. The reason we use tasklet is that we
|
* attached to the box. The reason we use tasklet is that we
|
||||||
@ -1255,7 +1332,7 @@ static void kbd_keycode(unsigned int keycode, int down, int hw_raw)
|
|||||||
if (rc == NOTIFY_STOP || !key_map) {
|
if (rc == NOTIFY_STOP || !key_map) {
|
||||||
atomic_notifier_call_chain(&keyboard_notifier_list,
|
atomic_notifier_call_chain(&keyboard_notifier_list,
|
||||||
KBD_UNBOUND_KEYCODE, ¶m);
|
KBD_UNBOUND_KEYCODE, ¶m);
|
||||||
compute_shiftstate();
|
do_compute_shiftstate();
|
||||||
kbd->slockstate = 0;
|
kbd->slockstate = 0;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1615,3 +1692,495 @@ int vt_do_diacrit(unsigned int cmd, void __user *up, int perm)
|
|||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_do_kdskbmode - set keyboard mode ioctl
|
||||||
|
* @console: the console to use
|
||||||
|
* @arg: the requested mode
|
||||||
|
*
|
||||||
|
* Update the keyboard mode bits while holding the correct locks.
|
||||||
|
* Return 0 for success or an error code.
|
||||||
|
*/
|
||||||
|
int vt_do_kdskbmode(int console, unsigned int arg)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
int ret = 0;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
switch(arg) {
|
||||||
|
case K_RAW:
|
||||||
|
kbd->kbdmode = VC_RAW;
|
||||||
|
break;
|
||||||
|
case K_MEDIUMRAW:
|
||||||
|
kbd->kbdmode = VC_MEDIUMRAW;
|
||||||
|
break;
|
||||||
|
case K_XLATE:
|
||||||
|
kbd->kbdmode = VC_XLATE;
|
||||||
|
do_compute_shiftstate();
|
||||||
|
break;
|
||||||
|
case K_UNICODE:
|
||||||
|
kbd->kbdmode = VC_UNICODE;
|
||||||
|
do_compute_shiftstate();
|
||||||
|
break;
|
||||||
|
case K_OFF:
|
||||||
|
kbd->kbdmode = VC_OFF;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ret = -EINVAL;
|
||||||
|
}
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_do_kdskbmeta - set keyboard meta state
|
||||||
|
* @console: the console to use
|
||||||
|
* @arg: the requested meta state
|
||||||
|
*
|
||||||
|
* Update the keyboard meta bits while holding the correct locks.
|
||||||
|
* Return 0 for success or an error code.
|
||||||
|
*/
|
||||||
|
int vt_do_kdskbmeta(int console, unsigned int arg)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
int ret = 0;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
switch(arg) {
|
||||||
|
case K_METABIT:
|
||||||
|
clr_vc_kbd_mode(kbd, VC_META);
|
||||||
|
break;
|
||||||
|
case K_ESCPREFIX:
|
||||||
|
set_vc_kbd_mode(kbd, VC_META);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
ret = -EINVAL;
|
||||||
|
}
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int vt_do_kbkeycode_ioctl(int cmd, struct kbkeycode __user *user_kbkc,
|
||||||
|
int perm)
|
||||||
|
{
|
||||||
|
struct kbkeycode tmp;
|
||||||
|
int kc = 0;
|
||||||
|
|
||||||
|
if (copy_from_user(&tmp, user_kbkc, sizeof(struct kbkeycode)))
|
||||||
|
return -EFAULT;
|
||||||
|
switch (cmd) {
|
||||||
|
case KDGETKEYCODE:
|
||||||
|
kc = getkeycode(tmp.scancode);
|
||||||
|
if (kc >= 0)
|
||||||
|
kc = put_user(kc, &user_kbkc->keycode);
|
||||||
|
break;
|
||||||
|
case KDSETKEYCODE:
|
||||||
|
if (!perm)
|
||||||
|
return -EPERM;
|
||||||
|
kc = setkeycode(tmp.scancode, tmp.keycode);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return kc;
|
||||||
|
}
|
||||||
|
|
||||||
|
#define i (tmp.kb_index)
|
||||||
|
#define s (tmp.kb_table)
|
||||||
|
#define v (tmp.kb_value)
|
||||||
|
|
||||||
|
int vt_do_kdsk_ioctl(int cmd, struct kbentry __user *user_kbe, int perm,
|
||||||
|
int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
struct kbentry tmp;
|
||||||
|
ushort *key_map, *new_map, val, ov;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
if (copy_from_user(&tmp, user_kbe, sizeof(struct kbentry)))
|
||||||
|
return -EFAULT;
|
||||||
|
|
||||||
|
if (!capable(CAP_SYS_TTY_CONFIG))
|
||||||
|
perm = 0;
|
||||||
|
|
||||||
|
switch (cmd) {
|
||||||
|
case KDGKBENT:
|
||||||
|
/* Ensure another thread doesn't free it under us */
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
key_map = key_maps[s];
|
||||||
|
if (key_map) {
|
||||||
|
val = U(key_map[i]);
|
||||||
|
if (kbd->kbdmode != VC_UNICODE && KTYP(val) >= NR_TYPES)
|
||||||
|
val = K_HOLE;
|
||||||
|
} else
|
||||||
|
val = (i ? K_HOLE : K_NOSUCHMAP);
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
return put_user(val, &user_kbe->kb_value);
|
||||||
|
case KDSKBENT:
|
||||||
|
if (!perm)
|
||||||
|
return -EPERM;
|
||||||
|
if (!i && v == K_NOSUCHMAP) {
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
/* deallocate map */
|
||||||
|
key_map = key_maps[s];
|
||||||
|
if (s && key_map) {
|
||||||
|
key_maps[s] = NULL;
|
||||||
|
if (key_map[0] == U(K_ALLOCATED)) {
|
||||||
|
kfree(key_map);
|
||||||
|
keymap_count--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (KTYP(v) < NR_TYPES) {
|
||||||
|
if (KVAL(v) > max_vals[KTYP(v)])
|
||||||
|
return -EINVAL;
|
||||||
|
} else
|
||||||
|
if (kbd->kbdmode != VC_UNICODE)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
/* ++Geert: non-PC keyboards may generate keycode zero */
|
||||||
|
#if !defined(__mc68000__) && !defined(__powerpc__)
|
||||||
|
/* assignment to entry 0 only tests validity of args */
|
||||||
|
if (!i)
|
||||||
|
break;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
new_map = kmalloc(sizeof(plain_map), GFP_KERNEL);
|
||||||
|
if (!new_map)
|
||||||
|
return -ENOMEM;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
key_map = key_maps[s];
|
||||||
|
if (key_map == NULL) {
|
||||||
|
int j;
|
||||||
|
|
||||||
|
if (keymap_count >= MAX_NR_OF_USER_KEYMAPS &&
|
||||||
|
!capable(CAP_SYS_RESOURCE)) {
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
kfree(new_map);
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
key_maps[s] = new_map;
|
||||||
|
key_map[0] = U(K_ALLOCATED);
|
||||||
|
for (j = 1; j < NR_KEYS; j++)
|
||||||
|
key_map[j] = U(K_HOLE);
|
||||||
|
keymap_count++;
|
||||||
|
} else
|
||||||
|
kfree(new_map);
|
||||||
|
|
||||||
|
ov = U(key_map[i]);
|
||||||
|
if (v == ov)
|
||||||
|
goto out;
|
||||||
|
/*
|
||||||
|
* Attention Key.
|
||||||
|
*/
|
||||||
|
if (((ov == K_SAK) || (v == K_SAK)) && !capable(CAP_SYS_ADMIN)) {
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
return -EPERM;
|
||||||
|
}
|
||||||
|
key_map[i] = U(v);
|
||||||
|
if (!s && (KTYP(ov) == KT_SHIFT || KTYP(v) == KT_SHIFT))
|
||||||
|
do_compute_shiftstate();
|
||||||
|
out:
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
#undef i
|
||||||
|
#undef s
|
||||||
|
#undef v
|
||||||
|
|
||||||
|
/* FIXME: This one needs untangling and locking */
|
||||||
|
int vt_do_kdgkb_ioctl(int cmd, struct kbsentry __user *user_kdgkb, int perm)
|
||||||
|
{
|
||||||
|
struct kbsentry *kbs;
|
||||||
|
char *p;
|
||||||
|
u_char *q;
|
||||||
|
u_char __user *up;
|
||||||
|
int sz;
|
||||||
|
int delta;
|
||||||
|
char *first_free, *fj, *fnw;
|
||||||
|
int i, j, k;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
if (!capable(CAP_SYS_TTY_CONFIG))
|
||||||
|
perm = 0;
|
||||||
|
|
||||||
|
kbs = kmalloc(sizeof(*kbs), GFP_KERNEL);
|
||||||
|
if (!kbs) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* we mostly copy too much here (512bytes), but who cares ;) */
|
||||||
|
if (copy_from_user(kbs, user_kdgkb, sizeof(struct kbsentry))) {
|
||||||
|
ret = -EFAULT;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
kbs->kb_string[sizeof(kbs->kb_string)-1] = '\0';
|
||||||
|
i = kbs->kb_func;
|
||||||
|
|
||||||
|
switch (cmd) {
|
||||||
|
case KDGKBSENT:
|
||||||
|
sz = sizeof(kbs->kb_string) - 1; /* sz should have been
|
||||||
|
a struct member */
|
||||||
|
up = user_kdgkb->kb_string;
|
||||||
|
p = func_table[i];
|
||||||
|
if(p)
|
||||||
|
for ( ; *p && sz; p++, sz--)
|
||||||
|
if (put_user(*p, up++)) {
|
||||||
|
ret = -EFAULT;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
if (put_user('\0', up)) {
|
||||||
|
ret = -EFAULT;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
kfree(kbs);
|
||||||
|
return ((p && *p) ? -EOVERFLOW : 0);
|
||||||
|
case KDSKBSENT:
|
||||||
|
if (!perm) {
|
||||||
|
ret = -EPERM;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
|
||||||
|
q = func_table[i];
|
||||||
|
first_free = funcbufptr + (funcbufsize - funcbufleft);
|
||||||
|
for (j = i+1; j < MAX_NR_FUNC && !func_table[j]; j++)
|
||||||
|
;
|
||||||
|
if (j < MAX_NR_FUNC)
|
||||||
|
fj = func_table[j];
|
||||||
|
else
|
||||||
|
fj = first_free;
|
||||||
|
|
||||||
|
delta = (q ? -strlen(q) : 1) + strlen(kbs->kb_string);
|
||||||
|
if (delta <= funcbufleft) { /* it fits in current buf */
|
||||||
|
if (j < MAX_NR_FUNC) {
|
||||||
|
memmove(fj + delta, fj, first_free - fj);
|
||||||
|
for (k = j; k < MAX_NR_FUNC; k++)
|
||||||
|
if (func_table[k])
|
||||||
|
func_table[k] += delta;
|
||||||
|
}
|
||||||
|
if (!q)
|
||||||
|
func_table[i] = fj;
|
||||||
|
funcbufleft -= delta;
|
||||||
|
} else { /* allocate a larger buffer */
|
||||||
|
sz = 256;
|
||||||
|
while (sz < funcbufsize - funcbufleft + delta)
|
||||||
|
sz <<= 1;
|
||||||
|
fnw = kmalloc(sz, GFP_KERNEL);
|
||||||
|
if(!fnw) {
|
||||||
|
ret = -ENOMEM;
|
||||||
|
goto reterr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!q)
|
||||||
|
func_table[i] = fj;
|
||||||
|
if (fj > funcbufptr)
|
||||||
|
memmove(fnw, funcbufptr, fj - funcbufptr);
|
||||||
|
for (k = 0; k < j; k++)
|
||||||
|
if (func_table[k])
|
||||||
|
func_table[k] = fnw + (func_table[k] - funcbufptr);
|
||||||
|
|
||||||
|
if (first_free > fj) {
|
||||||
|
memmove(fnw + (fj - funcbufptr) + delta, fj, first_free - fj);
|
||||||
|
for (k = j; k < MAX_NR_FUNC; k++)
|
||||||
|
if (func_table[k])
|
||||||
|
func_table[k] = fnw + (func_table[k] - funcbufptr) + delta;
|
||||||
|
}
|
||||||
|
if (funcbufptr != func_buf)
|
||||||
|
kfree(funcbufptr);
|
||||||
|
funcbufptr = fnw;
|
||||||
|
funcbufleft = funcbufleft - delta + sz - funcbufsize;
|
||||||
|
funcbufsize = sz;
|
||||||
|
}
|
||||||
|
strcpy(func_table[i], kbs->kb_string);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
ret = 0;
|
||||||
|
reterr:
|
||||||
|
kfree(kbs);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
int vt_do_kdskled(int console, int cmd, unsigned long arg, int perm)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
unsigned char ucval;
|
||||||
|
|
||||||
|
switch(cmd) {
|
||||||
|
/* the ioctls below read/set the flags usually shown in the leds */
|
||||||
|
/* don't use them - they will go away without warning */
|
||||||
|
case KDGKBLED:
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
ucval = kbd->ledflagstate | (kbd->default_ledflagstate << 4);
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
return put_user(ucval, (char __user *)arg);
|
||||||
|
|
||||||
|
case KDSKBLED:
|
||||||
|
if (!perm)
|
||||||
|
return -EPERM;
|
||||||
|
if (arg & ~0x77)
|
||||||
|
return -EINVAL;
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
kbd->ledflagstate = (arg & 7);
|
||||||
|
kbd->default_ledflagstate = ((arg >> 4) & 7);
|
||||||
|
set_leds();
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
break;
|
||||||
|
|
||||||
|
/* the ioctls below only set the lights, not the functions */
|
||||||
|
/* for those, see KDGKBLED and KDSKBLED above */
|
||||||
|
case KDGETLED:
|
||||||
|
ucval = getledstate();
|
||||||
|
return put_user(ucval, (char __user *)arg);
|
||||||
|
|
||||||
|
case KDSETLED:
|
||||||
|
if (!perm)
|
||||||
|
return -EPERM;
|
||||||
|
setledstate(kbd, arg);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return -ENOIOCTLCMD;
|
||||||
|
}
|
||||||
|
|
||||||
|
int vt_do_kdgkbmode(int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
/* This is a spot read so needs no locking */
|
||||||
|
switch (kbd->kbdmode) {
|
||||||
|
case VC_RAW:
|
||||||
|
return K_RAW;
|
||||||
|
case VC_MEDIUMRAW:
|
||||||
|
return K_MEDIUMRAW;
|
||||||
|
case VC_UNICODE:
|
||||||
|
return K_UNICODE;
|
||||||
|
case VC_OFF:
|
||||||
|
return K_OFF;
|
||||||
|
default:
|
||||||
|
return K_XLATE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_do_kdgkbmeta - report meta status
|
||||||
|
* @console: console to report
|
||||||
|
*
|
||||||
|
* Report the meta flag status of this console
|
||||||
|
*/
|
||||||
|
int vt_do_kdgkbmeta(int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
/* Again a spot read so no locking */
|
||||||
|
return vc_kbd_mode(kbd, VC_META) ? K_ESCPREFIX : K_METABIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_reset_unicode - reset the unicode status
|
||||||
|
* @console: console being reset
|
||||||
|
*
|
||||||
|
* Restore the unicode console state to its default
|
||||||
|
*/
|
||||||
|
void vt_reset_unicode(int console)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
kbd_table[console].kbdmode = default_utf8 ? VC_UNICODE : VC_XLATE;
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_get_shiftstate - shift bit state
|
||||||
|
*
|
||||||
|
* Report the shift bits from the keyboard state. We have to export
|
||||||
|
* this to support some oddities in the vt layer.
|
||||||
|
*/
|
||||||
|
int vt_get_shift_state(void)
|
||||||
|
{
|
||||||
|
/* Don't lock as this is a transient report */
|
||||||
|
return shift_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_reset_keyboard - reset keyboard state
|
||||||
|
* @console: console to reset
|
||||||
|
*
|
||||||
|
* Reset the keyboard bits for a console as part of a general console
|
||||||
|
* reset event
|
||||||
|
*/
|
||||||
|
void vt_reset_keyboard(int console)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
set_vc_kbd_mode(kbd, VC_REPEAT);
|
||||||
|
clr_vc_kbd_mode(kbd, VC_CKMODE);
|
||||||
|
clr_vc_kbd_mode(kbd, VC_APPLIC);
|
||||||
|
clr_vc_kbd_mode(kbd, VC_CRLF);
|
||||||
|
kbd->lockstate = 0;
|
||||||
|
kbd->slockstate = 0;
|
||||||
|
kbd->ledmode = LED_SHOW_FLAGS;
|
||||||
|
kbd->ledflagstate = kbd->default_ledflagstate;
|
||||||
|
/* do not do set_leds here because this causes an endless tasklet loop
|
||||||
|
when the keyboard hasn't been initialized yet */
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_get_kbd_mode_bit - read keyboard status bits
|
||||||
|
* @console: console to read from
|
||||||
|
* @bit: mode bit to read
|
||||||
|
*
|
||||||
|
* Report back a vt mode bit. We do this without locking so the
|
||||||
|
* caller must be sure that there are no synchronization needs
|
||||||
|
*/
|
||||||
|
|
||||||
|
int vt_get_kbd_mode_bit(int console, int bit)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
return vc_kbd_mode(kbd, bit);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_set_kbd_mode_bit - read keyboard status bits
|
||||||
|
* @console: console to read from
|
||||||
|
* @bit: mode bit to read
|
||||||
|
*
|
||||||
|
* Set a vt mode bit. We do this without locking so the
|
||||||
|
* caller must be sure that there are no synchronization needs
|
||||||
|
*/
|
||||||
|
|
||||||
|
void vt_set_kbd_mode_bit(int console, int bit)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
set_vc_kbd_mode(kbd, bit);
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* vt_clr_kbd_mode_bit - read keyboard status bits
|
||||||
|
* @console: console to read from
|
||||||
|
* @bit: mode bit to read
|
||||||
|
*
|
||||||
|
* Report back a vt mode bit. We do this without locking so the
|
||||||
|
* caller must be sure that there are no synchronization needs
|
||||||
|
*/
|
||||||
|
|
||||||
|
void vt_clr_kbd_mode_bit(int console, int bit)
|
||||||
|
{
|
||||||
|
struct kbd_struct * kbd = kbd_table + console;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
spin_lock_irqsave(&kbd_event_lock, flags);
|
||||||
|
clr_vc_kbd_mode(kbd, bit);
|
||||||
|
spin_unlock_irqrestore(&kbd_event_lock, flags);
|
||||||
|
}
|
||||||
|
@ -30,6 +30,7 @@
|
|||||||
|
|
||||||
extern void poke_blanked_console(void);
|
extern void poke_blanked_console(void);
|
||||||
|
|
||||||
|
/* FIXME: all this needs locking */
|
||||||
/* Variables for selection control. */
|
/* Variables for selection control. */
|
||||||
/* Use a dynamic buffer, instead of static (Dec 1994) */
|
/* Use a dynamic buffer, instead of static (Dec 1994) */
|
||||||
struct vc_data *sel_cons; /* must not be deallocated */
|
struct vc_data *sel_cons; /* must not be deallocated */
|
||||||
@ -138,7 +139,7 @@ int set_selection(const struct tiocl_selection __user *sel, struct tty_struct *t
|
|||||||
char *bp, *obp;
|
char *bp, *obp;
|
||||||
int i, ps, pe, multiplier;
|
int i, ps, pe, multiplier;
|
||||||
u16 c;
|
u16 c;
|
||||||
struct kbd_struct *kbd = kbd_table + fg_console;
|
int mode;
|
||||||
|
|
||||||
poke_blanked_console();
|
poke_blanked_console();
|
||||||
|
|
||||||
@ -182,7 +183,11 @@ int set_selection(const struct tiocl_selection __user *sel, struct tty_struct *t
|
|||||||
clear_selection();
|
clear_selection();
|
||||||
sel_cons = vc_cons[fg_console].d;
|
sel_cons = vc_cons[fg_console].d;
|
||||||
}
|
}
|
||||||
use_unicode = kbd && kbd->kbdmode == VC_UNICODE;
|
mode = vt_do_kdgkbmode(fg_console);
|
||||||
|
if (mode == K_UNICODE)
|
||||||
|
use_unicode = 1;
|
||||||
|
else
|
||||||
|
use_unicode = 0;
|
||||||
|
|
||||||
switch (sel_mode)
|
switch (sel_mode)
|
||||||
{
|
{
|
||||||
|
@ -1028,9 +1028,9 @@ void vc_deallocate(unsigned int currcons)
|
|||||||
* VT102 emulator
|
* VT102 emulator
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define set_kbd(vc, x) set_vc_kbd_mode(kbd_table + (vc)->vc_num, (x))
|
#define set_kbd(vc, x) vt_set_kbd_mode_bit((vc)->vc_num, (x))
|
||||||
#define clr_kbd(vc, x) clr_vc_kbd_mode(kbd_table + (vc)->vc_num, (x))
|
#define clr_kbd(vc, x) vt_clr_kbd_mode_bit((vc)->vc_num, (x))
|
||||||
#define is_kbd(vc, x) vc_kbd_mode(kbd_table + (vc)->vc_num, (x))
|
#define is_kbd(vc, x) vt_get_kbd_mode_bit((vc)->vc_num, (x))
|
||||||
|
|
||||||
#define decarm VC_REPEAT
|
#define decarm VC_REPEAT
|
||||||
#define decckm VC_CKMODE
|
#define decckm VC_CKMODE
|
||||||
@ -1652,16 +1652,7 @@ static void reset_terminal(struct vc_data *vc, int do_clear)
|
|||||||
vc->vc_deccm = global_cursor_default;
|
vc->vc_deccm = global_cursor_default;
|
||||||
vc->vc_decim = 0;
|
vc->vc_decim = 0;
|
||||||
|
|
||||||
set_kbd(vc, decarm);
|
vt_reset_keyboard(vc->vc_num);
|
||||||
clr_kbd(vc, decckm);
|
|
||||||
clr_kbd(vc, kbdapplic);
|
|
||||||
clr_kbd(vc, lnm);
|
|
||||||
kbd_table[vc->vc_num].lockstate = 0;
|
|
||||||
kbd_table[vc->vc_num].slockstate = 0;
|
|
||||||
kbd_table[vc->vc_num].ledmode = LED_SHOW_FLAGS;
|
|
||||||
kbd_table[vc->vc_num].ledflagstate = kbd_table[vc->vc_num].default_ledflagstate;
|
|
||||||
/* do not do set_leds here because this causes an endless tasklet loop
|
|
||||||
when the keyboard hasn't been initialized yet */
|
|
||||||
|
|
||||||
vc->vc_cursor_type = cur_default;
|
vc->vc_cursor_type = cur_default;
|
||||||
vc->vc_complement_mask = vc->vc_s_complement_mask;
|
vc->vc_complement_mask = vc->vc_s_complement_mask;
|
||||||
@ -1979,7 +1970,7 @@ static void do_con_trol(struct tty_struct *tty, struct vc_data *vc, int c)
|
|||||||
case 'q': /* DECLL - but only 3 leds */
|
case 'q': /* DECLL - but only 3 leds */
|
||||||
/* map 0,1,2,3 to 0,1,2,4 */
|
/* map 0,1,2,3 to 0,1,2,4 */
|
||||||
if (vc->vc_par[0] < 4)
|
if (vc->vc_par[0] < 4)
|
||||||
setledstate(kbd_table + vc->vc_num,
|
vt_set_led_state(vc->vc_num,
|
||||||
(vc->vc_par[0] < 3) ? vc->vc_par[0] : 4);
|
(vc->vc_par[0] < 3) ? vc->vc_par[0] : 4);
|
||||||
return;
|
return;
|
||||||
case 'r':
|
case 'r':
|
||||||
@ -2642,7 +2633,7 @@ int tioclinux(struct tty_struct *tty, unsigned long arg)
|
|||||||
* kernel-internal variable; programs not closely
|
* kernel-internal variable; programs not closely
|
||||||
* related to the kernel should not use this.
|
* related to the kernel should not use this.
|
||||||
*/
|
*/
|
||||||
data = shift_state;
|
data = vt_get_shift_state();
|
||||||
ret = __put_user(data, p);
|
ret = __put_user(data, p);
|
||||||
break;
|
break;
|
||||||
case TIOCL_GETMOUSEREPORTING:
|
case TIOCL_GETMOUSEREPORTING:
|
||||||
@ -2753,8 +2744,7 @@ static void con_stop(struct tty_struct *tty)
|
|||||||
console_num = tty->index;
|
console_num = tty->index;
|
||||||
if (!vc_cons_allocated(console_num))
|
if (!vc_cons_allocated(console_num))
|
||||||
return;
|
return;
|
||||||
set_vc_kbd_led(kbd_table + console_num, VC_SCROLLOCK);
|
vt_kbd_con_stop(console_num);
|
||||||
set_leds();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -2768,8 +2758,7 @@ static void con_start(struct tty_struct *tty)
|
|||||||
console_num = tty->index;
|
console_num = tty->index;
|
||||||
if (!vc_cons_allocated(console_num))
|
if (!vc_cons_allocated(console_num))
|
||||||
return;
|
return;
|
||||||
clr_vc_kbd_led(kbd_table + console_num, VC_SCROLLOCK);
|
vt_kbd_con_start(console_num);
|
||||||
set_leds();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void con_flush_chars(struct tty_struct *tty)
|
static void con_flush_chars(struct tty_struct *tty)
|
||||||
|
@ -195,232 +195,7 @@ int vt_waitactive(int n)
|
|||||||
#define GPLAST 0x3df
|
#define GPLAST 0x3df
|
||||||
#define GPNUM (GPLAST - GPFIRST + 1)
|
#define GPNUM (GPLAST - GPFIRST + 1)
|
||||||
|
|
||||||
#define i (tmp.kb_index)
|
|
||||||
#define s (tmp.kb_table)
|
|
||||||
#define v (tmp.kb_value)
|
|
||||||
static inline int
|
|
||||||
do_kdsk_ioctl(int cmd, struct kbentry __user *user_kbe, int perm, struct kbd_struct *kbd)
|
|
||||||
{
|
|
||||||
struct kbentry tmp;
|
|
||||||
ushort *key_map, val, ov;
|
|
||||||
|
|
||||||
if (copy_from_user(&tmp, user_kbe, sizeof(struct kbentry)))
|
|
||||||
return -EFAULT;
|
|
||||||
|
|
||||||
if (!capable(CAP_SYS_TTY_CONFIG))
|
|
||||||
perm = 0;
|
|
||||||
|
|
||||||
switch (cmd) {
|
|
||||||
case KDGKBENT:
|
|
||||||
key_map = key_maps[s];
|
|
||||||
if (key_map) {
|
|
||||||
val = U(key_map[i]);
|
|
||||||
if (kbd->kbdmode != VC_UNICODE && KTYP(val) >= NR_TYPES)
|
|
||||||
val = K_HOLE;
|
|
||||||
} else
|
|
||||||
val = (i ? K_HOLE : K_NOSUCHMAP);
|
|
||||||
return put_user(val, &user_kbe->kb_value);
|
|
||||||
case KDSKBENT:
|
|
||||||
if (!perm)
|
|
||||||
return -EPERM;
|
|
||||||
if (!i && v == K_NOSUCHMAP) {
|
|
||||||
/* deallocate map */
|
|
||||||
key_map = key_maps[s];
|
|
||||||
if (s && key_map) {
|
|
||||||
key_maps[s] = NULL;
|
|
||||||
if (key_map[0] == U(K_ALLOCATED)) {
|
|
||||||
kfree(key_map);
|
|
||||||
keymap_count--;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (KTYP(v) < NR_TYPES) {
|
|
||||||
if (KVAL(v) > max_vals[KTYP(v)])
|
|
||||||
return -EINVAL;
|
|
||||||
} else
|
|
||||||
if (kbd->kbdmode != VC_UNICODE)
|
|
||||||
return -EINVAL;
|
|
||||||
|
|
||||||
/* ++Geert: non-PC keyboards may generate keycode zero */
|
|
||||||
#if !defined(__mc68000__) && !defined(__powerpc__)
|
|
||||||
/* assignment to entry 0 only tests validity of args */
|
|
||||||
if (!i)
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (!(key_map = key_maps[s])) {
|
|
||||||
int j;
|
|
||||||
|
|
||||||
if (keymap_count >= MAX_NR_OF_USER_KEYMAPS &&
|
|
||||||
!capable(CAP_SYS_RESOURCE))
|
|
||||||
return -EPERM;
|
|
||||||
|
|
||||||
key_map = kmalloc(sizeof(plain_map),
|
|
||||||
GFP_KERNEL);
|
|
||||||
if (!key_map)
|
|
||||||
return -ENOMEM;
|
|
||||||
key_maps[s] = key_map;
|
|
||||||
key_map[0] = U(K_ALLOCATED);
|
|
||||||
for (j = 1; j < NR_KEYS; j++)
|
|
||||||
key_map[j] = U(K_HOLE);
|
|
||||||
keymap_count++;
|
|
||||||
}
|
|
||||||
ov = U(key_map[i]);
|
|
||||||
if (v == ov)
|
|
||||||
break; /* nothing to do */
|
|
||||||
/*
|
|
||||||
* Attention Key.
|
|
||||||
*/
|
|
||||||
if (((ov == K_SAK) || (v == K_SAK)) && !capable(CAP_SYS_ADMIN))
|
|
||||||
return -EPERM;
|
|
||||||
key_map[i] = U(v);
|
|
||||||
if (!s && (KTYP(ov) == KT_SHIFT || KTYP(v) == KT_SHIFT))
|
|
||||||
compute_shiftstate();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#undef i
|
|
||||||
#undef s
|
|
||||||
#undef v
|
|
||||||
|
|
||||||
static inline int
|
|
||||||
do_kbkeycode_ioctl(int cmd, struct kbkeycode __user *user_kbkc, int perm)
|
|
||||||
{
|
|
||||||
struct kbkeycode tmp;
|
|
||||||
int kc = 0;
|
|
||||||
|
|
||||||
if (copy_from_user(&tmp, user_kbkc, sizeof(struct kbkeycode)))
|
|
||||||
return -EFAULT;
|
|
||||||
switch (cmd) {
|
|
||||||
case KDGETKEYCODE:
|
|
||||||
kc = getkeycode(tmp.scancode);
|
|
||||||
if (kc >= 0)
|
|
||||||
kc = put_user(kc, &user_kbkc->keycode);
|
|
||||||
break;
|
|
||||||
case KDSETKEYCODE:
|
|
||||||
if (!perm)
|
|
||||||
return -EPERM;
|
|
||||||
kc = setkeycode(tmp.scancode, tmp.keycode);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return kc;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline int
|
|
||||||
do_kdgkb_ioctl(int cmd, struct kbsentry __user *user_kdgkb, int perm)
|
|
||||||
{
|
|
||||||
struct kbsentry *kbs;
|
|
||||||
char *p;
|
|
||||||
u_char *q;
|
|
||||||
u_char __user *up;
|
|
||||||
int sz;
|
|
||||||
int delta;
|
|
||||||
char *first_free, *fj, *fnw;
|
|
||||||
int i, j, k;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (!capable(CAP_SYS_TTY_CONFIG))
|
|
||||||
perm = 0;
|
|
||||||
|
|
||||||
kbs = kmalloc(sizeof(*kbs), GFP_KERNEL);
|
|
||||||
if (!kbs) {
|
|
||||||
ret = -ENOMEM;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* we mostly copy too much here (512bytes), but who cares ;) */
|
|
||||||
if (copy_from_user(kbs, user_kdgkb, sizeof(struct kbsentry))) {
|
|
||||||
ret = -EFAULT;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
kbs->kb_string[sizeof(kbs->kb_string)-1] = '\0';
|
|
||||||
i = kbs->kb_func;
|
|
||||||
|
|
||||||
switch (cmd) {
|
|
||||||
case KDGKBSENT:
|
|
||||||
sz = sizeof(kbs->kb_string) - 1; /* sz should have been
|
|
||||||
a struct member */
|
|
||||||
up = user_kdgkb->kb_string;
|
|
||||||
p = func_table[i];
|
|
||||||
if(p)
|
|
||||||
for ( ; *p && sz; p++, sz--)
|
|
||||||
if (put_user(*p, up++)) {
|
|
||||||
ret = -EFAULT;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
if (put_user('\0', up)) {
|
|
||||||
ret = -EFAULT;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
kfree(kbs);
|
|
||||||
return ((p && *p) ? -EOVERFLOW : 0);
|
|
||||||
case KDSKBSENT:
|
|
||||||
if (!perm) {
|
|
||||||
ret = -EPERM;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
|
|
||||||
q = func_table[i];
|
|
||||||
first_free = funcbufptr + (funcbufsize - funcbufleft);
|
|
||||||
for (j = i+1; j < MAX_NR_FUNC && !func_table[j]; j++)
|
|
||||||
;
|
|
||||||
if (j < MAX_NR_FUNC)
|
|
||||||
fj = func_table[j];
|
|
||||||
else
|
|
||||||
fj = first_free;
|
|
||||||
|
|
||||||
delta = (q ? -strlen(q) : 1) + strlen(kbs->kb_string);
|
|
||||||
if (delta <= funcbufleft) { /* it fits in current buf */
|
|
||||||
if (j < MAX_NR_FUNC) {
|
|
||||||
memmove(fj + delta, fj, first_free - fj);
|
|
||||||
for (k = j; k < MAX_NR_FUNC; k++)
|
|
||||||
if (func_table[k])
|
|
||||||
func_table[k] += delta;
|
|
||||||
}
|
|
||||||
if (!q)
|
|
||||||
func_table[i] = fj;
|
|
||||||
funcbufleft -= delta;
|
|
||||||
} else { /* allocate a larger buffer */
|
|
||||||
sz = 256;
|
|
||||||
while (sz < funcbufsize - funcbufleft + delta)
|
|
||||||
sz <<= 1;
|
|
||||||
fnw = kmalloc(sz, GFP_KERNEL);
|
|
||||||
if(!fnw) {
|
|
||||||
ret = -ENOMEM;
|
|
||||||
goto reterr;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!q)
|
|
||||||
func_table[i] = fj;
|
|
||||||
if (fj > funcbufptr)
|
|
||||||
memmove(fnw, funcbufptr, fj - funcbufptr);
|
|
||||||
for (k = 0; k < j; k++)
|
|
||||||
if (func_table[k])
|
|
||||||
func_table[k] = fnw + (func_table[k] - funcbufptr);
|
|
||||||
|
|
||||||
if (first_free > fj) {
|
|
||||||
memmove(fnw + (fj - funcbufptr) + delta, fj, first_free - fj);
|
|
||||||
for (k = j; k < MAX_NR_FUNC; k++)
|
|
||||||
if (func_table[k])
|
|
||||||
func_table[k] = fnw + (func_table[k] - funcbufptr) + delta;
|
|
||||||
}
|
|
||||||
if (funcbufptr != func_buf)
|
|
||||||
kfree(funcbufptr);
|
|
||||||
funcbufptr = fnw;
|
|
||||||
funcbufleft = funcbufleft - delta + sz - funcbufsize;
|
|
||||||
funcbufsize = sz;
|
|
||||||
}
|
|
||||||
strcpy(func_table[i], kbs->kb_string);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
ret = 0;
|
|
||||||
reterr:
|
|
||||||
kfree(kbs);
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline int
|
static inline int
|
||||||
do_fontx_ioctl(int cmd, struct consolefontdesc __user *user_cfd, int perm, struct console_font_op *op)
|
do_fontx_ioctl(int cmd, struct consolefontdesc __user *user_cfd, int perm, struct console_font_op *op)
|
||||||
@ -497,7 +272,6 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
{
|
{
|
||||||
struct vc_data *vc = tty->driver_data;
|
struct vc_data *vc = tty->driver_data;
|
||||||
struct console_font_op op; /* used in multiple places here */
|
struct console_font_op op; /* used in multiple places here */
|
||||||
struct kbd_struct * kbd;
|
|
||||||
unsigned int console;
|
unsigned int console;
|
||||||
unsigned char ucval;
|
unsigned char ucval;
|
||||||
unsigned int uival;
|
unsigned int uival;
|
||||||
@ -523,7 +297,6 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
if (current->signal->tty == tty || capable(CAP_SYS_TTY_CONFIG))
|
if (current->signal->tty == tty || capable(CAP_SYS_TTY_CONFIG))
|
||||||
perm = 1;
|
perm = 1;
|
||||||
|
|
||||||
kbd = kbd_table + console;
|
|
||||||
switch (cmd) {
|
switch (cmd) {
|
||||||
case TIOCLINUX:
|
case TIOCLINUX:
|
||||||
ret = tioclinux(tty, arg);
|
ret = tioclinux(tty, arg);
|
||||||
@ -565,7 +338,8 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
* this is naive.
|
* this is naive.
|
||||||
*/
|
*/
|
||||||
ucval = KB_101;
|
ucval = KB_101;
|
||||||
goto setchar;
|
ret = put_user(ucval, (char __user *)arg);
|
||||||
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* These cannot be implemented on any machine that implements
|
* These cannot be implemented on any machine that implements
|
||||||
@ -670,68 +444,25 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
case KDSKBMODE:
|
case KDSKBMODE:
|
||||||
if (!perm)
|
if (!perm)
|
||||||
goto eperm;
|
goto eperm;
|
||||||
switch(arg) {
|
ret = vt_do_kdskbmode(console, arg);
|
||||||
case K_RAW:
|
if (ret == 0)
|
||||||
kbd->kbdmode = VC_RAW;
|
|
||||||
break;
|
|
||||||
case K_MEDIUMRAW:
|
|
||||||
kbd->kbdmode = VC_MEDIUMRAW;
|
|
||||||
break;
|
|
||||||
case K_XLATE:
|
|
||||||
kbd->kbdmode = VC_XLATE;
|
|
||||||
compute_shiftstate();
|
|
||||||
break;
|
|
||||||
case K_UNICODE:
|
|
||||||
kbd->kbdmode = VC_UNICODE;
|
|
||||||
compute_shiftstate();
|
|
||||||
break;
|
|
||||||
case K_OFF:
|
|
||||||
kbd->kbdmode = VC_OFF;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
ret = -EINVAL;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
tty_ldisc_flush(tty);
|
tty_ldisc_flush(tty);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KDGKBMODE:
|
case KDGKBMODE:
|
||||||
switch (kbd->kbdmode) {
|
uival = vt_do_kdgkbmode(console);
|
||||||
case VC_RAW:
|
ret = put_user(uival, (int __user *)arg);
|
||||||
uival = K_RAW;
|
|
||||||
break;
|
break;
|
||||||
case VC_MEDIUMRAW:
|
|
||||||
uival = K_MEDIUMRAW;
|
|
||||||
break;
|
|
||||||
case VC_UNICODE:
|
|
||||||
uival = K_UNICODE;
|
|
||||||
break;
|
|
||||||
case VC_OFF:
|
|
||||||
uival = K_OFF;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
uival = K_XLATE;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
goto setint;
|
|
||||||
|
|
||||||
/* this could be folded into KDSKBMODE, but for compatibility
|
/* this could be folded into KDSKBMODE, but for compatibility
|
||||||
reasons it is not so easy to fold KDGKBMETA into KDGKBMODE */
|
reasons it is not so easy to fold KDGKBMETA into KDGKBMODE */
|
||||||
case KDSKBMETA:
|
case KDSKBMETA:
|
||||||
switch(arg) {
|
ret = vt_do_kdskbmeta(console, arg);
|
||||||
case K_METABIT:
|
|
||||||
clr_vc_kbd_mode(kbd, VC_META);
|
|
||||||
break;
|
|
||||||
case K_ESCPREFIX:
|
|
||||||
set_vc_kbd_mode(kbd, VC_META);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
ret = -EINVAL;
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KDGKBMETA:
|
case KDGKBMETA:
|
||||||
uival = (vc_kbd_mode(kbd, VC_META) ? K_ESCPREFIX : K_METABIT);
|
/* FIXME: should review whether this is worth locking */
|
||||||
|
uival = vt_do_kdgkbmeta(console);
|
||||||
setint:
|
setint:
|
||||||
ret = put_user(uival, (int __user *)arg);
|
ret = put_user(uival, (int __user *)arg);
|
||||||
break;
|
break;
|
||||||
@ -740,17 +471,17 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
case KDSETKEYCODE:
|
case KDSETKEYCODE:
|
||||||
if(!capable(CAP_SYS_TTY_CONFIG))
|
if(!capable(CAP_SYS_TTY_CONFIG))
|
||||||
perm = 0;
|
perm = 0;
|
||||||
ret = do_kbkeycode_ioctl(cmd, up, perm);
|
ret = vt_do_kbkeycode_ioctl(cmd, up, perm);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KDGKBENT:
|
case KDGKBENT:
|
||||||
case KDSKBENT:
|
case KDSKBENT:
|
||||||
ret = do_kdsk_ioctl(cmd, up, perm, kbd);
|
ret = vt_do_kdsk_ioctl(cmd, up, perm, console);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case KDGKBSENT:
|
case KDGKBSENT:
|
||||||
case KDSKBSENT:
|
case KDSKBSENT:
|
||||||
ret = do_kdgkb_ioctl(cmd, up, perm);
|
ret = vt_do_kdgkb_ioctl(cmd, up, perm);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/* Diacritical processing. Handled in keyboard.c as it has
|
/* Diacritical processing. Handled in keyboard.c as it has
|
||||||
@ -765,33 +496,10 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
/* the ioctls below read/set the flags usually shown in the leds */
|
/* the ioctls below read/set the flags usually shown in the leds */
|
||||||
/* don't use them - they will go away without warning */
|
/* don't use them - they will go away without warning */
|
||||||
case KDGKBLED:
|
case KDGKBLED:
|
||||||
ucval = kbd->ledflagstate | (kbd->default_ledflagstate << 4);
|
|
||||||
goto setchar;
|
|
||||||
|
|
||||||
case KDSKBLED:
|
case KDSKBLED:
|
||||||
if (!perm)
|
|
||||||
goto eperm;
|
|
||||||
if (arg & ~0x77) {
|
|
||||||
ret = -EINVAL;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
kbd->ledflagstate = (arg & 7);
|
|
||||||
kbd->default_ledflagstate = ((arg >> 4) & 7);
|
|
||||||
set_leds();
|
|
||||||
break;
|
|
||||||
|
|
||||||
/* the ioctls below only set the lights, not the functions */
|
|
||||||
/* for those, see KDGKBLED and KDSKBLED above */
|
|
||||||
case KDGETLED:
|
case KDGETLED:
|
||||||
ucval = getledstate();
|
|
||||||
setchar:
|
|
||||||
ret = put_user(ucval, (char __user *)arg);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case KDSETLED:
|
case KDSETLED:
|
||||||
if (!perm)
|
ret = vt_do_kdskled(console, cmd, arg, perm);
|
||||||
goto eperm;
|
|
||||||
setledstate(kbd, arg);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1286,7 +994,7 @@ int vt_ioctl(struct tty_struct *tty,
|
|||||||
void reset_vc(struct vc_data *vc)
|
void reset_vc(struct vc_data *vc)
|
||||||
{
|
{
|
||||||
vc->vc_mode = KD_TEXT;
|
vc->vc_mode = KD_TEXT;
|
||||||
kbd_table[vc->vc_num].kbdmode = default_utf8 ? VC_UNICODE : VC_XLATE;
|
vt_reset_unicode(vc->vc_num);
|
||||||
vc->vt_mode.mode = VT_AUTO;
|
vc->vt_mode.mode = VT_AUTO;
|
||||||
vc->vt_mode.waitv = 0;
|
vc->vt_mode.waitv = 0;
|
||||||
vc->vt_mode.relsig = 0;
|
vc->vt_mode.relsig = 0;
|
||||||
@ -1309,6 +1017,7 @@ void vc_SAK(struct work_struct *work)
|
|||||||
console_lock();
|
console_lock();
|
||||||
vc = vc_con->d;
|
vc = vc_con->d;
|
||||||
if (vc) {
|
if (vc) {
|
||||||
|
/* FIXME: review tty ref counting */
|
||||||
tty = vc->port.tty;
|
tty = vc->port.tty;
|
||||||
/*
|
/*
|
||||||
* SAK should also work in all raw modes and reset
|
* SAK should also work in all raw modes and reset
|
||||||
|
@ -7,8 +7,6 @@
|
|||||||
|
|
||||||
extern struct tasklet_struct keyboard_tasklet;
|
extern struct tasklet_struct keyboard_tasklet;
|
||||||
|
|
||||||
extern int shift_state;
|
|
||||||
|
|
||||||
extern char *func_table[MAX_NR_FUNC];
|
extern char *func_table[MAX_NR_FUNC];
|
||||||
extern char func_buf[];
|
extern char func_buf[];
|
||||||
extern char *funcbufptr;
|
extern char *funcbufptr;
|
||||||
@ -65,8 +63,6 @@ struct kbd_struct {
|
|||||||
#define VC_META 4 /* 0 - meta, 1 - meta=prefix with ESC */
|
#define VC_META 4 /* 0 - meta, 1 - meta=prefix with ESC */
|
||||||
};
|
};
|
||||||
|
|
||||||
extern struct kbd_struct kbd_table[];
|
|
||||||
|
|
||||||
extern int kbd_init(void);
|
extern int kbd_init(void);
|
||||||
|
|
||||||
extern unsigned char getledstate(void);
|
extern unsigned char getledstate(void);
|
||||||
@ -79,6 +75,7 @@ extern void (*kbd_ledfunc)(unsigned int led);
|
|||||||
extern int set_console(int nr);
|
extern int set_console(int nr);
|
||||||
extern void schedule_console_callback(void);
|
extern void schedule_console_callback(void);
|
||||||
|
|
||||||
|
/* FIXME: review locking for vt.c callers */
|
||||||
static inline void set_leds(void)
|
static inline void set_leds(void)
|
||||||
{
|
{
|
||||||
tasklet_schedule(&keyboard_tasklet);
|
tasklet_schedule(&keyboard_tasklet);
|
||||||
@ -142,8 +139,6 @@ static inline void chg_vc_kbd_led(struct kbd_struct * kbd, int flag)
|
|||||||
|
|
||||||
struct console;
|
struct console;
|
||||||
|
|
||||||
int getkeycode(unsigned int scancode);
|
|
||||||
int setkeycode(unsigned int scancode, unsigned int keycode);
|
|
||||||
void compute_shiftstate(void);
|
void compute_shiftstate(void);
|
||||||
|
|
||||||
/* defkeymap.c */
|
/* defkeymap.c */
|
||||||
|
@ -24,8 +24,6 @@
|
|||||||
|
|
||||||
#ifdef __KERNEL__
|
#ifdef __KERNEL__
|
||||||
struct notifier_block;
|
struct notifier_block;
|
||||||
extern const int NR_TYPES;
|
|
||||||
extern const int max_vals[];
|
|
||||||
extern unsigned short *key_maps[MAX_NR_KEYMAPS];
|
extern unsigned short *key_maps[MAX_NR_KEYMAPS];
|
||||||
extern unsigned short plain_map[NR_KEYS];
|
extern unsigned short plain_map[NR_KEYS];
|
||||||
|
|
||||||
|
@ -169,5 +169,28 @@ extern void hide_boot_cursor(bool hide);
|
|||||||
|
|
||||||
/* keyboard provided interfaces */
|
/* keyboard provided interfaces */
|
||||||
extern int vt_do_diacrit(unsigned int cmd, void __user *up, int eperm);
|
extern int vt_do_diacrit(unsigned int cmd, void __user *up, int eperm);
|
||||||
|
extern int vt_do_kdskbmode(int console, unsigned int arg);
|
||||||
|
extern int vt_do_kdskbmeta(int console, unsigned int arg);
|
||||||
|
extern int vt_do_kbkeycode_ioctl(int cmd, struct kbkeycode __user *user_kbkc,
|
||||||
|
int perm);
|
||||||
|
extern int vt_do_kdsk_ioctl(int cmd, struct kbentry __user *user_kbe,
|
||||||
|
int perm, int console);
|
||||||
|
extern int vt_do_kdgkb_ioctl(int cmd, struct kbsentry __user *user_kdgkb,
|
||||||
|
int perm);
|
||||||
|
extern int vt_do_kdskled(int console, int cmd, unsigned long arg, int perm);
|
||||||
|
extern int vt_do_kdgkbmode(int console);
|
||||||
|
extern int vt_do_kdgkbmeta(int console);
|
||||||
|
extern void vt_reset_unicode(int console);
|
||||||
|
extern int vt_get_shift_state(void);
|
||||||
|
extern void vt_reset_keyboard(int console);
|
||||||
|
extern int vt_get_leds(int console, int flag);
|
||||||
|
extern int vt_get_kbd_mode_bit(int console, int bit);
|
||||||
|
extern void vt_set_kbd_mode_bit(int console, int bit);
|
||||||
|
extern void vt_clr_kbd_mode_bit(int console, int bit);
|
||||||
|
extern void vt_set_led_state(int console, int leds);
|
||||||
|
extern void vt_set_led_state(int console, int leds);
|
||||||
|
extern void vt_kbd_con_start(int console);
|
||||||
|
extern void vt_kbd_con_stop(int console);
|
||||||
|
|
||||||
|
|
||||||
#endif /* _VT_KERN_H */
|
#endif /* _VT_KERN_H */
|
||||||
|
Loading…
Reference in New Issue
Block a user