Merge tag 'tty-5.9-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/tty
authorLinus Torvalds <torvalds@linux-foundation.org>
Thu, 6 Aug 2020 21:56:11 +0000 (14:56 -0700)
committerLinus Torvalds <torvalds@linux-foundation.org>
Thu, 6 Aug 2020 21:56:11 +0000 (14:56 -0700)
Pull tty/serial updates from Greg KH:
 "Here is the large set of TTY and Serial driver patches for 5.9-rc1.

  Lots of bugfixes in here, thanks to syzbot fuzzing for serial and vt
  and console code.

  Other highlights include:

   - much needed vt/vc code cleanup from Jiri Slaby

   - 8250 driver fixes and additions

   - various serial driver updates and feature enhancements

   - locking cleanup for serial/console initializations

   - other minor cleanups

  All of these have been in linux-next with no reported issues"

* tag 'tty-5.9-rc1' of git://git.kernel.org/pub/scm/linux/kernel/git/gregkh/tty: (90 commits)
  MAINTAINERS: enlist Greg formally for console stuff
  vgacon: Fix for missing check in scrollback handling
  Revert "serial: 8250: Let serial core initialise spin lock"
  serial: 8250: Let serial core initialise spin lock
  tty: keyboard, do not speculate on func_table index
  serial: stm32: Add RS485 RTS GPIO control
  serial: 8250_dw: Fix common clocks usage race condition
  serial: 8250_dw: Pass the same rate to the clk round and set rate methods
  serial: 8250_dw: Simplify the ref clock rate setting procedure
  serial: 8250: Add 8250 port clock update method
  tty: serial: imx: add imx earlycon driver
  tty: serial: imx: enable imx serial console port as module
  tty/synclink: remove leftover bits of non-PCI card support
  tty: Use the preferred form for passing the size of a structure type
  tty: Fix identation issues in struct serial_struct32
  tty: Avoid the use of one-element arrays
  serial: msm_serial: add sparse context annotation
  serial: pmac_zilog: add sparse context annotation
  newport_con: vc_color is now in state
  serial: imx: use hrtimers for rs485 delays
  ...

1  2 
MAINTAINERS
drivers/accessibility/speakup/main.c
drivers/tty/serial/qcom_geni_serial.c
drivers/tty/vt/keyboard.c
drivers/usb/misc/sisusbvga/sisusb_con.c
drivers/video/console/newport_con.c
drivers/video/fbdev/core/fbcon.c

