1 // SPDX-License-Identifier: GPL-2.0+
2 /* speakup.c
3 * review functions for the speakup screen review package.
4 * originally written by: Kirk Reiser and Andy Berdan.
5 *
6 * extensively modified by David Borowski.
7 *
8 ** Copyright (C) 1998 Kirk Reiser.
9 * Copyright (C) 2003 David Borowski.
10 */
11
12 #include <linux/kernel.h>
13 #include <linux/vt.h>
14 #include <linux/tty.h>
15 #include <linux/mm.h> /* __get_free_page() and friends */
16 #include <linux/vt_kern.h>
17 #include <linux/ctype.h>
18 #include <linux/selection.h>
19 #include <linux/unistd.h>
20 #include <linux/jiffies.h>
21 #include <linux/kthread.h>
22 #include <linux/keyboard.h> /* for KT_SHIFT */
23 #include <linux/kbd_kern.h> /* for vc_kbd_* and friends */
24 #include <linux/input.h>
25 #include <linux/kmod.h>
26
27 /* speakup_*_selection */
28 #include <linux/module.h>
29 #include <linux/sched.h>
30 #include <linux/slab.h>
31 #include <linux/types.h>
32 #include <linux/consolemap.h>
33
34 #include <linux/spinlock.h>
35 #include <linux/notifier.h>
36
37 #include <linux/uaccess.h> /* copy_from|to|user() and others */
38
39 #include "spk_priv.h"
40 #include "speakup.h"
41
42 #define MAX_DELAY msecs_to_jiffies(500)
43 #define MINECHOCHAR SPACE
44
45 MODULE_AUTHOR("Kirk Reiser <kirk@braille.uwo.ca>");
46 MODULE_AUTHOR("Daniel Drake <dsd@gentoo.org>");
47 MODULE_DESCRIPTION("Speakup console speech");
48 MODULE_LICENSE("GPL");
49 MODULE_VERSION(SPEAKUP_VERSION);
50
51 char *synth_name;
52 module_param_named(synth, synth_name, charp, 0444);
53 module_param_named(quiet, spk_quiet_boot, bool, 0444);
54
55 MODULE_PARM_DESC(synth, "Synth to start if speakup is built in.");
56 MODULE_PARM_DESC(quiet, "Do not announce when the synthesizer is found.");
57
58 special_func spk_special_handler;
59
60 short spk_pitch_shift, synth_flags;
61 static u16 buf[256];
62 int spk_attrib_bleep, spk_bleeps, spk_bleep_time = 10;
63 int spk_no_intr, spk_spell_delay;
64 int spk_key_echo, spk_say_word_ctl;
65 int spk_say_ctrl, spk_bell_pos;
66 short spk_punc_mask;
67 int spk_punc_level, spk_reading_punc;
68 char spk_str_caps_start[MAXVARLEN + 1] = "\0";
69 char spk_str_caps_stop[MAXVARLEN + 1] = "\0";
70 char spk_str_pause[MAXVARLEN + 1] = "\0";
71 bool spk_paused;
72 const struct st_bits_data spk_punc_info[] = {
73 {"none", "", 0},
74 {"some", "/$%&@", SOME},
75 {"most", "$%&#()=+*/@^<>|\\", MOST},
76 {"all", "!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~", PUNC},
77 {"delimiters", "", B_WDLM},
78 {"repeats", "()", CH_RPT},
79 {"extended numeric", "", B_EXNUM},
80 {"symbols", "", B_SYM},
81 {NULL, NULL}
82 };
83
84 static char mark_cut_flag;
85 #define MAX_KEY 160
86 static u_char *spk_shift_table;
87 u_char *spk_our_keys[MAX_KEY];
88 u_char spk_key_buf[600];
89 const u_char spk_key_defaults[] = {
90 #include "speakupmap.h"
91 };
92
93 /* Speakup Cursor Track Variables */
94 static int cursor_track = 1, prev_cursor_track = 1;
95
96 /* cursor track modes, must be ordered same as cursor_msgs */
97 enum {
98 CT_Off = 0,
99 CT_On,
100 CT_Highlight,
101 CT_Window,
102 CT_Max
103 };
104
105 #define read_all_mode CT_Max
106
107 static struct tty_struct *tty;
108
109 static void spkup_write(const u16 *in_buf, int count);
110
111 static char *phonetic[] = {
112 "alfa", "bravo", "charlie", "delta", "echo", "foxtrot", "golf", "hotel",
113 "india", "juliett", "keelo", "leema", "mike", "november", "oscar",
114 "papa",
115 "keh beck", "romeo", "sierra", "tango", "uniform", "victer", "whiskey",
116 "x ray", "yankee", "zulu"
117 };
118
119 /* array of 256 char pointers (one for each character description)
120 * initialized to default_chars and user selectable via
121 * /proc/speakup/characters
122 */
123 char *spk_characters[256];
124
125 char *spk_default_chars[256] = {
126 /*000*/ "null", "^a", "^b", "^c", "^d", "^e", "^f", "^g",
127 /*008*/ "^h", "^i", "^j", "^k", "^l", "^m", "^n", "^o",
128 /*016*/ "^p", "^q", "^r", "^s", "^t", "^u", "^v", "^w",
129 /*024*/ "^x", "^y", "^z", "control", "control", "control", "control",
130 "control",
131 /*032*/ "space", "bang!", "quote", "number", "dollar", "percent", "and",
132 "tick",
133 /*040*/ "left paren", "right paren", "star", "plus", "comma", "dash",
134 "dot",
135 "slash",
136 /*048*/ "zero", "one", "two", "three", "four", "five", "six", "seven",
137 "eight", "nine",
138 /*058*/ "colon", "semmy", "less", "equals", "greater", "question", "at",
139 /*065*/ "EIGH", "B", "C", "D", "E", "F", "G",
140 /*072*/ "H", "I", "J", "K", "L", "M", "N", "O",
141 /*080*/ "P", "Q", "R", "S", "T", "U", "V", "W", "X",
142 /*089*/ "Y", "ZED", "left bracket", "backslash", "right bracket",
143 "caret",
144 "line",
145 /*096*/ "accent", "a", "b", "c", "d", "e", "f", "g",
146 /*104*/ "h", "i", "j", "k", "l", "m", "n", "o",
147 /*112*/ "p", "q", "r", "s", "t", "u", "v", "w",
148 /*120*/ "x", "y", "zed", "left brace", "bar", "right brace", "tihlduh",
149 /*127*/ "del", "control", "control", "control", "control", "control",
150 "control", "control", "control", "control", "control",
151 /*138*/ "control", "control", "control", "control", "control",
152 "control", "control", "control", "control", "control",
153 "control", "control",
154 /*150*/ "control", "control", "control", "control", "control",
155 "control", "control", "control", "control", "control",
156 /*160*/ "nbsp", "inverted bang",
157 /*162*/ "cents", "pounds", "currency", "yen", "broken bar", "section",
158 /*168*/ "diaeresis", "copyright", "female ordinal", "double left angle",
159 /*172*/ "not", "soft hyphen", "registered", "macron",
160 /*176*/ "degrees", "plus or minus", "super two", "super three",
161 /*180*/ "acute accent", "micro", "pilcrow", "middle dot",
162 /*184*/ "cedilla", "super one", "male ordinal", "double right angle",
163 /*188*/ "one quarter", "one half", "three quarters",
164 "inverted question",
165 /*192*/ "A GRAVE", "A ACUTE", "A CIRCUMFLEX", "A TILDE", "A OOMLAUT",
166 "A RING",
167 /*198*/ "AE", "C CIDELLA", "E GRAVE", "E ACUTE", "E CIRCUMFLEX",
168 "E OOMLAUT",
169 /*204*/ "I GRAVE", "I ACUTE", "I CIRCUMFLEX", "I OOMLAUT", "ETH",
170 "N TILDE",
171 /*210*/ "O GRAVE", "O ACUTE", "O CIRCUMFLEX", "O TILDE", "O OOMLAUT",
172 /*215*/ "multiplied by", "O STROKE", "U GRAVE", "U ACUTE",
173 "U CIRCUMFLEX",
174 /*220*/ "U OOMLAUT", "Y ACUTE", "THORN", "sharp s", "a grave",
175 /*225*/ "a acute", "a circumflex", "a tilde", "a oomlaut", "a ring",
176 /*230*/ "ae", "c cidella", "e grave", "e acute",
177 /*234*/ "e circumflex", "e oomlaut", "i grave", "i acute",
178 "i circumflex",
179 /*239*/ "i oomlaut", "eth", "n tilde", "o grave", "o acute",
180 "o circumflex",
181 /*245*/ "o tilde", "o oomlaut", "divided by", "o stroke", "u grave",
182 "u acute",
183 /* 251 */ "u circumflex", "u oomlaut", "y acute", "thorn", "y oomlaut"
184 };
185
186 /* array of 256 u_short (one for each character)
187 * initialized to default_chartab and user selectable via
188 * /sys/module/speakup/parameters/chartab
189 */
190 u_short spk_chartab[256];
191
192 static u_short default_chartab[256] = {
193 B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 0-7 */
194 B_CTL, B_CTL, A_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 8-15 */
195 B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /*16-23 */
196 B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 24-31 */
197 WDLM, A_PUNC, PUNC, PUNC, PUNC, PUNC, PUNC, A_PUNC, /* !"#$%&' */
198 PUNC, PUNC, PUNC, PUNC, A_PUNC, A_PUNC, A_PUNC, PUNC, /* ()*+, -./ */
199 NUM, NUM, NUM, NUM, NUM, NUM, NUM, NUM, /* 01234567 */
200 NUM, NUM, A_PUNC, PUNC, PUNC, PUNC, PUNC, A_PUNC, /* 89:;<=>? */
201 PUNC, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* @ABCDEFG */
202 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* HIJKLMNO */
203 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* PQRSTUVW */
204 A_CAP, A_CAP, A_CAP, PUNC, PUNC, PUNC, PUNC, PUNC, /* XYZ[\]^_ */
205 PUNC, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* `abcdefg */
206 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* hijklmno */
207 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* pqrstuvw */
208 ALPHA, ALPHA, ALPHA, PUNC, PUNC, PUNC, PUNC, 0, /* xyz{|}~ */
209 B_CAPSYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 128-134 */
210 B_SYM, /* 135 */
211 B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 136-142 */
212 B_CAPSYM, /* 143 */
213 B_CAPSYM, B_CAPSYM, B_SYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, /* 144-150 */
214 B_SYM, /* 151 */
215 B_SYM, B_SYM, B_CAPSYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, /*152-158 */
216 B_SYM, /* 159 */
217 WDLM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_CAPSYM, /* 160-166 */
218 B_SYM, /* 167 */
219 B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 168-175 */
220 B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 176-183 */
221 B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 184-191 */
222 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* 192-199 */
223 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* 200-207 */
224 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, B_SYM, /* 208-215 */
225 A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, ALPHA, /* 216-223 */
226 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* 224-231 */
227 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* 232-239 */
228 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, B_SYM, /* 240-247 */
229 ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA /* 248-255 */
230 };
231
232 struct task_struct *speakup_task;
233 struct bleep spk_unprocessed_sound;
234 static int spk_keydown;
235 static u16 spk_lastkey;
236 static u_char spk_close_press, keymap_flags;
237 static u_char last_keycode, this_speakup_key;
238 static u_long last_spk_jiffy;
239
240 struct st_spk_t *speakup_console[MAX_NR_CONSOLES];
241
242 DEFINE_MUTEX(spk_mutex);
243
244 static int keyboard_notifier_call(struct notifier_block *,
245 unsigned long code, void *param);
246
247 static struct notifier_block keyboard_notifier_block = {
248 .notifier_call = keyboard_notifier_call,
249 };
250
251 static int vt_notifier_call(struct notifier_block *,
252 unsigned long code, void *param);
253
254 static struct notifier_block vt_notifier_block = {
255 .notifier_call = vt_notifier_call,
256 };
257
get_attributes(struct vc_data * vc,u16 * pos)258 static unsigned char get_attributes(struct vc_data *vc, u16 *pos)
259 {
260 pos = screen_pos(vc, pos - (u16 *)vc->vc_origin, true);
261 return (scr_readw(pos) & ~vc->vc_hi_font_mask) >> 8;
262 }
263
speakup_date(struct vc_data * vc)264 static void speakup_date(struct vc_data *vc)
265 {
266 spk_x = spk_cx = vc->state.x;
267 spk_y = spk_cy = vc->state.y;
268 spk_pos = spk_cp = vc->vc_pos;
269 spk_old_attr = spk_attr;
270 spk_attr = get_attributes(vc, (u_short *)spk_pos);
271 }
272
bleep(u_short val)273 static void bleep(u_short val)
274 {
275 static const short vals[] = {
276 350, 370, 392, 414, 440, 466, 491, 523, 554, 587, 619, 659
277 };
278 short freq;
279 int time = spk_bleep_time;
280
281 freq = vals[val % 12];
282 if (val > 11)
283 freq *= (1 << (val / 12));
284 spk_unprocessed_sound.freq = freq;
285 spk_unprocessed_sound.jiffies = msecs_to_jiffies(time);
286 spk_unprocessed_sound.active = 1;
287 /* We can only have 1 active sound at a time. */
288 }
289
speakup_shut_up(struct vc_data * vc)290 static void speakup_shut_up(struct vc_data *vc)
291 {
292 if (spk_killed)
293 return;
294 spk_shut_up |= 0x01;
295 spk_parked &= 0xfe;
296 speakup_date(vc);
297 if (synth)
298 spk_do_flush();
299 }
300
speech_kill(struct vc_data * vc)301 static void speech_kill(struct vc_data *vc)
302 {
303 char val = synth->is_alive(synth);
304
305 if (val == 0)
306 return;
307
308 /* re-enables synth, if disabled */
309 if (val == 2 || spk_killed) {
310 /* dead */
311 spk_shut_up &= ~0x40;
312 synth_printf("%s\n", spk_msg_get(MSG_IAM_ALIVE));
313 } else {
314 synth_printf("%s\n", spk_msg_get(MSG_YOU_KILLED_SPEAKUP));
315 spk_shut_up |= 0x40;
316 }
317 }
318
speakup_off(struct vc_data * vc)319 static void speakup_off(struct vc_data *vc)
320 {
321 if (spk_shut_up & 0x80) {
322 spk_shut_up &= 0x7f;
323 synth_printf("%s\n", spk_msg_get(MSG_HEY_THATS_BETTER));
324 } else {
325 spk_shut_up |= 0x80;
326 synth_printf("%s\n", spk_msg_get(MSG_YOU_TURNED_ME_OFF));
327 }
328 speakup_date(vc);
329 }
330
speakup_parked(struct vc_data * vc)331 static void speakup_parked(struct vc_data *vc)
332 {
333 if (spk_parked & 0x80) {
334 spk_parked = 0;
335 synth_printf("%s\n", spk_msg_get(MSG_UNPARKED));
336 } else {
337 spk_parked |= 0x80;
338 synth_printf("%s\n", spk_msg_get(MSG_PARKED));
339 }
340 }
341
speakup_cut(struct vc_data * vc)342 static void speakup_cut(struct vc_data *vc)
343 {
344 static const char err_buf[] = "set selection failed";
345 int ret;
346
347 if (!mark_cut_flag) {
348 mark_cut_flag = 1;
349 spk_xs = (u_short)spk_x;
350 spk_ys = (u_short)spk_y;
351 spk_sel_cons = vc;
352 synth_printf("%s\n", spk_msg_get(MSG_MARK));
353 return;
354 }
355 spk_xe = (u_short)spk_x;
356 spk_ye = (u_short)spk_y;
357 mark_cut_flag = 0;
358 synth_printf("%s\n", spk_msg_get(MSG_CUT));
359
360 ret = speakup_set_selection(tty);
361
362 switch (ret) {
363 case 0:
364 break; /* no error */
365 case -EFAULT:
366 pr_warn("%sEFAULT\n", err_buf);
367 break;
368 case -EINVAL:
369 pr_warn("%sEINVAL\n", err_buf);
370 break;
371 case -ENOMEM:
372 pr_warn("%sENOMEM\n", err_buf);
373 break;
374 }
375 }
376
speakup_paste(struct vc_data * vc)377 static void speakup_paste(struct vc_data *vc)
378 {
379 if (mark_cut_flag) {
380 mark_cut_flag = 0;
381 synth_printf("%s\n", spk_msg_get(MSG_MARK_CLEARED));
382 } else {
383 synth_printf("%s\n", spk_msg_get(MSG_PASTE));
384 speakup_paste_selection(tty);
385 }
386 }
387
say_attributes(struct vc_data * vc)388 static void say_attributes(struct vc_data *vc)
389 {
390 int fg = spk_attr & 0x0f;
391 int bg = spk_attr >> 4;
392
393 if (fg > 8) {
394 synth_printf("%s ", spk_msg_get(MSG_BRIGHT));
395 fg -= 8;
396 }
397 synth_printf("%s", spk_msg_get(MSG_COLORS_START + fg));
398 if (bg > 7) {
399 synth_printf(" %s ", spk_msg_get(MSG_ON_BLINKING));
400 bg -= 8;
401 } else {
402 synth_printf(" %s ", spk_msg_get(MSG_ON));
403 }
404 synth_printf("%s\n", spk_msg_get(MSG_COLORS_START + bg));
405 }
406
407 enum {
408 edge_top = 1,
409 edge_bottom,
410 edge_left,
411 edge_right,
412 edge_quiet
413 };
414
announce_edge(struct vc_data * vc,int msg_id)415 static void announce_edge(struct vc_data *vc, int msg_id)
416 {
417 if (spk_bleeps & 1)
418 bleep(spk_y);
419 if ((spk_bleeps & 2) && (msg_id < edge_quiet))
420 synth_printf("%s\n",
421 spk_msg_get(MSG_EDGE_MSGS_START + msg_id - 1));
422 }
423
speak_char(u16 ch)424 static void speak_char(u16 ch)
425 {
426 char *cp;
427 struct var_t *direct = spk_get_var(DIRECT);
428
429 if (ch >= 0x100 || (direct && direct->u.n.value)) {
430 if (ch < 0x100 && IS_CHAR(ch, B_CAP)) {
431 spk_pitch_shift++;
432 synth_printf("%s", spk_str_caps_start);
433 }
434 synth_putwc_s(ch);
435 if (ch < 0x100 && IS_CHAR(ch, B_CAP))
436 synth_printf("%s", spk_str_caps_stop);
437 return;
438 }
439
440 cp = spk_characters[ch];
441 if (!cp) {
442 pr_info("%s: cp == NULL!\n", __func__);
443 return;
444 }
445 if (IS_CHAR(ch, B_CAP)) {
446 spk_pitch_shift++;
447 synth_printf("%s %s %s",
448 spk_str_caps_start, cp, spk_str_caps_stop);
449 } else {
450 if (*cp == '^') {
451 cp++;
452 synth_printf(" %s%s ", spk_msg_get(MSG_CTRL), cp);
453 } else {
454 synth_printf(" %s ", cp);
455 }
456 }
457 }
458
get_char(struct vc_data * vc,u16 * pos,u_char * attribs)459 static u16 get_char(struct vc_data *vc, u16 *pos, u_char *attribs)
460 {
461 u16 ch = ' ';
462
463 if (vc && pos) {
464 u16 w;
465 u16 c;
466
467 pos = screen_pos(vc, pos - (u16 *)vc->vc_origin, true);
468 w = scr_readw(pos);
469 c = w & 0xff;
470
471 if (w & vc->vc_hi_font_mask) {
472 w &= ~vc->vc_hi_font_mask;
473 c |= 0x100;
474 }
475
476 ch = inverse_translate(vc, c, 1);
477 *attribs = (w & 0xff00) >> 8;
478 }
479 return ch;
480 }
481
say_char(struct vc_data * vc)482 static void say_char(struct vc_data *vc)
483 {
484 u16 ch;
485
486 spk_old_attr = spk_attr;
487 ch = get_char(vc, (u_short *)spk_pos, &spk_attr);
488 if (spk_attr != spk_old_attr) {
489 if (spk_attrib_bleep & 1)
490 bleep(spk_y);
491 if (spk_attrib_bleep & 2)
492 say_attributes(vc);
493 }
494 speak_char(ch);
495 }
496
say_phonetic_char(struct vc_data * vc)497 static void say_phonetic_char(struct vc_data *vc)
498 {
499 u16 ch;
500
501 spk_old_attr = spk_attr;
502 ch = get_char(vc, (u_short *)spk_pos, &spk_attr);
503 if (ch <= 0x7f && isalpha(ch)) {
504 ch &= 0x1f;
505 synth_printf("%s\n", phonetic[--ch]);
506 } else {
507 if (ch < 0x100 && IS_CHAR(ch, B_NUM))
508 synth_printf("%s ", spk_msg_get(MSG_NUMBER));
509 speak_char(ch);
510 }
511 }
512
say_prev_char(struct vc_data * vc)513 static void say_prev_char(struct vc_data *vc)
514 {
515 spk_parked |= 0x01;
516 if (spk_x == 0) {
517 announce_edge(vc, edge_left);
518 return;
519 }
520 spk_x--;
521 spk_pos -= 2;
522 say_char(vc);
523 }
524
say_next_char(struct vc_data * vc)525 static void say_next_char(struct vc_data *vc)
526 {
527 spk_parked |= 0x01;
528 if (spk_x == vc->vc_cols - 1) {
529 announce_edge(vc, edge_right);
530 return;
531 }
532 spk_x++;
533 spk_pos += 2;
534 say_char(vc);
535 }
536
537 /* get_word - will first check to see if the character under the
538 * reading cursor is a space and if spk_say_word_ctl is true it will
539 * return the word space. If spk_say_word_ctl is not set it will check to
540 * see if there is a word starting on the next position to the right
541 * and return that word if it exists. If it does not exist it will
542 * move left to the beginning of any previous word on the line or the
543 * beginning off the line whichever comes first..
544 */
545
get_word(struct vc_data * vc)546 static u_long get_word(struct vc_data *vc)
547 {
548 u_long cnt = 0, tmpx = spk_x, tmp_pos = spk_pos;
549 u16 ch;
550 u16 attr_ch;
551 u_char temp;
552
553 spk_old_attr = spk_attr;
554 ch = get_char(vc, (u_short *)tmp_pos, &temp);
555
556 /* decided to take out the sayword if on a space (mis-information */
557 if (spk_say_word_ctl && ch == SPACE) {
558 *buf = '\0';
559 synth_printf("%s\n", spk_msg_get(MSG_SPACE));
560 return 0;
561 } else if (tmpx < vc->vc_cols - 2 &&
562 (ch == SPACE || ch == 0 || (ch < 0x100 && IS_WDLM(ch))) &&
563 get_char(vc, (u_short *)tmp_pos + 1, &temp) > SPACE) {
564 tmp_pos += 2;
565 tmpx++;
566 } else {
567 while (tmpx > 0) {
568 ch = get_char(vc, (u_short *)tmp_pos - 1, &temp);
569 if ((ch == SPACE || ch == 0 ||
570 (ch < 0x100 && IS_WDLM(ch))) &&
571 get_char(vc, (u_short *)tmp_pos, &temp) > SPACE)
572 break;
573 tmp_pos -= 2;
574 tmpx--;
575 }
576 }
577 attr_ch = get_char(vc, (u_short *)tmp_pos, &spk_attr);
578 buf[cnt++] = attr_ch;
579 while (tmpx < vc->vc_cols - 1) {
580 tmp_pos += 2;
581 tmpx++;
582 ch = get_char(vc, (u_short *)tmp_pos, &temp);
583 if (ch == SPACE || ch == 0 ||
584 (buf[cnt - 1] < 0x100 && IS_WDLM(buf[cnt - 1]) &&
585 ch > SPACE))
586 break;
587 buf[cnt++] = ch;
588 }
589 buf[cnt] = '\0';
590 return cnt;
591 }
592
say_word(struct vc_data * vc)593 static void say_word(struct vc_data *vc)
594 {
595 u_long cnt = get_word(vc);
596 u_short saved_punc_mask = spk_punc_mask;
597
598 if (cnt == 0)
599 return;
600 spk_punc_mask = PUNC;
601 buf[cnt++] = SPACE;
602 spkup_write(buf, cnt);
603 spk_punc_mask = saved_punc_mask;
604 }
605
say_prev_word(struct vc_data * vc)606 static void say_prev_word(struct vc_data *vc)
607 {
608 u_char temp;
609 u16 ch;
610 u_short edge_said = 0, last_state = 0, state = 0;
611
612 spk_parked |= 0x01;
613
614 if (spk_x == 0) {
615 if (spk_y == 0) {
616 announce_edge(vc, edge_top);
617 return;
618 }
619 spk_y--;
620 spk_x = vc->vc_cols;
621 edge_said = edge_quiet;
622 }
623 while (1) {
624 if (spk_x == 0) {
625 if (spk_y == 0) {
626 edge_said = edge_top;
627 break;
628 }
629 if (edge_said != edge_quiet)
630 edge_said = edge_left;
631 if (state > 0)
632 break;
633 spk_y--;
634 spk_x = vc->vc_cols - 1;
635 } else {
636 spk_x--;
637 }
638 spk_pos -= 2;
639 ch = get_char(vc, (u_short *)spk_pos, &temp);
640 if (ch == SPACE || ch == 0)
641 state = 0;
642 else if (ch < 0x100 && IS_WDLM(ch))
643 state = 1;
644 else
645 state = 2;
646 if (state < last_state) {
647 spk_pos += 2;
648 spk_x++;
649 break;
650 }
651 last_state = state;
652 }
653 if (spk_x == 0 && edge_said == edge_quiet)
654 edge_said = edge_left;
655 if (edge_said > 0 && edge_said < edge_quiet)
656 announce_edge(vc, edge_said);
657 say_word(vc);
658 }
659
say_next_word(struct vc_data * vc)660 static void say_next_word(struct vc_data *vc)
661 {
662 u_char temp;
663 u16 ch;
664 u_short edge_said = 0, last_state = 2, state = 0;
665
666 spk_parked |= 0x01;
667 if (spk_x == vc->vc_cols - 1 && spk_y == vc->vc_rows - 1) {
668 announce_edge(vc, edge_bottom);
669 return;
670 }
671 while (1) {
672 ch = get_char(vc, (u_short *)spk_pos, &temp);
673 if (ch == SPACE || ch == 0)
674 state = 0;
675 else if (ch < 0x100 && IS_WDLM(ch))
676 state = 1;
677 else
678 state = 2;
679 if (state > last_state)
680 break;
681 if (spk_x >= vc->vc_cols - 1) {
682 if (spk_y == vc->vc_rows - 1) {
683 edge_said = edge_bottom;
684 break;
685 }
686 state = 0;
687 spk_y++;
688 spk_x = 0;
689 edge_said = edge_right;
690 } else {
691 spk_x++;
692 }
693 spk_pos += 2;
694 last_state = state;
695 }
696 if (edge_said > 0)
697 announce_edge(vc, edge_said);
698 say_word(vc);
699 }
700
spell_word(struct vc_data * vc)701 static void spell_word(struct vc_data *vc)
702 {
703 static char const *delay_str[] = { "", ",", ".", ". .", ". . ." };
704 u16 *cp = buf;
705 char *cp1;
706 char *str_cap = spk_str_caps_stop;
707 char *last_cap = spk_str_caps_stop;
708 struct var_t *direct = spk_get_var(DIRECT);
709 u16 ch;
710
711 if (!get_word(vc))
712 return;
713 while ((ch = *cp)) {
714 if (cp != buf)
715 synth_printf(" %s ", delay_str[spk_spell_delay]);
716 /* FIXME: Non-latin1 considered as lower case */
717 if (ch < 0x100 && IS_CHAR(ch, B_CAP)) {
718 str_cap = spk_str_caps_start;
719 if (*spk_str_caps_stop)
720 spk_pitch_shift++;
721 else /* synth has no pitch */
722 last_cap = spk_str_caps_stop;
723 } else {
724 str_cap = spk_str_caps_stop;
725 }
726 if (str_cap != last_cap) {
727 synth_printf("%s", str_cap);
728 last_cap = str_cap;
729 }
730 if (ch >= 0x100 || (direct && direct->u.n.value)) {
731 synth_putwc_s(ch);
732 } else if (this_speakup_key == SPELL_PHONETIC &&
733 ch <= 0x7f && isalpha(ch)) {
734 ch &= 0x1f;
735 cp1 = phonetic[--ch];
736 synth_printf("%s", cp1);
737 } else {
738 cp1 = spk_characters[ch];
739 if (*cp1 == '^') {
740 synth_printf("%s", spk_msg_get(MSG_CTRL));
741 cp1++;
742 }
743 synth_printf("%s", cp1);
744 }
745 cp++;
746 }
747 if (str_cap != spk_str_caps_stop)
748 synth_printf("%s", spk_str_caps_stop);
749 }
750
get_line(struct vc_data * vc)751 static int get_line(struct vc_data *vc)
752 {
753 u_long tmp = spk_pos - (spk_x * 2);
754 int i = 0;
755 u_char tmp2;
756
757 spk_old_attr = spk_attr;
758 spk_attr = get_attributes(vc, (u_short *)spk_pos);
759 for (i = 0; i < vc->vc_cols; i++) {
760 buf[i] = get_char(vc, (u_short *)tmp, &tmp2);
761 tmp += 2;
762 }
763 for (--i; i >= 0; i--)
764 if (buf[i] != SPACE)
765 break;
766 return ++i;
767 }
768
say_line(struct vc_data * vc)769 static void say_line(struct vc_data *vc)
770 {
771 int i = get_line(vc);
772 u16 *cp;
773 u_short saved_punc_mask = spk_punc_mask;
774
775 if (i == 0) {
776 synth_printf("%s\n", spk_msg_get(MSG_BLANK));
777 return;
778 }
779 buf[i++] = '\n';
780 if (this_speakup_key == SAY_LINE_INDENT) {
781 cp = buf;
782 while (*cp == SPACE)
783 cp++;
784 synth_printf("%zd, ", (cp - buf) + 1);
785 }
786 spk_punc_mask = spk_punc_masks[spk_reading_punc];
787 spkup_write(buf, i);
788 spk_punc_mask = saved_punc_mask;
789 }
790
say_prev_line(struct vc_data * vc)791 static void say_prev_line(struct vc_data *vc)
792 {
793 spk_parked |= 0x01;
794 if (spk_y == 0) {
795 announce_edge(vc, edge_top);
796 return;
797 }
798 spk_y--;
799 spk_pos -= vc->vc_size_row;
800 say_line(vc);
801 }
802
say_next_line(struct vc_data * vc)803 static void say_next_line(struct vc_data *vc)
804 {
805 spk_parked |= 0x01;
806 if (spk_y == vc->vc_rows - 1) {
807 announce_edge(vc, edge_bottom);
808 return;
809 }
810 spk_y++;
811 spk_pos += vc->vc_size_row;
812 say_line(vc);
813 }
814
say_from_to(struct vc_data * vc,u_long from,u_long to,int read_punc)815 static int say_from_to(struct vc_data *vc, u_long from, u_long to,
816 int read_punc)
817 {
818 int i = 0;
819 u_char tmp;
820 u_short saved_punc_mask = spk_punc_mask;
821
822 spk_old_attr = spk_attr;
823 spk_attr = get_attributes(vc, (u_short *)from);
824 while (from < to) {
825 buf[i++] = get_char(vc, (u_short *)from, &tmp);
826 from += 2;
827 if (i >= vc->vc_size_row)
828 break;
829 }
830 for (--i; i >= 0; i--)
831 if (buf[i] != SPACE)
832 break;
833 buf[++i] = SPACE;
834 buf[++i] = '\0';
835 if (i < 1)
836 return i;
837 if (read_punc)
838 spk_punc_mask = spk_punc_info[spk_reading_punc].mask;
839 spkup_write(buf, i);
840 if (read_punc)
841 spk_punc_mask = saved_punc_mask;
842 return i - 1;
843 }
844
say_line_from_to(struct vc_data * vc,u_long from,u_long to,int read_punc)845 static void say_line_from_to(struct vc_data *vc, u_long from, u_long to,
846 int read_punc)
847 {
848 u_long start = vc->vc_origin + (spk_y * vc->vc_size_row);
849 u_long end = start + (to * 2);
850
851 start += from * 2;
852 if (say_from_to(vc, start, end, read_punc) <= 0)
853 if (cursor_track != read_all_mode)
854 synth_printf("%s\n", spk_msg_get(MSG_BLANK));
855 }
856
857 /* Sentence Reading Commands */
858
859 static int currsentence;
860 static int numsentences[2];
861 static u16 *sentbufend[2];
862 static u16 *sentmarks[2][10];
863 static int currbuf;
864 static int bn;
865 static u16 sentbuf[2][256];
866
say_sentence_num(int num,int prev)867 static int say_sentence_num(int num, int prev)
868 {
869 bn = currbuf;
870 currsentence = num + 1;
871 if (prev && --bn == -1)
872 bn = 1;
873
874 if (num > numsentences[bn])
875 return 0;
876
877 spkup_write(sentmarks[bn][num], sentbufend[bn] - sentmarks[bn][num]);
878 return 1;
879 }
880
get_sentence_buf(struct vc_data * vc,int read_punc)881 static int get_sentence_buf(struct vc_data *vc, int read_punc)
882 {
883 u_long start, end;
884 int i, bn;
885 u_char tmp;
886
887 currbuf++;
888 if (currbuf == 2)
889 currbuf = 0;
890 bn = currbuf;
891 start = vc->vc_origin + ((spk_y) * vc->vc_size_row);
892 end = vc->vc_origin + ((spk_y) * vc->vc_size_row) + vc->vc_cols * 2;
893
894 numsentences[bn] = 0;
895 sentmarks[bn][0] = &sentbuf[bn][0];
896 i = 0;
897 spk_old_attr = spk_attr;
898 spk_attr = get_attributes(vc, (u_short *)start);
899
900 while (start < end) {
901 sentbuf[bn][i] = get_char(vc, (u_short *)start, &tmp);
902 if (i > 0) {
903 if (sentbuf[bn][i] == SPACE &&
904 sentbuf[bn][i - 1] == '.' &&
905 numsentences[bn] < 9) {
906 /* Sentence Marker */
907 numsentences[bn]++;
908 sentmarks[bn][numsentences[bn]] =
909 &sentbuf[bn][i];
910 }
911 }
912 i++;
913 start += 2;
914 if (i >= vc->vc_size_row)
915 break;
916 }
917
918 for (--i; i >= 0; i--)
919 if (sentbuf[bn][i] != SPACE)
920 break;
921
922 if (i < 1)
923 return -1;
924
925 sentbuf[bn][++i] = SPACE;
926 sentbuf[bn][++i] = '\0';
927
928 sentbufend[bn] = &sentbuf[bn][i];
929 return numsentences[bn];
930 }
931
say_screen_from_to(struct vc_data * vc,u_long from,u_long to)932 static void say_screen_from_to(struct vc_data *vc, u_long from, u_long to)
933 {
934 u_long start = vc->vc_origin, end;
935
936 if (from > 0)
937 start += from * vc->vc_size_row;
938 if (to > vc->vc_rows)
939 to = vc->vc_rows;
940 end = vc->vc_origin + (to * vc->vc_size_row);
941 for (from = start; from < end; from = to) {
942 to = from + vc->vc_size_row;
943 say_from_to(vc, from, to, 1);
944 }
945 }
946
say_screen(struct vc_data * vc)947 static void say_screen(struct vc_data *vc)
948 {
949 say_screen_from_to(vc, 0, vc->vc_rows);
950 }
951
speakup_win_say(struct vc_data * vc)952 static void speakup_win_say(struct vc_data *vc)
953 {
954 u_long start, end, from, to;
955
956 if (win_start < 2) {
957 synth_printf("%s\n", spk_msg_get(MSG_NO_WINDOW));
958 return;
959 }
960 start = vc->vc_origin + (win_top * vc->vc_size_row);
961 end = vc->vc_origin + (win_bottom * vc->vc_size_row);
962 while (start <= end) {
963 from = start + (win_left * 2);
964 to = start + (win_right * 2);
965 say_from_to(vc, from, to, 1);
966 start += vc->vc_size_row;
967 }
968 }
969
top_edge(struct vc_data * vc)970 static void top_edge(struct vc_data *vc)
971 {
972 spk_parked |= 0x01;
973 spk_pos = vc->vc_origin + 2 * spk_x;
974 spk_y = 0;
975 say_line(vc);
976 }
977
bottom_edge(struct vc_data * vc)978 static void bottom_edge(struct vc_data *vc)
979 {
980 spk_parked |= 0x01;
981 spk_pos += (vc->vc_rows - spk_y - 1) * vc->vc_size_row;
982 spk_y = vc->vc_rows - 1;
983 say_line(vc);
984 }
985
left_edge(struct vc_data * vc)986 static void left_edge(struct vc_data *vc)
987 {
988 spk_parked |= 0x01;
989 spk_pos -= spk_x * 2;
990 spk_x = 0;
991 say_char(vc);
992 }
993
right_edge(struct vc_data * vc)994 static void right_edge(struct vc_data *vc)
995 {
996 spk_parked |= 0x01;
997 spk_pos += (vc->vc_cols - spk_x - 1) * 2;
998 spk_x = vc->vc_cols - 1;
999 say_char(vc);
1000 }
1001
say_first_char(struct vc_data * vc)1002 static void say_first_char(struct vc_data *vc)
1003 {
1004 int i, len = get_line(vc);
1005 u16 ch;
1006
1007 spk_parked |= 0x01;
1008 if (len == 0) {
1009 synth_printf("%s\n", spk_msg_get(MSG_BLANK));
1010 return;
1011 }
1012 for (i = 0; i < len; i++)
1013 if (buf[i] != SPACE)
1014 break;
1015 ch = buf[i];
1016 spk_pos -= (spk_x - i) * 2;
1017 spk_x = i;
1018 synth_printf("%d, ", ++i);
1019 speak_char(ch);
1020 }
1021
say_last_char(struct vc_data * vc)1022 static void say_last_char(struct vc_data *vc)
1023 {
1024 int len = get_line(vc);
1025 u16 ch;
1026
1027 spk_parked |= 0x01;
1028 if (len == 0) {
1029 synth_printf("%s\n", spk_msg_get(MSG_BLANK));
1030 return;
1031 }
1032 ch = buf[--len];
1033 spk_pos -= (spk_x - len) * 2;
1034 spk_x = len;
1035 synth_printf("%d, ", ++len);
1036 speak_char(ch);
1037 }
1038
say_position(struct vc_data * vc)1039 static void say_position(struct vc_data *vc)
1040 {
1041 synth_printf(spk_msg_get(MSG_POS_INFO), spk_y + 1, spk_x + 1,
1042 vc->vc_num + 1);
1043 synth_printf("\n");
1044 }
1045
1046 /* Added by brianb */
say_char_num(struct vc_data * vc)1047 static void say_char_num(struct vc_data *vc)
1048 {
1049 u_char tmp;
1050 u16 ch = get_char(vc, (u_short *)spk_pos, &tmp);
1051
1052 synth_printf(spk_msg_get(MSG_CHAR_INFO), ch, ch);
1053 }
1054
1055 /* these are stub functions to keep keyboard.c happy. */
1056
say_from_top(struct vc_data * vc)1057 static void say_from_top(struct vc_data *vc)
1058 {
1059 say_screen_from_to(vc, 0, spk_y);
1060 }
1061
say_to_bottom(struct vc_data * vc)1062 static void say_to_bottom(struct vc_data *vc)
1063 {
1064 say_screen_from_to(vc, spk_y, vc->vc_rows);
1065 }
1066
say_from_left(struct vc_data * vc)1067 static void say_from_left(struct vc_data *vc)
1068 {
1069 say_line_from_to(vc, 0, spk_x, 1);
1070 }
1071
say_to_right(struct vc_data * vc)1072 static void say_to_right(struct vc_data *vc)
1073 {
1074 say_line_from_to(vc, spk_x, vc->vc_cols, 1);
1075 }
1076
1077 /* end of stub functions. */
1078
spkup_write(const u16 * in_buf,int count)1079 static void spkup_write(const u16 *in_buf, int count)
1080 {
1081 static int rep_count;
1082 static u16 ch = '\0', old_ch = '\0';
1083 static u_short char_type, last_type;
1084 int in_count = count;
1085
1086 spk_keydown = 0;
1087 while (count--) {
1088 if (cursor_track == read_all_mode) {
1089 /* Insert Sentence Index */
1090 if ((in_buf == sentmarks[bn][currsentence]) &&
1091 (currsentence <= numsentences[bn]))
1092 synth_insert_next_index(currsentence++);
1093 }
1094 ch = *in_buf++;
1095 if (ch < 0x100)
1096 char_type = spk_chartab[ch];
1097 else
1098 char_type = ALPHA;
1099 if (ch == old_ch && !(char_type & B_NUM)) {
1100 if (++rep_count > 2)
1101 continue;
1102 } else {
1103 if ((last_type & CH_RPT) && rep_count > 2) {
1104 synth_printf(" ");
1105 synth_printf(spk_msg_get(MSG_REPEAT_DESC),
1106 ++rep_count);
1107 synth_printf(" ");
1108 }
1109 rep_count = 0;
1110 }
1111 if (ch == spk_lastkey) {
1112 rep_count = 0;
1113 if (spk_key_echo == 1 && ch >= MINECHOCHAR)
1114 speak_char(ch);
1115 } else if (char_type & B_ALPHA) {
1116 if ((synth_flags & SF_DEC) && (last_type & PUNC))
1117 synth_buffer_add(SPACE);
1118 synth_putwc_s(ch);
1119 } else if (char_type & B_NUM) {
1120 rep_count = 0;
1121 synth_putwc_s(ch);
1122 } else if (char_type & spk_punc_mask) {
1123 speak_char(ch);
1124 char_type &= ~PUNC; /* for dec nospell processing */
1125 } else if (char_type & SYNTH_OK) {
1126 /* these are usually puncts like . and , which synth
1127 * needs for expression.
1128 * suppress multiple to get rid of long pauses and
1129 * clear repeat count
1130 * so if someone has
1131 * repeats on you don't get nothing repeated count
1132 */
1133 if (ch != old_ch)
1134 synth_putwc_s(ch);
1135 else
1136 rep_count = 0;
1137 } else {
1138 /* send space and record position, if next is num overwrite space */
1139 if (old_ch != ch)
1140 synth_buffer_add(SPACE);
1141 else
1142 rep_count = 0;
1143 }
1144 old_ch = ch;
1145 last_type = char_type;
1146 }
1147 spk_lastkey = 0;
1148 if (in_count > 2 && rep_count > 2) {
1149 if (last_type & CH_RPT) {
1150 synth_printf(" ");
1151 synth_printf(spk_msg_get(MSG_REPEAT_DESC2),
1152 ++rep_count);
1153 synth_printf(" ");
1154 }
1155 rep_count = 0;
1156 }
1157 }
1158
1159 static const int NUM_CTL_LABELS = (MSG_CTL_END - MSG_CTL_START + 1);
1160
1161 static void read_all_doc(struct vc_data *vc);
1162 static void cursor_done(struct timer_list *unused);
1163 static DEFINE_TIMER(cursor_timer, cursor_done);
1164
do_handle_shift(struct vc_data * vc,u_char value,char up_flag)1165 static void do_handle_shift(struct vc_data *vc, u_char value, char up_flag)
1166 {
1167 unsigned long flags;
1168
1169 if (!synth || up_flag || spk_killed)
1170 return;
1171 spin_lock_irqsave(&speakup_info.spinlock, flags);
1172 if (cursor_track == read_all_mode) {
1173 switch (value) {
1174 case KVAL(K_SHIFT):
1175 del_timer(&cursor_timer);
1176 spk_shut_up &= 0xfe;
1177 spk_do_flush();
1178 read_all_doc(vc);
1179 break;
1180 case KVAL(K_CTRL):
1181 del_timer(&cursor_timer);
1182 cursor_track = prev_cursor_track;
1183 spk_shut_up &= 0xfe;
1184 spk_do_flush();
1185 break;
1186 }
1187 } else {
1188 spk_shut_up &= 0xfe;
1189 spk_do_flush();
1190 }
1191 if (spk_say_ctrl && value < NUM_CTL_LABELS)
1192 synth_printf("%s", spk_msg_get(MSG_CTL_START + value));
1193 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1194 }
1195
do_handle_latin(struct vc_data * vc,u_char value,char up_flag)1196 static void do_handle_latin(struct vc_data *vc, u_char value, char up_flag)
1197 {
1198 unsigned long flags;
1199
1200 spin_lock_irqsave(&speakup_info.spinlock, flags);
1201 if (up_flag) {
1202 spk_lastkey = 0;
1203 spk_keydown = 0;
1204 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1205 return;
1206 }
1207 if (!synth || spk_killed) {
1208 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1209 return;
1210 }
1211 spk_shut_up &= 0xfe;
1212 spk_lastkey = value;
1213 spk_keydown++;
1214 spk_parked &= 0xfe;
1215 if (spk_key_echo == 2 && value >= MINECHOCHAR)
1216 speak_char(value);
1217 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1218 }
1219
spk_set_key_info(const u_char * key_info,u_char * k_buffer)1220 int spk_set_key_info(const u_char *key_info, u_char *k_buffer)
1221 {
1222 int i = 0, states, key_data_len;
1223 const u_char *cp = key_info;
1224 u_char *cp1 = k_buffer;
1225 u_char ch, version, num_keys;
1226
1227 version = *cp++;
1228 if (version != KEY_MAP_VER) {
1229 pr_debug("version found %d should be %d\n",
1230 version, KEY_MAP_VER);
1231 return -EINVAL;
1232 }
1233 num_keys = *cp;
1234 states = (int)cp[1];
1235 key_data_len = (states + 1) * (num_keys + 1);
1236 if (key_data_len + SHIFT_TBL_SIZE + 4 >= sizeof(spk_key_buf)) {
1237 pr_debug("too many key_infos (%d over %u)\n",
1238 key_data_len + SHIFT_TBL_SIZE + 4,
1239 (unsigned int)(sizeof(spk_key_buf)));
1240 return -EINVAL;
1241 }
1242 memset(k_buffer, 0, SHIFT_TBL_SIZE);
1243 memset(spk_our_keys, 0, sizeof(spk_our_keys));
1244 spk_shift_table = k_buffer;
1245 spk_our_keys[0] = spk_shift_table;
1246 cp1 += SHIFT_TBL_SIZE;
1247 memcpy(cp1, cp, key_data_len + 3);
1248 /* get num_keys, states and data */
1249 cp1 += 2; /* now pointing at shift states */
1250 for (i = 1; i <= states; i++) {
1251 ch = *cp1++;
1252 if (ch >= SHIFT_TBL_SIZE) {
1253 pr_debug("(%d) not valid shift state (max_allowed = %d)\n",
1254 ch, SHIFT_TBL_SIZE);
1255 return -EINVAL;
1256 }
1257 spk_shift_table[ch] = i;
1258 }
1259 keymap_flags = *cp1++;
1260 while ((ch = *cp1)) {
1261 if (ch >= MAX_KEY) {
1262 pr_debug("(%d), not valid key, (max_allowed = %d)\n",
1263 ch, MAX_KEY);
1264 return -EINVAL;
1265 }
1266 spk_our_keys[ch] = cp1;
1267 cp1 += states + 1;
1268 }
1269 return 0;
1270 }
1271
1272 static struct var_t spk_vars[] = {
1273 /* bell must be first to set high limit */
1274 {BELL_POS, .u.n = {NULL, 0, 0, 0, 0, 0, NULL} },
1275 {SPELL_DELAY, .u.n = {NULL, 0, 0, 4, 0, 0, NULL} },
1276 {ATTRIB_BLEEP, .u.n = {NULL, 1, 0, 3, 0, 0, NULL} },
1277 {BLEEPS, .u.n = {NULL, 3, 0, 3, 0, 0, NULL} },
1278 {BLEEP_TIME, .u.n = {NULL, 30, 1, 200, 0, 0, NULL} },
1279 {PUNC_LEVEL, .u.n = {NULL, 1, 0, 4, 0, 0, NULL} },
1280 {READING_PUNC, .u.n = {NULL, 1, 0, 4, 0, 0, NULL} },
1281 {CURSOR_TIME, .u.n = {NULL, 120, 50, 600, 0, 0, NULL} },
1282 {SAY_CONTROL, TOGGLE_0},
1283 {SAY_WORD_CTL, TOGGLE_0},
1284 {NO_INTERRUPT, TOGGLE_0},
1285 {KEY_ECHO, .u.n = {NULL, 1, 0, 2, 0, 0, NULL} },
1286 V_LAST_VAR
1287 };
1288
toggle_cursoring(struct vc_data * vc)1289 static void toggle_cursoring(struct vc_data *vc)
1290 {
1291 if (cursor_track == read_all_mode)
1292 cursor_track = prev_cursor_track;
1293 if (++cursor_track >= CT_Max)
1294 cursor_track = 0;
1295 synth_printf("%s\n", spk_msg_get(MSG_CURSOR_MSGS_START + cursor_track));
1296 }
1297
spk_reset_default_chars(void)1298 void spk_reset_default_chars(void)
1299 {
1300 int i;
1301
1302 /* First, free any non-default */
1303 for (i = 0; i < 256; i++) {
1304 if (spk_characters[i] &&
1305 (spk_characters[i] != spk_default_chars[i]))
1306 kfree(spk_characters[i]);
1307 }
1308
1309 memcpy(spk_characters, spk_default_chars, sizeof(spk_default_chars));
1310 }
1311
spk_reset_default_chartab(void)1312 void spk_reset_default_chartab(void)
1313 {
1314 memcpy(spk_chartab, default_chartab, sizeof(default_chartab));
1315 }
1316
1317 static const struct st_bits_data *pb_edit;
1318
edit_bits(struct vc_data * vc,u_char type,u_char ch,u_short key)1319 static int edit_bits(struct vc_data *vc, u_char type, u_char ch, u_short key)
1320 {
1321 short mask = pb_edit->mask, ch_type = spk_chartab[ch];
1322
1323 if (type != KT_LATIN || (ch_type & B_NUM) || ch < SPACE)
1324 return -1;
1325 if (ch == SPACE) {
1326 synth_printf("%s\n", spk_msg_get(MSG_EDIT_DONE));
1327 spk_special_handler = NULL;
1328 return 1;
1329 }
1330 if (mask < PUNC && !(ch_type & PUNC))
1331 return -1;
1332 spk_chartab[ch] ^= mask;
1333 speak_char(ch);
1334 synth_printf(" %s\n",
1335 (spk_chartab[ch] & mask) ? spk_msg_get(MSG_ON) :
1336 spk_msg_get(MSG_OFF));
1337 return 1;
1338 }
1339
1340 /* Allocation concurrency is protected by the console semaphore */
speakup_allocate(struct vc_data * vc,gfp_t gfp_flags)1341 static int speakup_allocate(struct vc_data *vc, gfp_t gfp_flags)
1342 {
1343 int vc_num;
1344
1345 vc_num = vc->vc_num;
1346 if (!speakup_console[vc_num]) {
1347 speakup_console[vc_num] = kzalloc(sizeof(*speakup_console[0]),
1348 gfp_flags);
1349 if (!speakup_console[vc_num])
1350 return -ENOMEM;
1351 speakup_date(vc);
1352 } else if (!spk_parked) {
1353 speakup_date(vc);
1354 }
1355
1356 return 0;
1357 }
1358
speakup_deallocate(struct vc_data * vc)1359 static void speakup_deallocate(struct vc_data *vc)
1360 {
1361 int vc_num;
1362
1363 vc_num = vc->vc_num;
1364 kfree(speakup_console[vc_num]);
1365 speakup_console[vc_num] = NULL;
1366 }
1367
1368 static u_char is_cursor;
1369 static u_long old_cursor_pos, old_cursor_x, old_cursor_y;
1370 static int cursor_con;
1371
1372 static void reset_highlight_buffers(struct vc_data *);
1373
1374 static int read_all_key;
1375
1376 static int in_keyboard_notifier;
1377
1378 static void start_read_all_timer(struct vc_data *vc, int command);
1379
1380 enum {
1381 RA_NOTHING,
1382 RA_NEXT_SENT,
1383 RA_PREV_LINE,
1384 RA_NEXT_LINE,
1385 RA_PREV_SENT,
1386 RA_DOWN_ARROW,
1387 RA_TIMER,
1388 RA_FIND_NEXT_SENT,
1389 RA_FIND_PREV_SENT,
1390 };
1391
kbd_fakekey2(struct vc_data * vc,int command)1392 static void kbd_fakekey2(struct vc_data *vc, int command)
1393 {
1394 del_timer(&cursor_timer);
1395 speakup_fake_down_arrow();
1396 start_read_all_timer(vc, command);
1397 }
1398
read_all_doc(struct vc_data * vc)1399 static void read_all_doc(struct vc_data *vc)
1400 {
1401 if ((vc->vc_num != fg_console) || !synth || spk_shut_up)
1402 return;
1403 if (!synth_supports_indexing())
1404 return;
1405 if (cursor_track != read_all_mode)
1406 prev_cursor_track = cursor_track;
1407 cursor_track = read_all_mode;
1408 spk_reset_index_count(0);
1409 if (get_sentence_buf(vc, 0) == -1) {
1410 del_timer(&cursor_timer);
1411 if (!in_keyboard_notifier)
1412 speakup_fake_down_arrow();
1413 start_read_all_timer(vc, RA_DOWN_ARROW);
1414 } else {
1415 say_sentence_num(0, 0);
1416 synth_insert_next_index(0);
1417 start_read_all_timer(vc, RA_TIMER);
1418 }
1419 }
1420
stop_read_all(struct vc_data * vc)1421 static void stop_read_all(struct vc_data *vc)
1422 {
1423 del_timer(&cursor_timer);
1424 cursor_track = prev_cursor_track;
1425 spk_shut_up &= 0xfe;
1426 spk_do_flush();
1427 }
1428
start_read_all_timer(struct vc_data * vc,int command)1429 static void start_read_all_timer(struct vc_data *vc, int command)
1430 {
1431 struct var_t *cursor_timeout;
1432
1433 cursor_con = vc->vc_num;
1434 read_all_key = command;
1435 cursor_timeout = spk_get_var(CURSOR_TIME);
1436 mod_timer(&cursor_timer,
1437 jiffies + msecs_to_jiffies(cursor_timeout->u.n.value));
1438 }
1439
handle_cursor_read_all(struct vc_data * vc,int command)1440 static void handle_cursor_read_all(struct vc_data *vc, int command)
1441 {
1442 int indcount, sentcount, rv, sn;
1443
1444 switch (command) {
1445 case RA_NEXT_SENT:
1446 /* Get Current Sentence */
1447 spk_get_index_count(&indcount, &sentcount);
1448 /*printk("%d %d ", indcount, sentcount); */
1449 spk_reset_index_count(sentcount + 1);
1450 if (indcount == 1) {
1451 if (!say_sentence_num(sentcount + 1, 0)) {
1452 kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
1453 return;
1454 }
1455 synth_insert_next_index(0);
1456 } else {
1457 sn = 0;
1458 if (!say_sentence_num(sentcount + 1, 1)) {
1459 sn = 1;
1460 spk_reset_index_count(sn);
1461 } else {
1462 synth_insert_next_index(0);
1463 }
1464 if (!say_sentence_num(sn, 0)) {
1465 kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
1466 return;
1467 }
1468 synth_insert_next_index(0);
1469 }
1470 start_read_all_timer(vc, RA_TIMER);
1471 break;
1472 case RA_PREV_SENT:
1473 break;
1474 case RA_NEXT_LINE:
1475 read_all_doc(vc);
1476 break;
1477 case RA_PREV_LINE:
1478 break;
1479 case RA_DOWN_ARROW:
1480 if (get_sentence_buf(vc, 0) == -1) {
1481 kbd_fakekey2(vc, RA_DOWN_ARROW);
1482 } else {
1483 say_sentence_num(0, 0);
1484 synth_insert_next_index(0);
1485 start_read_all_timer(vc, RA_TIMER);
1486 }
1487 break;
1488 case RA_FIND_NEXT_SENT:
1489 rv = get_sentence_buf(vc, 0);
1490 if (rv == -1)
1491 read_all_doc(vc);
1492 if (rv == 0) {
1493 kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
1494 } else {
1495 say_sentence_num(1, 0);
1496 synth_insert_next_index(0);
1497 start_read_all_timer(vc, RA_TIMER);
1498 }
1499 break;
1500 case RA_FIND_PREV_SENT:
1501 break;
1502 case RA_TIMER:
1503 spk_get_index_count(&indcount, &sentcount);
1504 if (indcount < 2)
1505 kbd_fakekey2(vc, RA_DOWN_ARROW);
1506 else
1507 start_read_all_timer(vc, RA_TIMER);
1508 break;
1509 }
1510 }
1511
pre_handle_cursor(struct vc_data * vc,u_char value,char up_flag)1512 static int pre_handle_cursor(struct vc_data *vc, u_char value, char up_flag)
1513 {
1514 unsigned long flags;
1515
1516 spin_lock_irqsave(&speakup_info.spinlock, flags);
1517 if (cursor_track == read_all_mode) {
1518 spk_parked &= 0xfe;
1519 if (!synth || up_flag || spk_shut_up) {
1520 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1521 return NOTIFY_STOP;
1522 }
1523 del_timer(&cursor_timer);
1524 spk_shut_up &= 0xfe;
1525 spk_do_flush();
1526 start_read_all_timer(vc, value + 1);
1527 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1528 return NOTIFY_STOP;
1529 }
1530 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1531 return NOTIFY_OK;
1532 }
1533
do_handle_cursor(struct vc_data * vc,u_char value,char up_flag)1534 static void do_handle_cursor(struct vc_data *vc, u_char value, char up_flag)
1535 {
1536 unsigned long flags;
1537 struct var_t *cursor_timeout;
1538
1539 spin_lock_irqsave(&speakup_info.spinlock, flags);
1540 spk_parked &= 0xfe;
1541 if (!synth || up_flag || spk_shut_up || cursor_track == CT_Off) {
1542 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1543 return;
1544 }
1545 spk_shut_up &= 0xfe;
1546 if (spk_no_intr)
1547 spk_do_flush();
1548 /* the key press flushes if !no_inter but we want to flush on cursor
1549 * moves regardless of no_inter state
1550 */
1551 is_cursor = value + 1;
1552 old_cursor_pos = vc->vc_pos;
1553 old_cursor_x = vc->state.x;
1554 old_cursor_y = vc->state.y;
1555 speakup_console[vc->vc_num]->ht.cy = vc->state.y;
1556 cursor_con = vc->vc_num;
1557 if (cursor_track == CT_Highlight)
1558 reset_highlight_buffers(vc);
1559 cursor_timeout = spk_get_var(CURSOR_TIME);
1560 mod_timer(&cursor_timer,
1561 jiffies + msecs_to_jiffies(cursor_timeout->u.n.value));
1562 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1563 }
1564
update_color_buffer(struct vc_data * vc,const u16 * ic,int len)1565 static void update_color_buffer(struct vc_data *vc, const u16 *ic, int len)
1566 {
1567 int i, bi, hi;
1568 int vc_num = vc->vc_num;
1569
1570 bi = (vc->vc_attr & 0x70) >> 4;
1571 hi = speakup_console[vc_num]->ht.highsize[bi];
1572
1573 i = 0;
1574 if (speakup_console[vc_num]->ht.highsize[bi] == 0) {
1575 speakup_console[vc_num]->ht.rpos[bi] = vc->vc_pos;
1576 speakup_console[vc_num]->ht.rx[bi] = vc->state.x;
1577 speakup_console[vc_num]->ht.ry[bi] = vc->state.y;
1578 }
1579 while ((hi < COLOR_BUFFER_SIZE) && (i < len)) {
1580 if (ic[i] > 32) {
1581 speakup_console[vc_num]->ht.highbuf[bi][hi] = ic[i];
1582 hi++;
1583 } else if ((ic[i] == 32) && (hi != 0)) {
1584 if (speakup_console[vc_num]->ht.highbuf[bi][hi - 1] !=
1585 32) {
1586 speakup_console[vc_num]->ht.highbuf[bi][hi] =
1587 ic[i];
1588 hi++;
1589 }
1590 }
1591 i++;
1592 }
1593 speakup_console[vc_num]->ht.highsize[bi] = hi;
1594 }
1595
reset_highlight_buffers(struct vc_data * vc)1596 static void reset_highlight_buffers(struct vc_data *vc)
1597 {
1598 int i;
1599 int vc_num = vc->vc_num;
1600
1601 for (i = 0; i < 8; i++)
1602 speakup_console[vc_num]->ht.highsize[i] = 0;
1603 }
1604
count_highlight_color(struct vc_data * vc)1605 static int count_highlight_color(struct vc_data *vc)
1606 {
1607 int i, bg;
1608 int cc;
1609 int vc_num = vc->vc_num;
1610 u16 ch;
1611 u16 *start = (u16 *)vc->vc_origin;
1612
1613 for (i = 0; i < 8; i++)
1614 speakup_console[vc_num]->ht.bgcount[i] = 0;
1615
1616 for (i = 0; i < vc->vc_rows; i++) {
1617 u16 *end = start + vc->vc_cols * 2;
1618 u16 *ptr;
1619
1620 for (ptr = start; ptr < end; ptr++) {
1621 ch = get_attributes(vc, ptr);
1622 bg = (ch & 0x70) >> 4;
1623 speakup_console[vc_num]->ht.bgcount[bg]++;
1624 }
1625 start += vc->vc_size_row;
1626 }
1627
1628 cc = 0;
1629 for (i = 0; i < 8; i++)
1630 if (speakup_console[vc_num]->ht.bgcount[i] > 0)
1631 cc++;
1632 return cc;
1633 }
1634
get_highlight_color(struct vc_data * vc)1635 static int get_highlight_color(struct vc_data *vc)
1636 {
1637 int i, j;
1638 unsigned int cptr[8];
1639 int vc_num = vc->vc_num;
1640
1641 for (i = 0; i < 8; i++)
1642 cptr[i] = i;
1643
1644 for (i = 0; i < 7; i++)
1645 for (j = i + 1; j < 8; j++)
1646 if (speakup_console[vc_num]->ht.bgcount[cptr[i]] >
1647 speakup_console[vc_num]->ht.bgcount[cptr[j]])
1648 swap(cptr[i], cptr[j]);
1649
1650 for (i = 0; i < 8; i++)
1651 if (speakup_console[vc_num]->ht.bgcount[cptr[i]] != 0)
1652 if (speakup_console[vc_num]->ht.highsize[cptr[i]] > 0)
1653 return cptr[i];
1654 return -1;
1655 }
1656
speak_highlight(struct vc_data * vc)1657 static int speak_highlight(struct vc_data *vc)
1658 {
1659 int hc, d;
1660 int vc_num = vc->vc_num;
1661
1662 if (count_highlight_color(vc) == 1)
1663 return 0;
1664 hc = get_highlight_color(vc);
1665 if (hc != -1) {
1666 d = vc->state.y - speakup_console[vc_num]->ht.cy;
1667 if ((d == 1) || (d == -1))
1668 if (speakup_console[vc_num]->ht.ry[hc] != vc->state.y)
1669 return 0;
1670 spk_parked |= 0x01;
1671 spk_do_flush();
1672 spkup_write(speakup_console[vc_num]->ht.highbuf[hc],
1673 speakup_console[vc_num]->ht.highsize[hc]);
1674 spk_pos = spk_cp = speakup_console[vc_num]->ht.rpos[hc];
1675 spk_x = spk_cx = speakup_console[vc_num]->ht.rx[hc];
1676 spk_y = spk_cy = speakup_console[vc_num]->ht.ry[hc];
1677 return 1;
1678 }
1679 return 0;
1680 }
1681
cursor_done(struct timer_list * unused)1682 static void cursor_done(struct timer_list *unused)
1683 {
1684 struct vc_data *vc = vc_cons[cursor_con].d;
1685 unsigned long flags;
1686
1687 del_timer(&cursor_timer);
1688 spin_lock_irqsave(&speakup_info.spinlock, flags);
1689 if (cursor_con != fg_console) {
1690 is_cursor = 0;
1691 goto out;
1692 }
1693 speakup_date(vc);
1694 if (win_enabled) {
1695 if (vc->state.x >= win_left && vc->state.x <= win_right &&
1696 vc->state.y >= win_top && vc->state.y <= win_bottom) {
1697 spk_keydown = 0;
1698 is_cursor = 0;
1699 goto out;
1700 }
1701 }
1702 if (cursor_track == read_all_mode) {
1703 handle_cursor_read_all(vc, read_all_key);
1704 goto out;
1705 }
1706 if (cursor_track == CT_Highlight) {
1707 if (speak_highlight(vc)) {
1708 spk_keydown = 0;
1709 is_cursor = 0;
1710 goto out;
1711 }
1712 }
1713 if (cursor_track == CT_Window)
1714 speakup_win_say(vc);
1715 else if (is_cursor == 1 || is_cursor == 4)
1716 say_line_from_to(vc, 0, vc->vc_cols, 0);
1717 else
1718 say_char(vc);
1719 spk_keydown = 0;
1720 is_cursor = 0;
1721 out:
1722 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1723 }
1724
1725 /* called by: vt_notifier_call() */
speakup_bs(struct vc_data * vc)1726 static void speakup_bs(struct vc_data *vc)
1727 {
1728 unsigned long flags;
1729
1730 if (!speakup_console[vc->vc_num])
1731 return;
1732 if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
1733 /* Speakup output, discard */
1734 return;
1735 if (!spk_parked)
1736 speakup_date(vc);
1737 if (spk_shut_up || !synth) {
1738 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1739 return;
1740 }
1741 if (vc->vc_num == fg_console && spk_keydown) {
1742 spk_keydown = 0;
1743 if (!is_cursor)
1744 say_char(vc);
1745 }
1746 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1747 }
1748
1749 /* called by: vt_notifier_call() */
speakup_con_write(struct vc_data * vc,u16 * str,int len)1750 static void speakup_con_write(struct vc_data *vc, u16 *str, int len)
1751 {
1752 unsigned long flags;
1753
1754 if ((vc->vc_num != fg_console) || spk_shut_up || !synth)
1755 return;
1756 if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
1757 /* Speakup output, discard */
1758 return;
1759 if (spk_bell_pos && spk_keydown && (vc->state.x == spk_bell_pos - 1))
1760 bleep(3);
1761 if ((is_cursor) || (cursor_track == read_all_mode)) {
1762 if (cursor_track == CT_Highlight)
1763 update_color_buffer(vc, str, len);
1764 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1765 return;
1766 }
1767 if (win_enabled) {
1768 if (vc->state.x >= win_left && vc->state.x <= win_right &&
1769 vc->state.y >= win_top && vc->state.y <= win_bottom) {
1770 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1771 return;
1772 }
1773 }
1774
1775 spkup_write(str, len);
1776 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1777 }
1778
speakup_con_update(struct vc_data * vc)1779 static void speakup_con_update(struct vc_data *vc)
1780 {
1781 unsigned long flags;
1782
1783 if (!speakup_console[vc->vc_num] || spk_parked)
1784 return;
1785 if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
1786 /* Speakup output, discard */
1787 return;
1788 speakup_date(vc);
1789 if (vc->vc_mode == KD_GRAPHICS && !spk_paused && spk_str_pause[0]) {
1790 synth_printf("%s", spk_str_pause);
1791 spk_paused = true;
1792 }
1793 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1794 }
1795
do_handle_spec(struct vc_data * vc,u_char value,char up_flag)1796 static void do_handle_spec(struct vc_data *vc, u_char value, char up_flag)
1797 {
1798 unsigned long flags;
1799 int on_off = 2;
1800 char *label;
1801
1802 if (!synth || up_flag || spk_killed)
1803 return;
1804 spin_lock_irqsave(&speakup_info.spinlock, flags);
1805 spk_shut_up &= 0xfe;
1806 if (spk_no_intr)
1807 spk_do_flush();
1808 switch (value) {
1809 case KVAL(K_CAPS):
1810 label = spk_msg_get(MSG_KEYNAME_CAPSLOCK);
1811 on_off = vt_get_leds(fg_console, VC_CAPSLOCK);
1812 break;
1813 case KVAL(K_NUM):
1814 label = spk_msg_get(MSG_KEYNAME_NUMLOCK);
1815 on_off = vt_get_leds(fg_console, VC_NUMLOCK);
1816 break;
1817 case KVAL(K_HOLD):
1818 label = spk_msg_get(MSG_KEYNAME_SCROLLLOCK);
1819 on_off = vt_get_leds(fg_console, VC_SCROLLOCK);
1820 if (speakup_console[vc->vc_num])
1821 speakup_console[vc->vc_num]->tty_stopped = on_off;
1822 break;
1823 default:
1824 spk_parked &= 0xfe;
1825 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1826 return;
1827 }
1828 if (on_off < 2)
1829 synth_printf("%s %s\n",
1830 label, spk_msg_get(MSG_STATUS_START + on_off));
1831 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
1832 }
1833
inc_dec_var(u_char value)1834 static int inc_dec_var(u_char value)
1835 {
1836 struct st_var_header *p_header;
1837 struct var_t *var_data;
1838 char num_buf[32];
1839 char *cp = num_buf;
1840 char *pn;
1841 int var_id = (int)value - VAR_START;
1842 int how = (var_id & 1) ? E_INC : E_DEC;
1843
1844 var_id = var_id / 2 + FIRST_SET_VAR;
1845 p_header = spk_get_var_header(var_id);
1846 if (!p_header)
1847 return -1;
1848 if (p_header->var_type != VAR_NUM)
1849 return -1;
1850 var_data = p_header->data;
1851 if (spk_set_num_var(1, p_header, how) != 0)
1852 return -1;
1853 if (!spk_close_press) {
1854 for (pn = p_header->name; *pn; pn++) {
1855 if (*pn == '_')
1856 *cp = SPACE;
1857 else
1858 *cp++ = *pn;
1859 }
1860 }
1861 snprintf(cp, sizeof(num_buf) - (cp - num_buf), " %d ",
1862 var_data->u.n.value);
1863 synth_printf("%s", num_buf);
1864 return 0;
1865 }
1866
speakup_win_set(struct vc_data * vc)1867 static void speakup_win_set(struct vc_data *vc)
1868 {
1869 char info[40];
1870
1871 if (win_start > 1) {
1872 synth_printf("%s\n", spk_msg_get(MSG_WINDOW_ALREADY_SET));
1873 return;
1874 }
1875 if (spk_x < win_left || spk_y < win_top) {
1876 synth_printf("%s\n", spk_msg_get(MSG_END_BEFORE_START));
1877 return;
1878 }
1879 if (win_start && spk_x == win_left && spk_y == win_top) {
1880 win_left = 0;
1881 win_right = vc->vc_cols - 1;
1882 win_bottom = spk_y;
1883 snprintf(info, sizeof(info), spk_msg_get(MSG_WINDOW_LINE),
1884 (int)win_top + 1);
1885 } else {
1886 if (!win_start) {
1887 win_top = spk_y;
1888 win_left = spk_x;
1889 } else {
1890 win_bottom = spk_y;
1891 win_right = spk_x;
1892 }
1893 snprintf(info, sizeof(info), spk_msg_get(MSG_WINDOW_BOUNDARY),
1894 (win_start) ?
1895 spk_msg_get(MSG_END) : spk_msg_get(MSG_START),
1896 (int)spk_y + 1, (int)spk_x + 1);
1897 }
1898 synth_printf("%s\n", info);
1899 win_start++;
1900 }
1901
speakup_win_clear(struct vc_data * vc)1902 static void speakup_win_clear(struct vc_data *vc)
1903 {
1904 win_top = 0;
1905 win_bottom = 0;
1906 win_left = 0;
1907 win_right = 0;
1908 win_start = 0;
1909 synth_printf("%s\n", spk_msg_get(MSG_WINDOW_CLEARED));
1910 }
1911
speakup_win_enable(struct vc_data * vc)1912 static void speakup_win_enable(struct vc_data *vc)
1913 {
1914 if (win_start < 2) {
1915 synth_printf("%s\n", spk_msg_get(MSG_NO_WINDOW));
1916 return;
1917 }
1918 win_enabled ^= 1;
1919 if (win_enabled)
1920 synth_printf("%s\n", spk_msg_get(MSG_WINDOW_SILENCED));
1921 else
1922 synth_printf("%s\n", spk_msg_get(MSG_WINDOW_SILENCE_DISABLED));
1923 }
1924
speakup_bits(struct vc_data * vc)1925 static void speakup_bits(struct vc_data *vc)
1926 {
1927 int val = this_speakup_key - (FIRST_EDIT_BITS - 1);
1928
1929 if (spk_special_handler || val < 1 || val > 6) {
1930 synth_printf("%s\n", spk_msg_get(MSG_ERROR));
1931 return;
1932 }
1933 pb_edit = &spk_punc_info[val];
1934 synth_printf(spk_msg_get(MSG_EDIT_PROMPT), pb_edit->name);
1935 spk_special_handler = edit_bits;
1936 }
1937
handle_goto(struct vc_data * vc,u_char type,u_char ch,u_short key)1938 static int handle_goto(struct vc_data *vc, u_char type, u_char ch, u_short key)
1939 {
1940 static u_char goto_buf[8];
1941 static int num;
1942 int maxlen;
1943 char *cp;
1944 u16 wch;
1945
1946 if (type == KT_SPKUP && ch == SPEAKUP_GOTO)
1947 goto do_goto;
1948 if (type == KT_LATIN && ch == '\n')
1949 goto do_goto;
1950 if (type != 0)
1951 goto oops;
1952 if (ch == 8) {
1953 u16 wch;
1954
1955 if (num == 0)
1956 return -1;
1957 wch = goto_buf[--num];
1958 goto_buf[num] = '\0';
1959 spkup_write(&wch, 1);
1960 return 1;
1961 }
1962 if (ch < '+' || ch > 'y')
1963 goto oops;
1964 wch = ch;
1965 goto_buf[num++] = ch;
1966 goto_buf[num] = '\0';
1967 spkup_write(&wch, 1);
1968 maxlen = (*goto_buf >= '0') ? 3 : 4;
1969 if ((ch == '+' || ch == '-') && num == 1)
1970 return 1;
1971 if (ch >= '0' && ch <= '9' && num < maxlen)
1972 return 1;
1973 if (num < maxlen - 1 || num > maxlen)
1974 goto oops;
1975 if (ch < 'x' || ch > 'y') {
1976 oops:
1977 if (!spk_killed)
1978 synth_printf(" %s\n", spk_msg_get(MSG_GOTO_CANCELED));
1979 goto_buf[num = 0] = '\0';
1980 spk_special_handler = NULL;
1981 return 1;
1982 }
1983
1984 /* Do not replace with kstrtoul: here we need cp to be updated */
1985 goto_pos = simple_strtoul(goto_buf, &cp, 10);
1986
1987 if (*cp == 'x') {
1988 if (*goto_buf < '0')
1989 goto_pos += spk_x;
1990 else if (goto_pos > 0)
1991 goto_pos--;
1992
1993 if (goto_pos >= vc->vc_cols)
1994 goto_pos = vc->vc_cols - 1;
1995 goto_x = 1;
1996 } else {
1997 if (*goto_buf < '0')
1998 goto_pos += spk_y;
1999 else if (goto_pos > 0)
2000 goto_pos--;
2001
2002 if (goto_pos >= vc->vc_rows)
2003 goto_pos = vc->vc_rows - 1;
2004 goto_x = 0;
2005 }
2006 goto_buf[num = 0] = '\0';
2007 do_goto:
2008 spk_special_handler = NULL;
2009 spk_parked |= 0x01;
2010 if (goto_x) {
2011 spk_pos -= spk_x * 2;
2012 spk_x = goto_pos;
2013 spk_pos += goto_pos * 2;
2014 say_word(vc);
2015 } else {
2016 spk_y = goto_pos;
2017 spk_pos = vc->vc_origin + (goto_pos * vc->vc_size_row);
2018 say_line(vc);
2019 }
2020 return 1;
2021 }
2022
speakup_goto(struct vc_data * vc)2023 static void speakup_goto(struct vc_data *vc)
2024 {
2025 if (spk_special_handler) {
2026 synth_printf("%s\n", spk_msg_get(MSG_ERROR));
2027 return;
2028 }
2029 synth_printf("%s\n", spk_msg_get(MSG_GOTO));
2030 spk_special_handler = handle_goto;
2031 }
2032
speakup_help(struct vc_data * vc)2033 static void speakup_help(struct vc_data *vc)
2034 {
2035 spk_handle_help(vc, KT_SPKUP, SPEAKUP_HELP, 0);
2036 }
2037
do_nothing(struct vc_data * vc)2038 static void do_nothing(struct vc_data *vc)
2039 {
2040 return; /* flush done in do_spkup */
2041 }
2042
2043 static u_char key_speakup, spk_key_locked;
2044
speakup_lock(struct vc_data * vc)2045 static void speakup_lock(struct vc_data *vc)
2046 {
2047 if (!spk_key_locked) {
2048 spk_key_locked = 16;
2049 key_speakup = 16;
2050 } else {
2051 spk_key_locked = 0;
2052 key_speakup = 0;
2053 }
2054 }
2055
2056 typedef void (*spkup_hand) (struct vc_data *);
2057 static spkup_hand spkup_handler[] = {
2058 /* must be ordered same as defines in speakup.h */
2059 do_nothing, speakup_goto, speech_kill, speakup_shut_up,
2060 speakup_cut, speakup_paste, say_first_char, say_last_char,
2061 say_char, say_prev_char, say_next_char,
2062 say_word, say_prev_word, say_next_word,
2063 say_line, say_prev_line, say_next_line,
2064 top_edge, bottom_edge, left_edge, right_edge,
2065 spell_word, spell_word, say_screen,
2066 say_position, say_attributes,
2067 speakup_off, speakup_parked, say_line, /* this is for indent */
2068 say_from_top, say_to_bottom,
2069 say_from_left, say_to_right,
2070 say_char_num, speakup_bits, speakup_bits, say_phonetic_char,
2071 speakup_bits, speakup_bits, speakup_bits,
2072 speakup_win_set, speakup_win_clear, speakup_win_enable, speakup_win_say,
2073 speakup_lock, speakup_help, toggle_cursoring, read_all_doc, NULL
2074 };
2075
do_spkup(struct vc_data * vc,u_char value)2076 static void do_spkup(struct vc_data *vc, u_char value)
2077 {
2078 if (spk_killed && value != SPEECH_KILL)
2079 return;
2080 spk_keydown = 0;
2081 spk_lastkey = 0;
2082 spk_shut_up &= 0xfe;
2083 this_speakup_key = value;
2084 if (value < SPKUP_MAX_FUNC && spkup_handler[value]) {
2085 spk_do_flush();
2086 (*spkup_handler[value]) (vc);
2087 } else {
2088 if (inc_dec_var(value) < 0)
2089 bleep(9);
2090 }
2091 }
2092
2093 static const char *pad_chars = "0123456789+-*/\015,.?()";
2094
2095 static int
speakup_key(struct vc_data * vc,int shift_state,int keycode,u_short keysym,int up_flag)2096 speakup_key(struct vc_data *vc, int shift_state, int keycode, u_short keysym,
2097 int up_flag)
2098 {
2099 unsigned long flags;
2100 int kh;
2101 u_char *key_info;
2102 u_char type = KTYP(keysym), value = KVAL(keysym), new_key = 0;
2103 u_char shift_info, offset;
2104 int ret = 0;
2105
2106 if (!synth)
2107 return 0;
2108
2109 spin_lock_irqsave(&speakup_info.spinlock, flags);
2110 tty = vc->port.tty;
2111 if (type >= 0xf0)
2112 type -= 0xf0;
2113 if (type == KT_PAD &&
2114 (vt_get_leds(fg_console, VC_NUMLOCK))) {
2115 if (up_flag) {
2116 spk_keydown = 0;
2117 goto out;
2118 }
2119 value = pad_chars[value];
2120 spk_lastkey = value;
2121 spk_keydown++;
2122 spk_parked &= 0xfe;
2123 goto no_map;
2124 }
2125 if (keycode >= MAX_KEY)
2126 goto no_map;
2127 key_info = spk_our_keys[keycode];
2128 if (!key_info)
2129 goto no_map;
2130 /* Check valid read all mode keys */
2131 if ((cursor_track == read_all_mode) && (!up_flag)) {
2132 switch (value) {
2133 case KVAL(K_DOWN):
2134 case KVAL(K_UP):
2135 case KVAL(K_LEFT):
2136 case KVAL(K_RIGHT):
2137 case KVAL(K_PGUP):
2138 case KVAL(K_PGDN):
2139 break;
2140 default:
2141 stop_read_all(vc);
2142 break;
2143 }
2144 }
2145 shift_info = (shift_state & 0x0f) + key_speakup;
2146 offset = spk_shift_table[shift_info];
2147 if (offset) {
2148 new_key = key_info[offset];
2149 if (new_key) {
2150 ret = 1;
2151 if (new_key == SPK_KEY) {
2152 if (!spk_key_locked)
2153 key_speakup = (up_flag) ? 0 : 16;
2154 if (up_flag || spk_killed)
2155 goto out;
2156 spk_shut_up &= 0xfe;
2157 spk_do_flush();
2158 goto out;
2159 }
2160 if (up_flag)
2161 goto out;
2162 if (last_keycode == keycode &&
2163 time_after(last_spk_jiffy + MAX_DELAY, jiffies)) {
2164 spk_close_press = 1;
2165 offset = spk_shift_table[shift_info + 32];
2166 /* double press? */
2167 if (offset && key_info[offset])
2168 new_key = key_info[offset];
2169 }
2170 last_keycode = keycode;
2171 last_spk_jiffy = jiffies;
2172 type = KT_SPKUP;
2173 value = new_key;
2174 }
2175 }
2176 no_map:
2177 if (type == KT_SPKUP && !spk_special_handler) {
2178 do_spkup(vc, new_key);
2179 spk_close_press = 0;
2180 ret = 1;
2181 goto out;
2182 }
2183 if (up_flag || spk_killed || type == KT_SHIFT)
2184 goto out;
2185 spk_shut_up &= 0xfe;
2186 kh = (value == KVAL(K_DOWN)) ||
2187 (value == KVAL(K_UP)) ||
2188 (value == KVAL(K_LEFT)) ||
2189 (value == KVAL(K_RIGHT));
2190 if ((cursor_track != read_all_mode) || !kh)
2191 if (!spk_no_intr)
2192 spk_do_flush();
2193 if (spk_special_handler) {
2194 if (type == KT_SPEC && value == 1) {
2195 value = '\n';
2196 type = KT_LATIN;
2197 } else if (type == KT_LETTER) {
2198 type = KT_LATIN;
2199 } else if (value == 0x7f) {
2200 value = 8; /* make del = backspace */
2201 }
2202 ret = (*spk_special_handler) (vc, type, value, keycode);
2203 spk_close_press = 0;
2204 if (ret < 0)
2205 bleep(9);
2206 goto out;
2207 }
2208 last_keycode = 0;
2209 out:
2210 spin_unlock_irqrestore(&speakup_info.spinlock, flags);
2211 return ret;
2212 }
2213
keyboard_notifier_call(struct notifier_block * nb,unsigned long code,void * _param)2214 static int keyboard_notifier_call(struct notifier_block *nb,
2215 unsigned long code, void *_param)
2216 {
2217 struct keyboard_notifier_param *param = _param;
2218 struct vc_data *vc = param->vc;
2219 int up = !param->down;
2220 int ret = NOTIFY_OK;
2221 static int keycode; /* to hold the current keycode */
2222
2223 in_keyboard_notifier = 1;
2224
2225 if (vc->vc_mode == KD_GRAPHICS)
2226 goto out;
2227
2228 /*
2229 * First, determine whether we are handling a fake keypress on
2230 * the current processor. If we are, then return NOTIFY_OK,
2231 * to pass the keystroke up the chain. This prevents us from
2232 * trying to take the Speakup lock while it is held by the
2233 * processor on which the simulated keystroke was generated.
2234 * Also, the simulated keystrokes should be ignored by Speakup.
2235 */
2236
2237 if (speakup_fake_key_pressed())
2238 goto out;
2239
2240 switch (code) {
2241 case KBD_KEYCODE:
2242 /* speakup requires keycode and keysym currently */
2243 keycode = param->value;
2244 break;
2245 case KBD_UNBOUND_KEYCODE:
2246 /* not used yet */
2247 break;
2248 case KBD_UNICODE:
2249 /* not used yet */
2250 break;
2251 case KBD_KEYSYM:
2252 if (speakup_key(vc, param->shift, keycode, param->value, up))
2253 ret = NOTIFY_STOP;
2254 else if (KTYP(param->value) == KT_CUR)
2255 ret = pre_handle_cursor(vc, KVAL(param->value), up);
2256 break;
2257 case KBD_POST_KEYSYM:{
2258 unsigned char type = KTYP(param->value) - 0xf0;
2259 unsigned char val = KVAL(param->value);
2260
2261 switch (type) {
2262 case KT_SHIFT:
2263 do_handle_shift(vc, val, up);
2264 break;
2265 case KT_LATIN:
2266 case KT_LETTER:
2267 do_handle_latin(vc, val, up);
2268 break;
2269 case KT_CUR:
2270 do_handle_cursor(vc, val, up);
2271 break;
2272 case KT_SPEC:
2273 do_handle_spec(vc, val, up);
2274 break;
2275 }
2276 break;
2277 }
2278 }
2279 out:
2280 in_keyboard_notifier = 0;
2281 return ret;
2282 }
2283
vt_notifier_call(struct notifier_block * nb,unsigned long code,void * _param)2284 static int vt_notifier_call(struct notifier_block *nb,
2285 unsigned long code, void *_param)
2286 {
2287 struct vt_notifier_param *param = _param;
2288 struct vc_data *vc = param->vc;
2289
2290 switch (code) {
2291 case VT_ALLOCATE:
2292 if (vc->vc_mode == KD_TEXT)
2293 speakup_allocate(vc, GFP_ATOMIC);
2294 break;
2295 case VT_DEALLOCATE:
2296 speakup_deallocate(vc);
2297 break;
2298 case VT_WRITE:
2299 if (param->c == '\b') {
2300 speakup_bs(vc);
2301 } else {
2302 u16 d = param->c;
2303
2304 speakup_con_write(vc, &d, 1);
2305 }
2306 break;
2307 case VT_UPDATE:
2308 speakup_con_update(vc);
2309 break;
2310 }
2311 return NOTIFY_OK;
2312 }
2313
2314 /* called by: module_exit() */
speakup_exit(void)2315 static void __exit speakup_exit(void)
2316 {
2317 int i;
2318
2319 unregister_keyboard_notifier(&keyboard_notifier_block);
2320 unregister_vt_notifier(&vt_notifier_block);
2321 speakup_unregister_devsynth();
2322 speakup_cancel_selection();
2323 speakup_cancel_paste();
2324 del_timer_sync(&cursor_timer);
2325 kthread_stop(speakup_task);
2326 speakup_task = NULL;
2327 mutex_lock(&spk_mutex);
2328 synth_release();
2329 mutex_unlock(&spk_mutex);
2330 spk_ttyio_unregister_ldisc();
2331
2332 speakup_kobj_exit();
2333
2334 for (i = 0; i < MAX_NR_CONSOLES; i++)
2335 kfree(speakup_console[i]);
2336
2337 speakup_remove_virtual_keyboard();
2338
2339 for (i = 0; i < MAXVARS; i++)
2340 speakup_unregister_var(i);
2341
2342 for (i = 0; i < 256; i++) {
2343 if (spk_characters[i] != spk_default_chars[i])
2344 kfree(spk_characters[i]);
2345 }
2346
2347 spk_free_user_msgs();
2348 }
2349
2350 /* call by: module_init() */
speakup_init(void)2351 static int __init speakup_init(void)
2352 {
2353 int i;
2354 long err = 0;
2355 struct vc_data *vc = vc_cons[fg_console].d;
2356 struct var_t *var;
2357
2358 /* These first few initializations cannot fail. */
2359 spk_initialize_msgs(); /* Initialize arrays for i18n. */
2360 spk_reset_default_chars();
2361 spk_reset_default_chartab();
2362 spk_strlwr(synth_name);
2363 spk_vars[0].u.n.high = vc->vc_cols;
2364 for (var = spk_vars; var->var_id != MAXVARS; var++)
2365 speakup_register_var(var);
2366 for (var = synth_time_vars;
2367 (var->var_id >= 0) && (var->var_id < MAXVARS); var++)
2368 speakup_register_var(var);
2369 for (i = 1; spk_punc_info[i].mask != 0; i++)
2370 spk_set_mask_bits(NULL, i, 2);
2371
2372 spk_set_key_info(spk_key_defaults, spk_key_buf);
2373
2374 /* From here on out, initializations can fail. */
2375 err = speakup_add_virtual_keyboard();
2376 if (err)
2377 goto error_virtkeyboard;
2378
2379 for (i = 0; i < MAX_NR_CONSOLES; i++)
2380 if (vc_cons[i].d) {
2381 err = speakup_allocate(vc_cons[i].d, GFP_KERNEL);
2382 if (err)
2383 goto error_kobjects;
2384 }
2385
2386 if (spk_quiet_boot)
2387 spk_shut_up |= 0x01;
2388
2389 err = speakup_kobj_init();
2390 if (err)
2391 goto error_kobjects;
2392
2393 spk_ttyio_register_ldisc();
2394 synth_init(synth_name);
2395 speakup_register_devsynth();
2396 /*
2397 * register_devsynth might fail, but this error is not fatal.
2398 * /dev/synth is an extra feature; the rest of Speakup
2399 * will work fine without it.
2400 */
2401
2402 err = register_keyboard_notifier(&keyboard_notifier_block);
2403 if (err)
2404 goto error_kbdnotifier;
2405 err = register_vt_notifier(&vt_notifier_block);
2406 if (err)
2407 goto error_vtnotifier;
2408
2409 speakup_task = kthread_create(speakup_thread, NULL, "speakup");
2410
2411 if (IS_ERR(speakup_task)) {
2412 err = PTR_ERR(speakup_task);
2413 goto error_task;
2414 }
2415
2416 set_user_nice(speakup_task, 10);
2417 wake_up_process(speakup_task);
2418
2419 pr_info("speakup %s: initialized\n", SPEAKUP_VERSION);
2420 pr_info("synth name on entry is: %s\n", synth_name);
2421 goto out;
2422
2423 error_task:
2424 unregister_vt_notifier(&vt_notifier_block);
2425
2426 error_vtnotifier:
2427 unregister_keyboard_notifier(&keyboard_notifier_block);
2428 del_timer(&cursor_timer);
2429
2430 error_kbdnotifier:
2431 speakup_unregister_devsynth();
2432 mutex_lock(&spk_mutex);
2433 synth_release();
2434 mutex_unlock(&spk_mutex);
2435 speakup_kobj_exit();
2436
2437 error_kobjects:
2438 for (i = 0; i < MAX_NR_CONSOLES; i++)
2439 kfree(speakup_console[i]);
2440
2441 speakup_remove_virtual_keyboard();
2442
2443 error_virtkeyboard:
2444 for (i = 0; i < MAXVARS; i++)
2445 speakup_unregister_var(i);
2446
2447 for (i = 0; i < 256; i++) {
2448 if (spk_characters[i] != spk_default_chars[i])
2449 kfree(spk_characters[i]);
2450 }
2451
2452 spk_free_user_msgs();
2453
2454 out:
2455 return err;
2456 }
2457
2458 module_init(speakup_init);
2459 module_exit(speakup_exit);
2460