2 * Minimalistic braille device kernel support.
4 * By default, shows console messages on the braille device.
5 * Pressing Insert switches to VC browsing.
7 * Copyright (C) Samuel Thibault <samuel.thibault@ens-lyon.org>
9 * This program is free software ; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation ; either version 2 of the License, or
12 * (at your option) any later version.
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY ; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with the program ; if not, write to the Free Software
21 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include <linux/autoconf.h>
25 #include <linux/kernel.h>
26 #include <linux/module.h>
27 #include <linux/moduleparam.h>
28 #include <linux/console.h>
29 #include <linux/notifier.h>
31 #include <linux/selection.h>
32 #include <linux/vt_kern.h>
33 #include <linux/consolemap.h>
35 #include <linux/keyboard.h>
36 #include <linux/kbd_kern.h>
37 #include <linux/input.h>
39 MODULE_AUTHOR("samuel.thibault@ens-lyon.org");
40 MODULE_DESCRIPTION("braille device");
41 MODULE_LICENSE("GPL");
44 * Braille device support part.
47 /* Emit various sounds */
49 module_param(sound
, bool, 0);
50 MODULE_PARM_DESC(sound
, "emit sounds");
52 static void beep(unsigned int freq
)
55 kd_mksound(freq
, HZ
/10);
60 #define BRAILLE_KEY KEY_INSERT
61 static u16 console_buf
[WIDTH
];
62 static int console_cursor
;
65 static int vc_x
, vc_y
, lastvc_x
, lastvc_y
;
67 /* show console ? (or show VC) */
68 static int console_show
= 1;
69 /* pending newline ? */
70 static int console_newline
= 1;
71 static int lastVC
= -1;
73 static struct console
*braille_co
;
75 /* Very VisioBraille-specific */
76 static void braille_write(u16
*buf
)
78 static u16 lastwrite
[WIDTH
];
79 unsigned char data
[1 + 1 + 2*WIDTH
+ 2 + 1], csum
= 0, *c
;
86 if (!memcmp(lastwrite
, buf
, WIDTH
* sizeof(*buf
)))
88 memcpy(lastwrite
, buf
, WIDTH
* sizeof(*buf
));
99 for (i
= 0; i
< WIDTH
; i
++) {
103 else if (out
== 0x00)
120 braille_co
->write(braille_co
, data
, c
- data
);
123 /* Follow the VC cursor*/
124 static void vc_follow_cursor(struct vc_data
*vc
)
126 vc_x
= vc
->vc_x
- (vc
->vc_x
% WIDTH
);
132 /* Maybe the VC cursor moved, if so follow it */
133 static void vc_maybe_cursor_moved(struct vc_data
*vc
)
135 if (vc
->vc_x
!= lastvc_x
|| vc
->vc_y
!= lastvc_y
)
136 vc_follow_cursor(vc
);
139 /* Show portion of VC at vc_x, vc_y */
140 static void vc_refresh(struct vc_data
*vc
)
145 for (i
= 0; i
< WIDTH
; i
++) {
146 u16 glyph
= screen_glyph(vc
,
147 2 * (vc_x
+ i
) + vc_y
* vc
->vc_size_row
);
148 buf
[i
] = inverse_translate(vc
, glyph
, 1);
157 static int keyboard_notifier_call(struct notifier_block
*blk
,
158 unsigned long code
, void *_param
)
160 struct keyboard_notifier_param
*param
= _param
;
161 struct vc_data
*vc
= param
->vc
;
170 if (param
->value
== BRAILLE_KEY
) {
173 vc_maybe_cursor_moved(vc
);
179 switch (param
->value
) {
184 braille_write(console_buf
);
191 } else if (vc_y
>= 1) {
194 vc_x
= vc
->vc_cols
-WIDTH
;
199 if (vc_x
+ WIDTH
< vc
->vc_cols
) {
201 } else if (vc_y
+ 1 < vc
->vc_rows
) {
209 if (vc_y
+ 1 < vc
->vc_rows
)
221 vc_follow_cursor(vc
);
229 vc_y
= vc
->vc_rows
-1;
235 if (ret
== NOTIFY_STOP
)
239 case KBD_POST_KEYSYM
:
241 unsigned char type
= KTYP(param
->value
) - 0xf0;
242 if (type
== KT_SPEC
) {
243 unsigned char val
= KVAL(param
->value
);
248 on_off
= vc_kbd_led(kbd_table
+ fg_console
,
252 on_off
= vc_kbd_led(kbd_table
+ fg_console
,
256 on_off
= vc_kbd_led(kbd_table
+ fg_console
,
262 else if (on_off
== 0)
266 case KBD_UNBOUND_KEYCODE
:
275 static struct notifier_block keyboard_notifier_block
= {
276 .notifier_call
= keyboard_notifier_call
,
279 static int vt_notifier_call(struct notifier_block
*blk
,
280 unsigned long code
, void *_param
)
282 struct vt_notifier_param
*param
= _param
;
283 struct vc_data
*vc
= param
->vc
;
291 unsigned char c
= param
->c
;
292 if (vc
->vc_num
!= fg_console
)
297 if (console_cursor
> 0) {
299 console_buf
[console_cursor
] = ' ';
313 /* Ignore other control sequences */
315 if (console_newline
) {
316 memset(console_buf
, 0, sizeof(console_buf
));
320 if (console_cursor
== WIDTH
)
321 memmove(console_buf
, &console_buf
[1],
322 (WIDTH
-1) * sizeof(*console_buf
));
325 console_buf
[console_cursor
-1] = c
;
329 braille_write(console_buf
);
331 vc_maybe_cursor_moved(vc
);
337 /* Maybe a VT switch, flush */
339 if (vc
->vc_num
!= lastVC
) {
341 memset(console_buf
, 0, sizeof(console_buf
));
343 braille_write(console_buf
);
346 vc_maybe_cursor_moved(vc
);
354 static struct notifier_block vt_notifier_block
= {
355 .notifier_call
= vt_notifier_call
,
359 * Called from printk.c when console=brl is given
362 int braille_register_console(struct console
*console
, int index
,
363 char *console_options
, char *braille_options
)
366 if (!console_options
)
367 /* Only support VisioBraille for now */
368 console_options
= "57600o8";
371 if (console
->setup
) {
372 ret
= console
->setup(console
, console_options
);
376 console
->flags
|= CON_ENABLED
;
377 console
->index
= index
;
378 braille_co
= console
;
379 register_keyboard_notifier(&keyboard_notifier_block
);
380 register_vt_notifier(&vt_notifier_block
);
384 int braille_unregister_console(struct console
*console
)
386 if (braille_co
!= console
)
388 unregister_keyboard_notifier(&keyboard_notifier_block
);
389 unregister_vt_notifier(&vt_notifier_block
);