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
					
				
			
		
					 10 changed files with 660 additions and 377 deletions
				
			
		| 
						 | 
					@ -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 @@ eperm:
 | 
				
			||||||
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…
	
	Add table
		Add a link
		
	
		Reference in a new issue