diff --cc MAINTAINERS
Simple merge
index 02471d932d71f698bb596380f4cbceabef74bd12,0000000000000000000000000000000000000000..ddfd12afe3b9a6f1303ef39108a2f2ba7611ac95
mode 100644,000000..100644
--- /dev/null
@@@ -1,2460 -1,0 +1,2460 @@@
-       spk_x = spk_cx = vc->vc_x;
-       spk_y = spk_cy = vc->vc_y;
 +// SPDX-License-Identifier: GPL-2.0+
 +/* speakup.c
 + * review functions for the speakup screen review package.
 + * originally written by: Kirk Reiser and Andy Berdan.
 + *
 + * extensively modified by David Borowski.
 + *
 + ** Copyright (C) 1998  Kirk Reiser.
 + *  Copyright (C) 2003  David Borowski.
 + */
 +
 +#include <linux/kernel.h>
 +#include <linux/vt.h>
 +#include <linux/tty.h>
 +#include <linux/mm.h>         /* __get_free_page() and friends */
 +#include <linux/vt_kern.h>
 +#include <linux/ctype.h>
 +#include <linux/selection.h>
 +#include <linux/unistd.h>
 +#include <linux/jiffies.h>
 +#include <linux/kthread.h>
 +#include <linux/keyboard.h>   /* for KT_SHIFT */
 +#include <linux/kbd_kern.h>   /* for vc_kbd_* and friends */
 +#include <linux/input.h>
 +#include <linux/kmod.h>
 +
 +/* speakup_*_selection */
 +#include <linux/module.h>
 +#include <linux/sched.h>
 +#include <linux/slab.h>
 +#include <linux/types.h>
 +#include <linux/consolemap.h>
 +
 +#include <linux/spinlock.h>
 +#include <linux/notifier.h>
 +
 +#include <linux/uaccess.h>    /* copy_from|to|user() and others */
 +
 +#include "spk_priv.h"
 +#include "speakup.h"
 +
 +#define MAX_DELAY msecs_to_jiffies(500)
 +#define MINECHOCHAR SPACE
 +
 +MODULE_AUTHOR("Kirk Reiser <kirk@braille.uwo.ca>");
 +MODULE_AUTHOR("Daniel Drake <dsd@gentoo.org>");
 +MODULE_DESCRIPTION("Speakup console speech");
 +MODULE_LICENSE("GPL");
 +MODULE_VERSION(SPEAKUP_VERSION);
 +
 +char *synth_name;
 +module_param_named(synth, synth_name, charp, 0444);
 +module_param_named(quiet, spk_quiet_boot, bool, 0444);
 +
 +MODULE_PARM_DESC(synth, "Synth to start if speakup is built in.");
 +MODULE_PARM_DESC(quiet, "Do not announce when the synthesizer is found.");
 +
 +special_func spk_special_handler;
 +
 +short spk_pitch_shift, synth_flags;
 +static u16 buf[256];
 +int spk_attrib_bleep, spk_bleeps, spk_bleep_time = 10;
 +int spk_no_intr, spk_spell_delay;
 +int spk_key_echo, spk_say_word_ctl;
 +int spk_say_ctrl, spk_bell_pos;
 +short spk_punc_mask;
 +int spk_punc_level, spk_reading_punc;
 +char spk_str_caps_start[MAXVARLEN + 1] = "\0";
 +char spk_str_caps_stop[MAXVARLEN + 1] = "\0";
 +char spk_str_pause[MAXVARLEN + 1] = "\0";
 +bool spk_paused;
 +const struct st_bits_data spk_punc_info[] = {
 +      {"none", "", 0},
 +      {"some", "/$%&@", SOME},
 +      {"most", "$%&#()=+*/@^<>|\\", MOST},
 +      {"all", "!\"#$%&'()*+,-./:;<=>?@[\\]^_`{|}~", PUNC},
 +      {"delimiters", "", B_WDLM},
 +      {"repeats", "()", CH_RPT},
 +      {"extended numeric", "", B_EXNUM},
 +      {"symbols", "", B_SYM},
 +      {NULL, NULL}
 +};
 +
 +static char mark_cut_flag;
 +#define MAX_KEY 160
 +static u_char *spk_shift_table;
 +u_char *spk_our_keys[MAX_KEY];
 +u_char spk_key_buf[600];
 +const u_char spk_key_defaults[] = {
 +#include "speakupmap.h"
 +};
 +
 +/* Speakup Cursor Track Variables */
 +static int cursor_track = 1, prev_cursor_track = 1;
 +
 +/* cursor track modes, must be ordered same as cursor_msgs */
 +enum {
 +      CT_Off = 0,
 +      CT_On,
 +      CT_Highlight,
 +      CT_Window,
 +      CT_Max
 +};
 +
 +#define read_all_mode CT_Max
 +
 +static struct tty_struct *tty;
 +
 +static void spkup_write(const u16 *in_buf, int count);
 +
 +static char *phonetic[] = {
 +      "alfa", "bravo", "charlie", "delta", "echo", "foxtrot", "golf", "hotel",
 +      "india", "juliett", "keelo", "leema", "mike", "november", "oscar",
 +          "papa",
 +      "keh beck", "romeo", "sierra", "tango", "uniform", "victer", "whiskey",
 +      "x ray", "yankee", "zulu"
 +};
 +
 +/* array of 256 char pointers (one for each character description)
 + * initialized to default_chars and user selectable via
 + * /proc/speakup/characters
 + */
 +char *spk_characters[256];
 +
 +char *spk_default_chars[256] = {
 +/*000*/ "null", "^a", "^b", "^c", "^d", "^e", "^f", "^g",
 +/*008*/ "^h", "^i", "^j", "^k", "^l", "^m", "^n", "^o",
 +/*016*/ "^p", "^q", "^r", "^s", "^t", "^u", "^v", "^w",
 +/*024*/ "^x", "^y", "^z", "control", "control", "control", "control",
 +          "control",
 +/*032*/ "space", "bang!", "quote", "number", "dollar", "percent", "and",
 +          "tick",
 +/*040*/ "left paren", "right paren", "star", "plus", "comma", "dash",
 +          "dot",
 +      "slash",
 +/*048*/ "zero", "one", "two", "three", "four", "five", "six", "seven",
 +      "eight", "nine",
 +/*058*/ "colon", "semmy", "less", "equals", "greater", "question", "at",
 +/*065*/ "EIGH", "B", "C", "D", "E", "F", "G",
 +/*072*/ "H", "I", "J", "K", "L", "M", "N", "O",
 +/*080*/ "P", "Q", "R", "S", "T", "U", "V", "W", "X",
 +/*089*/ "Y", "ZED", "left bracket", "backslash", "right bracket",
 +          "caret",
 +      "line",
 +/*096*/ "accent", "a", "b", "c", "d", "e", "f", "g",
 +/*104*/ "h", "i", "j", "k", "l", "m", "n", "o",
 +/*112*/ "p", "q", "r", "s", "t", "u", "v", "w",
 +/*120*/ "x", "y", "zed", "left brace", "bar", "right brace", "tihlduh",
 +/*127*/ "del", "control", "control", "control", "control", "control",
 +          "control", "control", "control", "control", "control",
 +/*138*/ "control", "control", "control", "control", "control",
 +          "control", "control", "control", "control", "control",
 +          "control", "control",
 +/*150*/ "control", "control", "control", "control", "control",
 +          "control", "control", "control", "control", "control",
 +/*160*/ "nbsp", "inverted bang",
 +/*162*/ "cents", "pounds", "currency", "yen", "broken bar", "section",
 +/*168*/ "diaeresis", "copyright", "female ordinal", "double left angle",
 +/*172*/ "not", "soft hyphen", "registered", "macron",
 +/*176*/ "degrees", "plus or minus", "super two", "super three",
 +/*180*/ "acute accent", "micro", "pilcrow", "middle dot",
 +/*184*/ "cedilla", "super one", "male ordinal", "double right angle",
 +/*188*/ "one quarter", "one half", "three quarters",
 +          "inverted question",
 +/*192*/ "A GRAVE", "A ACUTE", "A CIRCUMFLEX", "A TILDE", "A OOMLAUT",
 +          "A RING",
 +/*198*/ "AE", "C CIDELLA", "E GRAVE", "E ACUTE", "E CIRCUMFLEX",
 +          "E OOMLAUT",
 +/*204*/ "I GRAVE", "I ACUTE", "I CIRCUMFLEX", "I OOMLAUT", "ETH",
 +          "N TILDE",
 +/*210*/ "O GRAVE", "O ACUTE", "O CIRCUMFLEX", "O TILDE", "O OOMLAUT",
 +/*215*/ "multiplied by", "O STROKE", "U GRAVE", "U ACUTE",
 +          "U CIRCUMFLEX",
 +/*220*/ "U OOMLAUT", "Y ACUTE", "THORN", "sharp s", "a grave",
 +/*225*/ "a acute", "a circumflex", "a tilde", "a oomlaut", "a ring",
 +/*230*/ "ae", "c cidella", "e grave", "e acute",
 +/*234*/ "e circumflex", "e oomlaut", "i grave", "i acute",
 +          "i circumflex",
 +/*239*/ "i oomlaut", "eth", "n tilde", "o grave", "o acute",
 +          "o circumflex",
 +/*245*/ "o tilde", "o oomlaut", "divided by", "o stroke", "u grave",
 +          "u acute",
 +/* 251 */ "u circumflex", "u oomlaut", "y acute", "thorn", "y oomlaut"
 +};
 +
 +/* array of 256 u_short (one for each character)
 + * initialized to default_chartab and user selectable via
 + * /sys/module/speakup/parameters/chartab
 + */
 +u_short spk_chartab[256];
 +
 +static u_short default_chartab[256] = {
 +      B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 0-7 */
 +      B_CTL, B_CTL, A_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 8-15 */
 +      B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /*16-23 */
 +      B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, B_CTL, /* 24-31 */
 +      WDLM, A_PUNC, PUNC, PUNC, PUNC, PUNC, PUNC, A_PUNC,     /*  !"#$%&' */
 +      PUNC, PUNC, PUNC, PUNC, A_PUNC, A_PUNC, A_PUNC, PUNC,   /* ()*+, -./ */
 +      NUM, NUM, NUM, NUM, NUM, NUM, NUM, NUM, /* 01234567 */
 +      NUM, NUM, A_PUNC, PUNC, PUNC, PUNC, PUNC, A_PUNC,       /* 89:;<=>? */
 +      PUNC, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP,  /* @ABCDEFG */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* HIJKLMNO */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* PQRSTUVW */
 +      A_CAP, A_CAP, A_CAP, PUNC, PUNC, PUNC, PUNC, PUNC,      /* XYZ[\]^_ */
 +      PUNC, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA,  /* `abcdefg */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* hijklmno */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* pqrstuvw */
 +      ALPHA, ALPHA, ALPHA, PUNC, PUNC, PUNC, PUNC, 0, /* xyz{|}~ */
 +      B_CAPSYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 128-134 */
 +      B_SYM,  /* 135 */
 +      B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 136-142 */
 +      B_CAPSYM,       /* 143 */
 +      B_CAPSYM, B_CAPSYM, B_SYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, /* 144-150 */
 +      B_SYM,  /* 151 */
 +      B_SYM, B_SYM, B_CAPSYM, B_CAPSYM, B_SYM, B_SYM, B_SYM, /*152-158 */
 +      B_SYM,  /* 159 */
 +      WDLM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_CAPSYM, /* 160-166 */
 +      B_SYM,  /* 167 */
 +      B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 168-175 */
 +      B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 176-183 */
 +      B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, B_SYM, /* 184-191 */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* 192-199 */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, /* 200-207 */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, B_SYM, /* 208-215 */
 +      A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, A_CAP, ALPHA, /* 216-223 */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* 224-231 */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, /* 232-239 */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, B_SYM, /* 240-247 */
 +      ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA, ALPHA  /* 248-255 */
 +};
 +
 +struct task_struct *speakup_task;
 +struct bleep spk_unprocessed_sound;
 +static int spk_keydown;
 +static u16 spk_lastkey;
 +static u_char spk_close_press, keymap_flags;
 +static u_char last_keycode, this_speakup_key;
 +static u_long last_spk_jiffy;
 +
 +struct st_spk_t *speakup_console[MAX_NR_CONSOLES];
 +
 +DEFINE_MUTEX(spk_mutex);
 +
 +static int keyboard_notifier_call(struct notifier_block *,
 +                                unsigned long code, void *param);
 +
 +static struct notifier_block keyboard_notifier_block = {
 +      .notifier_call = keyboard_notifier_call,
 +};
 +
 +static int vt_notifier_call(struct notifier_block *,
 +                          unsigned long code, void *param);
 +
 +static struct notifier_block vt_notifier_block = {
 +      .notifier_call = vt_notifier_call,
 +};
 +
 +static unsigned char get_attributes(struct vc_data *vc, u16 *pos)
 +{
 +      pos = screen_pos(vc, pos - (u16 *)vc->vc_origin, 1);
 +      return (scr_readw(pos) & ~vc->vc_hi_font_mask) >> 8;
 +}
 +
 +static void speakup_date(struct vc_data *vc)
 +{
-       old_cursor_x = vc->vc_x;
-       old_cursor_y = vc->vc_y;
-       speakup_console[vc->vc_num]->ht.cy = vc->vc_y;
++      spk_x = spk_cx = vc->state.x;
++      spk_y = spk_cy = vc->state.y;
 +      spk_pos = spk_cp = vc->vc_pos;
 +      spk_old_attr = spk_attr;
 +      spk_attr = get_attributes(vc, (u_short *)spk_pos);
 +}
 +
 +static void bleep(u_short val)
 +{
 +      static const short vals[] = {
 +              350, 370, 392, 414, 440, 466, 491, 523, 554, 587, 619, 659
 +      };
 +      short freq;
 +      int time = spk_bleep_time;
 +
 +      freq = vals[val % 12];
 +      if (val > 11)
 +              freq *= (1 << (val / 12));
 +      spk_unprocessed_sound.freq = freq;
 +      spk_unprocessed_sound.jiffies = msecs_to_jiffies(time);
 +      spk_unprocessed_sound.active = 1;
 +      /* We can only have 1 active sound at a time. */
 +}
 +
 +static void speakup_shut_up(struct vc_data *vc)
 +{
 +      if (spk_killed)
 +              return;
 +      spk_shut_up |= 0x01;
 +      spk_parked &= 0xfe;
 +      speakup_date(vc);
 +      if (synth)
 +              spk_do_flush();
 +}
 +
 +static void speech_kill(struct vc_data *vc)
 +{
 +      char val = synth->is_alive(synth);
 +
 +      if (val == 0)
 +              return;
 +
 +      /* re-enables synth, if disabled */
 +      if (val == 2 || spk_killed) {
 +              /* dead */
 +              spk_shut_up &= ~0x40;
 +              synth_printf("%s\n", spk_msg_get(MSG_IAM_ALIVE));
 +      } else {
 +              synth_printf("%s\n", spk_msg_get(MSG_YOU_KILLED_SPEAKUP));
 +              spk_shut_up |= 0x40;
 +      }
 +}
 +
 +static void speakup_off(struct vc_data *vc)
 +{
 +      if (spk_shut_up & 0x80) {
 +              spk_shut_up &= 0x7f;
 +              synth_printf("%s\n", spk_msg_get(MSG_HEY_THATS_BETTER));
 +      } else {
 +              spk_shut_up |= 0x80;
 +              synth_printf("%s\n", spk_msg_get(MSG_YOU_TURNED_ME_OFF));
 +      }
 +      speakup_date(vc);
 +}
 +
 +static void speakup_parked(struct vc_data *vc)
 +{
 +      if (spk_parked & 0x80) {
 +              spk_parked = 0;
 +              synth_printf("%s\n", spk_msg_get(MSG_UNPARKED));
 +      } else {
 +              spk_parked |= 0x80;
 +              synth_printf("%s\n", spk_msg_get(MSG_PARKED));
 +      }
 +}
 +
 +static void speakup_cut(struct vc_data *vc)
 +{
 +      static const char err_buf[] = "set selection failed";
 +      int ret;
 +
 +      if (!mark_cut_flag) {
 +              mark_cut_flag = 1;
 +              spk_xs = (u_short)spk_x;
 +              spk_ys = (u_short)spk_y;
 +              spk_sel_cons = vc;
 +              synth_printf("%s\n", spk_msg_get(MSG_MARK));
 +              return;
 +      }
 +      spk_xe = (u_short)spk_x;
 +      spk_ye = (u_short)spk_y;
 +      mark_cut_flag = 0;
 +      synth_printf("%s\n", spk_msg_get(MSG_CUT));
 +
 +      speakup_clear_selection();
 +      ret = speakup_set_selection(tty);
 +
 +      switch (ret) {
 +      case 0:
 +              break;          /* no error */
 +      case -EFAULT:
 +              pr_warn("%sEFAULT\n", err_buf);
 +              break;
 +      case -EINVAL:
 +              pr_warn("%sEINVAL\n", err_buf);
 +              break;
 +      case -ENOMEM:
 +              pr_warn("%sENOMEM\n", err_buf);
 +              break;
 +      }
 +}
 +
 +static void speakup_paste(struct vc_data *vc)
 +{
 +      if (mark_cut_flag) {
 +              mark_cut_flag = 0;
 +              synth_printf("%s\n", spk_msg_get(MSG_MARK_CLEARED));
 +      } else {
 +              synth_printf("%s\n", spk_msg_get(MSG_PASTE));
 +              speakup_paste_selection(tty);
 +      }
 +}
 +
 +static void say_attributes(struct vc_data *vc)
 +{
 +      int fg = spk_attr & 0x0f;
 +      int bg = spk_attr >> 4;
 +
 +      if (fg > 8) {
 +              synth_printf("%s ", spk_msg_get(MSG_BRIGHT));
 +              fg -= 8;
 +      }
 +      synth_printf("%s", spk_msg_get(MSG_COLORS_START + fg));
 +      if (bg > 7) {
 +              synth_printf(" %s ", spk_msg_get(MSG_ON_BLINKING));
 +              bg -= 8;
 +      } else {
 +              synth_printf(" %s ", spk_msg_get(MSG_ON));
 +      }
 +      synth_printf("%s\n", spk_msg_get(MSG_COLORS_START + bg));
 +}
 +
 +enum {
 +      edge_top = 1,
 +      edge_bottom,
 +      edge_left,
 +      edge_right,
 +      edge_quiet
 +};
 +
 +static void announce_edge(struct vc_data *vc, int msg_id)
 +{
 +      if (spk_bleeps & 1)
 +              bleep(spk_y);
 +      if ((spk_bleeps & 2) && (msg_id < edge_quiet))
 +              synth_printf("%s\n",
 +                           spk_msg_get(MSG_EDGE_MSGS_START + msg_id - 1));
 +}
 +
 +static void speak_char(u16 ch)
 +{
 +      char *cp;
 +      struct var_t *direct = spk_get_var(DIRECT);
 +
 +      if (ch >= 0x100 || (direct && direct->u.n.value)) {
 +              if (ch < 0x100 && IS_CHAR(ch, B_CAP)) {
 +                      spk_pitch_shift++;
 +                      synth_printf("%s", spk_str_caps_start);
 +              }
 +              synth_putwc_s(ch);
 +              if (ch < 0x100 && IS_CHAR(ch, B_CAP))
 +                      synth_printf("%s", spk_str_caps_stop);
 +              return;
 +      }
 +
 +      cp = spk_characters[ch];
 +      if (!cp) {
 +              pr_info("%s: cp == NULL!\n", __func__);
 +              return;
 +      }
 +      if (IS_CHAR(ch, B_CAP)) {
 +              spk_pitch_shift++;
 +              synth_printf("%s %s %s",
 +                           spk_str_caps_start, cp, spk_str_caps_stop);
 +      } else {
 +              if (*cp == '^') {
 +                      cp++;
 +                      synth_printf(" %s%s ", spk_msg_get(MSG_CTRL), cp);
 +              } else {
 +                      synth_printf(" %s ", cp);
 +              }
 +      }
 +}
 +
 +static u16 get_char(struct vc_data *vc, u16 *pos, u_char *attribs)
 +{
 +      u16 ch = ' ';
 +
 +      if (vc && pos) {
 +              u16 w;
 +              u16 c;
 +
 +              pos = screen_pos(vc, pos - (u16 *)vc->vc_origin, 1);
 +              w = scr_readw(pos);
 +              c = w & 0xff;
 +
 +              if (w & vc->vc_hi_font_mask) {
 +                      w &= ~vc->vc_hi_font_mask;
 +                      c |= 0x100;
 +              }
 +
 +              ch = inverse_translate(vc, c, 1);
 +              *attribs = (w & 0xff00) >> 8;
 +      }
 +      return ch;
 +}
 +
 +static void say_char(struct vc_data *vc)
 +{
 +      u16 ch;
 +
 +      spk_old_attr = spk_attr;
 +      ch = get_char(vc, (u_short *)spk_pos, &spk_attr);
 +      if (spk_attr != spk_old_attr) {
 +              if (spk_attrib_bleep & 1)
 +                      bleep(spk_y);
 +              if (spk_attrib_bleep & 2)
 +                      say_attributes(vc);
 +      }
 +      speak_char(ch);
 +}
 +
 +static void say_phonetic_char(struct vc_data *vc)
 +{
 +      u16 ch;
 +
 +      spk_old_attr = spk_attr;
 +      ch = get_char(vc, (u_short *)spk_pos, &spk_attr);
 +      if (ch <= 0x7f && isalpha(ch)) {
 +              ch &= 0x1f;
 +              synth_printf("%s\n", phonetic[--ch]);
 +      } else {
 +              if (ch < 0x100 && IS_CHAR(ch, B_NUM))
 +                      synth_printf("%s ", spk_msg_get(MSG_NUMBER));
 +              speak_char(ch);
 +      }
 +}
 +
 +static void say_prev_char(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      if (spk_x == 0) {
 +              announce_edge(vc, edge_left);
 +              return;
 +      }
 +      spk_x--;
 +      spk_pos -= 2;
 +      say_char(vc);
 +}
 +
 +static void say_next_char(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      if (spk_x == vc->vc_cols - 1) {
 +              announce_edge(vc, edge_right);
 +              return;
 +      }
 +      spk_x++;
 +      spk_pos += 2;
 +      say_char(vc);
 +}
 +
 +/* get_word - will first check to see if the character under the
 + * reading cursor is a space and if spk_say_word_ctl is true it will
 + * return the word space.  If spk_say_word_ctl is not set it will check to
 + * see if there is a word starting on the next position to the right
 + * and return that word if it exists.  If it does not exist it will
 + * move left to the beginning of any previous word on the line or the
 + * beginning off the line whichever comes first..
 + */
 +
 +static u_long get_word(struct vc_data *vc)
 +{
 +      u_long cnt = 0, tmpx = spk_x, tmp_pos = spk_pos;
 +      u16 ch;
 +      u16 attr_ch;
 +      u_char temp;
 +
 +      spk_old_attr = spk_attr;
 +      ch = get_char(vc, (u_short *)tmp_pos, &temp);
 +
 +/* decided to take out the sayword if on a space (mis-information */
 +      if (spk_say_word_ctl && ch == SPACE) {
 +              *buf = '\0';
 +              synth_printf("%s\n", spk_msg_get(MSG_SPACE));
 +              return 0;
 +      } else if (tmpx < vc->vc_cols - 2 &&
 +                 (ch == SPACE || ch == 0 || (ch < 0x100 && IS_WDLM(ch))) &&
 +                 get_char(vc, (u_short *)tmp_pos + 1, &temp) > SPACE) {
 +              tmp_pos += 2;
 +              tmpx++;
 +      } else {
 +              while (tmpx > 0) {
 +                      ch = get_char(vc, (u_short *)tmp_pos - 1, &temp);
 +                      if ((ch == SPACE || ch == 0 ||
 +                           (ch < 0x100 && IS_WDLM(ch))) &&
 +                          get_char(vc, (u_short *)tmp_pos, &temp) > SPACE)
 +                              break;
 +                      tmp_pos -= 2;
 +                      tmpx--;
 +              }
 +      }
 +      attr_ch = get_char(vc, (u_short *)tmp_pos, &spk_attr);
 +      buf[cnt++] = attr_ch;
 +      while (tmpx < vc->vc_cols - 1) {
 +              tmp_pos += 2;
 +              tmpx++;
 +              ch = get_char(vc, (u_short *)tmp_pos, &temp);
 +              if (ch == SPACE || ch == 0 ||
 +                  (buf[cnt - 1] < 0x100 && IS_WDLM(buf[cnt - 1]) &&
 +                   ch > SPACE))
 +                      break;
 +              buf[cnt++] = ch;
 +      }
 +      buf[cnt] = '\0';
 +      return cnt;
 +}
 +
 +static void say_word(struct vc_data *vc)
 +{
 +      u_long cnt = get_word(vc);
 +      u_short saved_punc_mask = spk_punc_mask;
 +
 +      if (cnt == 0)
 +              return;
 +      spk_punc_mask = PUNC;
 +      buf[cnt++] = SPACE;
 +      spkup_write(buf, cnt);
 +      spk_punc_mask = saved_punc_mask;
 +}
 +
 +static void say_prev_word(struct vc_data *vc)
 +{
 +      u_char temp;
 +      u16 ch;
 +      u_short edge_said = 0, last_state = 0, state = 0;
 +
 +      spk_parked |= 0x01;
 +
 +      if (spk_x == 0) {
 +              if (spk_y == 0) {
 +                      announce_edge(vc, edge_top);
 +                      return;
 +              }
 +              spk_y--;
 +              spk_x = vc->vc_cols;
 +              edge_said = edge_quiet;
 +      }
 +      while (1) {
 +              if (spk_x == 0) {
 +                      if (spk_y == 0) {
 +                              edge_said = edge_top;
 +                              break;
 +                      }
 +                      if (edge_said != edge_quiet)
 +                              edge_said = edge_left;
 +                      if (state > 0)
 +                              break;
 +                      spk_y--;
 +                      spk_x = vc->vc_cols - 1;
 +              } else {
 +                      spk_x--;
 +              }
 +              spk_pos -= 2;
 +              ch = get_char(vc, (u_short *)spk_pos, &temp);
 +              if (ch == SPACE || ch == 0)
 +                      state = 0;
 +              else if (ch < 0x100 && IS_WDLM(ch))
 +                      state = 1;
 +              else
 +                      state = 2;
 +              if (state < last_state) {
 +                      spk_pos += 2;
 +                      spk_x++;
 +                      break;
 +              }
 +              last_state = state;
 +      }
 +      if (spk_x == 0 && edge_said == edge_quiet)
 +              edge_said = edge_left;
 +      if (edge_said > 0 && edge_said < edge_quiet)
 +              announce_edge(vc, edge_said);
 +      say_word(vc);
 +}
 +
 +static void say_next_word(struct vc_data *vc)
 +{
 +      u_char temp;
 +      u16 ch;
 +      u_short edge_said = 0, last_state = 2, state = 0;
 +
 +      spk_parked |= 0x01;
 +      if (spk_x == vc->vc_cols - 1 && spk_y == vc->vc_rows - 1) {
 +              announce_edge(vc, edge_bottom);
 +              return;
 +      }
 +      while (1) {
 +              ch = get_char(vc, (u_short *)spk_pos, &temp);
 +              if (ch == SPACE || ch == 0)
 +                      state = 0;
 +              else if (ch < 0x100 && IS_WDLM(ch))
 +                      state = 1;
 +              else
 +                      state = 2;
 +              if (state > last_state)
 +                      break;
 +              if (spk_x >= vc->vc_cols - 1) {
 +                      if (spk_y == vc->vc_rows - 1) {
 +                              edge_said = edge_bottom;
 +                              break;
 +                      }
 +                      state = 0;
 +                      spk_y++;
 +                      spk_x = 0;
 +                      edge_said = edge_right;
 +              } else {
 +                      spk_x++;
 +              }
 +              spk_pos += 2;
 +              last_state = state;
 +      }
 +      if (edge_said > 0)
 +              announce_edge(vc, edge_said);
 +      say_word(vc);
 +}
 +
 +static void spell_word(struct vc_data *vc)
 +{
 +      static char const *delay_str[] = { "", ",", ".", ". .", ". . ." };
 +      u16 *cp = buf;
 +      char *cp1;
 +      char *str_cap = spk_str_caps_stop;
 +      char *last_cap = spk_str_caps_stop;
 +      struct var_t *direct = spk_get_var(DIRECT);
 +      u16 ch;
 +
 +      if (!get_word(vc))
 +              return;
 +      while ((ch = *cp)) {
 +              if (cp != buf)
 +                      synth_printf(" %s ", delay_str[spk_spell_delay]);
 +              /* FIXME: Non-latin1 considered as lower case */
 +              if (ch < 0x100 && IS_CHAR(ch, B_CAP)) {
 +                      str_cap = spk_str_caps_start;
 +                      if (*spk_str_caps_stop)
 +                              spk_pitch_shift++;
 +                      else    /* synth has no pitch */
 +                              last_cap = spk_str_caps_stop;
 +              } else {
 +                      str_cap = spk_str_caps_stop;
 +              }
 +              if (str_cap != last_cap) {
 +                      synth_printf("%s", str_cap);
 +                      last_cap = str_cap;
 +              }
 +              if (ch >= 0x100 || (direct && direct->u.n.value)) {
 +                      synth_putwc_s(ch);
 +              } else if (this_speakup_key == SPELL_PHONETIC &&
 +                  ch <= 0x7f && isalpha(ch)) {
 +                      ch &= 0x1f;
 +                      cp1 = phonetic[--ch];
 +                      synth_printf("%s", cp1);
 +              } else {
 +                      cp1 = spk_characters[ch];
 +                      if (*cp1 == '^') {
 +                              synth_printf("%s", spk_msg_get(MSG_CTRL));
 +                              cp1++;
 +                      }
 +                      synth_printf("%s", cp1);
 +              }
 +              cp++;
 +      }
 +      if (str_cap != spk_str_caps_stop)
 +              synth_printf("%s", spk_str_caps_stop);
 +}
 +
 +static int get_line(struct vc_data *vc)
 +{
 +      u_long tmp = spk_pos - (spk_x * 2);
 +      int i = 0;
 +      u_char tmp2;
 +
 +      spk_old_attr = spk_attr;
 +      spk_attr = get_attributes(vc, (u_short *)spk_pos);
 +      for (i = 0; i < vc->vc_cols; i++) {
 +              buf[i] = get_char(vc, (u_short *)tmp, &tmp2);
 +              tmp += 2;
 +      }
 +      for (--i; i >= 0; i--)
 +              if (buf[i] != SPACE)
 +                      break;
 +      return ++i;
 +}
 +
 +static void say_line(struct vc_data *vc)
 +{
 +      int i = get_line(vc);
 +      u16 *cp;
 +      u_short saved_punc_mask = spk_punc_mask;
 +
 +      if (i == 0) {
 +              synth_printf("%s\n", spk_msg_get(MSG_BLANK));
 +              return;
 +      }
 +      buf[i++] = '\n';
 +      if (this_speakup_key == SAY_LINE_INDENT) {
 +              cp = buf;
 +              while (*cp == SPACE)
 +                      cp++;
 +              synth_printf("%zd, ", (cp - buf) + 1);
 +      }
 +      spk_punc_mask = spk_punc_masks[spk_reading_punc];
 +      spkup_write(buf, i);
 +      spk_punc_mask = saved_punc_mask;
 +}
 +
 +static void say_prev_line(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      if (spk_y == 0) {
 +              announce_edge(vc, edge_top);
 +              return;
 +      }
 +      spk_y--;
 +      spk_pos -= vc->vc_size_row;
 +      say_line(vc);
 +}
 +
 +static void say_next_line(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      if (spk_y == vc->vc_rows - 1) {
 +              announce_edge(vc, edge_bottom);
 +              return;
 +      }
 +      spk_y++;
 +      spk_pos += vc->vc_size_row;
 +      say_line(vc);
 +}
 +
 +static int say_from_to(struct vc_data *vc, u_long from, u_long to,
 +                     int read_punc)
 +{
 +      int i = 0;
 +      u_char tmp;
 +      u_short saved_punc_mask = spk_punc_mask;
 +
 +      spk_old_attr = spk_attr;
 +      spk_attr = get_attributes(vc, (u_short *)from);
 +      while (from < to) {
 +              buf[i++] = get_char(vc, (u_short *)from, &tmp);
 +              from += 2;
 +              if (i >= vc->vc_size_row)
 +                      break;
 +      }
 +      for (--i; i >= 0; i--)
 +              if (buf[i] != SPACE)
 +                      break;
 +      buf[++i] = SPACE;
 +      buf[++i] = '\0';
 +      if (i < 1)
 +              return i;
 +      if (read_punc)
 +              spk_punc_mask = spk_punc_info[spk_reading_punc].mask;
 +      spkup_write(buf, i);
 +      if (read_punc)
 +              spk_punc_mask = saved_punc_mask;
 +      return i - 1;
 +}
 +
 +static void say_line_from_to(struct vc_data *vc, u_long from, u_long to,
 +                           int read_punc)
 +{
 +      u_long start = vc->vc_origin + (spk_y * vc->vc_size_row);
 +      u_long end = start + (to * 2);
 +
 +      start += from * 2;
 +      if (say_from_to(vc, start, end, read_punc) <= 0)
 +              if (cursor_track != read_all_mode)
 +                      synth_printf("%s\n", spk_msg_get(MSG_BLANK));
 +}
 +
 +/* Sentence Reading Commands */
 +
 +static int currsentence;
 +static int numsentences[2];
 +static u16 *sentbufend[2];
 +static u16 *sentmarks[2][10];
 +static int currbuf;
 +static int bn;
 +static u16 sentbuf[2][256];
 +
 +static int say_sentence_num(int num, int prev)
 +{
 +      bn = currbuf;
 +      currsentence = num + 1;
 +      if (prev && --bn == -1)
 +              bn = 1;
 +
 +      if (num > numsentences[bn])
 +              return 0;
 +
 +      spkup_write(sentmarks[bn][num], sentbufend[bn] - sentmarks[bn][num]);
 +      return 1;
 +}
 +
 +static int get_sentence_buf(struct vc_data *vc, int read_punc)
 +{
 +      u_long start, end;
 +      int i, bn;
 +      u_char tmp;
 +
 +      currbuf++;
 +      if (currbuf == 2)
 +              currbuf = 0;
 +      bn = currbuf;
 +      start = vc->vc_origin + ((spk_y) * vc->vc_size_row);
 +      end = vc->vc_origin + ((spk_y) * vc->vc_size_row) + vc->vc_cols * 2;
 +
 +      numsentences[bn] = 0;
 +      sentmarks[bn][0] = &sentbuf[bn][0];
 +      i = 0;
 +      spk_old_attr = spk_attr;
 +      spk_attr = get_attributes(vc, (u_short *)start);
 +
 +      while (start < end) {
 +              sentbuf[bn][i] = get_char(vc, (u_short *)start, &tmp);
 +              if (i > 0) {
 +                      if (sentbuf[bn][i] == SPACE &&
 +                          sentbuf[bn][i - 1] == '.' &&
 +                          numsentences[bn] < 9) {
 +                              /* Sentence Marker */
 +                              numsentences[bn]++;
 +                              sentmarks[bn][numsentences[bn]] =
 +                                  &sentbuf[bn][i];
 +                      }
 +              }
 +              i++;
 +              start += 2;
 +              if (i >= vc->vc_size_row)
 +                      break;
 +      }
 +
 +      for (--i; i >= 0; i--)
 +              if (sentbuf[bn][i] != SPACE)
 +                      break;
 +
 +      if (i < 1)
 +              return -1;
 +
 +      sentbuf[bn][++i] = SPACE;
 +      sentbuf[bn][++i] = '\0';
 +
 +      sentbufend[bn] = &sentbuf[bn][i];
 +      return numsentences[bn];
 +}
 +
 +static void say_screen_from_to(struct vc_data *vc, u_long from, u_long to)
 +{
 +      u_long start = vc->vc_origin, end;
 +
 +      if (from > 0)
 +              start += from * vc->vc_size_row;
 +      if (to > vc->vc_rows)
 +              to = vc->vc_rows;
 +      end = vc->vc_origin + (to * vc->vc_size_row);
 +      for (from = start; from < end; from = to) {
 +              to = from + vc->vc_size_row;
 +              say_from_to(vc, from, to, 1);
 +      }
 +}
 +
 +static void say_screen(struct vc_data *vc)
 +{
 +      say_screen_from_to(vc, 0, vc->vc_rows);
 +}
 +
 +static void speakup_win_say(struct vc_data *vc)
 +{
 +      u_long start, end, from, to;
 +
 +      if (win_start < 2) {
 +              synth_printf("%s\n", spk_msg_get(MSG_NO_WINDOW));
 +              return;
 +      }
 +      start = vc->vc_origin + (win_top * vc->vc_size_row);
 +      end = vc->vc_origin + (win_bottom * vc->vc_size_row);
 +      while (start <= end) {
 +              from = start + (win_left * 2);
 +              to = start + (win_right * 2);
 +              say_from_to(vc, from, to, 1);
 +              start += vc->vc_size_row;
 +      }
 +}
 +
 +static void top_edge(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      spk_pos = vc->vc_origin + 2 * spk_x;
 +      spk_y = 0;
 +      say_line(vc);
 +}
 +
 +static void bottom_edge(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      spk_pos += (vc->vc_rows - spk_y - 1) * vc->vc_size_row;
 +      spk_y = vc->vc_rows - 1;
 +      say_line(vc);
 +}
 +
 +static void left_edge(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      spk_pos -= spk_x * 2;
 +      spk_x = 0;
 +      say_char(vc);
 +}
 +
 +static void right_edge(struct vc_data *vc)
 +{
 +      spk_parked |= 0x01;
 +      spk_pos += (vc->vc_cols - spk_x - 1) * 2;
 +      spk_x = vc->vc_cols - 1;
 +      say_char(vc);
 +}
 +
 +static void say_first_char(struct vc_data *vc)
 +{
 +      int i, len = get_line(vc);
 +      u16 ch;
 +
 +      spk_parked |= 0x01;
 +      if (len == 0) {
 +              synth_printf("%s\n", spk_msg_get(MSG_BLANK));
 +              return;
 +      }
 +      for (i = 0; i < len; i++)
 +              if (buf[i] != SPACE)
 +                      break;
 +      ch = buf[i];
 +      spk_pos -= (spk_x - i) * 2;
 +      spk_x = i;
 +      synth_printf("%d, ", ++i);
 +      speak_char(ch);
 +}
 +
 +static void say_last_char(struct vc_data *vc)
 +{
 +      int len = get_line(vc);
 +      u16 ch;
 +
 +      spk_parked |= 0x01;
 +      if (len == 0) {
 +              synth_printf("%s\n", spk_msg_get(MSG_BLANK));
 +              return;
 +      }
 +      ch = buf[--len];
 +      spk_pos -= (spk_x - len) * 2;
 +      spk_x = len;
 +      synth_printf("%d, ", ++len);
 +      speak_char(ch);
 +}
 +
 +static void say_position(struct vc_data *vc)
 +{
 +      synth_printf(spk_msg_get(MSG_POS_INFO), spk_y + 1, spk_x + 1,
 +                   vc->vc_num + 1);
 +      synth_printf("\n");
 +}
 +
 +/* Added by brianb */
 +static void say_char_num(struct vc_data *vc)
 +{
 +      u_char tmp;
 +      u16 ch = get_char(vc, (u_short *)spk_pos, &tmp);
 +
 +      synth_printf(spk_msg_get(MSG_CHAR_INFO), ch, ch);
 +}
 +
 +/* these are stub functions to keep keyboard.c happy. */
 +
 +static void say_from_top(struct vc_data *vc)
 +{
 +      say_screen_from_to(vc, 0, spk_y);
 +}
 +
 +static void say_to_bottom(struct vc_data *vc)
 +{
 +      say_screen_from_to(vc, spk_y, vc->vc_rows);
 +}
 +
 +static void say_from_left(struct vc_data *vc)
 +{
 +      say_line_from_to(vc, 0, spk_x, 1);
 +}
 +
 +static void say_to_right(struct vc_data *vc)
 +{
 +      say_line_from_to(vc, spk_x, vc->vc_cols, 1);
 +}
 +
 +/* end of stub functions. */
 +
 +static void spkup_write(const u16 *in_buf, int count)
 +{
 +      static int rep_count;
 +      static u16 ch = '\0', old_ch = '\0';
 +      static u_short char_type, last_type;
 +      int in_count = count;
 +
 +      spk_keydown = 0;
 +      while (count--) {
 +              if (cursor_track == read_all_mode) {
 +                      /* Insert Sentence Index */
 +                      if ((in_buf == sentmarks[bn][currsentence]) &&
 +                          (currsentence <= numsentences[bn]))
 +                              synth_insert_next_index(currsentence++);
 +              }
 +              ch = *in_buf++;
 +              if (ch < 0x100)
 +                      char_type = spk_chartab[ch];
 +              else
 +                      char_type = ALPHA;
 +              if (ch == old_ch && !(char_type & B_NUM)) {
 +                      if (++rep_count > 2)
 +                              continue;
 +              } else {
 +                      if ((last_type & CH_RPT) && rep_count > 2) {
 +                              synth_printf(" ");
 +                              synth_printf(spk_msg_get(MSG_REPEAT_DESC),
 +                                           ++rep_count);
 +                              synth_printf(" ");
 +                      }
 +                      rep_count = 0;
 +              }
 +              if (ch == spk_lastkey) {
 +                      rep_count = 0;
 +                      if (spk_key_echo == 1 && ch >= MINECHOCHAR)
 +                              speak_char(ch);
 +              } else if (char_type & B_ALPHA) {
 +                      if ((synth_flags & SF_DEC) && (last_type & PUNC))
 +                              synth_buffer_add(SPACE);
 +                      synth_putwc_s(ch);
 +              } else if (char_type & B_NUM) {
 +                      rep_count = 0;
 +                      synth_putwc_s(ch);
 +              } else if (char_type & spk_punc_mask) {
 +                      speak_char(ch);
 +                      char_type &= ~PUNC;     /* for dec nospell processing */
 +              } else if (char_type & SYNTH_OK) {
 +                      /* these are usually puncts like . and , which synth
 +                       * needs for expression.
 +                       * suppress multiple to get rid of long pauses and
 +                       * clear repeat count
 +                       * so if someone has
 +                       * repeats on you don't get nothing repeated count
 +                       */
 +                      if (ch != old_ch)
 +                              synth_putwc_s(ch);
 +                      else
 +                              rep_count = 0;
 +              } else {
 +/* send space and record position, if next is num overwrite space */
 +                      if (old_ch != ch)
 +                              synth_buffer_add(SPACE);
 +                      else
 +                              rep_count = 0;
 +              }
 +              old_ch = ch;
 +              last_type = char_type;
 +      }
 +      spk_lastkey = 0;
 +      if (in_count > 2 && rep_count > 2) {
 +              if (last_type & CH_RPT) {
 +                      synth_printf(" ");
 +                      synth_printf(spk_msg_get(MSG_REPEAT_DESC2),
 +                                   ++rep_count);
 +                      synth_printf(" ");
 +              }
 +              rep_count = 0;
 +      }
 +}
 +
 +static const int NUM_CTL_LABELS = (MSG_CTL_END - MSG_CTL_START + 1);
 +
 +static void read_all_doc(struct vc_data *vc);
 +static void cursor_done(struct timer_list *unused);
 +static DEFINE_TIMER(cursor_timer, cursor_done);
 +
 +static void do_handle_shift(struct vc_data *vc, u_char value, char up_flag)
 +{
 +      unsigned long flags;
 +
 +      if (!synth || up_flag || spk_killed)
 +              return;
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      if (cursor_track == read_all_mode) {
 +              switch (value) {
 +              case KVAL(K_SHIFT):
 +                      del_timer(&cursor_timer);
 +                      spk_shut_up &= 0xfe;
 +                      spk_do_flush();
 +                      read_all_doc(vc);
 +                      break;
 +              case KVAL(K_CTRL):
 +                      del_timer(&cursor_timer);
 +                      cursor_track = prev_cursor_track;
 +                      spk_shut_up &= 0xfe;
 +                      spk_do_flush();
 +                      break;
 +              }
 +      } else {
 +              spk_shut_up &= 0xfe;
 +              spk_do_flush();
 +      }
 +      if (spk_say_ctrl && value < NUM_CTL_LABELS)
 +              synth_printf("%s", spk_msg_get(MSG_CTL_START + value));
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +static void do_handle_latin(struct vc_data *vc, u_char value, char up_flag)
 +{
 +      unsigned long flags;
 +
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      if (up_flag) {
 +              spk_lastkey = 0;
 +              spk_keydown = 0;
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      if (!synth || spk_killed) {
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      spk_shut_up &= 0xfe;
 +      spk_lastkey = value;
 +      spk_keydown++;
 +      spk_parked &= 0xfe;
 +      if (spk_key_echo == 2 && value >= MINECHOCHAR)
 +              speak_char(value);
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +int spk_set_key_info(const u_char *key_info, u_char *k_buffer)
 +{
 +      int i = 0, states, key_data_len;
 +      const u_char *cp = key_info;
 +      u_char *cp1 = k_buffer;
 +      u_char ch, version, num_keys;
 +
 +      version = *cp++;
 +      if (version != KEY_MAP_VER) {
 +              pr_debug("version found %d should be %d\n",
 +                       version, KEY_MAP_VER);
 +              return -EINVAL;
 +      }
 +      num_keys = *cp;
 +      states = (int)cp[1];
 +      key_data_len = (states + 1) * (num_keys + 1);
 +      if (key_data_len + SHIFT_TBL_SIZE + 4 >= sizeof(spk_key_buf)) {
 +              pr_debug("too many key_infos (%d over %u)\n",
 +                       key_data_len + SHIFT_TBL_SIZE + 4,
 +                       (unsigned int)(sizeof(spk_key_buf)));
 +              return -EINVAL;
 +      }
 +      memset(k_buffer, 0, SHIFT_TBL_SIZE);
 +      memset(spk_our_keys, 0, sizeof(spk_our_keys));
 +      spk_shift_table = k_buffer;
 +      spk_our_keys[0] = spk_shift_table;
 +      cp1 += SHIFT_TBL_SIZE;
 +      memcpy(cp1, cp, key_data_len + 3);
 +      /* get num_keys, states and data */
 +      cp1 += 2;               /* now pointing at shift states */
 +      for (i = 1; i <= states; i++) {
 +              ch = *cp1++;
 +              if (ch >= SHIFT_TBL_SIZE) {
 +                      pr_debug("(%d) not valid shift state (max_allowed = %d)\n",
 +                               ch, SHIFT_TBL_SIZE);
 +                      return -EINVAL;
 +              }
 +              spk_shift_table[ch] = i;
 +      }
 +      keymap_flags = *cp1++;
 +      while ((ch = *cp1)) {
 +              if (ch >= MAX_KEY) {
 +                      pr_debug("(%d), not valid key, (max_allowed = %d)\n",
 +                               ch, MAX_KEY);
 +                      return -EINVAL;
 +              }
 +              spk_our_keys[ch] = cp1;
 +              cp1 += states + 1;
 +      }
 +      return 0;
 +}
 +
 +static struct var_t spk_vars[] = {
 +      /* bell must be first to set high limit */
 +      {BELL_POS, .u.n = {NULL, 0, 0, 0, 0, 0, NULL} },
 +      {SPELL_DELAY, .u.n = {NULL, 0, 0, 4, 0, 0, NULL} },
 +      {ATTRIB_BLEEP, .u.n = {NULL, 1, 0, 3, 0, 0, NULL} },
 +      {BLEEPS, .u.n = {NULL, 3, 0, 3, 0, 0, NULL} },
 +      {BLEEP_TIME, .u.n = {NULL, 30, 1, 200, 0, 0, NULL} },
 +      {PUNC_LEVEL, .u.n = {NULL, 1, 0, 4, 0, 0, NULL} },
 +      {READING_PUNC, .u.n = {NULL, 1, 0, 4, 0, 0, NULL} },
 +      {CURSOR_TIME, .u.n = {NULL, 120, 50, 600, 0, 0, NULL} },
 +      {SAY_CONTROL, TOGGLE_0},
 +      {SAY_WORD_CTL, TOGGLE_0},
 +      {NO_INTERRUPT, TOGGLE_0},
 +      {KEY_ECHO, .u.n = {NULL, 1, 0, 2, 0, 0, NULL} },
 +      V_LAST_VAR
 +};
 +
 +static void toggle_cursoring(struct vc_data *vc)
 +{
 +      if (cursor_track == read_all_mode)
 +              cursor_track = prev_cursor_track;
 +      if (++cursor_track >= CT_Max)
 +              cursor_track = 0;
 +      synth_printf("%s\n", spk_msg_get(MSG_CURSOR_MSGS_START + cursor_track));
 +}
 +
 +void spk_reset_default_chars(void)
 +{
 +      int i;
 +
 +      /* First, free any non-default */
 +      for (i = 0; i < 256; i++) {
 +              if (spk_characters[i] &&
 +                  (spk_characters[i] != spk_default_chars[i]))
 +                      kfree(spk_characters[i]);
 +      }
 +
 +      memcpy(spk_characters, spk_default_chars, sizeof(spk_default_chars));
 +}
 +
 +void spk_reset_default_chartab(void)
 +{
 +      memcpy(spk_chartab, default_chartab, sizeof(default_chartab));
 +}
 +
 +static const struct st_bits_data *pb_edit;
 +
 +static int edit_bits(struct vc_data *vc, u_char type, u_char ch, u_short key)
 +{
 +      short mask = pb_edit->mask, ch_type = spk_chartab[ch];
 +
 +      if (type != KT_LATIN || (ch_type & B_NUM) || ch < SPACE)
 +              return -1;
 +      if (ch == SPACE) {
 +              synth_printf("%s\n", spk_msg_get(MSG_EDIT_DONE));
 +              spk_special_handler = NULL;
 +              return 1;
 +      }
 +      if (mask < PUNC && !(ch_type & PUNC))
 +              return -1;
 +      spk_chartab[ch] ^= mask;
 +      speak_char(ch);
 +      synth_printf(" %s\n",
 +                   (spk_chartab[ch] & mask) ? spk_msg_get(MSG_ON) :
 +                   spk_msg_get(MSG_OFF));
 +      return 1;
 +}
 +
 +/* Allocation concurrency is protected by the console semaphore */
 +static int speakup_allocate(struct vc_data *vc, gfp_t gfp_flags)
 +{
 +      int vc_num;
 +
 +      vc_num = vc->vc_num;
 +      if (!speakup_console[vc_num]) {
 +              speakup_console[vc_num] = kzalloc(sizeof(*speakup_console[0]),
 +                                                gfp_flags);
 +              if (!speakup_console[vc_num])
 +                      return -ENOMEM;
 +              speakup_date(vc);
 +      } else if (!spk_parked) {
 +              speakup_date(vc);
 +      }
 +
 +      return 0;
 +}
 +
 +static void speakup_deallocate(struct vc_data *vc)
 +{
 +      int vc_num;
 +
 +      vc_num = vc->vc_num;
 +      kfree(speakup_console[vc_num]);
 +      speakup_console[vc_num] = NULL;
 +}
 +
 +static u_char is_cursor;
 +static u_long old_cursor_pos, old_cursor_x, old_cursor_y;
 +static int cursor_con;
 +
 +static void reset_highlight_buffers(struct vc_data *);
 +
 +static int read_all_key;
 +
 +static int in_keyboard_notifier;
 +
 +static void start_read_all_timer(struct vc_data *vc, int command);
 +
 +enum {
 +      RA_NOTHING,
 +      RA_NEXT_SENT,
 +      RA_PREV_LINE,
 +      RA_NEXT_LINE,
 +      RA_PREV_SENT,
 +      RA_DOWN_ARROW,
 +      RA_TIMER,
 +      RA_FIND_NEXT_SENT,
 +      RA_FIND_PREV_SENT,
 +};
 +
 +static void kbd_fakekey2(struct vc_data *vc, int command)
 +{
 +      del_timer(&cursor_timer);
 +      speakup_fake_down_arrow();
 +      start_read_all_timer(vc, command);
 +}
 +
 +static void read_all_doc(struct vc_data *vc)
 +{
 +      if ((vc->vc_num != fg_console) || !synth || spk_shut_up)
 +              return;
 +      if (!synth_supports_indexing())
 +              return;
 +      if (cursor_track != read_all_mode)
 +              prev_cursor_track = cursor_track;
 +      cursor_track = read_all_mode;
 +      spk_reset_index_count(0);
 +      if (get_sentence_buf(vc, 0) == -1) {
 +              del_timer(&cursor_timer);
 +              if (!in_keyboard_notifier)
 +                      speakup_fake_down_arrow();
 +              start_read_all_timer(vc, RA_DOWN_ARROW);
 +      } else {
 +              say_sentence_num(0, 0);
 +              synth_insert_next_index(0);
 +              start_read_all_timer(vc, RA_TIMER);
 +      }
 +}
 +
 +static void stop_read_all(struct vc_data *vc)
 +{
 +      del_timer(&cursor_timer);
 +      cursor_track = prev_cursor_track;
 +      spk_shut_up &= 0xfe;
 +      spk_do_flush();
 +}
 +
 +static void start_read_all_timer(struct vc_data *vc, int command)
 +{
 +      struct var_t *cursor_timeout;
 +
 +      cursor_con = vc->vc_num;
 +      read_all_key = command;
 +      cursor_timeout = spk_get_var(CURSOR_TIME);
 +      mod_timer(&cursor_timer,
 +                jiffies + msecs_to_jiffies(cursor_timeout->u.n.value));
 +}
 +
 +static void handle_cursor_read_all(struct vc_data *vc, int command)
 +{
 +      int indcount, sentcount, rv, sn;
 +
 +      switch (command) {
 +      case RA_NEXT_SENT:
 +              /* Get Current Sentence */
 +              spk_get_index_count(&indcount, &sentcount);
 +              /*printk("%d %d  ", indcount, sentcount); */
 +              spk_reset_index_count(sentcount + 1);
 +              if (indcount == 1) {
 +                      if (!say_sentence_num(sentcount + 1, 0)) {
 +                              kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
 +                              return;
 +                      }
 +                      synth_insert_next_index(0);
 +              } else {
 +                      sn = 0;
 +                      if (!say_sentence_num(sentcount + 1, 1)) {
 +                              sn = 1;
 +                              spk_reset_index_count(sn);
 +                      } else {
 +                              synth_insert_next_index(0);
 +                      }
 +                      if (!say_sentence_num(sn, 0)) {
 +                              kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
 +                              return;
 +                      }
 +                      synth_insert_next_index(0);
 +              }
 +              start_read_all_timer(vc, RA_TIMER);
 +              break;
 +      case RA_PREV_SENT:
 +              break;
 +      case RA_NEXT_LINE:
 +              read_all_doc(vc);
 +              break;
 +      case RA_PREV_LINE:
 +              break;
 +      case RA_DOWN_ARROW:
 +              if (get_sentence_buf(vc, 0) == -1) {
 +                      kbd_fakekey2(vc, RA_DOWN_ARROW);
 +              } else {
 +                      say_sentence_num(0, 0);
 +                      synth_insert_next_index(0);
 +                      start_read_all_timer(vc, RA_TIMER);
 +              }
 +              break;
 +      case RA_FIND_NEXT_SENT:
 +              rv = get_sentence_buf(vc, 0);
 +              if (rv == -1)
 +                      read_all_doc(vc);
 +              if (rv == 0) {
 +                      kbd_fakekey2(vc, RA_FIND_NEXT_SENT);
 +              } else {
 +                      say_sentence_num(1, 0);
 +                      synth_insert_next_index(0);
 +                      start_read_all_timer(vc, RA_TIMER);
 +              }
 +              break;
 +      case RA_FIND_PREV_SENT:
 +              break;
 +      case RA_TIMER:
 +              spk_get_index_count(&indcount, &sentcount);
 +              if (indcount < 2)
 +                      kbd_fakekey2(vc, RA_DOWN_ARROW);
 +              else
 +                      start_read_all_timer(vc, RA_TIMER);
 +              break;
 +      }
 +}
 +
 +static int pre_handle_cursor(struct vc_data *vc, u_char value, char up_flag)
 +{
 +      unsigned long flags;
 +
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      if (cursor_track == read_all_mode) {
 +              spk_parked &= 0xfe;
 +              if (!synth || up_flag || spk_shut_up) {
 +                      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +                      return NOTIFY_STOP;
 +              }
 +              del_timer(&cursor_timer);
 +              spk_shut_up &= 0xfe;
 +              spk_do_flush();
 +              start_read_all_timer(vc, value + 1);
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return NOTIFY_STOP;
 +      }
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +      return NOTIFY_OK;
 +}
 +
 +static void do_handle_cursor(struct vc_data *vc, u_char value, char up_flag)
 +{
 +      unsigned long flags;
 +      struct var_t *cursor_timeout;
 +
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      spk_parked &= 0xfe;
 +      if (!synth || up_flag || spk_shut_up || cursor_track == CT_Off) {
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      spk_shut_up &= 0xfe;
 +      if (spk_no_intr)
 +              spk_do_flush();
 +/* the key press flushes if !no_inter but we want to flush on cursor
 + * moves regardless of no_inter state
 + */
 +      is_cursor = value + 1;
 +      old_cursor_pos = vc->vc_pos;
-               speakup_console[vc_num]->ht.rx[bi] = vc->vc_x;
-               speakup_console[vc_num]->ht.ry[bi] = vc->vc_y;
++      old_cursor_x = vc->state.x;
++      old_cursor_y = vc->state.y;
++      speakup_console[vc->vc_num]->ht.cy = vc->state.y;
 +      cursor_con = vc->vc_num;
 +      if (cursor_track == CT_Highlight)
 +              reset_highlight_buffers(vc);
 +      cursor_timeout = spk_get_var(CURSOR_TIME);
 +      mod_timer(&cursor_timer,
 +                jiffies + msecs_to_jiffies(cursor_timeout->u.n.value));
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +static void update_color_buffer(struct vc_data *vc, const u16 *ic, int len)
 +{
 +      int i, bi, hi;
 +      int vc_num = vc->vc_num;
 +
 +      bi = (vc->vc_attr & 0x70) >> 4;
 +      hi = speakup_console[vc_num]->ht.highsize[bi];
 +
 +      i = 0;
 +      if (speakup_console[vc_num]->ht.highsize[bi] == 0) {
 +              speakup_console[vc_num]->ht.rpos[bi] = vc->vc_pos;
-               d = vc->vc_y - speakup_console[vc_num]->ht.cy;
++              speakup_console[vc_num]->ht.rx[bi] = vc->state.x;
++              speakup_console[vc_num]->ht.ry[bi] = vc->state.y;
 +      }
 +      while ((hi < COLOR_BUFFER_SIZE) && (i < len)) {
 +              if (ic[i] > 32) {
 +                      speakup_console[vc_num]->ht.highbuf[bi][hi] = ic[i];
 +                      hi++;
 +              } else if ((ic[i] == 32) && (hi != 0)) {
 +                      if (speakup_console[vc_num]->ht.highbuf[bi][hi - 1] !=
 +                          32) {
 +                              speakup_console[vc_num]->ht.highbuf[bi][hi] =
 +                                  ic[i];
 +                              hi++;
 +                      }
 +              }
 +              i++;
 +      }
 +      speakup_console[vc_num]->ht.highsize[bi] = hi;
 +}
 +
 +static void reset_highlight_buffers(struct vc_data *vc)
 +{
 +      int i;
 +      int vc_num = vc->vc_num;
 +
 +      for (i = 0; i < 8; i++)
 +              speakup_console[vc_num]->ht.highsize[i] = 0;
 +}
 +
 +static int count_highlight_color(struct vc_data *vc)
 +{
 +      int i, bg;
 +      int cc;
 +      int vc_num = vc->vc_num;
 +      u16 ch;
 +      u16 *start = (u16 *)vc->vc_origin;
 +
 +      for (i = 0; i < 8; i++)
 +              speakup_console[vc_num]->ht.bgcount[i] = 0;
 +
 +      for (i = 0; i < vc->vc_rows; i++) {
 +              u16 *end = start + vc->vc_cols * 2;
 +              u16 *ptr;
 +
 +              for (ptr = start; ptr < end; ptr++) {
 +                      ch = get_attributes(vc, ptr);
 +                      bg = (ch & 0x70) >> 4;
 +                      speakup_console[vc_num]->ht.bgcount[bg]++;
 +              }
 +              start += vc->vc_size_row;
 +      }
 +
 +      cc = 0;
 +      for (i = 0; i < 8; i++)
 +              if (speakup_console[vc_num]->ht.bgcount[i] > 0)
 +                      cc++;
 +      return cc;
 +}
 +
 +static int get_highlight_color(struct vc_data *vc)
 +{
 +      int i, j;
 +      unsigned int cptr[8];
 +      int vc_num = vc->vc_num;
 +
 +      for (i = 0; i < 8; i++)
 +              cptr[i] = i;
 +
 +      for (i = 0; i < 7; i++)
 +              for (j = i + 1; j < 8; j++)
 +                      if (speakup_console[vc_num]->ht.bgcount[cptr[i]] >
 +                          speakup_console[vc_num]->ht.bgcount[cptr[j]])
 +                              swap(cptr[i], cptr[j]);
 +
 +      for (i = 0; i < 8; i++)
 +              if (speakup_console[vc_num]->ht.bgcount[cptr[i]] != 0)
 +                      if (speakup_console[vc_num]->ht.highsize[cptr[i]] > 0)
 +                              return cptr[i];
 +      return -1;
 +}
 +
 +static int speak_highlight(struct vc_data *vc)
 +{
 +      int hc, d;
 +      int vc_num = vc->vc_num;
 +
 +      if (count_highlight_color(vc) == 1)
 +              return 0;
 +      hc = get_highlight_color(vc);
 +      if (hc != -1) {
-                       if (speakup_console[vc_num]->ht.ry[hc] != vc->vc_y)
++              d = vc->state.y - speakup_console[vc_num]->ht.cy;
 +              if ((d == 1) || (d == -1))
-               if (vc->vc_x >= win_left && vc->vc_x <= win_right &&
-                   vc->vc_y >= win_top && vc->vc_y <= win_bottom) {
++                      if (speakup_console[vc_num]->ht.ry[hc] != vc->state.y)
 +                              return 0;
 +              spk_parked |= 0x01;
 +              spk_do_flush();
 +              spkup_write(speakup_console[vc_num]->ht.highbuf[hc],
 +                          speakup_console[vc_num]->ht.highsize[hc]);
 +              spk_pos = spk_cp = speakup_console[vc_num]->ht.rpos[hc];
 +              spk_x = spk_cx = speakup_console[vc_num]->ht.rx[hc];
 +              spk_y = spk_cy = speakup_console[vc_num]->ht.ry[hc];
 +              return 1;
 +      }
 +      return 0;
 +}
 +
 +static void cursor_done(struct timer_list *unused)
 +{
 +      struct vc_data *vc = vc_cons[cursor_con].d;
 +      unsigned long flags;
 +
 +      del_timer(&cursor_timer);
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      if (cursor_con != fg_console) {
 +              is_cursor = 0;
 +              goto out;
 +      }
 +      speakup_date(vc);
 +      if (win_enabled) {
-       if (spk_bell_pos && spk_keydown && (vc->vc_x == spk_bell_pos - 1))
++              if (vc->state.x >= win_left && vc->state.x <= win_right &&
++                  vc->state.y >= win_top && vc->state.y <= win_bottom) {
 +                      spk_keydown = 0;
 +                      is_cursor = 0;
 +                      goto out;
 +              }
 +      }
 +      if (cursor_track == read_all_mode) {
 +              handle_cursor_read_all(vc, read_all_key);
 +              goto out;
 +      }
 +      if (cursor_track == CT_Highlight) {
 +              if (speak_highlight(vc)) {
 +                      spk_keydown = 0;
 +                      is_cursor = 0;
 +                      goto out;
 +              }
 +      }
 +      if (cursor_track == CT_Window)
 +              speakup_win_say(vc);
 +      else if (is_cursor == 1 || is_cursor == 4)
 +              say_line_from_to(vc, 0, vc->vc_cols, 0);
 +      else
 +              say_char(vc);
 +      spk_keydown = 0;
 +      is_cursor = 0;
 +out:
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +/* called by: vt_notifier_call() */
 +static void speakup_bs(struct vc_data *vc)
 +{
 +      unsigned long flags;
 +
 +      if (!speakup_console[vc->vc_num])
 +              return;
 +      if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
 +              /* Speakup output, discard */
 +              return;
 +      if (!spk_parked)
 +              speakup_date(vc);
 +      if (spk_shut_up || !synth) {
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      if (vc->vc_num == fg_console && spk_keydown) {
 +              spk_keydown = 0;
 +              if (!is_cursor)
 +                      say_char(vc);
 +      }
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +/* called by: vt_notifier_call() */
 +static void speakup_con_write(struct vc_data *vc, u16 *str, int len)
 +{
 +      unsigned long flags;
 +
 +      if ((vc->vc_num != fg_console) || spk_shut_up || !synth)
 +              return;
 +      if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
 +              /* Speakup output, discard */
 +              return;
-               if (vc->vc_x >= win_left && vc->vc_x <= win_right &&
-                   vc->vc_y >= win_top && vc->vc_y <= win_bottom) {
++      if (spk_bell_pos && spk_keydown && (vc->state.x == spk_bell_pos - 1))
 +              bleep(3);
 +      if ((is_cursor) || (cursor_track == read_all_mode)) {
 +              if (cursor_track == CT_Highlight)
 +                      update_color_buffer(vc, str, len);
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      if (win_enabled) {
++              if (vc->state.x >= win_left && vc->state.x <= win_right &&
++                  vc->state.y >= win_top && vc->state.y <= win_bottom) {
 +                      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +                      return;
 +              }
 +      }
 +
 +      spkup_write(str, len);
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +static void speakup_con_update(struct vc_data *vc)
 +{
 +      unsigned long flags;
 +
 +      if (!speakup_console[vc->vc_num] || spk_parked)
 +              return;
 +      if (!spin_trylock_irqsave(&speakup_info.spinlock, flags))
 +              /* Speakup output, discard */
 +              return;
 +      speakup_date(vc);
 +      if (vc->vc_mode == KD_GRAPHICS && !spk_paused && spk_str_pause[0]) {
 +              synth_printf("%s", spk_str_pause);
 +              spk_paused = true;
 +      }
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +static void do_handle_spec(struct vc_data *vc, u_char value, char up_flag)
 +{
 +      unsigned long flags;
 +      int on_off = 2;
 +      char *label;
 +
 +      if (!synth || up_flag || spk_killed)
 +              return;
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      spk_shut_up &= 0xfe;
 +      if (spk_no_intr)
 +              spk_do_flush();
 +      switch (value) {
 +      case KVAL(K_CAPS):
 +              label = spk_msg_get(MSG_KEYNAME_CAPSLOCK);
 +              on_off = vt_get_leds(fg_console, VC_CAPSLOCK);
 +              break;
 +      case KVAL(K_NUM):
 +              label = spk_msg_get(MSG_KEYNAME_NUMLOCK);
 +              on_off = vt_get_leds(fg_console, VC_NUMLOCK);
 +              break;
 +      case KVAL(K_HOLD):
 +              label = spk_msg_get(MSG_KEYNAME_SCROLLLOCK);
 +              on_off = vt_get_leds(fg_console, VC_SCROLLOCK);
 +              if (speakup_console[vc->vc_num])
 +                      speakup_console[vc->vc_num]->tty_stopped = on_off;
 +              break;
 +      default:
 +              spk_parked &= 0xfe;
 +              spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +              return;
 +      }
 +      if (on_off < 2)
 +              synth_printf("%s %s\n",
 +                           label, spk_msg_get(MSG_STATUS_START + on_off));
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +}
 +
 +static int inc_dec_var(u_char value)
 +{
 +      struct st_var_header *p_header;
 +      struct var_t *var_data;
 +      char num_buf[32];
 +      char *cp = num_buf;
 +      char *pn;
 +      int var_id = (int)value - VAR_START;
 +      int how = (var_id & 1) ? E_INC : E_DEC;
 +
 +      var_id = var_id / 2 + FIRST_SET_VAR;
 +      p_header = spk_get_var_header(var_id);
 +      if (!p_header)
 +              return -1;
 +      if (p_header->var_type != VAR_NUM)
 +              return -1;
 +      var_data = p_header->data;
 +      if (spk_set_num_var(1, p_header, how) != 0)
 +              return -1;
 +      if (!spk_close_press) {
 +              for (pn = p_header->name; *pn; pn++) {
 +                      if (*pn == '_')
 +                              *cp = SPACE;
 +                      else
 +                              *cp++ = *pn;
 +              }
 +      }
 +      snprintf(cp, sizeof(num_buf) - (cp - num_buf), " %d ",
 +               var_data->u.n.value);
 +      synth_printf("%s", num_buf);
 +      return 0;
 +}
 +
 +static void speakup_win_set(struct vc_data *vc)
 +{
 +      char info[40];
 +
 +      if (win_start > 1) {
 +              synth_printf("%s\n", spk_msg_get(MSG_WINDOW_ALREADY_SET));
 +              return;
 +      }
 +      if (spk_x < win_left || spk_y < win_top) {
 +              synth_printf("%s\n", spk_msg_get(MSG_END_BEFORE_START));
 +              return;
 +      }
 +      if (win_start && spk_x == win_left && spk_y == win_top) {
 +              win_left = 0;
 +              win_right = vc->vc_cols - 1;
 +              win_bottom = spk_y;
 +              snprintf(info, sizeof(info), spk_msg_get(MSG_WINDOW_LINE),
 +                       (int)win_top + 1);
 +      } else {
 +              if (!win_start) {
 +                      win_top = spk_y;
 +                      win_left = spk_x;
 +              } else {
 +                      win_bottom = spk_y;
 +                      win_right = spk_x;
 +              }
 +              snprintf(info, sizeof(info), spk_msg_get(MSG_WINDOW_BOUNDARY),
 +                       (win_start) ?
 +                              spk_msg_get(MSG_END) : spk_msg_get(MSG_START),
 +                       (int)spk_y + 1, (int)spk_x + 1);
 +      }
 +      synth_printf("%s\n", info);
 +      win_start++;
 +}
 +
 +static void speakup_win_clear(struct vc_data *vc)
 +{
 +      win_top = 0;
 +      win_bottom = 0;
 +      win_left = 0;
 +      win_right = 0;
 +      win_start = 0;
 +      synth_printf("%s\n", spk_msg_get(MSG_WINDOW_CLEARED));
 +}
 +
 +static void speakup_win_enable(struct vc_data *vc)
 +{
 +      if (win_start < 2) {
 +              synth_printf("%s\n", spk_msg_get(MSG_NO_WINDOW));
 +              return;
 +      }
 +      win_enabled ^= 1;
 +      if (win_enabled)
 +              synth_printf("%s\n", spk_msg_get(MSG_WINDOW_SILENCED));
 +      else
 +              synth_printf("%s\n", spk_msg_get(MSG_WINDOW_SILENCE_DISABLED));
 +}
 +
 +static void speakup_bits(struct vc_data *vc)
 +{
 +      int val = this_speakup_key - (FIRST_EDIT_BITS - 1);
 +
 +      if (spk_special_handler || val < 1 || val > 6) {
 +              synth_printf("%s\n", spk_msg_get(MSG_ERROR));
 +              return;
 +      }
 +      pb_edit = &spk_punc_info[val];
 +      synth_printf(spk_msg_get(MSG_EDIT_PROMPT), pb_edit->name);
 +      spk_special_handler = edit_bits;
 +}
 +
 +static int handle_goto(struct vc_data *vc, u_char type, u_char ch, u_short key)
 +{
 +      static u_char goto_buf[8];
 +      static int num;
 +      int maxlen;
 +      char *cp;
 +      u16 wch;
 +
 +      if (type == KT_SPKUP && ch == SPEAKUP_GOTO)
 +              goto do_goto;
 +      if (type == KT_LATIN && ch == '\n')
 +              goto do_goto;
 +      if (type != 0)
 +              goto oops;
 +      if (ch == 8) {
 +              u16 wch;
 +
 +              if (num == 0)
 +                      return -1;
 +              wch = goto_buf[--num];
 +              goto_buf[num] = '\0';
 +              spkup_write(&wch, 1);
 +              return 1;
 +      }
 +      if (ch < '+' || ch > 'y')
 +              goto oops;
 +      wch = ch;
 +      goto_buf[num++] = ch;
 +      goto_buf[num] = '\0';
 +      spkup_write(&wch, 1);
 +      maxlen = (*goto_buf >= '0') ? 3 : 4;
 +      if ((ch == '+' || ch == '-') && num == 1)
 +              return 1;
 +      if (ch >= '0' && ch <= '9' && num < maxlen)
 +              return 1;
 +      if (num < maxlen - 1 || num > maxlen)
 +              goto oops;
 +      if (ch < 'x' || ch > 'y') {
 +oops:
 +              if (!spk_killed)
 +                      synth_printf(" %s\n", spk_msg_get(MSG_GOTO_CANCELED));
 +              goto_buf[num = 0] = '\0';
 +              spk_special_handler = NULL;
 +              return 1;
 +      }
 +
 +      /* Do not replace with kstrtoul: here we need cp to be updated */
 +      goto_pos = simple_strtoul(goto_buf, &cp, 10);
 +
 +      if (*cp == 'x') {
 +              if (*goto_buf < '0')
 +                      goto_pos += spk_x;
 +              else if (goto_pos > 0)
 +                      goto_pos--;
 +
 +              if (goto_pos >= vc->vc_cols)
 +                      goto_pos = vc->vc_cols - 1;
 +              goto_x = 1;
 +      } else {
 +              if (*goto_buf < '0')
 +                      goto_pos += spk_y;
 +              else if (goto_pos > 0)
 +                      goto_pos--;
 +
 +              if (goto_pos >= vc->vc_rows)
 +                      goto_pos = vc->vc_rows - 1;
 +              goto_x = 0;
 +      }
 +      goto_buf[num = 0] = '\0';
 +do_goto:
 +      spk_special_handler = NULL;
 +      spk_parked |= 0x01;
 +      if (goto_x) {
 +              spk_pos -= spk_x * 2;
 +              spk_x = goto_pos;
 +              spk_pos += goto_pos * 2;
 +              say_word(vc);
 +      } else {
 +              spk_y = goto_pos;
 +              spk_pos = vc->vc_origin + (goto_pos * vc->vc_size_row);
 +              say_line(vc);
 +      }
 +      return 1;
 +}
 +
 +static void speakup_goto(struct vc_data *vc)
 +{
 +      if (spk_special_handler) {
 +              synth_printf("%s\n", spk_msg_get(MSG_ERROR));
 +              return;
 +      }
 +      synth_printf("%s\n", spk_msg_get(MSG_GOTO));
 +      spk_special_handler = handle_goto;
 +}
 +
 +static void speakup_help(struct vc_data *vc)
 +{
 +      spk_handle_help(vc, KT_SPKUP, SPEAKUP_HELP, 0);
 +}
 +
 +static void do_nothing(struct vc_data *vc)
 +{
 +      return;                 /* flush done in do_spkup */
 +}
 +
 +static u_char key_speakup, spk_key_locked;
 +
 +static void speakup_lock(struct vc_data *vc)
 +{
 +      if (!spk_key_locked) {
 +              spk_key_locked = 16;
 +              key_speakup = 16;
 +      } else {
 +              spk_key_locked = 0;
 +              key_speakup = 0;
 +      }
 +}
 +
 +typedef void (*spkup_hand) (struct vc_data *);
 +static spkup_hand spkup_handler[] = {
 +      /* must be ordered same as defines in speakup.h */
 +      do_nothing, speakup_goto, speech_kill, speakup_shut_up,
 +      speakup_cut, speakup_paste, say_first_char, say_last_char,
 +      say_char, say_prev_char, say_next_char,
 +      say_word, say_prev_word, say_next_word,
 +      say_line, say_prev_line, say_next_line,
 +      top_edge, bottom_edge, left_edge, right_edge,
 +      spell_word, spell_word, say_screen,
 +      say_position, say_attributes,
 +      speakup_off, speakup_parked, say_line,  /* this is for indent */
 +      say_from_top, say_to_bottom,
 +      say_from_left, say_to_right,
 +      say_char_num, speakup_bits, speakup_bits, say_phonetic_char,
 +      speakup_bits, speakup_bits, speakup_bits,
 +      speakup_win_set, speakup_win_clear, speakup_win_enable, speakup_win_say,
 +      speakup_lock, speakup_help, toggle_cursoring, read_all_doc, NULL
 +};
 +
 +static void do_spkup(struct vc_data *vc, u_char value)
 +{
 +      if (spk_killed && value != SPEECH_KILL)
 +              return;
 +      spk_keydown = 0;
 +      spk_lastkey = 0;
 +      spk_shut_up &= 0xfe;
 +      this_speakup_key = value;
 +      if (value < SPKUP_MAX_FUNC && spkup_handler[value]) {
 +              spk_do_flush();
 +              (*spkup_handler[value]) (vc);
 +      } else {
 +              if (inc_dec_var(value) < 0)
 +                      bleep(9);
 +      }
 +}
 +
 +static const char *pad_chars = "0123456789+-*/\015,.?()";
 +
 +static int
 +speakup_key(struct vc_data *vc, int shift_state, int keycode, u_short keysym,
 +          int up_flag)
 +{
 +      unsigned long flags;
 +      int kh;
 +      u_char *key_info;
 +      u_char type = KTYP(keysym), value = KVAL(keysym), new_key = 0;
 +      u_char shift_info, offset;
 +      int ret = 0;
 +
 +      if (!synth)
 +              return 0;
 +
 +      spin_lock_irqsave(&speakup_info.spinlock, flags);
 +      tty = vc->port.tty;
 +      if (type >= 0xf0)
 +              type -= 0xf0;
 +      if (type == KT_PAD &&
 +          (vt_get_leds(fg_console, VC_NUMLOCK))) {
 +              if (up_flag) {
 +                      spk_keydown = 0;
 +                      goto out;
 +              }
 +              value = pad_chars[value];
 +              spk_lastkey = value;
 +              spk_keydown++;
 +              spk_parked &= 0xfe;
 +              goto no_map;
 +      }
 +      if (keycode >= MAX_KEY)
 +              goto no_map;
 +      key_info = spk_our_keys[keycode];
 +      if (!key_info)
 +              goto no_map;
 +      /* Check valid read all mode keys */
 +      if ((cursor_track == read_all_mode) && (!up_flag)) {
 +              switch (value) {
 +              case KVAL(K_DOWN):
 +              case KVAL(K_UP):
 +              case KVAL(K_LEFT):
 +              case KVAL(K_RIGHT):
 +              case KVAL(K_PGUP):
 +              case KVAL(K_PGDN):
 +                      break;
 +              default:
 +                      stop_read_all(vc);
 +                      break;
 +              }
 +      }
 +      shift_info = (shift_state & 0x0f) + key_speakup;
 +      offset = spk_shift_table[shift_info];
 +      if (offset) {
 +              new_key = key_info[offset];
 +              if (new_key) {
 +                      ret = 1;
 +                      if (new_key == SPK_KEY) {
 +                              if (!spk_key_locked)
 +                                      key_speakup = (up_flag) ? 0 : 16;
 +                              if (up_flag || spk_killed)
 +                                      goto out;
 +                              spk_shut_up &= 0xfe;
 +                              spk_do_flush();
 +                              goto out;
 +                      }
 +                      if (up_flag)
 +                              goto out;
 +                      if (last_keycode == keycode &&
 +                          time_after(last_spk_jiffy + MAX_DELAY, jiffies)) {
 +                              spk_close_press = 1;
 +                              offset = spk_shift_table[shift_info + 32];
 +                              /* double press? */
 +                              if (offset && key_info[offset])
 +                                      new_key = key_info[offset];
 +                      }
 +                      last_keycode = keycode;
 +                      last_spk_jiffy = jiffies;
 +                      type = KT_SPKUP;
 +                      value = new_key;
 +              }
 +      }
 +no_map:
 +      if (type == KT_SPKUP && !spk_special_handler) {
 +              do_spkup(vc, new_key);
 +              spk_close_press = 0;
 +              ret = 1;
 +              goto out;
 +      }
 +      if (up_flag || spk_killed || type == KT_SHIFT)
 +              goto out;
 +      spk_shut_up &= 0xfe;
 +      kh = (value == KVAL(K_DOWN)) ||
 +          (value == KVAL(K_UP)) ||
 +          (value == KVAL(K_LEFT)) ||
 +          (value == KVAL(K_RIGHT));
 +      if ((cursor_track != read_all_mode) || !kh)
 +              if (!spk_no_intr)
 +                      spk_do_flush();
 +      if (spk_special_handler) {
 +              if (type == KT_SPEC && value == 1) {
 +                      value = '\n';
 +                      type = KT_LATIN;
 +              } else if (type == KT_LETTER) {
 +                      type = KT_LATIN;
 +              } else if (value == 0x7f) {
 +                      value = 8;      /* make del = backspace */
 +              }
 +              ret = (*spk_special_handler) (vc, type, value, keycode);
 +              spk_close_press = 0;
 +              if (ret < 0)
 +                      bleep(9);
 +              goto out;
 +      }
 +      last_keycode = 0;
 +out:
 +      spin_unlock_irqrestore(&speakup_info.spinlock, flags);
 +      return ret;
 +}
 +
 +static int keyboard_notifier_call(struct notifier_block *nb,
 +                                unsigned long code, void *_param)
 +{
 +      struct keyboard_notifier_param *param = _param;
 +      struct vc_data *vc = param->vc;
 +      int up = !param->down;
 +      int ret = NOTIFY_OK;
 +      static int keycode;     /* to hold the current keycode */
 +
 +      in_keyboard_notifier = 1;
 +
 +      if (vc->vc_mode == KD_GRAPHICS)
 +              goto out;
 +
 +      /*
 +       * First, determine whether we are handling a fake keypress on
 +       * the current processor.  If we are, then return NOTIFY_OK,
 +       * to pass the keystroke up the chain.  This prevents us from
 +       * trying to take the Speakup lock while it is held by the
 +       * processor on which the simulated keystroke was generated.
 +       * Also, the simulated keystrokes should be ignored by Speakup.
 +       */
 +
 +      if (speakup_fake_key_pressed())
 +              goto out;
 +
 +      switch (code) {
 +      case KBD_KEYCODE:
 +              /* speakup requires keycode and keysym currently */
 +              keycode = param->value;
 +              break;
 +      case KBD_UNBOUND_KEYCODE:
 +              /* not used yet */
 +              break;
 +      case KBD_UNICODE:
 +              /* not used yet */
 +              break;
 +      case KBD_KEYSYM:
 +              if (speakup_key(vc, param->shift, keycode, param->value, up))
 +                      ret = NOTIFY_STOP;
 +              else if (KTYP(param->value) == KT_CUR)
 +                      ret = pre_handle_cursor(vc, KVAL(param->value), up);
 +              break;
 +      case KBD_POST_KEYSYM:{
 +                      unsigned char type = KTYP(param->value) - 0xf0;
 +                      unsigned char val = KVAL(param->value);
 +
 +                      switch (type) {
 +                      case KT_SHIFT:
 +                              do_handle_shift(vc, val, up);
 +                              break;
 +                      case KT_LATIN:
 +                      case KT_LETTER:
 +                              do_handle_latin(vc, val, up);
 +                              break;
 +                      case KT_CUR:
 +                              do_handle_cursor(vc, val, up);
 +                              break;
 +                      case KT_SPEC:
 +                              do_handle_spec(vc, val, up);
 +                              break;
 +                      }
 +                      break;
 +              }
 +      }
 +out:
 +      in_keyboard_notifier = 0;
 +      return ret;
 +}
 +
 +static int vt_notifier_call(struct notifier_block *nb,
 +                          unsigned long code, void *_param)
 +{
 +      struct vt_notifier_param *param = _param;
 +      struct vc_data *vc = param->vc;
 +
 +      switch (code) {
 +      case VT_ALLOCATE:
 +              if (vc->vc_mode == KD_TEXT)
 +                      speakup_allocate(vc, GFP_ATOMIC);
 +              break;
 +      case VT_DEALLOCATE:
 +              speakup_deallocate(vc);
 +              break;
 +      case VT_WRITE:
 +              if (param->c == '\b') {
 +                      speakup_bs(vc);
 +              } else {
 +                      u16 d = param->c;
 +
 +                      speakup_con_write(vc, &d, 1);
 +              }
 +              break;
 +      case VT_UPDATE:
 +              speakup_con_update(vc);
 +              break;
 +      }
 +      return NOTIFY_OK;
 +}
 +
 +/* called by: module_exit() */
 +static void __exit speakup_exit(void)
 +{
 +      int i;
 +
 +      unregister_keyboard_notifier(&keyboard_notifier_block);
 +      unregister_vt_notifier(&vt_notifier_block);
 +      speakup_unregister_devsynth();
 +      speakup_cancel_selection();
 +      speakup_cancel_paste();
 +      del_timer_sync(&cursor_timer);
 +      kthread_stop(speakup_task);
 +      speakup_task = NULL;
 +      mutex_lock(&spk_mutex);
 +      synth_release();
 +      mutex_unlock(&spk_mutex);
 +      spk_ttyio_unregister_ldisc();
 +
 +      speakup_kobj_exit();
 +
 +      for (i = 0; i < MAX_NR_CONSOLES; i++)
 +              kfree(speakup_console[i]);
 +
 +      speakup_remove_virtual_keyboard();
 +
 +      for (i = 0; i < MAXVARS; i++)
 +              speakup_unregister_var(i);
 +
 +      for (i = 0; i < 256; i++) {
 +              if (spk_characters[i] != spk_default_chars[i])
 +                      kfree(spk_characters[i]);
 +      }
 +
 +      spk_free_user_msgs();
 +}
 +
 +/* call by: module_init() */
 +static int __init speakup_init(void)
 +{
 +      int i;
 +      long err = 0;
 +      struct vc_data *vc = vc_cons[fg_console].d;
 +      struct var_t *var;
 +
 +      /* These first few initializations cannot fail. */
 +      spk_initialize_msgs();  /* Initialize arrays for i18n. */
 +      spk_reset_default_chars();
 +      spk_reset_default_chartab();
 +      spk_strlwr(synth_name);
 +      spk_vars[0].u.n.high = vc->vc_cols;
 +      for (var = spk_vars; var->var_id != MAXVARS; var++)
 +              speakup_register_var(var);
 +      for (var = synth_time_vars;
 +           (var->var_id >= 0) && (var->var_id < MAXVARS); var++)
 +              speakup_register_var(var);
 +      for (i = 1; spk_punc_info[i].mask != 0; i++)
 +              spk_set_mask_bits(NULL, i, 2);
 +
 +      spk_set_key_info(spk_key_defaults, spk_key_buf);
 +
 +      /* From here on out, initializations can fail. */
 +      err = speakup_add_virtual_keyboard();
 +      if (err)
 +              goto error_virtkeyboard;
 +
 +      for (i = 0; i < MAX_NR_CONSOLES; i++)
 +              if (vc_cons[i].d) {
 +                      err = speakup_allocate(vc_cons[i].d, GFP_KERNEL);
 +                      if (err)
 +                              goto error_kobjects;
 +              }
 +
 +      if (spk_quiet_boot)
 +              spk_shut_up |= 0x01;
 +
 +      err = speakup_kobj_init();
 +      if (err)
 +              goto error_kobjects;
 +
 +      spk_ttyio_register_ldisc();
 +      synth_init(synth_name);
 +      speakup_register_devsynth();
 +      /*
 +       * register_devsynth might fail, but this error is not fatal.
 +       * /dev/synth is an extra feature; the rest of Speakup
 +       * will work fine without it.
 +       */
 +
 +      err = register_keyboard_notifier(&keyboard_notifier_block);
 +      if (err)
 +              goto error_kbdnotifier;
 +      err = register_vt_notifier(&vt_notifier_block);
 +      if (err)
 +              goto error_vtnotifier;
 +
 +      speakup_task = kthread_create(speakup_thread, NULL, "speakup");
 +
 +      if (IS_ERR(speakup_task)) {
 +              err = PTR_ERR(speakup_task);
 +              goto error_task;
 +      }
 +
 +      set_user_nice(speakup_task, 10);
 +      wake_up_process(speakup_task);
 +
 +      pr_info("speakup %s: initialized\n", SPEAKUP_VERSION);
 +      pr_info("synth name on entry is: %s\n", synth_name);
 +      goto out;
 +
 +error_task:
 +      unregister_vt_notifier(&vt_notifier_block);
 +
 +error_vtnotifier:
 +      unregister_keyboard_notifier(&keyboard_notifier_block);
 +      del_timer(&cursor_timer);
 +
 +error_kbdnotifier:
 +      speakup_unregister_devsynth();
 +      mutex_lock(&spk_mutex);
 +      synth_release();
 +      mutex_unlock(&spk_mutex);
 +      speakup_kobj_exit();
 +
 +error_kobjects:
 +      for (i = 0; i < MAX_NR_CONSOLES; i++)
 +              kfree(speakup_console[i]);
 +
 +      speakup_remove_virtual_keyboard();
 +
 +error_virtkeyboard:
 +      for (i = 0; i < MAXVARS; i++)
 +              speakup_unregister_var(i);
 +
 +      for (i = 0; i < 256; i++) {
 +              if (spk_characters[i] != spk_default_chars[i])
 +                      kfree(spk_characters[i]);
 +      }
 +
 +      spk_free_user_msgs();
 +
 +out:
 +      return err;
 +}
 +
 +module_init(speakup_init);
 +module_exit(speakup_exit);
index 07b7b6b05b8b2dcfc511729475b5c1ea0b35f61b,1ed3d354e16dec83ef1fbf5573b5b48c9794ad61..3aa29d201f54d38e1d24f63b883679d4a50ffa9a
@@@ -768,8 -718,8 +768,8 @@@ static void qcom_geni_serial_handle_tx(
                u8 buf[sizeof(u32)];
                int c;
  
-               memset(buf, 0, ARRAY_SIZE(buf));
+               memset(buf, 0, sizeof(buf));
 -              tx_bytes = min_t(size_t, remaining, port->tx_bytes_pw);
 +              tx_bytes = min_t(size_t, remaining, BYTES_PER_FIFO_WORD);
  
                for (c = 0; c < tx_bytes ; c++) {
                        buf[c] = xmit->buf[tail++];
Simple merge
Simple merge
index af9f5ab96f74ff8a0838efcbdf90ac2d6a578316,2997b29071721b52bd3580a423d7f38a6acede83..8a31fc2b2258a79c969d1913a7b629daa6983fc7
@@@ -676,8 -676,8 +676,8 @@@ static void fbcon_prepare_logo(struct v
                q = (unsigned short *) (vc->vc_origin +
                                        vc->vc_size_row *
                                        rows);
 -              scr_memcpyw(q, save, logo_lines * new_cols * 2);
 +              scr_memcpyw(q, save, array3_size(logo_lines, new_cols, 2));
-               vc->vc_y += logo_lines;
+               vc->state.y += logo_lines;
                vc->vc_pos += logo_lines * vc->vc_size_row;
                kfree(save);
        }