Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
35c5ca68
Commit
35c5ca68
authored
Jul 03, 2002
by
James Simmons
Browse files
Options
Browse Files
Download
Plain Diff
Merge heisenberg.transvirtual.com:/tmp/linus-2.5
into heisenberg.transvirtual.com:/tmp/console-2.5
parents
55eebc46
be0db904
Changes
10
Hide whitespace changes
Inline
Side-by-side
Showing
10 changed files
with
483 additions
and
457 deletions
+483
-457
arch/i386/kernel/apm.c
arch/i386/kernel/apm.c
+1
-1
arch/ppc/xmon/start.c
arch/ppc/xmon/start.c
+1
-1
arch/ppc64/xmon/start.c
arch/ppc64/xmon/start.c
+1
-1
drivers/acpi/system.c
drivers/acpi/system.c
+1
-1
drivers/char/console.c
drivers/char/console.c
+46
-36
drivers/char/keyboard.c
drivers/char/keyboard.c
+397
-378
drivers/char/sysrq.c
drivers/char/sysrq.c
+30
-16
drivers/char/tty_io.c
drivers/char/tty_io.c
+2
-13
include/linux/console_struct.h
include/linux/console_struct.h
+1
-0
include/linux/sysrq.h
include/linux/sysrq.h
+3
-10
No files found.
arch/i386/kernel/apm.c
View file @
35c5ca68
...
...
@@ -923,7 +923,7 @@ static void apm_power_off(void)
*/
static
void
handle_poweroff
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
struct
tty_struct
*
tty
)
{
apm_power_off
();
}
...
...
arch/ppc/xmon/start.c
View file @
35c5ca68
...
...
@@ -98,7 +98,7 @@ static unsigned long chrp_find_phys_io_base(void)
#ifdef CONFIG_MAGIC_SYSRQ
static
void
sysrq_handle_xmon
(
int
key
,
struct
pt_regs
*
regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
struct
tty_struct
*
tty
)
{
xmon
(
regs
);
}
...
...
arch/ppc64/xmon/start.c
View file @
35c5ca68
...
...
@@ -52,7 +52,7 @@ void buf_access(void)
}
#endif
static
void
sysrq_handle_xmon
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
static
void
sysrq_handle_xmon
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
tty_struct
*
tty
)
{
xmon
(
pt_regs
);
}
...
...
drivers/acpi/system.c
View file @
35c5ca68
...
...
@@ -1174,7 +1174,7 @@ acpi_system_remove_fs (
/* Simple wrapper calling power down function. */
static
void
acpi_sysrq_power_off
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
struct
tty_struct
*
tty
)
{
acpi_power_off
();
}
...
...
drivers/char/console.c
View file @
35c5ca68
...
...
@@ -126,6 +126,7 @@ const struct consw *conswitchp;
#define DEFAULT_BELL_DURATION (HZ/8)
extern
void
vcs_make_devfs
(
unsigned
int
index
,
int
unregister
);
extern
void
console_map_init
(
void
);
#ifndef MIN
#define MIN(a,b) ((a) < (b) ? (a) : (b))
...
...
@@ -2381,6 +2382,7 @@ static int con_open(struct tty_struct *tty, struct file * filp)
vt_cons
[
currcons
]
->
vc_num
=
currcons
;
tty
->
driver_data
=
vt_cons
[
currcons
];
vc_cons
[
currcons
].
d
->
vc_tty
=
tty
;
if
(
!
tty
->
winsize
.
ws_row
&&
!
tty
->
winsize
.
ws_col
)
{
tty
->
winsize
.
ws_row
=
video_num_lines
;
...
...
@@ -2445,42 +2447,6 @@ void __init con_init(void)
return
;
}
memset
(
&
console_driver
,
0
,
sizeof
(
struct
tty_driver
));
console_driver
.
magic
=
TTY_DRIVER_MAGIC
;
console_driver
.
name
=
"vc/%d"
;
console_driver
.
name_base
=
1
;
console_driver
.
major
=
TTY_MAJOR
;
console_driver
.
minor_start
=
1
;
console_driver
.
num
=
MAX_NR_CONSOLES
;
console_driver
.
type
=
TTY_DRIVER_TYPE_CONSOLE
;
console_driver
.
init_termios
=
tty_std_termios
;
console_driver
.
flags
=
TTY_DRIVER_REAL_RAW
|
TTY_DRIVER_RESET_TERMIOS
;
/* Tell tty_register_driver() to skip consoles because they are
* registered before kmalloc() is ready. We'll patch them in later.
* See comments at console_init(); see also con_init_devfs().
*/
console_driver
.
flags
|=
TTY_DRIVER_NO_DEVFS
;
console_driver
.
refcount
=
&
console_refcount
;
console_driver
.
table
=
console_table
;
console_driver
.
termios
=
console_termios
;
console_driver
.
termios_locked
=
console_termios_locked
;
console_driver
.
open
=
con_open
;
console_driver
.
close
=
con_close
;
console_driver
.
write
=
con_write
;
console_driver
.
write_room
=
con_write_room
;
console_driver
.
put_char
=
con_put_char
;
console_driver
.
flush_chars
=
con_flush_chars
;
console_driver
.
chars_in_buffer
=
con_chars_in_buffer
;
console_driver
.
ioctl
=
vt_ioctl
;
console_driver
.
stop
=
con_stop
;
console_driver
.
start
=
con_start
;
console_driver
.
throttle
=
con_throttle
;
console_driver
.
unthrottle
=
con_unthrottle
;
if
(
tty_register_driver
(
&
console_driver
))
panic
(
"Couldn't register console driver
\n
"
);
init_timer
(
&
console_timer
);
console_timer
.
function
=
blank_screen
;
if
(
blankinterval
)
{
...
...
@@ -2519,6 +2485,50 @@ void __init con_init(void)
#endif
}
int
__init
vty_init
(
void
)
{
memset
(
&
console_driver
,
0
,
sizeof
(
struct
tty_driver
));
console_driver
.
magic
=
TTY_DRIVER_MAGIC
;
console_driver
.
name
=
"vc/%d"
;
console_driver
.
name_base
=
1
;
console_driver
.
major
=
TTY_MAJOR
;
console_driver
.
minor_start
=
1
;
console_driver
.
num
=
MAX_NR_CONSOLES
;
console_driver
.
type
=
TTY_DRIVER_TYPE_CONSOLE
;
console_driver
.
init_termios
=
tty_std_termios
;
console_driver
.
flags
=
TTY_DRIVER_REAL_RAW
|
TTY_DRIVER_RESET_TERMIOS
;
/* Tell tty_register_driver() to skip consoles because they are
* registered before kmalloc() is ready. We'll patch them in later.
* See comments at console_init(); see also con_init_devfs().
*/
console_driver
.
flags
|=
TTY_DRIVER_NO_DEVFS
;
console_driver
.
refcount
=
&
console_refcount
;
console_driver
.
table
=
console_table
;
console_driver
.
termios
=
console_termios
;
console_driver
.
termios_locked
=
console_termios_locked
;
console_driver
.
open
=
con_open
;
console_driver
.
close
=
con_close
;
console_driver
.
write
=
con_write
;
console_driver
.
write_room
=
con_write_room
;
console_driver
.
put_char
=
con_put_char
;
console_driver
.
flush_chars
=
con_flush_chars
;
console_driver
.
chars_in_buffer
=
con_chars_in_buffer
;
console_driver
.
ioctl
=
vt_ioctl
;
console_driver
.
stop
=
con_stop
;
console_driver
.
start
=
con_start
;
console_driver
.
throttle
=
con_throttle
;
console_driver
.
unthrottle
=
con_unthrottle
;
if
(
tty_register_driver
(
&
console_driver
))
panic
(
"Couldn't register console driver
\n
"
);
kbd_init
();
console_map_init
();
vcs_init
();
return
0
;
}
#ifndef VT_SINGLE_DRIVER
static
void
clear_buffer_attributes
(
int
currcons
)
...
...
drivers/char/keyboard.c
View file @
35c5ca68
...
...
@@ -36,6 +36,7 @@
#include <asm/keyboard.h>
#include <asm/bitops.h>
#include <linux/console_struct.h>
#include <linux/kbd_kern.h>
#include <linux/kbd_diacr.h>
#include <linux/vt_kern.h>
...
...
@@ -43,8 +44,14 @@
#include <linux/sysrq.h>
#include <linux/pm.h>
extern
void
ctrl_alt_del
(
void
);
#define SIZE(x) (sizeof(x)/sizeof((x)[0]))
/*
* Exported functions/variables
*/
#ifndef KBD_DEFMODE
#define KBD_DEFMODE ((1 << VC_REPEAT) | (1 << VC_META))
#endif
...
...
@@ -64,277 +71,121 @@
void
(
*
kbd_ledfunc
)(
unsigned
int
led
);
EXPORT_SYMBOL
(
handle_scancode
);
EXPORT_SYMBOL
(
kbd_ledfunc
);
extern
void
ctrl_alt_del
(
void
);
struct
console
;
/* kbd_pt_regs - set by keyboard_interrupt(), used by show_ptregs() */
struct
pt_regs
*
kbd_pt_regs
;
void
compute_shiftstate
(
void
);
/*
* global state includes the following, and various static variables
* in this module: prev_scancode, shift_state, diacr, npadch, dead_key_next.
* (last_console is now a global variable)
*/
/* shift state counters.. */
static
unsigned
char
k_down
[
NR_SHIFT
];
/* keyboard key bitmap */
static
unsigned
long
key_down
[
256
/
BITS_PER_LONG
];
static
int
dead_key_next
;
/*
* In order to retrieve the shift_state (for the mouse server), either
* the variable must be global, or a new procedure must be created to
* return the value. I chose the former way.
* Handler Tables.
*/
int
shift_state
;
static
int
npadch
=
-
1
;
/* -1 or number assembled on pad */
static
unsigned
char
diacr
;
static
char
rep
;
/* flag telling character repeat */
struct
kbd_struct
kbd_table
[
MAX_NR_CONSOLES
];
static
struct
tty_struct
**
ttytab
;
static
struct
kbd_struct
*
kbd
=
kbd_table
;
static
struct
tty_struct
*
tty
;
void
compute_shiftstate
(
void
);
typedef
void
(
*
k_hand
)(
unsigned
char
value
,
char
up_flag
);
typedef
void
(
k_handfn
)(
unsigned
char
value
,
char
up_flag
);
static
k_handfn
do_self
,
do_fn
,
do_spec
,
do_pad
,
do_dead
,
do_cons
,
do_cur
,
do_shift
,
do_meta
,
do_ascii
,
do_lock
,
do_lowercase
,
do_slock
,
do_dead2
,
do_ignore
;
static
k_hand
key_handler
[
16
]
=
{
do_self
,
do_fn
,
do_spec
,
do_pad
,
do_dead
,
do_cons
,
do_cur
,
do_shift
,
do_meta
,
do_ascii
,
do_lock
,
do_lowercase
,
do_slock
,
do_dead2
,
do_ignore
,
do_ignore
};
/* Key types processed even in raw modes */
#define TYPES_ALLOWED_IN_RAW_MODE ((1 << KT_SPEC) | (1 << KT_SHIFT))
#define SPECIALS_ALLOWED_IN_RAW_MODE (1 << KVAL(K_SAK))
typedef
void
(
*
void_fnp
)(
void
);
typedef
void
(
void_fn
)(
void
);
#define K_HANDLERS\
k_self, k_fn, k_spec, k_pad,\
k_dead, k_cons, k_cur, k_shift,\
k_meta, k_ascii, k_lock, k_lowercase,\
k_slock, k_dead2, k_ignore, k_ignore
static
void_fn
do_null
,
enter
,
show_ptregs
,
send_intr
,
lastcons
,
caps_toggle
,
num
,
hold
,
scroll_forw
,
scroll_back
,
boot_it
,
caps_on
,
compose
,
SAK
,
decr_console
,
incr_console
,
spawn_console
,
bare_num
;
typedef
void
(
k_handler_fn
)(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
);
static
k_handler_fn
K_HANDLERS
;
static
k_handler_fn
*
k_handler
[
16
]
=
{
K_HANDLERS
};
static
void_fnp
spec_fn_table
[]
=
{
do_null
,
enter
,
show_ptregs
,
show_mem
,
show_state
,
send_intr
,
lastcons
,
caps_toggle
,
num
,
hold
,
scroll_forw
,
scroll_back
,
boot_it
,
caps_on
,
compose
,
SAK
,
decr_console
,
incr_console
,
spawn_console
,
bare_num
};
#define FN_HANDLERS\
fn_null, fn_enter, fn_show_ptregs, fn_show_mem,\
fn_show_state, fn_send_intr, fn_lastcons, fn_caps_toggle,\
fn_num, fn_hold, fn_scroll_forw, fn_scroll_back,\
fn_boot_it, fn_caps_on, fn_compose, fn_SAK,\
fn_dec_console, fn_inc_console, fn_spawn_con, fn_bare_num
#define SPECIALS_ALLOWED_IN_RAW_MODE (1 << KVAL(K_SAK))
typedef
void
(
fn_handler_fn
)(
struct
vc_data
*
vc
);
static
fn_handler_fn
FN_HANDLERS
;
static
fn_handler_fn
*
fn_handler
[]
=
{
FN_HANDLERS
};
/*
* Variables/functions exported for vt.c
*/
/* maximum values each key_handler can handle */
const
int
max_vals
[]
=
{
255
,
SIZE
(
func_table
)
-
1
,
SIZE
(
spec_fn_table
)
-
1
,
NR_PAD
-
1
,
NR_DEAD
-
1
,
255
,
3
,
NR_SHIFT
-
1
,
255
,
NR_ASCII
-
1
,
NR_LOCK
-
1
,
255
,
NR_LOCK
-
1
,
255
255
,
SIZE
(
func_table
)
-
1
,
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
};
const
int
NR_TYPES
=
SIZE
(
max_vals
);
/* N.B. drivers/macintosh/mac_keyb.c needs to call put_queue */
void
put_queue
(
int
);
static
unsigned
char
handle_diacr
(
unsigned
char
);
/* kbd_pt_regs - set by keyboard_interrupt(), used by show_ptregs() */
struct
pt_regs
*
kbd_pt_regs
;
#ifdef CONFIG_MAGIC_SYSRQ
static
int
sysrq_pressed
;
#endif
static
struct
pm_dev
*
pm_kbd
;
/*
* Many other routines do put_queue, but I think either
* they produce ASCII, or they produce some user-assigned
* string, and in both cases we might assume that it is
* in utf-8 already.
*/
void
to_utf8
(
ushort
c
)
{
if
(
c
<
0x80
)
put_queue
(
c
);
/* 0******* */
else
if
(
c
<
0x800
)
{
put_queue
(
0xc0
|
(
c
>>
6
));
/* 110***** 10****** */
put_queue
(
0x80
|
(
c
&
0x3f
));
}
else
{
put_queue
(
0xe0
|
(
c
>>
12
));
/* 1110**** 10****** 10****** */
put_queue
(
0x80
|
((
c
>>
6
)
&
0x3f
));
put_queue
(
0x80
|
(
c
&
0x3f
));
}
/* UTF-8 is defined for words of up to 31 bits,
but we need only 16 bits here */
}
int
spawnpid
,
spawnsig
;
/*
* Translation of escaped scancodes to keycodes.
* This is now user-settable (for machines were it makes sense).
*/
int
setkeycode
(
unsigned
int
scancode
,
unsigned
int
keycode
)
{
return
kbd_setkeycode
(
scancode
,
keycode
);
}
int
getkeycode
(
unsigned
int
scancode
)
{
return
kbd_getkeycode
(
scancode
);
}
void
handle_scancode
(
unsigned
char
scancode
,
int
down
)
int
setkeycode
(
unsigned
int
scancode
,
unsigned
int
keycode
)
{
unsigned
char
keycode
;
char
up_flag
=
down
?
0
:
0200
;
char
raw_mode
;
pm_access
(
pm_kbd
);
add_keyboard_randomness
(
scancode
|
up_flag
);
tty
=
ttytab
?
ttytab
[
fg_console
]
:
NULL
;
if
(
tty
&&
(
!
tty
->
driver_data
))
{
/*
* We touch the tty structure via the ttytab array
* without knowing whether or not tty is open, which
* is inherently dangerous. We currently rely on that
* fact that console_open sets tty->driver_data when
* it opens it, and clears it when it closes it.
*/
tty
=
NULL
;
}
kbd
=
kbd_table
+
fg_console
;
if
((
raw_mode
=
(
kbd
->
kbdmode
==
VC_RAW
)))
{
put_queue
(
scancode
|
up_flag
);
/* we do not return yet, because we want to maintain
the key_down array, so that we have the correct
values when finishing RAW mode or when changing VT's */
}
/*
* Convert scancode to keycode
*/
if
(
!
kbd_translate
(
scancode
,
&
keycode
,
raw_mode
))
goto
out
;
/*
* At this point the variable `keycode' contains the keycode.
* Note: the keycode must not be 0 (++Geert: on m68k 0 is valid).
* We keep track of the up/down status of the key, and
* return the keycode if in MEDIUMRAW mode.
*/
if
(
up_flag
)
{
rep
=
0
;
if
(
!
test_and_clear_bit
(
keycode
,
key_down
))
up_flag
=
kbd_unexpected_up
(
keycode
);
}
else
rep
=
test_and_set_bit
(
keycode
,
key_down
);
#ifdef CONFIG_MAGIC_SYSRQ
/* Handle the SysRq Hack */
if
(
keycode
==
SYSRQ_KEY
)
{
sysrq_pressed
=
!
up_flag
;
goto
out
;
}
else
if
(
sysrq_pressed
)
{
if
(
!
up_flag
)
{
handle_sysrq
(
kbd_sysrq_xlate
[
keycode
],
kbd_pt_regs
,
kbd
,
tty
);
goto
out
;
}
}
#endif
return
kbd_setkeycode
(
scancode
,
keycode
);
}
if
(
kbd
->
kbdmode
==
VC_MEDIUMRAW
)
{
/* soon keycodes will require more than one byte */
put_queue
(
keycode
+
up_flag
);
raw_mode
=
1
;
/* Most key classes will be ignored */
}
/*
* Variables/function exported for console.c
*/
int
shift_state
=
0
;
/*
* Small change in philosophy: earlier we defined repetition by
* rep = keycode == prev_keycode;
* prev_keycode = keycode;
* but now by the fact that the depressed key was down already.
* Does this ever make a difference? Yes.
*/
/*
* Internal Data.
*/
/*
* Repeat a key only if the input buffers are empty or the
* characters get echoed locally. This makes key repeat usable
* with slow applications and under heavy loads.
*/
if
(
!
rep
||
(
vc_kbd_mode
(
kbd
,
VC_REPEAT
)
&&
tty
&&
(
L_ECHO
(
tty
)
||
(
tty
->
driver
.
chars_in_buffer
(
tty
)
==
0
))))
{
u_short
keysym
;
u_char
type
;
static
unsigned
long
key_down
[
256
/
BITS_PER_LONG
];
/* keyboard key bitmap */
static
unsigned
char
shift_down
[
NR_SHIFT
];
/* shift state counters.. */
static
int
dead_key_next
;
static
int
npadch
=
-
1
;
/* -1 or number assembled on pad */
static
unsigned
char
diacr
;
static
char
rep
;
/* flag telling character repeat */
pm_callback
pm_kbd_request_override
=
NULL
;
typedef
void
(
pm_kbd_func
)
(
void
);
static
struct
pm_dev
*
pm_kbd
;
/* the XOR below used to be an OR */
int
shift_final
=
(
shift_state
|
kbd
->
slockstate
)
^
kbd
->
lockstate
;
ushort
*
key_map
=
key_maps
[
shift_final
];
static
unsigned
char
ledstate
=
0xff
;
/* undefined */
static
unsigned
char
ledioctl
;
if
(
key_map
!=
NULL
)
{
keysym
=
key_map
[
keycode
];
type
=
KTYP
(
keysym
);
static
struct
ledptr
{
unsigned
int
*
addr
;
unsigned
int
mask
;
unsigned
char
valid
:
1
;
}
ledptrs
[
3
];
if
(
type
>=
0xf0
)
{
type
-=
0xf0
;
if
(
raw_mode
&&
!
(
TYPES_ALLOWED_IN_RAW_MODE
&
(
1
<<
type
)))
goto
out
;
if
(
type
==
KT_LETTER
)
{
type
=
KT_LATIN
;
if
(
vc_kbd_led
(
kbd
,
VC_CAPSLOCK
))
{
key_map
=
key_maps
[
shift_final
^
(
1
<<
KG_SHIFT
)];
if
(
key_map
)
keysym
=
key_map
[
keycode
];
}
}
(
*
key_handler
[
type
])(
keysym
&
0xff
,
up_flag
);
if
(
type
!=
KT_SLOCK
)
kbd
->
slockstate
=
0
;
}
else
{
/* maybe only if (kbd->kbdmode == VC_UNICODE) ? */
if
(
!
up_flag
&&
!
raw_mode
)
to_utf8
(
keysym
);
}
}
else
{
/* maybe beep? */
/* we have at least to update shift_state */
#if 1
/* how? two almost equivalent choices follow */
compute_shiftstate
();
kbd
->
slockstate
=
0
;
/* play it safe */
#else
keysym
=
U
(
plain_map
[
keycode
]);
type
=
KTYP
(
keysym
);
if
(
type
==
KT_SHIFT
)
(
*
key_handler
[
type
])(
keysym
&
0xff
,
up_flag
);
struct
kbd_struct
kbd_table
[
MAX_NR_CONSOLES
];
static
struct
kbd_struct
*
kbd
=
kbd_table
;
#ifdef CONFIG_MAGIC_SYSRQ
static
int
sysrq_pressed
;
#endif
}
}
out:
do_poke_blanked_console
=
1
;
schedule_console_callback
();
}
void
put_queue
(
int
ch
)
/*
* Helper Functions.
*/
void
put_queue
(
struct
vc_data
*
vc
,
int
ch
)
{
struct
tty_struct
*
tty
=
vc
->
vc_tty
;
if
(
tty
)
{
tty_insert_flip_char
(
tty
,
ch
,
0
);
con_schedule_flip
(
tty
);
}
}
static
void
puts_queue
(
char
*
cp
)
static
void
puts_queue
(
struct
vc_data
*
vc
,
char
*
cp
)
{
struct
tty_struct
*
tty
=
vc
->
vc_tty
;
if
(
!
tty
)
return
;
...
...
@@ -345,48 +196,137 @@ static void puts_queue(char *cp)
con_schedule_flip
(
tty
);
}
static
void
applkey
(
int
key
,
char
mode
)
static
void
applkey
(
struct
vc_data
*
vc
,
int
key
,
char
mode
)
{
static
char
buf
[]
=
{
0x1b
,
'O'
,
0x00
,
0x00
};
buf
[
1
]
=
(
mode
?
'O'
:
'['
);
buf
[
2
]
=
key
;
puts_queue
(
buf
);
puts_queue
(
vc
,
buf
);
}
static
void
enter
(
void
)
/*
* Many other routines do put_queue, but I think either
* they produce ASCII, or they produce some user-assigned
* string, and in both cases we might assume that it is
* in utf-8 already. UTF-8 is defined for words of up to 31 bits,
* but we need only 16 bits here
*/
void
to_utf8
(
struct
vc_data
*
vc
,
ushort
c
)
{
if
(
c
<
0x80
)
/* 0******* */
put_queue
(
vc
,
c
);
else
if
(
c
<
0x800
)
{
/* 110***** 10****** */
put_queue
(
vc
,
0xc0
|
(
c
>>
6
));
put_queue
(
vc
,
0x80
|
(
c
&
0x3f
));
}
else
{
/* 1110**** 10****** 10*******/
put_queue
(
vc
,
0xe0
|
(
c
>>
12
));
put_queue
(
vc
,
0x80
|
((
c
>>
6
)
&
0x3f
));
put_queue
(
vc
,
0x80
|
(
c
&
0x3f
));
}
}
/* called after returning from RAW mode or when changing consoles -
recompute shift_down[] and shift_state from key_down[] */
/* maybe called when keymap is undefined, so that shiftkey release is seen */
void
compute_shiftstate
(
void
)
{
int
i
,
j
,
sym
,
val
;
shift_state
=
0
;
memset
(
shift_down
,
0
,
sizeof
(
shift_down
));
for
(
i
=
0
;
i
<
SIZE
(
key_down
);
i
+=
BITS_PER_LONG
)
{
if
(
!
key_down
[
i
/
BITS_PER_LONG
])
continue
;
for
(
j
=
0
;
j
<
BITS_PER_LONG
;
j
++
)
{
if
(
!
test_bit
(
i
+
j
,
key_down
))
continue
;
sym
=
U
(
plain_map
[
i
+
j
]);
if
(
KTYP
(
sym
)
!=
KT_SHIFT
&&
KTYP
(
sym
)
!=
KT_SLOCK
)
continue
;
val
=
KVAL
(
sym
);
if
(
val
==
KVAL
(
K_CAPSSHIFT
))
val
=
KVAL
(
K_SHIFT
);
shift_down
[
val
]
++
;
shift_state
|=
(
1
<<
val
);
}
}
}
/*
* We have a combining character DIACR here, followed by the character CH.
* If the combination occurs in the table, return the corresponding value.
* Otherwise, if CH is a space or equals DIACR, return DIACR.
* Otherwise, conclude that DIACR was not combining after all,
* queue it and return CH.
*/
unsigned
char
handle_diacr
(
struct
vc_data
*
vc
,
unsigned
char
ch
)
{
int
d
=
diacr
;
int
i
;
diacr
=
0
;
for
(
i
=
0
;
i
<
accent_table_size
;
i
++
)
{
if
(
accent_table
[
i
].
diacr
==
d
&&
accent_table
[
i
].
base
==
ch
)
return
accent_table
[
i
].
result
;
}
if
(
ch
==
' '
||
ch
==
d
)
return
d
;
put_queue
(
vc
,
d
);
return
ch
;
}
/*
* Special function handlers
*/
static
void
fn_enter
(
struct
vc_data
*
vc
)
{
if
(
diacr
)
{
put_queue
(
diacr
);
put_queue
(
vc
,
diacr
);
diacr
=
0
;
}
put_queue
(
13
);
if
(
vc_kbd_mode
(
kbd
,
VC_CRLF
))
put_queue
(
10
);
put_queue
(
vc
,
13
);
if
(
vc_kbd_mode
(
kbd
,
VC_CRLF
))
put_queue
(
vc
,
10
);
}
static
void
caps_toggle
(
void
)
static
void
fn_caps_toggle
(
struct
vc_data
*
vc
)
{
if
(
rep
)
return
;
chg_vc_kbd_led
(
kbd
,
VC_CAPSLOCK
);
}
static
void
caps_on
(
void
)
static
void
fn_caps_on
(
struct
vc_data
*
vc
)
{
if
(
rep
)
return
;
set_vc_kbd_led
(
kbd
,
VC_CAPSLOCK
);
}
static
void
show_ptregs
(
void
)
static
void
fn_show_ptregs
(
struct
vc_data
*
vc
)
{
if
(
kbd_pt_regs
)
show_regs
(
kbd_pt_regs
);
}
static
void
hold
(
void
)
static
void
fn_hold
(
struct
vc_data
*
vc
)
{
struct
tty_struct
*
tty
=
vc
->
vc_tty
;
if
(
rep
||
!
tty
)
return
;
...
...
@@ -401,12 +341,12 @@ static void hold(void)
stop_tty
(
tty
);
}
static
void
num
(
void
)
static
void
fn_num
(
struct
vc_data
*
vc
)
{
if
(
vc_kbd_mode
(
kbd
,
VC_APPLIC
))
applkey
(
'P'
,
1
);
applkey
(
vc
,
'P'
,
1
);
else
bare_num
(
);
fn_bare_num
(
vc
);
}
/*
...
...
@@ -415,19 +355,19 @@ static void num(void)
* Bind this to NumLock if you prefer that the NumLock key always
* changes the NumLock flag.
*/
static
void
bare_num
(
void
)
static
void
fn_bare_num
(
struct
vc_data
*
vc
)
{
if
(
!
rep
)
chg_vc_kbd_led
(
kbd
,
VC_NUMLOCK
);
}
static
void
lastcons
(
void
)
static
void
fn_lastcons
(
struct
vc_data
*
vc
)
{
/* switch to the last used console, ChN */
set_console
(
last_console
);
}
static
void
decr_console
(
void
)
static
void
fn_dec_console
(
struct
vc_data
*
vc
)
{
int
i
;
...
...
@@ -440,7 +380,7 @@ static void decr_console(void)
set_console
(
i
);
}
static
void
incr_console
(
void
)
static
void
fn_inc_console
(
struct
vc_data
*
vc
)
{
int
i
;
...
...
@@ -453,98 +393,112 @@ static void incr_console(void)
set_console
(
i
);
}
static
void
send_intr
(
void
)
static
void
fn_send_intr
(
struct
vc_data
*
vc
)
{
struct
tty_struct
*
tty
=
vc
->
vc_tty
;
if
(
!
tty
)
return
;
tty_insert_flip_char
(
tty
,
0
,
TTY_BREAK
);
con_schedule_flip
(
tty
);
}
static
void
scroll_forw
(
void
)
static
void
fn_scroll_forw
(
struct
vc_data
*
vc
)
{
scrollfront
(
0
);
}
static
void
scroll_back
(
void
)
static
void
fn_scroll_back
(
struct
vc_data
*
vc
)
{
scrollback
(
0
);
}
static
void
boot_it
(
void
)
static
void
fn_show_mem
(
struct
vc_data
*
vc
)
{
show_mem
();
}
static
void
fn_show_state
(
struct
vc_data
*
vc
)
{
show_state
();
}
static
void
fn_boot_it
(
struct
vc_data
*
vc
)
{
ctrl_alt_del
();
}
static
void
compose
(
void
)
static
void
fn_compose
(
struct
vc_data
*
vc
)
{
dead_key_next
=
1
;
}
int
spawnpid
,
spawnsig
;
static
void
spawn_console
(
void
)
static
void
fn_spawn_con
(
struct
vc_data
*
vc
)
{
if
(
spawnpid
)
if
(
kill_proc
(
spawnpid
,
spawnsig
,
1
))
spawnpid
=
0
;
}
static
void
SAK
(
void
)
static
void
fn_SAK
(
struct
vc_data
*
vc
)
{
struct
tty_struct
*
tty
=
vc
->
vc_tty
;
/*
* SAK should also work in all raw modes and reset
* them properly.
*/
do_SAK
(
tty
);
if
(
tty
)
do_SAK
(
tty
);
reset_vc
(
fg_console
);
#if 0
do_unblank_screen(); /* not in interrupt routine? */
#endif
}
static
void
do_ignore
(
unsigned
char
value
,
char
up_flag
)
static
void
fn_null
(
struct
vc_data
*
vc
)
{
compute_shiftstate
();
}
static
void
do_null
()
/*
* Special key handlers
*/
static
void
k_ignore
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
compute_shiftstate
();
}
static
void
do_spec
(
unsigned
char
value
,
char
up_flag
)
static
void
k_spec
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
if
(
value
>=
SIZE
(
spec_fn_table
))
if
(
value
>=
SIZE
(
fn_handler
))
return
;
if
((
kbd
->
kbdmode
==
VC_RAW
||
kbd
->
kbdmode
==
VC_MEDIUMRAW
)
&&
!
(
SPECIALS_ALLOWED_IN_RAW_MODE
&
(
1
<<
value
)))
return
;
spec_fn_table
[
value
](
);
fn_handler
[
value
](
vc
);
}
static
void
do_lowercase
(
unsigned
char
value
,
char
up_flag
)
static
void
k_lowercase
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
printk
(
KERN_ERR
"keyboard.c: do_lowercase was called - impossible
\n
"
);
}
static
void
do_self
(
unsigned
char
value
,
char
up_flag
)
static
void
k_self
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
/* no action, if this is a key release */
if
(
diacr
)
value
=
handle_diacr
(
value
);
value
=
handle_diacr
(
v
c
,
v
alue
);
if
(
dead_key_next
)
{
dead_key_next
=
0
;
diacr
=
value
;
return
;
}
put_queue
(
value
);
put_queue
(
vc
,
value
);
}
#define A_GRAVE '`'
...
...
@@ -557,10 +511,10 @@ static unsigned char ret_diacr[NR_DEAD] =
{
A_GRAVE
,
A_ACUTE
,
A_CFLEX
,
A_TILDE
,
A_DIAER
,
A_CEDIL
};
/* Obsolete - for backwards compatibility only */
static
void
do_dead
(
unsigned
char
value
,
char
up_flag
)
static
void
k_dead
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
value
=
ret_diacr
[
value
];
do_dead2
(
value
,
up_flag
);
k_dead2
(
vc
,
value
,
up_flag
);
}
/*
...
...
@@ -568,60 +522,33 @@ static void do_dead(unsigned char value, char up_flag)
* dead keys modifying the same character. Very useful
* for Vietnamese.
*/
static
void
do_dead2
(
unsigned
char
value
,
char
up_flag
)
static
void
k_dead2
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
diacr
=
(
diacr
?
handle_diacr
(
value
)
:
value
);
}
/*
* We have a combining character DIACR here, followed by the character CH.
* If the combination occurs in the table, return the corresponding value.
* Otherwise, if CH is a space or equals DIACR, return DIACR.
* Otherwise, conclude that DIACR was not combining after all,
* queue it and return CH.
*/
unsigned
char
handle_diacr
(
unsigned
char
ch
)
{
int
d
=
diacr
;
int
i
;
diacr
=
0
;
for
(
i
=
0
;
i
<
accent_table_size
;
i
++
)
{
if
(
accent_table
[
i
].
diacr
==
d
&&
accent_table
[
i
].
base
==
ch
)
return
accent_table
[
i
].
result
;
}
if
(
ch
==
' '
||
ch
==
d
)
return
d
;
put_queue
(
d
);
return
ch
;
diacr
=
(
diacr
?
handle_diacr
(
vc
,
value
)
:
value
);
}
static
void
do_cons
(
unsigned
char
value
,
char
up_flag
)
static
void
k_cons
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
set_console
(
value
);
}
static
void
do_fn
(
unsigned
char
value
,
char
up_flag
)
static
void
k_fn
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
if
(
value
<
SIZE
(
func_table
))
{
if
(
func_table
[
value
])
puts_queue
(
func_table
[
value
]);
puts_queue
(
vc
,
func_table
[
value
]);
}
else
printk
(
KERN_ERR
"
do
_fn called with value=%d
\n
"
,
value
);
printk
(
KERN_ERR
"
k
_fn called with value=%d
\n
"
,
value
);
}
static
void
do_pad
(
unsigned
char
value
,
char
up_flag
)
static
void
k_pad
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
static
const
char
*
pad_chars
=
"0123456789+-*/
\015
,.?()"
;
static
const
char
*
app_map
=
"pqrstuvwxylSRQMnnmPQ"
;
...
...
@@ -630,8 +557,8 @@ static void do_pad(unsigned char value, char up_flag)
return
;
/* no action, if this is a key release */
/* kludge... shift forces cursor/number keys */
if
(
vc_kbd_mode
(
kbd
,
VC_APPLIC
)
&&
!
k
_down
[
KG_SHIFT
])
{
applkey
(
app_map
[
value
],
1
);
if
(
vc_kbd_mode
(
kbd
,
VC_APPLIC
)
&&
!
shift
_down
[
KG_SHIFT
])
{
applkey
(
vc
,
app_map
[
value
],
1
);
return
;
}
...
...
@@ -639,55 +566,55 @@ static void do_pad(unsigned char value, char up_flag)
switch
(
value
)
{
case
KVAL
(
K_PCOMMA
):
case
KVAL
(
K_PDOT
):
do_fn
(
KVAL
(
K_REMOVE
),
0
);
k_fn
(
vc
,
KVAL
(
K_REMOVE
),
0
);
return
;
case
KVAL
(
K_P0
):
do_fn
(
KVAL
(
K_INSERT
),
0
);
k_fn
(
vc
,
KVAL
(
K_INSERT
),
0
);
return
;
case
KVAL
(
K_P1
):
do_fn
(
KVAL
(
K_SELECT
),
0
);
k_fn
(
vc
,
KVAL
(
K_SELECT
),
0
);
return
;
case
KVAL
(
K_P2
):
do_cur
(
KVAL
(
K_DOWN
),
0
);
k_cur
(
vc
,
KVAL
(
K_DOWN
),
0
);
return
;
case
KVAL
(
K_P3
):
do_fn
(
KVAL
(
K_PGDN
),
0
);
k_fn
(
vc
,
KVAL
(
K_PGDN
),
0
);
return
;
case
KVAL
(
K_P4
):
do_cur
(
KVAL
(
K_LEFT
),
0
);
k_cur
(
vc
,
KVAL
(
K_LEFT
),
0
);
return
;
case
KVAL
(
K_P6
):
do_cur
(
KVAL
(
K_RIGHT
),
0
);
k_cur
(
vc
,
KVAL
(
K_RIGHT
),
0
);
return
;
case
KVAL
(
K_P7
):
do_fn
(
KVAL
(
K_FIND
),
0
);
k_fn
(
vc
,
KVAL
(
K_FIND
),
0
);
return
;
case
KVAL
(
K_P8
):
do_cur
(
KVAL
(
K_UP
),
0
);
k_cur
(
vc
,
KVAL
(
K_UP
),
0
);
return
;
case
KVAL
(
K_P9
):
do_fn
(
KVAL
(
K_PGUP
),
0
);
k_fn
(
vc
,
KVAL
(
K_PGUP
),
0
);
return
;
case
KVAL
(
K_P5
):
applkey
(
'G'
,
vc_kbd_mode
(
kbd
,
VC_APPLIC
));
applkey
(
vc
,
'G'
,
vc_kbd_mode
(
kbd
,
VC_APPLIC
));
return
;
}
put_queue
(
pad_chars
[
value
]);
put_queue
(
vc
,
pad_chars
[
value
]);
if
(
value
==
KVAL
(
K_PENTER
)
&&
vc_kbd_mode
(
kbd
,
VC_CRLF
))
put_queue
(
10
);
put_queue
(
vc
,
10
);
}
static
void
do_cur
(
unsigned
char
value
,
char
up_flag
)
static
void
k_cur
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
static
const
char
*
cur_chars
=
"BDCA"
;
if
(
up_flag
)
return
;
applkey
(
cur_chars
[
value
],
vc_kbd_mode
(
kbd
,
VC_CKMODE
));
applkey
(
vc
,
cur_chars
[
value
],
vc_kbd_mode
(
kbd
,
VC_CKMODE
));
}
static
void
do_shift
(
unsigned
char
value
,
char
up_flag
)
static
void
k_shift
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
int
old_state
=
shift_state
;
...
...
@@ -705,12 +632,12 @@ static void do_shift(unsigned char value, char up_flag)
if
(
up_flag
)
{
/* handle the case that two shift or control
keys are depressed simultaneously */
if
(
k
_down
[
value
])
k
_down
[
value
]
--
;
if
(
shift
_down
[
value
])
shift
_down
[
value
]
--
;
}
else
k
_down
[
value
]
++
;
shift
_down
[
value
]
++
;
if
(
k
_down
[
value
])
if
(
shift
_down
[
value
])
shift_state
|=
(
1
<<
value
);
else
shift_state
&=
~
(
1
<<
value
);
...
...
@@ -718,54 +645,26 @@ static void do_shift(unsigned char value, char up_flag)
/* kludge */
if
(
up_flag
&&
shift_state
!=
old_state
&&
npadch
!=
-
1
)
{
if
(
kbd
->
kbdmode
==
VC_UNICODE
)
to_utf8
(
npadch
&
0xffff
);
to_utf8
(
vc
,
npadch
&
0xffff
);
else
put_queue
(
npadch
&
0xff
);
put_queue
(
vc
,
npadch
&
0xff
);
npadch
=
-
1
;
}
}
/* called after returning from RAW mode or when changing consoles -
recompute k_down[] and shift_state from key_down[] */
/* maybe called when keymap is undefined, so that shiftkey release is seen */
void
compute_shiftstate
(
void
)
{
int
i
,
j
,
k
,
sym
,
val
;
shift_state
=
0
;
for
(
i
=
0
;
i
<
SIZE
(
k_down
);
i
++
)
k_down
[
i
]
=
0
;
for
(
i
=
0
;
i
<
SIZE
(
key_down
);
i
++
)
if
(
key_down
[
i
])
{
/* skip this word if not a single bit on */
k
=
i
*
BITS_PER_LONG
;
for
(
j
=
0
;
j
<
BITS_PER_LONG
;
j
++
,
k
++
)
if
(
test_bit
(
k
,
key_down
))
{
sym
=
U
(
plain_map
[
k
]);
if
(
KTYP
(
sym
)
==
KT_SHIFT
||
KTYP
(
sym
)
==
KT_SLOCK
)
{
val
=
KVAL
(
sym
);
if
(
val
==
KVAL
(
K_CAPSSHIFT
))
val
=
KVAL
(
K_SHIFT
);
k_down
[
val
]
++
;
shift_state
|=
(
1
<<
val
);
}
}
}
}
static
void
do_meta
(
unsigned
char
value
,
char
up_flag
)
static
void
k_meta
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
)
return
;
if
(
vc_kbd_mode
(
kbd
,
VC_META
))
{
put_queue
(
'\033'
);
put_queue
(
value
);
put_queue
(
vc
,
'\033'
);
put_queue
(
v
c
,
v
alue
);
}
else
put_queue
(
value
|
0x80
);
put_queue
(
v
c
,
v
alue
|
0x80
);
}
static
void
do_ascii
(
unsigned
char
value
,
char
up_flag
)
static
void
k_ascii
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
int
base
;
...
...
@@ -785,16 +684,16 @@ static void do_ascii(unsigned char value, char up_flag)
npadch
=
npadch
*
base
+
value
;
}
static
void
do_lock
(
unsigned
char
value
,
char
up_flag
)
static
void
k_lock
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
if
(
up_flag
||
rep
)
return
;
chg_vc_kbd_lock
(
kbd
,
value
);
}
static
void
do_slock
(
unsigned
char
value
,
char
up_flag
)
static
void
k_slock
(
struct
vc_data
*
vc
,
unsigned
char
value
,
char
up_flag
)
{
do_shift
(
value
,
up_flag
);
k_shift
(
vc
,
value
,
up_flag
);
if
(
up_flag
||
rep
)
return
;
chg_vc_kbd_slock
(
kbd
,
value
);
...
...
@@ -810,10 +709,6 @@ static void do_slock(unsigned char value, char up_flag)
* or (ii) whatever pattern of lights people want to show using KDSETLED,
* or (iii) specified bits of specified words in kernel memory.
*/
static
unsigned
char
ledstate
=
0xff
;
/* undefined */
static
unsigned
char
ledioctl
;
unsigned
char
getledstate
(
void
)
{
return
ledstate
;
}
...
...
@@ -827,12 +722,6 @@ void setledstate(struct kbd_struct *kbd, unsigned int led) {
set_leds
();
}
static
struct
ledptr
{
unsigned
int
*
addr
;
unsigned
int
mask
;
unsigned
char
valid
:
1
;
}
ledptrs
[
3
];
void
register_leds
(
int
console
,
unsigned
int
led
,
unsigned
int
*
addr
,
unsigned
int
mask
)
{
struct
kbd_struct
*
kbd
=
kbd_table
+
console
;
...
...
@@ -888,6 +777,7 @@ static inline unsigned char getleds(void){
* used, but this allows for easy and efficient race-condition
* prevention later on.
*/
static
void
kbd_bh
(
unsigned
long
dummy
)
{
unsigned
char
leds
=
getleds
();
...
...
@@ -902,15 +792,146 @@ static void kbd_bh(unsigned long dummy)
EXPORT_SYMBOL
(
keyboard_tasklet
);
DECLARE_TASKLET_DISABLED
(
keyboard_tasklet
,
kbd_bh
,
0
);
typedef
void
(
pm_kbd_func
)
(
void
);
void
handle_scancode
(
unsigned
char
scancode
,
int
down
)
{
struct
vc_data
*
vc
=
vc_cons
[
fg_console
].
d
;
char
up_flag
=
down
?
0
:
0200
;
struct
tty_struct
*
tty
;
unsigned
char
keycode
;
char
raw_mode
;
pm_callback
pm_kbd_request_override
=
NULL
;
pm_access
(
pm_kbd
);
add_keyboard_randomness
(
scancode
|
up_flag
);
tty
=
vc
->
vc_tty
;
if
(
tty
&&
(
!
tty
->
driver_data
))
{
/*
* We touch the tty structure via the ttytab array
* without knowing whether or not tty is open, which
* is inherently dangerous. We currently rely on that
* fact that console_open sets tty->driver_data when
* it opens it, and clears it when it closes it.
*/
tty
=
NULL
;
}
kbd
=
kbd_table
+
fg_console
;
if
((
raw_mode
=
(
kbd
->
kbdmode
==
VC_RAW
)))
{
put_queue
(
vc
,
scancode
|
up_flag
);
/* we do not return yet, because we want to maintain
the key_down array, so that we have the correct
values when finishing RAW mode or when changing VT's */
}
/*
* Convert scancode to keycode
*/
if
(
!
kbd_translate
(
scancode
,
&
keycode
,
raw_mode
))
goto
out
;
/*
* At this point the variable `keycode' contains the keycode.
* Note: the keycode must not be 0 (++Geert: on m68k 0 is valid).
* We keep track of the up/down status of the key, and
* return the keycode if in MEDIUMRAW mode.
*/
if
(
up_flag
)
{
rep
=
0
;
if
(
!
test_and_clear_bit
(
keycode
,
key_down
))
up_flag
=
kbd_unexpected_up
(
keycode
);
}
else
rep
=
test_and_set_bit
(
keycode
,
key_down
);
#ifdef CONFIG_MAGIC_SYSRQ
/* Handle the SysRq Hack */
if
(
keycode
==
SYSRQ_KEY
)
{
sysrq_pressed
=
!
up_flag
;
goto
out
;
}
else
if
(
sysrq_pressed
)
{
if
(
!
up_flag
)
{
handle_sysrq
(
kbd_sysrq_xlate
[
keycode
],
kbd_pt_regs
,
tty
);
goto
out
;
}
}
#endif
if
(
kbd
->
kbdmode
==
VC_MEDIUMRAW
)
{
/* soon keycodes will require more than one byte */
put_queue
(
vc
,
keycode
+
up_flag
);
raw_mode
=
1
;
/* Most key classes will be ignored */
}
/*
* Small change in philosophy: earlier we defined repetition by
* rep = keycode == prev_keycode;
* prev_keycode = keycode;
* but now by the fact that the depressed key was down already.
* Does this ever make a difference? Yes.
*/
/*
* Repeat a key only if the input buffers are empty or the
* characters get echoed locally. This makes key repeat usable
* with slow applications and under heavy loads.
*/
if
(
!
rep
||
(
vc_kbd_mode
(
kbd
,
VC_REPEAT
)
&&
tty
&&
(
L_ECHO
(
tty
)
||
(
tty
->
driver
.
chars_in_buffer
(
tty
)
==
0
))))
{
u_short
keysym
;
u_char
type
;
/* the XOR below used to be an OR */
int
shift_final
=
(
shift_state
|
kbd
->
slockstate
)
^
kbd
->
lockstate
;
ushort
*
key_map
=
key_maps
[
shift_final
];
if
(
key_map
!=
NULL
)
{
keysym
=
key_map
[
keycode
];
type
=
KTYP
(
keysym
);
if
(
type
>=
0xf0
)
{
type
-=
0xf0
;
if
(
raw_mode
&&
!
(
TYPES_ALLOWED_IN_RAW_MODE
&
(
1
<<
type
)))
goto
out
;
if
(
type
==
KT_LETTER
)
{
type
=
KT_LATIN
;
if
(
vc_kbd_led
(
kbd
,
VC_CAPSLOCK
))
{
key_map
=
key_maps
[
shift_final
^
(
1
<<
KG_SHIFT
)];
if
(
key_map
)
keysym
=
key_map
[
keycode
];
}
}
(
*
k_handler
[
type
])(
vc
,
keysym
&
0xff
,
up_flag
);
if
(
type
!=
KT_SLOCK
)
kbd
->
slockstate
=
0
;
}
else
{
/* maybe only if (kbd->kbdmode == VC_UNICODE) ? */
if
(
!
up_flag
&&
!
raw_mode
)
to_utf8
(
vc
,
keysym
);
}
}
else
{
/* maybe beep? */
/* we have at least to update shift_state */
#if 1
/* how? two almost equivalent choices follow */
compute_shiftstate
();
kbd
->
slockstate
=
0
;
/* play it safe */
#else
keysym
=
U
(
plain_map
[
keycode
]);
type
=
KTYP
(
keysym
);
if
(
type
==
KT_SHIFT
)
(
*
key_handler
[
type
])(
keysym
&
0xff
,
up_flag
);
#endif
}
}
out:
do_poke_blanked_console
=
1
;
schedule_console_callback
();
}
int
__init
kbd_init
(
void
)
{
int
i
;
struct
kbd_struct
kbd0
;
extern
struct
tty_driver
console_driver
;
kbd0
.
ledflagstate
=
kbd0
.
default_ledflagstate
=
KBD_DEFLEDS
;
kbd0
.
ledmode
=
LED_SHOW_FLAGS
;
...
...
@@ -922,8 +943,6 @@ int __init kbd_init(void)
for
(
i
=
0
;
i
<
MAX_NR_CONSOLES
;
i
++
)
kbd_table
[
i
]
=
kbd0
;
ttytab
=
console_driver
.
table
;
kbd_init_hw
();
tasklet_enable
(
&
keyboard_tasklet
);
...
...
drivers/char/sysrq.c
View file @
35c5ca68
...
...
@@ -45,7 +45,8 @@ void (*sysrq_power_off)(void);
/* Loglevel sysrq handler */
static
void
sysrq_handle_loglevel
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
int
i
;
i
=
key
-
'0'
;
console_loglevel
=
7
;
...
...
@@ -62,7 +63,8 @@ static struct sysrq_key_op sysrq_loglevel_op = {
/* SAK sysrq handler */
#ifdef CONFIG_VT
static
void
sysrq_handle_SAK
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
if
(
tty
)
do_SAK
(
tty
);
reset_vc
(
fg_console
);
...
...
@@ -77,7 +79,10 @@ static struct sysrq_key_op sysrq_SAK_op = {
/* unraw sysrq handler */
static
void
sysrq_handle_unraw
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
struct
kbd_struct
*
kbd
=
&
kbd_table
[
fg_console
];
if
(
kbd
)
kbd
->
kbdmode
=
VC_XLATE
;
}
...
...
@@ -90,7 +95,8 @@ static struct sysrq_key_op sysrq_unraw_op = {
/* reboot sysrq handler */
static
void
sysrq_handle_reboot
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
machine_restart
(
NULL
);
}
static
struct
sysrq_key_op
sysrq_reboot_op
=
{
...
...
@@ -218,7 +224,8 @@ void do_emergency_sync(void) {
}
static
void
sysrq_handle_sync
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
emergency_sync_scheduled
=
EMERG_SYNC
;
wakeup_bdflush
();
}
...
...
@@ -229,7 +236,8 @@ static struct sysrq_key_op sysrq_sync_op = {
};
static
void
sysrq_handle_mountro
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
emergency_sync_scheduled
=
EMERG_REMOUNT
;
wakeup_bdflush
();
}
...
...
@@ -245,7 +253,8 @@ static struct sysrq_key_op sysrq_mountro_op = {
/* SHOW SYSRQ HANDLERS BLOCK */
static
void
sysrq_handle_showregs
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
if
(
pt_regs
)
show_regs
(
pt_regs
);
}
...
...
@@ -257,7 +266,8 @@ static struct sysrq_key_op sysrq_showregs_op = {
static
void
sysrq_handle_showstate
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
show_state
();
}
static
struct
sysrq_key_op
sysrq_showstate_op
=
{
...
...
@@ -268,7 +278,8 @@ static struct sysrq_key_op sysrq_showstate_op = {
static
void
sysrq_handle_showmem
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
show_mem
();
}
static
struct
sysrq_key_op
sysrq_showmem_op
=
{
...
...
@@ -296,7 +307,8 @@ static void send_sig_all(int sig)
}
static
void
sysrq_handle_term
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
send_sig_all
(
SIGTERM
);
console_loglevel
=
8
;
}
...
...
@@ -307,7 +319,8 @@ static struct sysrq_key_op sysrq_term_op = {
};
static
void
sysrq_handle_kill
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
send_sig_all
(
SIGKILL
);
console_loglevel
=
8
;
}
...
...
@@ -416,13 +429,13 @@ void __sysrq_put_key_op (int key, struct sysrq_key_op *op_p) {
* and any other keycode arrives.
*/
void
handle_sysrq
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
void
handle_sysrq
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
tty_struct
*
tty
)
{
if
(
!
sysrq_enabled
)
return
;
__sysrq_lock_table
();
__handle_sysrq_nolock
(
key
,
pt_regs
,
kbd
,
tty
);
__handle_sysrq_nolock
(
key
,
pt_regs
,
tty
);
__sysrq_unlock_table
();
}
...
...
@@ -433,7 +446,8 @@ void handle_sysrq(int key, struct pt_regs *pt_regs,
*/
void
__handle_sysrq_nolock
(
int
key
,
struct
pt_regs
*
pt_regs
,
struct
kbd_struct
*
kbd
,
struct
tty_struct
*
tty
)
{
struct
tty_struct
*
tty
)
{
struct
sysrq_key_op
*
op_p
;
int
orig_log_level
;
int
i
,
j
;
...
...
@@ -449,7 +463,7 @@ void __handle_sysrq_nolock(int key, struct pt_regs *pt_regs,
if
(
op_p
)
{
printk
(
"%s
\n
"
,
op_p
->
action_msg
);
console_loglevel
=
orig_log_level
;
op_p
->
handler
(
key
,
pt_regs
,
kbd
,
tty
);
op_p
->
handler
(
key
,
pt_regs
,
tty
);
}
else
{
printk
(
"HELP : "
);
/* Only print the help msg once per handler */
...
...
drivers/char/tty_io.c
View file @
35c5ca68
...
...
@@ -2245,9 +2245,8 @@ static struct tty_driver dev_tty_driver, dev_syscons_driver;
static
struct
tty_driver
dev_ptmx_driver
;
#endif
#ifdef CONFIG_VT
extern
void
con_init_devfs
(
void
);
extern
void
console_map_init
(
void
);
static
struct
tty_driver
dev_console_driver
;
extern
int
vty_init
(
void
);
#endif
/*
...
...
@@ -2287,13 +2286,6 @@ void __init tty_init(void)
if
(
tty_register_driver
(
&
dev_syscons_driver
))
panic
(
"Couldn't register /dev/console driver
\n
"
);
/* console calls tty_register_driver() before kmalloc() works.
* Thus, we can't devfs_register() then. Do so now, instead.
*/
#ifdef CONFIG_VT
con_init_devfs
();
#endif
#ifdef CONFIG_UNIX98_PTYS
dev_ptmx_driver
=
dev_tty_driver
;
dev_ptmx_driver
.
driver_name
=
"/dev/ptmx"
;
...
...
@@ -2317,10 +2309,7 @@ void __init tty_init(void)
if
(
tty_register_driver
(
&
dev_console_driver
))
panic
(
"Couldn't register /dev/tty0 driver
\n
"
);
vcs_init
();
kbd_init
();
console_map_init
();
vty_init
();
#endif
#ifdef CONFIG_ESPSERIAL
/* init ESP before rs, so rs doesn't see the port */
...
...
include/linux/console_struct.h
View file @
35c5ca68
...
...
@@ -33,6 +33,7 @@ struct vc_data {
unsigned
int
vc_top
,
vc_bottom
;
/* Scrolling region */
unsigned
int
vc_state
;
/* Escape sequence parser state */
unsigned
int
vc_npar
,
vc_par
[
NPAR
];
/* Parameters of current escape sequence */
struct
tty_struct
*
vc_tty
;
/* TTY we are attached to */
unsigned
long
vc_origin
;
/* [!] Start of real screen */
unsigned
long
vc_scr_end
;
/* [!] End of real screen */
unsigned
long
vc_visible_origin
;
/* [!] Top of visible window */
...
...
include/linux/sysrq.h
View file @
35c5ca68
...
...
@@ -14,12 +14,10 @@
#include <linux/config.h>
struct
pt_regs
;
struct
kbd_struct
;
struct
tty_struct
;
struct
sysrq_key_op
{
void
(
*
handler
)(
int
,
struct
pt_regs
*
,
struct
kbd_struct
*
,
struct
tty_struct
*
);
void
(
*
handler
)(
int
,
struct
pt_regs
*
,
struct
tty_struct
*
);
char
*
help_msg
;
char
*
action_msg
;
};
...
...
@@ -31,19 +29,14 @@ struct sysrq_key_op {
* are available -- else NULL's).
*/
void
handle_sysrq
(
int
,
struct
pt_regs
*
,
struct
kbd_struct
*
,
struct
tty_struct
*
);
void
handle_sysrq
(
int
,
struct
pt_regs
*
,
struct
tty_struct
*
);
/*
* Nonlocking version of handle sysrq, used by sysrq handlers that need to
* call sysrq handlers
*/
void
__handle_sysrq_nolock
(
int
,
struct
pt_regs
*
,
struct
kbd_struct
*
,
struct
tty_struct
*
);
void
__handle_sysrq_nolock
(
int
,
struct
pt_regs
*
,
struct
tty_struct
*
);
/*
* Sysrq registration manipulation functions
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment