Quellcodebibliothek Statistik Leitseite products/Sources/formale Sprachen/C/Linux/drivers/tty/vt/   (Open Source Betriebssystem Version 6.17.9©)  Datei vom 24.10.2025 mit Größe 118 kB image not shown  

Quelle  vt.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0
/*
 *  Copyright (C) 1991, 1992  Linus Torvalds
 */


/*
 * Hopefully this will be a rather complete VT102 implementation.
 *
 * Beeping thanks to John T Kohl.
 *
 * Virtual Consoles, Screen Blanking, Screen Dumping, Color, Graphics
 *   Chars, and VT100 enhancements by Peter MacDonald.
 *
 * Copy and paste function by Andrew Haylett,
 *   some enhancements by Alessandro Rubini.
 *
 * Code to check for different video-cards mostly by Galen Hunt,
 * <g-hunt@ee.utah.edu>
 *
 * Rudimentary ISO 10646/Unicode/UTF-8 character set support by
 * Markus Kuhn, <mskuhn@immd4.informatik.uni-erlangen.de>.
 *
 * Dynamic allocation of consoles, aeb@cwi.nl, May 1994
 * Resizing of consoles, aeb, 940926
 *
 * Code for xterm like mouse click reporting by Peter Orbaek 20-Jul-94
 * <poe@daimi.aau.dk>
 *
 * User-defined bell sound, new setterm control sequences and printk
 * redirection by Martin Mares <mj@k332.feld.cvut.cz> 19-Nov-95
 *
 * APM screenblank bug fixed Takashi Manabe <manabe@roy.dsl.tutics.tut.jp>
 *
 * Merge with the abstract console driver by Geert Uytterhoeven
 * <geert@linux-m68k.org>, Jan 1997.
 *
 *   Original m68k console driver modifications by
 *
 *     - Arno Griffioen <arno@usn.nl>
 *     - David Carter <carter@cs.bris.ac.uk>
 * 
 *   The abstract console driver provides a generic interface for a text
 *   console. It supports VGA text mode, frame buffer based graphical consoles
 *   and special graphics processors that are only accessible through some
 *   registers (e.g. a TMS340x0 GSP).
 *
 *   The interface to the hardware is specified using a special structure
 *   (struct consw) which contains function pointers to console operations
 *   (see <linux/console.h> for more information).
 *
 * Support for changeable cursor shape
 * by Pavel Machek <pavel@atrey.karlin.mff.cuni.cz>, August 1997
 *
 * Ported to i386 and con_scrolldelta fixed
 * by Emmanuel Marty <core@ggi-project.org>, April 1998
 *
 * Resurrected character buffers in videoram plus lots of other trickery
 * by Martin Mares <mj@atrey.karlin.mff.cuni.cz>, July 1998
 *
 * Removed old-style timers, introduced console_timer, made timer
 * deletion SMP-safe.  17Jun00, Andrew Morton
 *
 * Removed console_lock, enabled interrupts across all console operations
 * 13 March 2001, Andrew Morton
 *
 * Fixed UTF-8 mode so alternate charset modes always work according
 * to control sequences interpreted in do_con_trol function
 * preserving backward VT100 semigraphics compatibility,
 * malformed UTF sequences represented as sequences of replacement glyphs,
 * original codes or '?' as a last resort if replacement glyph is undefined
 * by Adam Tla/lka <atlka@pg.gda.pl>, Aug 2006
 */


#include <linux/module.h>
#include <linux/types.h>
#include <linux/sched/signal.h>
#include <linux/tty.h>
#include <linux/tty_flip.h>
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/errno.h>
#include <linux/kd.h>
#include <linux/slab.h>
#include <linux/vmalloc.h>
#include <linux/major.h>
#include <linux/mm.h>
#include <linux/console.h>
#include <linux/init.h>
#include <linux/mutex.h>
#include <linux/vt_kern.h>
#include <linux/selection.h>
#include <linux/tiocl.h>
#include <linux/kbd_kern.h>
#include <linux/consolemap.h>
#include <linux/timer.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>
#include <linux/pm.h>
#include <linux/font.h>
#include <linux/bitops.h>
#include <linux/notifier.h>
#include <linux/device.h>
#include <linux/io.h>
#include <linux/uaccess.h>
#include <linux/kdb.h>
#include <linux/ctype.h>
#include <linux/gcd.h>

#define MAX_NR_CON_DRIVER 16

#define CON_DRIVER_FLAG_MODULE 1
#define CON_DRIVER_FLAG_INIT   2
#define CON_DRIVER_FLAG_ATTR   4
#define CON_DRIVER_FLAG_ZOMBIE 8

struct con_driver {
 const struct consw *con;
 const char *desc;
 struct device *dev;
 int node;
 int first;
 int last;
 int flag;
};

static struct con_driver registered_con_driver[MAX_NR_CON_DRIVER];
const struct consw *conswitchp;

/*
 * Here is the default bell parameters: 750HZ, 1/8th of a second
 */

#define DEFAULT_BELL_PITCH 750
#define DEFAULT_BELL_DURATION (HZ/8)
#define DEFAULT_CURSOR_BLINK_MS 200

struct vc vc_cons [MAX_NR_CONSOLES];
EXPORT_SYMBOL(vc_cons);

static const struct consw *con_driver_map[MAX_NR_CONSOLES];

static int con_open(struct tty_struct *, struct file *);
static void vc_init(struct vc_data *vc, int do_clear);
static void gotoxy(struct vc_data *vc, int new_x, int new_y);
static void save_cur(struct vc_data *vc);
static void reset_terminal(struct vc_data *vc, int do_clear);
static void con_flush_chars(struct tty_struct *tty);
static int set_vesa_blanking(u8 __user *mode);
static void set_cursor(struct vc_data *vc);
static void hide_cursor(struct vc_data *vc);
static void console_callback(struct work_struct *ignored);
static void con_driver_unregister_callback(struct work_struct *ignored);
static void blank_screen_t(struct timer_list *unused);
static void set_palette(struct vc_data *vc);
static void unblank_screen(void);

#define vt_get_kmsg_redirect() vt_kmsg_redirect(-1)

int default_utf8 = true;
module_param(default_utf8, int, S_IRUGO | S_IWUSR);
int global_cursor_default = -1;
module_param(global_cursor_default, int, S_IRUGO | S_IWUSR);
EXPORT_SYMBOL(global_cursor_default);

static int cur_default = CUR_UNDERLINE;
module_param(cur_default, int, S_IRUGO | S_IWUSR);

/*
 * ignore_poke: don't unblank the screen when things are typed.  This is
 * mainly for the privacy of braille terminal users.
 */

static int ignore_poke;

int do_poke_blanked_console;
int console_blanked;
EXPORT_SYMBOL(console_blanked);

static enum vesa_blank_mode vesa_blank_mode;
static int vesa_off_interval;
static int blankinterval;
core_param(consoleblank, blankinterval, int, 0444);

static DECLARE_WORK(console_work, console_callback);
static DECLARE_WORK(con_driver_unregister_work, con_driver_unregister_callback);

/*
 * fg_console is the current virtual console,
 * last_console is the last used one,
 * want_console is the console we want to switch to,
 * saved_* variants are for save/restore around kernel debugger enter/leave
 */

int fg_console;
EXPORT_SYMBOL(fg_console);
int last_console;
int want_console = -1;

static int saved_fg_console;
static int saved_last_console;
static int saved_want_console;
static int saved_vc_mode;
static int saved_console_blanked;

/*
 * For each existing display, we have a pointer to console currently visible
 * on that display, allowing consoles other than fg_console to be refreshed
 * appropriately. Unless the low-level driver supplies its own display_fg
 * variable, we use this one for the "master display".
 */

static struct vc_data *master_display_fg;

/*
 * Unfortunately, we need to delay tty echo when we're currently writing to the
 * console since the code is (and always was) not re-entrant, so we schedule
 * all flip requests to process context with schedule-task() and run it from
 * console_callback().
 */


/*
 * For the same reason, we defer scrollback to the console callback.
 */

static int scrollback_delta;

/*
 * Hook so that the power management routines can (un)blank
 * the console on our behalf.
 */

int (*console_blank_hook)(int);
EXPORT_SYMBOL(console_blank_hook);

static DEFINE_TIMER(console_timer, blank_screen_t);
static int blank_state;
static int blank_timer_expired;
enum {
 blank_off = 0,
 blank_normal_wait,
 blank_vesa_wait,
};

/*
 * /sys/class/tty/tty0/
 *
 * the attribute 'active' contains the name of the current vc
 * console and it supports poll() to detect vc switches
 */

static struct device *tty0dev;

/*
 * Notifier list for console events.
 */

static ATOMIC_NOTIFIER_HEAD(vt_notifier_list);

int register_vt_notifier(struct notifier_block *nb)
{
 return atomic_notifier_chain_register(&vt_notifier_list, nb);
}
EXPORT_SYMBOL_GPL(register_vt_notifier);

int unregister_vt_notifier(struct notifier_block *nb)
{
 return atomic_notifier_chain_unregister(&vt_notifier_list, nb);
}
EXPORT_SYMBOL_GPL(unregister_vt_notifier);

static void notify_write(struct vc_data *vc, unsigned int unicode)
{
 struct vt_notifier_param param = { .vc = vc, .c = unicode };
 atomic_notifier_call_chain(&vt_notifier_list, VT_WRITE, ¶m);
}

static void notify_update(struct vc_data *vc)
{
 struct vt_notifier_param param = { .vc = vc };
 atomic_notifier_call_chain(&vt_notifier_list, VT_UPDATE, ¶m);
}
/*
 * Low-Level Functions
 */


static inline bool con_is_fg(const struct vc_data *vc)
{
 return vc->vc_num == fg_console;
}

static inline bool con_should_update(const struct vc_data *vc)
{
 return con_is_visible(vc) && !console_blanked;
}

static inline u16 *screenpos(const struct vc_data *vc, unsigned int offset,
        bool viewed)
{
 unsigned long origin = viewed ? vc->vc_visible_origin : vc->vc_origin;

 return (u16 *)(origin + offset);
}

static void con_putc(struct vc_data *vc, u16 ca, unsigned int y, unsigned int x)
{
 if (vc->vc_sw->con_putc)
  vc->vc_sw->con_putc(vc, ca, y, x);
 else
  vc->vc_sw->con_putcs(vc, &ca, 1, y, x);
}

/* Called  from the keyboard irq path.. */
static inline void scrolldelta(int lines)
{
 /* FIXME */
 /* scrolldelta needs some kind of consistency lock, but the BKL was
   and still is not protecting versus the scheduled back end */

 scrollback_delta += lines;
 schedule_console_callback();
}

void schedule_console_callback(void)
{
 schedule_work(&console_work);
}

/*
 * Code to manage unicode-based screen buffers
 */


/*
 * Our screen buffer is preceded by an array of line pointers so that
 * scrolling only implies some pointer shuffling.
 */


static u32 **vc_uniscr_alloc(unsigned int cols, unsigned int rows)
{
 u32 **uni_lines;
 void *p;
 unsigned int memsize, i, col_size = cols * sizeof(**uni_lines);

 /* allocate everything in one go */
 memsize = col_size * rows;
 memsize += rows * sizeof(*uni_lines);
 uni_lines = vzalloc(memsize);
 if (!uni_lines)
  return NULL;

 /* initial line pointers */
 p = uni_lines + rows;
 for (i = 0; i < rows; i++) {
  uni_lines[i] = p;
  p += col_size;
 }

 return uni_lines;
}

static void vc_uniscr_free(u32 **uni_lines)
{
 vfree(uni_lines);
}

static void vc_uniscr_set(struct vc_data *vc, u32 **new_uni_lines)
{
 vc_uniscr_free(vc->vc_uni_lines);
 vc->vc_uni_lines = new_uni_lines;
}

static void vc_uniscr_putc(struct vc_data *vc, u32 uc)
{
 if (vc->vc_uni_lines)
  vc->vc_uni_lines[vc->state.y][vc->state.x] = uc;
}

static void vc_uniscr_insert(struct vc_data *vc, unsigned int nr)
{
 if (vc->vc_uni_lines) {
  u32 *ln = vc->vc_uni_lines[vc->state.y];
  unsigned int x = vc->state.x, cols = vc->vc_cols;

  memmove(&ln[x + nr], &ln[x], (cols - x - nr) * sizeof(*ln));
  memset32(&ln[x], ' ', nr);
 }
}

static void vc_uniscr_delete(struct vc_data *vc, unsigned int nr)
{
 if (vc->vc_uni_lines) {
  u32 *ln = vc->vc_uni_lines[vc->state.y];
  unsigned int x = vc->state.x, cols = vc->vc_cols;

  memmove(&ln[x], &ln[x + nr], (cols - x - nr) * sizeof(*ln));
  memset32(&ln[cols - nr], ' ', nr);
 }
}

static void vc_uniscr_clear_line(struct vc_data *vc, unsigned int x,
     unsigned int nr)
{
 if (vc->vc_uni_lines)
  memset32(&vc->vc_uni_lines[vc->state.y][x], ' ', nr);
}

static void vc_uniscr_clear_lines(struct vc_data *vc, unsigned int y,
      unsigned int nr)
{
 if (vc->vc_uni_lines)
  while (nr--)
   memset32(vc->vc_uni_lines[y++], ' ', vc->vc_cols);
}

/* juggling array rotation algorithm (complexity O(N), size complexity O(1)) */
static void juggle_array(u32 **array, unsigned int size, unsigned int nr)
{
 unsigned int gcd_idx;

 for (gcd_idx = 0; gcd_idx < gcd(nr, size); gcd_idx++) {
  u32 *gcd_idx_val = array[gcd_idx];
  unsigned int dst_idx = gcd_idx;

  while (1) {
   unsigned int src_idx = (dst_idx + nr) % size;
   if (src_idx == gcd_idx)
    break;

   array[dst_idx] = array[src_idx];
   dst_idx = src_idx;
  }

  array[dst_idx] = gcd_idx_val;
 }
}

static void vc_uniscr_scroll(struct vc_data *vc, unsigned int top,
        unsigned int bottom, enum con_scroll dir,
        unsigned int nr)
{
 u32 **uni_lines = vc->vc_uni_lines;
 unsigned int size = bottom - top;

 if (!uni_lines)
  return;

 if (dir == SM_DOWN) {
  juggle_array(&uni_lines[top], size, size - nr);
  vc_uniscr_clear_lines(vc, top, nr);
 } else {
  juggle_array(&uni_lines[top], size, nr);
  vc_uniscr_clear_lines(vc, bottom - nr, nr);
 }
}

static u32 vc_uniscr_getc(struct vc_data *vc, int relative_pos)
{
 int pos = vc->state.x + vc->vc_need_wrap + relative_pos;

 if (vc->vc_uni_lines && in_range(pos, 0, vc->vc_cols))
  return vc->vc_uni_lines[vc->state.y][pos];
 return 0;
}

static void vc_uniscr_copy_area(u32 **dst_lines,
    unsigned int dst_cols,
    unsigned int dst_rows,
    u32 **src_lines,
    unsigned int src_cols,
    unsigned int src_top_row,
    unsigned int src_bot_row)
{
 unsigned int dst_row = 0;

 if (!dst_lines)
  return;

 while (src_top_row < src_bot_row) {
  u32 *src_line = src_lines[src_top_row];
  u32 *dst_line = dst_lines[dst_row];

  memcpy(dst_line, src_line, src_cols * sizeof(*src_line));
  if (dst_cols - src_cols)
   memset32(dst_line + src_cols, ' ', dst_cols - src_cols);
  src_top_row++;
  dst_row++;
 }
 while (dst_row < dst_rows) {
  u32 *dst_line = dst_lines[dst_row];

  memset32(dst_line, ' ', dst_cols);
  dst_row++;
 }
}

/*
 * Called from vcs_read() to make sure unicode screen retrieval is possible.
 * This will initialize the unicode screen buffer if not already done.
 * This returns 0 if OK, or a negative error code otherwise.
 * In particular, -ENODATA is returned if the console is not in UTF-8 mode.
 */

int vc_uniscr_check(struct vc_data *vc)
{
 u32 **uni_lines;
 unsigned short *p;
 int x, y, mask;

 WARN_CONSOLE_UNLOCKED();

 if (!vc->vc_utf)
  return -ENODATA;

 if (vc->vc_uni_lines)
  return 0;

 uni_lines = vc_uniscr_alloc(vc->vc_cols, vc->vc_rows);
 if (!uni_lines)
  return -ENOMEM;

 /*
 * Let's populate it initially with (imperfect) reverse translation.
 * This is the next best thing we can do short of having it enabled
 * from the start even when no users rely on this functionality. True
 * unicode content will be available after a complete screen refresh.
 */

 p = (unsigned short *)vc->vc_origin;
 mask = vc->vc_hi_font_mask | 0xff;
 for (y = 0; y < vc->vc_rows; y++) {
  u32 *line = uni_lines[y];
  for (x = 0; x < vc->vc_cols; x++) {
   u16 glyph = scr_readw(p++) & mask;
   line[x] = inverse_translate(vc, glyph, true);
  }
 }

 vc->vc_uni_lines = uni_lines;

 return 0;
}

/*
 * Called from vcs_read() to get the unicode data from the screen.
 * This must be preceded by a successful call to vc_uniscr_check() once
 * the console lock has been taken.
 */

void vc_uniscr_copy_line(const struct vc_data *vc, void *dest, bool viewed,
    unsigned int row, unsigned int col, unsigned int nr)
{
 u32 **uni_lines = vc->vc_uni_lines;
 int offset = row * vc->vc_size_row + col * 2;
 unsigned long pos;

 if (WARN_ON_ONCE(!uni_lines))
  return;

 pos = (unsigned long)screenpos(vc, offset, viewed);
 if (pos >= vc->vc_origin && pos < vc->vc_scr_end) {
  /*
 * Desired position falls in the main screen buffer.
 * However the actual row/col might be different if
 * scrollback is active.
 */

  row = (pos - vc->vc_origin) / vc->vc_size_row;
  col = ((pos - vc->vc_origin) % vc->vc_size_row) / 2;
  memcpy(dest, &uni_lines[row][col], nr * sizeof(u32));
 } else {
  /*
 * Scrollback is active. For now let's simply backtranslate
 * the screen glyphs until the unicode screen buffer does
 * synchronize with console display drivers for a scrollback
 * buffer of its own.
 */

  u16 *p = (u16 *)pos;
  int mask = vc->vc_hi_font_mask | 0xff;
  u32 *uni_buf = dest;
  while (nr--) {
   u16 glyph = scr_readw(p++) & mask;
   *uni_buf++ = inverse_translate(vc, glyph, true);
  }
 }
}

static void con_scroll(struct vc_data *vc, unsigned int top,
         unsigned int bottom, enum con_scroll dir,
         unsigned int nr)
{
 unsigned int rows = bottom - top;
 u16 *clear, *dst, *src;

 if (top + nr >= bottom)
  nr = rows - 1;
 if (bottom > vc->vc_rows || top >= bottom || nr < 1)
  return;

 vc_uniscr_scroll(vc, top, bottom, dir, nr);
 if (con_is_visible(vc) &&
   vc->vc_sw->con_scroll(vc, top, bottom, dir, nr))
  return;

 src = clear = (u16 *)(vc->vc_origin + vc->vc_size_row * top);
 dst = (u16 *)(vc->vc_origin + vc->vc_size_row * (top + nr));

 if (dir == SM_UP) {
  clear = src + (rows - nr) * vc->vc_cols;
  swap(src, dst);
 }
 scr_memmovew(dst, src, (rows - nr) * vc->vc_size_row);
 scr_memsetw(clear, vc->vc_video_erase_char, vc->vc_size_row * nr);
}

static void do_update_region(struct vc_data *vc, unsigned long start, int count)
{
 unsigned int xx, yy, offset;
 u16 *p = (u16 *)start;

 offset = (start - vc->vc_origin) / 2;
 xx = offset % vc->vc_cols;
 yy = offset / vc->vc_cols;

 for(;;) {
  u16 attrib = scr_readw(p) & 0xff00;
  int startx = xx;
  u16 *q = p;
  while (xx < vc->vc_cols && count) {
   if (attrib != (scr_readw(p) & 0xff00)) {
    if (p > q)
     vc->vc_sw->con_putcs(vc, q, p-q, yy, startx);
    startx = xx;
    q = p;
    attrib = scr_readw(p) & 0xff00;
   }
   p++;
   xx++;
   count--;
  }
  if (p > q)
   vc->vc_sw->con_putcs(vc, q, p-q, yy, startx);
  if (!count)
   break;
  xx = 0;
  yy++;
 }
}

void update_region(struct vc_data *vc, unsigned long start, int count)
{
 WARN_CONSOLE_UNLOCKED();

 if (con_should_update(vc)) {
  hide_cursor(vc);
  do_update_region(vc, start, count);
  set_cursor(vc);
 }
}
EXPORT_SYMBOL(update_region);

/* Structure of attributes is hardware-dependent */

static u8 build_attr(struct vc_data *vc, u8 _color,
  enum vc_intensity _intensity, bool _blink, bool _underline,
  bool _reverse, bool _italic)
{
 if (vc->vc_sw->con_build_attr)
  return vc->vc_sw->con_build_attr(vc, _color, _intensity,
         _blink, _underline, _reverse, _italic);

/*
 * ++roman: I completely changed the attribute format for monochrome
 * mode (!can_do_color). The formerly used MDA (monochrome display
 * adapter) format didn't allow the combination of certain effects.
 * Now the attribute is just a bit vector:
 *  Bit 0..1: intensity (0..2)
 *  Bit 2   : underline
 *  Bit 3   : reverse
 *  Bit 7   : blink
 */

 {
 u8 a = _color;
 if (!vc->vc_can_do_color)
  return _intensity |
         (_italic    << 1) |
         (_underline << 2) |
         (_reverse   << 3) |
         (_blink     << 7);
 if (_italic)
  a = (a & 0xF0) | vc->vc_itcolor;
 else if (_underline)
  a = (a & 0xf0) | vc->vc_ulcolor;
 else if (_intensity == VCI_HALF_BRIGHT)
  a = (a & 0xf0) | vc->vc_halfcolor;
 if (_reverse)
  a = (a & 0x88) | (((a >> 4) | (a << 4)) & 0x77);
 if (_blink)
  a ^= 0x80;
 if (_intensity == VCI_BOLD)
  a ^= 0x08;
 if (vc->vc_hi_font_mask == 0x100)
  a <<= 1;
 return a;
 }
}

static void update_attr(struct vc_data *vc)
{
 vc->vc_attr = build_attr(vc, vc->state.color, vc->state.intensity,
               vc->state.blink, vc->state.underline,
               vc->state.reverse ^ vc->vc_decscnm, vc->state.italic);
 vc->vc_video_erase_char = ' ' | (build_attr(vc, vc->state.color,
    VCI_NORMAL, vc->state.blink, false,
    vc->vc_decscnm, false) << 8);
}

/* Note: inverting the screen twice should revert to the original state */
void invert_screen(struct vc_data *vc, int offset, int count, bool viewed)
{
 u16 *p;

 WARN_CONSOLE_UNLOCKED();

 count /= 2;
 p = screenpos(vc, offset, viewed);
 if (vc->vc_sw->con_invert_region) {
  vc->vc_sw->con_invert_region(vc, p, count);
 } else {
  u16 *q = p;
  int cnt = count;
  u16 a;

  if (!vc->vc_can_do_color) {
   while (cnt--) {
       a = scr_readw(q);
       a ^= 0x0800;
       scr_writew(a, q);
       q++;
   }
  } else if (vc->vc_hi_font_mask == 0x100) {
   while (cnt--) {
    a = scr_readw(q);
    a = (a & 0x11ff) |
       ((a & 0xe000) >> 4) |
       ((a & 0x0e00) << 4);
    scr_writew(a, q);
    q++;
   }
  } else {
   while (cnt--) {
    a = scr_readw(q);
    a = (a & 0x88ff) |
       ((a & 0x7000) >> 4) |
       ((a & 0x0700) << 4);
    scr_writew(a, q);
    q++;
   }
  }
 }

 if (con_should_update(vc))
  do_update_region(vc, (unsigned long) p, count);
 notify_update(vc);
}

/* used by selection: complement pointer position */
void complement_pos(struct vc_data *vc, int offset)
{
 static int old_offset = -1;
 static unsigned short old;
 static unsigned short oldx, oldy;

 WARN_CONSOLE_UNLOCKED();

 if (old_offset != -1 && old_offset >= 0 &&
     old_offset < vc->vc_screenbuf_size) {
  scr_writew(old, screenpos(vc, old_offset, true));
  if (con_should_update(vc))
   con_putc(vc, old, oldy, oldx);
  notify_update(vc);
 }

 old_offset = offset;

 if (offset != -1 && offset >= 0 &&
     offset < vc->vc_screenbuf_size) {
  unsigned short new;
  u16 *p = screenpos(vc, offset, true);
  old = scr_readw(p);
  new = old ^ vc->vc_complement_mask;
  scr_writew(new, p);
  if (con_should_update(vc)) {
   oldx = (offset >> 1) % vc->vc_cols;
   oldy = (offset >> 1) / vc->vc_cols;
   con_putc(vc, new, oldy, oldx);
  }
  notify_update(vc);
 }
}

static void insert_char(struct vc_data *vc, unsigned int nr)
{
 unsigned short *p = (unsigned short *) vc->vc_pos;

 vc_uniscr_insert(vc, nr);
 scr_memmovew(p + nr, p, (vc->vc_cols - vc->state.x - nr) * 2);
 scr_memsetw(p, vc->vc_video_erase_char, nr * 2);
 vc->vc_need_wrap = 0;
 if (con_should_update(vc))
  do_update_region(vc, (unsigned long) p,
   vc->vc_cols - vc->state.x);
}

static void delete_char(struct vc_data *vc, unsigned int nr)
{
 unsigned short *p = (unsigned short *) vc->vc_pos;

 vc_uniscr_delete(vc, nr);
 scr_memmovew(p, p + nr, (vc->vc_cols - vc->state.x - nr) * 2);
 scr_memsetw(p + vc->vc_cols - vc->state.x - nr, vc->vc_video_erase_char,
   nr * 2);
 vc->vc_need_wrap = 0;
 if (con_should_update(vc))
  do_update_region(vc, (unsigned long) p,
   vc->vc_cols - vc->state.x);
}

static int softcursor_original = -1;

static void add_softcursor(struct vc_data *vc)
{
 int i = scr_readw((u16 *) vc->vc_pos);
 u32 type = vc->vc_cursor_type;

 if (!(type & CUR_SW))
  return;
 if (softcursor_original != -1)
  return;
 softcursor_original = i;
 i |= CUR_SET(type);
 i ^= CUR_CHANGE(type);
 if ((type & CUR_ALWAYS_BG) &&
   (softcursor_original & CUR_BG) == (i & CUR_BG))
  i ^= CUR_BG;
 if ((type & CUR_INVERT_FG_BG) && (i & CUR_FG) == ((i & CUR_BG) >> 4))
  i ^= CUR_FG;
 scr_writew(i, (u16 *)vc->vc_pos);
 if (con_should_update(vc))
  con_putc(vc, i, vc->state.y, vc->state.x);
}

static void hide_softcursor(struct vc_data *vc)
{
 if (softcursor_original != -1) {
  scr_writew(softcursor_original, (u16 *)vc->vc_pos);
  if (con_should_update(vc))
   con_putc(vc, softcursor_original, vc->state.y,
     vc->state.x);
  softcursor_original = -1;
 }
}

static void hide_cursor(struct vc_data *vc)
{
 if (vc_is_sel(vc))
  clear_selection();

 vc->vc_sw->con_cursor(vc, false);
 hide_softcursor(vc);
}

static void set_cursor(struct vc_data *vc)
{
 if (!con_is_fg(vc) || console_blanked || vc->vc_mode == KD_GRAPHICS)
  return;
 if (vc->vc_deccm) {
  if (vc_is_sel(vc))
   clear_selection();
  add_softcursor(vc);
  if (CUR_SIZE(vc->vc_cursor_type) != CUR_NONE)
   vc->vc_sw->con_cursor(vc, true);
 } else
  hide_cursor(vc);
}

static void set_origin(struct vc_data *vc)
{
 WARN_CONSOLE_UNLOCKED();

 if (!con_is_visible(vc) ||
     !vc->vc_sw->con_set_origin ||
     !vc->vc_sw->con_set_origin(vc))
  vc->vc_origin = (unsigned long)vc->vc_screenbuf;
 vc->vc_visible_origin = vc->vc_origin;
 vc->vc_scr_end = vc->vc_origin + vc->vc_screenbuf_size;
 vc->vc_pos = vc->vc_origin + vc->vc_size_row * vc->state.y +
  2 * vc->state.x;
}

static void save_screen(struct vc_data *vc)
{
 WARN_CONSOLE_UNLOCKED();

 if (vc->vc_sw->con_save_screen)
  vc->vc_sw->con_save_screen(vc);
}

static void flush_scrollback(struct vc_data *vc)
{
 WARN_CONSOLE_UNLOCKED();

 set_origin(vc);
 if (!con_is_visible(vc))
  return;

 /*
 * The legacy way for flushing the scrollback buffer is to use a side
 * effect of the con_switch method. We do it only on the foreground
 * console as background consoles have no scrollback buffers in that
 * case and we obviously don't want to switch to them.
 */

 hide_cursor(vc);
 vc->vc_sw->con_switch(vc);
 set_cursor(vc);
}

/*
 * Redrawing of screen
 */


void clear_buffer_attributes(struct vc_data *vc)
{
 unsigned short *p = (unsigned short *)vc->vc_origin;
 int count = vc->vc_screenbuf_size / 2;
 int mask = vc->vc_hi_font_mask | 0xff;

 for (; count > 0; count--, p++) {
  scr_writew((scr_readw(p)&mask) | (vc->vc_video_erase_char & ~mask), p);
 }
}

void redraw_screen(struct vc_data *vc, int is_switch)
{
 int redraw = 0;

 WARN_CONSOLE_UNLOCKED();

 if (!vc) {
  /* strange ... */
  /* printk("redraw_screen: tty %d not allocated ??\n", new_console+1); */
  return;
 }

 if (is_switch) {
  struct vc_data *old_vc = vc_cons[fg_console].d;
  if (old_vc == vc)
   return;
  if (!con_is_visible(vc))
   redraw = 1;
  *vc->vc_display_fg = vc;
  fg_console = vc->vc_num;
  hide_cursor(old_vc);
  if (!con_is_visible(old_vc)) {
   save_screen(old_vc);
   set_origin(old_vc);
  }
  if (tty0dev)
   sysfs_notify(&tty0dev->kobj, NULL, "active");
 } else {
  hide_cursor(vc);
  redraw = 1;
 }

 if (redraw) {
  bool update;
  int old_was_color = vc->vc_can_do_color;

  set_origin(vc);
  update = vc->vc_sw->con_switch(vc);
  set_palette(vc);
  /*
 * If console changed from mono<->color, the best we can do
 * is to clear the buffer attributes. As it currently stands,
 * rebuilding new attributes from the old buffer is not doable
 * without overly complex code.
 */

  if (old_was_color != vc->vc_can_do_color) {
   update_attr(vc);
   clear_buffer_attributes(vc);
  }

  if (update && vc->vc_mode != KD_GRAPHICS)
   do_update_region(vc, vc->vc_origin, vc->vc_screenbuf_size / 2);
 }
 set_cursor(vc);
 if (is_switch) {
  vt_set_leds_compute_shiftstate();
  notify_update(vc);
 }
}
EXPORT_SYMBOL(redraw_screen);

/*
 * Allocation, freeing and resizing of VTs.
 */


int vc_cons_allocated(unsigned int i)
{
 return (i < MAX_NR_CONSOLES && vc_cons[i].d);
}

static void visual_init(struct vc_data *vc, int num, bool init)
{
 /* ++Geert: vc->vc_sw->con_init determines console size */
 if (vc->vc_sw)
  module_put(vc->vc_sw->owner);
 vc->vc_sw = conswitchp;

 if (con_driver_map[num])
  vc->vc_sw = con_driver_map[num];

 __module_get(vc->vc_sw->owner);
 vc->vc_num = num;
 vc->vc_display_fg = &master_display_fg;
 if (vc->uni_pagedict_loc)
  con_free_unimap(vc);
 vc->uni_pagedict_loc = &vc->uni_pagedict;
 vc->uni_pagedict = NULL;
 vc->vc_hi_font_mask = 0;
 vc->vc_complement_mask = 0;
 vc->vc_can_do_color = 0;
 vc->vc_cur_blink_ms = DEFAULT_CURSOR_BLINK_MS;
 vc->vc_sw->con_init(vc, init);
 if (!vc->vc_complement_mask)
  vc->vc_complement_mask = vc->vc_can_do_color ? 0x7700 : 0x0800;
 vc->vc_s_complement_mask = vc->vc_complement_mask;
 vc->vc_size_row = vc->vc_cols << 1;
 vc->vc_screenbuf_size = vc->vc_rows * vc->vc_size_row;
}


static void visual_deinit(struct vc_data *vc)
{
 vc->vc_sw->con_deinit(vc);
 module_put(vc->vc_sw->owner);
}

static void vc_port_destruct(struct tty_port *port)
{
 struct vc_data *vc = container_of(port, struct vc_data, port);

 kfree(vc);
}

static const struct tty_port_operations vc_port_ops = {
 .destruct = vc_port_destruct,
};

/*
 * Change # of rows and columns (0 means unchanged/the size of fg_console)
 * [this is to be used together with some user program
 * like resize that changes the hardware videomode]
 */

#define VC_MAXCOL (32767)
#define VC_MAXROW (32767)

int vc_allocate(unsigned int currcons) /* return 0 on success */
{
 struct vt_notifier_param param;
 struct vc_data *vc;
 int err;

 WARN_CONSOLE_UNLOCKED();

 if (currcons >= MAX_NR_CONSOLES)
  return -ENXIO;

 if (vc_cons[currcons].d)
  return 0;

 /* due to the granularity of kmalloc, we waste some memory here */
 /* the alloc is done in two steps, to optimize the common situation
   of a 25x80 console (structsize=216, screenbuf_size=4000) */

 /* although the numbers above are not valid since long ago, the
   point is still up-to-date and the comment still has its value
   even if only as a historical artifact.  --mj, July 1998 */

 param.vc = vc = kzalloc(sizeof(struct vc_data), GFP_KERNEL);
 if (!vc)
  return -ENOMEM;

 vc_cons[currcons].d = vc;
 tty_port_init(&vc->port);
 vc->port.ops = &vc_port_ops;
 INIT_WORK(&vc_cons[currcons].SAK_work, vc_SAK);

 visual_init(vc, currcons, true);

 if (!*vc->uni_pagedict_loc)
  con_set_default_unimap(vc);

 err = -EINVAL;
 if (vc->vc_cols > VC_MAXCOL || vc->vc_rows > VC_MAXROW ||
     vc->vc_screenbuf_size > KMALLOC_MAX_SIZE || !vc->vc_screenbuf_size)
  goto err_free;
 err = -ENOMEM;
 vc->vc_screenbuf = kzalloc(vc->vc_screenbuf_size, GFP_KERNEL);
 if (!vc->vc_screenbuf)
  goto err_free;

 /* If no drivers have overridden us and the user didn't pass a
   boot option, default to displaying the cursor */

 if (global_cursor_default == -1)
  global_cursor_default = 1;

 vc_init(vc, 1);
 vcs_make_sysfs(currcons);
 atomic_notifier_call_chain(&vt_notifier_list, VT_ALLOCATE, ¶m);

 return 0;
err_free:
 visual_deinit(vc);
 kfree(vc);
 vc_cons[currcons].d = NULL;
 return err;
}

static inline int resize_screen(struct vc_data *vc, int width, int height,
    bool from_user)
{
 /* Resizes the resolution of the display adapater */
 int err = 0;

 if (vc->vc_sw->con_resize)
  err = vc->vc_sw->con_resize(vc, width, height, from_user);

 return err;
}

/**
 * vc_do_resize - resizing method for the tty
 * @tty: tty being resized
 * @vc: virtual console private data
 * @cols: columns
 * @lines: lines
 * @from_user: invoked by a user?
 *
 * Resize a virtual console, clipping according to the actual constraints. If
 * the caller passes a tty structure then update the termios winsize
 * information and perform any necessary signal handling.
 *
 * Locking: Caller must hold the console semaphore. Takes the termios rwsem and
 * ctrl.lock of the tty IFF a tty is passed.
 */

static int vc_do_resize(struct tty_struct *tty, struct vc_data *vc,
   unsigned int cols, unsigned int lines, bool from_user)
{
 unsigned long old_origin, new_origin, new_scr_end, rlth, rrem, err = 0;
 unsigned long end;
 unsigned int old_rows, old_row_size, first_copied_row;
 unsigned int new_cols, new_rows, new_row_size, new_screen_size;
 unsigned short *oldscreen, *newscreen;
 u32 **new_uniscr = NULL;

 WARN_CONSOLE_UNLOCKED();

 if (cols > VC_MAXCOL || lines > VC_MAXROW)
  return -EINVAL;

 new_cols = (cols ? cols : vc->vc_cols);
 new_rows = (lines ? lines : vc->vc_rows);
 new_row_size = new_cols << 1;
 new_screen_size = new_row_size * new_rows;

 if (new_cols == vc->vc_cols && new_rows == vc->vc_rows) {
  /*
 * This function is being called here to cover the case
 * where the userspace calls the FBIOPUT_VSCREENINFO twice,
 * passing the same fb_var_screeninfo containing the fields
 * yres/xres equal to a number non-multiple of vc_font.height
 * and yres_virtual/xres_virtual equal to number lesser than the
 * vc_font.height and yres/xres.
 * In the second call, the struct fb_var_screeninfo isn't
 * being modified by the underlying driver because of the
 * if above, and this causes the fbcon_display->vrows to become
 * negative and it eventually leads to out-of-bound
 * access by the imageblit function.
 * To give the correct values to the struct and to not have
 * to deal with possible errors from the code below, we call
 * the resize_screen here as well.
 */

  return resize_screen(vc, new_cols, new_rows, from_user);
 }

 if (new_screen_size > KMALLOC_MAX_SIZE || !new_screen_size)
  return -EINVAL;
 newscreen = kzalloc(new_screen_size, GFP_USER);
 if (!newscreen)
  return -ENOMEM;

 if (vc->vc_uni_lines) {
  new_uniscr = vc_uniscr_alloc(new_cols, new_rows);
  if (!new_uniscr) {
   kfree(newscreen);
   return -ENOMEM;
  }
 }

 if (vc_is_sel(vc))
  clear_selection();

 old_rows = vc->vc_rows;
 old_row_size = vc->vc_size_row;

 err = resize_screen(vc, new_cols, new_rows, from_user);
 if (err) {
  kfree(newscreen);
  vc_uniscr_free(new_uniscr);
  return err;
 }

 vc->vc_rows = new_rows;
 vc->vc_cols = new_cols;
 vc->vc_size_row = new_row_size;
 vc->vc_screenbuf_size = new_screen_size;

 rlth = min(old_row_size, new_row_size);
 rrem = new_row_size - rlth;
 old_origin = vc->vc_origin;
 new_origin = (long) newscreen;
 new_scr_end = new_origin + new_screen_size;

 if (vc->state.y > new_rows) {
  if (old_rows - vc->state.y < new_rows) {
   /*
 * Cursor near the bottom, copy contents from the
 * bottom of buffer
 */

   first_copied_row = (old_rows - new_rows);
  } else {
   /*
 * Cursor is in no man's land, copy 1/2 screenful
 * from the top and bottom of cursor position
 */

   first_copied_row = (vc->state.y - new_rows/2);
  }
  old_origin += first_copied_row * old_row_size;
 } else
  first_copied_row = 0;
 end = old_origin + old_row_size * min(old_rows, new_rows);

 vc_uniscr_copy_area(new_uniscr, new_cols, new_rows,
       vc->vc_uni_lines, rlth/2, first_copied_row,
       min(old_rows, new_rows));
 vc_uniscr_set(vc, new_uniscr);

 update_attr(vc);

 while (old_origin < end) {
  scr_memcpyw((unsigned short *) new_origin,
       (unsigned short *) old_origin, rlth);
  if (rrem)
   scr_memsetw((void *)(new_origin + rlth),
        vc->vc_video_erase_char, rrem);
  old_origin += old_row_size;
  new_origin += new_row_size;
 }
 if (new_scr_end > new_origin)
  scr_memsetw((void *)new_origin, vc->vc_video_erase_char,
       new_scr_end - new_origin);
 oldscreen = vc->vc_screenbuf;
 vc->vc_screenbuf = newscreen;
 vc->vc_screenbuf_size = new_screen_size;
 set_origin(vc);
 kfree(oldscreen);

 /* do part of a reset_terminal() */
 vc->vc_top = 0;
 vc->vc_bottom = vc->vc_rows;
 gotoxy(vc, vc->state.x, vc->state.y);
 save_cur(vc);

 if (tty) {
  /* Rewrite the requested winsize data with the actual
   resulting sizes */

  struct winsize ws;
  memset(&ws, 0, sizeof(ws));
  ws.ws_row = vc->vc_rows;
  ws.ws_col = vc->vc_cols;
  ws.ws_ypixel = vc->vc_scan_lines;
  tty_do_resize(tty, &ws);
 }

 if (con_is_visible(vc))
  update_screen(vc);
 vt_event_post(VT_EVENT_RESIZE, vc->vc_num, vc->vc_num);
 notify_update(vc);
 return err;
}

/**
 * __vc_resize - resize a VT
 * @vc: virtual console
 * @cols: columns
 * @rows: rows
 * @from_user: invoked by a user?
 *
 * Resize a virtual console as seen from the console end of things. We use the
 * common vc_do_resize() method to update the structures.
 *
 * Locking: The caller must hold the console sem to protect console internals
 * and @vc->port.tty.
 */

int __vc_resize(struct vc_data *vc, unsigned int cols, unsigned int rows,
  bool from_user)
{
 return vc_do_resize(vc->port.tty, vc, cols, rows, from_user);
}
EXPORT_SYMBOL(__vc_resize);

/**
 * vt_resize - resize a VT
 * @tty: tty to resize
 * @ws: winsize attributes
 *
 * Resize a virtual terminal. This is called by the tty layer as we register
 * our own handler for resizing. The mutual helper does all the actual work.
 *
 * Locking: Takes the console sem and the called methods then take the tty
 * termios_rwsem and the tty ctrl.lock in that order.
 */

static int vt_resize(struct tty_struct *tty, struct winsize *ws)
{
 struct vc_data *vc = tty->driver_data;
 int ret;

 console_lock();
 ret = vc_do_resize(tty, vc, ws->ws_col, ws->ws_row, false);
 console_unlock();
 return ret;
}

struct vc_data *vc_deallocate(unsigned int currcons)
{
 struct vc_data *vc = NULL;

 WARN_CONSOLE_UNLOCKED();

 if (vc_cons_allocated(currcons)) {
  struct vt_notifier_param param;

  param.vc = vc = vc_cons[currcons].d;
  atomic_notifier_call_chain(&vt_notifier_list, VT_DEALLOCATE, ¶m);
  vcs_remove_sysfs(currcons);
  visual_deinit(vc);
  con_free_unimap(vc);
  put_pid(vc->vt_pid);
  vc_uniscr_set(vc, NULL);
  kfree(vc->vc_screenbuf);
  vc_cons[currcons].d = NULL;
 }
 return vc;
}

/*
 * VT102 emulator
 */


enum { EPecma = 0, EPdec, EPeq, EPgt, EPlt};

#define set_kbd(vc, x) vt_set_kbd_mode_bit((vc)->vc_num, (x))
#define clr_kbd(vc, x) vt_clr_kbd_mode_bit((vc)->vc_num, (x))
#define is_kbd(vc, x) vt_get_kbd_mode_bit((vc)->vc_num, (x))

#define decarm  VC_REPEAT
#define decckm  VC_CKMODE
#define kbdapplic VC_APPLIC
#define lnm  VC_CRLF

const unsigned char color_table[] = { 0, 4, 2, 6, 1, 5, 3, 7,
           8,12,10,14, 9,13,11,15 };
EXPORT_SYMBOL(color_table);

/* the default colour table, for VGA+ colour systems */
unsigned char default_red[] = {
 0x00, 0xaa, 0x00, 0xaa, 0x00, 0xaa, 0x00, 0xaa,
 0x55, 0xff, 0x55, 0xff, 0x55, 0xff, 0x55, 0xff
};
module_param_array(default_red, byte, NULL, S_IRUGO | S_IWUSR);
EXPORT_SYMBOL(default_red);

unsigned char default_grn[] = {
 0x00, 0x00, 0xaa, 0x55, 0x00, 0x00, 0xaa, 0xaa,
 0x55, 0x55, 0xff, 0xff, 0x55, 0x55, 0xff, 0xff
};
module_param_array(default_grn, byte, NULL, S_IRUGO | S_IWUSR);
EXPORT_SYMBOL(default_grn);

unsigned char default_blu[] = {
 0x00, 0x00, 0x00, 0x00, 0xaa, 0xaa, 0xaa, 0xaa,
 0x55, 0x55, 0x55, 0x55, 0xff, 0xff, 0xff, 0xff
};
module_param_array(default_blu, byte, NULL, S_IRUGO | S_IWUSR);
EXPORT_SYMBOL(default_blu);

/*
 * gotoxy() must verify all boundaries, because the arguments
 * might also be negative. If the given position is out of
 * bounds, the cursor is placed at the nearest margin.
 */

static void gotoxy(struct vc_data *vc, int new_x, int new_y)
{
 int min_y, max_y;

 if (new_x < 0)
  vc->state.x = 0;
 else {
  if (new_x >= vc->vc_cols)
   vc->state.x = vc->vc_cols - 1;
  else
   vc->state.x = new_x;
 }

  if (vc->vc_decom) {
  min_y = vc->vc_top;
  max_y = vc->vc_bottom;
 } else {
  min_y = 0;
  max_y = vc->vc_rows;
 }
 if (new_y < min_y)
  vc->state.y = min_y;
 else if (new_y >= max_y)
  vc->state.y = max_y - 1;
 else
  vc->state.y = new_y;
 vc->vc_pos = vc->vc_origin + vc->state.y * vc->vc_size_row +
  (vc->state.x << 1);
 vc->vc_need_wrap = 0;
}

/* for absolute user moves, when decom is set */
static void gotoxay(struct vc_data *vc, int new_x, int new_y)
{
 gotoxy(vc, new_x, vc->vc_decom ? (vc->vc_top + new_y) : new_y);
}

void scrollback(struct vc_data *vc)
{
 scrolldelta(-(vc->vc_rows / 2));
}

void scrollfront(struct vc_data *vc, int lines)
{
 if (!lines)
  lines = vc->vc_rows / 2;
 scrolldelta(lines);
}

static void lf(struct vc_data *vc)
{
     /* don't scroll if above bottom of scrolling region, or
 * if below scrolling region
 */

 if (vc->state.y + 1 == vc->vc_bottom)
  con_scroll(vc, vc->vc_top, vc->vc_bottom, SM_UP, 1);
 else if (vc->state.y < vc->vc_rows - 1) {
  vc->state.y++;
  vc->vc_pos += vc->vc_size_row;
 }
 vc->vc_need_wrap = 0;
 notify_write(vc, '\n');
}

static void ri(struct vc_data *vc)
{
     /* don't scroll if below top of scrolling region, or
 * if above scrolling region
 */

 if (vc->state.y == vc->vc_top)
  con_scroll(vc, vc->vc_top, vc->vc_bottom, SM_DOWN, 1);
 else if (vc->state.y > 0) {
  vc->state.y--;
  vc->vc_pos -= vc->vc_size_row;
 }
 vc->vc_need_wrap = 0;
}

static inline void cr(struct vc_data *vc)
{
 vc->vc_pos -= vc->state.x << 1;
 vc->vc_need_wrap = vc->state.x = 0;
 notify_write(vc, '\r');
}

static inline void bs(struct vc_data *vc)
{
 if (vc->state.x) {
  vc->vc_pos -= 2;
  vc->state.x--;
  vc->vc_need_wrap = 0;
  notify_write(vc, '\b');
 }
}

static inline void del(struct vc_data *vc)
{
 /* ignored */
}

enum CSI_J {
 CSI_J_CURSOR_TO_END = 0,
 CSI_J_START_TO_CURSOR = 1,
 CSI_J_VISIBLE  = 2,
 CSI_J_FULL  = 3,
};

static void csi_J(struct vc_data *vc, enum CSI_J vpar)
{
 unsigned short *start;
 unsigned int count;

 switch (vpar) {
 case CSI_J_CURSOR_TO_END:
  vc_uniscr_clear_line(vc, vc->state.x,
         vc->vc_cols - vc->state.x);
  vc_uniscr_clear_lines(vc, vc->state.y + 1,
          vc->vc_rows - vc->state.y - 1);
  count = (vc->vc_scr_end - vc->vc_pos) >> 1;
  start = (unsigned short *)vc->vc_pos;
  break;
 case CSI_J_START_TO_CURSOR:
  vc_uniscr_clear_line(vc, 0, vc->state.x + 1);
  vc_uniscr_clear_lines(vc, 0, vc->state.y);
  count = ((vc->vc_pos - vc->vc_origin) >> 1) + 1;
  start = (unsigned short *)vc->vc_origin;
  break;
 case CSI_J_FULL:
  flush_scrollback(vc);
  fallthrough;
 case CSI_J_VISIBLE:
  vc_uniscr_clear_lines(vc, 0, vc->vc_rows);
  count = vc->vc_cols * vc->vc_rows;
  start = (unsigned short *)vc->vc_origin;
  break;
 default:
  return;
 }
 scr_memsetw(start, vc->vc_video_erase_char, 2 * count);
 if (con_should_update(vc))
  do_update_region(vc, (unsigned long) start, count);
 vc->vc_need_wrap = 0;
}

enum {
 CSI_K_CURSOR_TO_LINEEND  = 0,
 CSI_K_LINESTART_TO_CURSOR = 1,
 CSI_K_LINE   = 2,
};

static void csi_K(struct vc_data *vc)
{
 unsigned int count;
 unsigned short *start = (unsigned short *)vc->vc_pos;
 int offset;

 switch (vc->vc_par[0]) {
 case CSI_K_CURSOR_TO_LINEEND:
  offset = 0;
  count = vc->vc_cols - vc->state.x;
  break;
 case CSI_K_LINESTART_TO_CURSOR:
  offset = -vc->state.x;
  count = vc->state.x + 1;
  break;
 case CSI_K_LINE:
  offset = -vc->state.x;
  count = vc->vc_cols;
  break;
 default:
  return;
 }
 vc_uniscr_clear_line(vc, vc->state.x + offset, count);
 scr_memsetw(start + offset, vc->vc_video_erase_char, 2 * count);
 vc->vc_need_wrap = 0;
 if (con_should_update(vc))
  do_update_region(vc, (unsigned long)(start + offset), count);
}

/* erase the following count positions */
static void csi_X(struct vc_data *vc)
{       /* not vt100? */
 unsigned int count = clamp(vc->vc_par[0], 1, vc->vc_cols - vc->state.x);

 vc_uniscr_clear_line(vc, vc->state.x, count);
 scr_memsetw((unsigned short *)vc->vc_pos, vc->vc_video_erase_char, 2 * count);
 if (con_should_update(vc))
  vc->vc_sw->con_clear(vc, vc->state.y, vc->state.x, count);
 vc->vc_need_wrap = 0;
}

static void default_attr(struct vc_data *vc)
{
 vc->state.intensity = VCI_NORMAL;
 vc->state.italic = false;
 vc->state.underline = false;
 vc->state.reverse = false;
 vc->state.blink = false;
 vc->state.color = vc->vc_def_color;
}

struct rgb { u8 r; u8 g; u8 b; };

static void rgb_from_256(unsigned int i, struct rgb *c)
{
 if (i < 8) {            /* Standard colours. */
  c->r = i&1 ? 0xaa : 0x00;
  c->g = i&2 ? 0xaa : 0x00;
  c->b = i&4 ? 0xaa : 0x00;
 } else if (i < 16) {
  c->r = i&1 ? 0xff : 0x55;
  c->g = i&2 ? 0xff : 0x55;
  c->b = i&4 ? 0xff : 0x55;
 } else if (i < 232) {   /* 6x6x6 colour cube. */
  i -= 16;
  c->b = i % 6 * 255 / 6;
  i /= 6;
  c->g = i % 6 * 255 / 6;
  i /= 6;
  c->r = i     * 255 / 6;
 } else                  /* Grayscale ramp. */
  c->r = c->g = c->b = i * 10 - 2312;
}

static void rgb_foreground(struct vc_data *vc, const struct rgb *c)
{
 u8 hue = 0, max = max3(c->r, c->g, c->b);

 if (c->r > max / 2)
  hue |= 4;
 if (c->g > max / 2)
  hue |= 2;
 if (c->b > max / 2)
  hue |= 1;

 if (hue == 7 && max <= 0x55) {
  hue = 0;
  vc->state.intensity = VCI_BOLD;
 } else if (max > 0xaa)
  vc->state.intensity = VCI_BOLD;
 else
  vc->state.intensity = VCI_NORMAL;

 vc->state.color = (vc->state.color & 0xf0) | hue;
}

static void rgb_background(struct vc_data *vc, const struct rgb *c)
{
 /* For backgrounds, err on the dark side. */
 vc->state.color = (vc->state.color & 0x0f)
  | (c->r&0x80) >> 1 | (c->g&0x80) >> 2 | (c->b&0x80) >> 3;
}

/*
 * ITU T.416 Higher colour modes. They break the usual properties of SGR codes
 * and thus need to be detected and ignored by hand. That standard also
 * wants : rather than ; as separators but sequences containing : are currently
 * completely ignored by the parser.
 *
 * Subcommands 3 (CMY) and 4 (CMYK) are so insane there's no point in
 * supporting them.
 */

static int vc_t416_color(struct vc_data *vc, int i,
  void(*set_color)(struct vc_data *vc, const struct rgb *c))
{
 struct rgb c;

 i++;
 if (i > vc->vc_npar)
  return i;

 if (vc->vc_par[i] == 5 && i + 1 <= vc->vc_npar) {
  /* 256 colours */
  i++;
  rgb_from_256(vc->vc_par[i], &c);
 } else if (vc->vc_par[i] == 2 && i + 3 <= vc->vc_npar) {
  /* 24 bit */
  c.r = vc->vc_par[i + 1];
  c.g = vc->vc_par[i + 2];
  c.b = vc->vc_par[i + 3];
  i += 3;
 } else
  return i;

 set_color(vc, &c);

 return i;
}

enum {
 CSI_m_DEFAULT   = 0,
 CSI_m_BOLD   = 1,
 CSI_m_HALF_BRIGHT  = 2,
 CSI_m_ITALIC   = 3,
 CSI_m_UNDERLINE   = 4,
 CSI_m_BLINK   = 5,
 CSI_m_REVERSE   = 7,
 CSI_m_PRI_FONT   = 10,
 CSI_m_ALT_FONT1   = 11,
 CSI_m_ALT_FONT2   = 12,
 CSI_m_DOUBLE_UNDERLINE  = 21,
 CSI_m_NORMAL_INTENSITY  = 22,
 CSI_m_NO_ITALIC   = 23,
 CSI_m_NO_UNDERLINE  = 24,
 CSI_m_NO_BLINK   = 25,
 CSI_m_NO_REVERSE  = 27,
 CSI_m_FG_COLOR_BEG  = 30,
 CSI_m_FG_COLOR_END  = 37,
 CSI_m_FG_COLOR   = 38,
 CSI_m_DEFAULT_FG_COLOR  = 39,
 CSI_m_BG_COLOR_BEG  = 40,
 CSI_m_BG_COLOR_END  = 47,
 CSI_m_BG_COLOR   = 48,
 CSI_m_DEFAULT_BG_COLOR  = 49,
 CSI_m_BRIGHT_FG_COLOR_BEG = 90,
 CSI_m_BRIGHT_FG_COLOR_END = 97,
 CSI_m_BRIGHT_FG_COLOR_OFF = CSI_m_BRIGHT_FG_COLOR_BEG - CSI_m_FG_COLOR_BEG,
 CSI_m_BRIGHT_BG_COLOR_BEG = 100,
 CSI_m_BRIGHT_BG_COLOR_END = 107,
 CSI_m_BRIGHT_BG_COLOR_OFF = CSI_m_BRIGHT_BG_COLOR_BEG - CSI_m_BG_COLOR_BEG,
};

/* console_lock is held */
static void csi_m(struct vc_data *vc)
{
 int i;

 for (i = 0; i <= vc->vc_npar; i++)
  switch (vc->vc_par[i]) {
  case CSI_m_DEFAULT: /* all attributes off */
   default_attr(vc);
   break;
  case CSI_m_BOLD:
   vc->state.intensity = VCI_BOLD;
   break;
  case CSI_m_HALF_BRIGHT:
   vc->state.intensity = VCI_HALF_BRIGHT;
   break;
  case CSI_m_ITALIC:
   vc->state.italic = true;
   break;
  case CSI_m_DOUBLE_UNDERLINE:
   /*
 * No console drivers support double underline, so
 * convert it to a single underline.
 */

  case CSI_m_UNDERLINE:
   vc->state.underline = true;
   break;
  case CSI_m_BLINK:
   vc->state.blink = true;
   break;
  case CSI_m_REVERSE:
   vc->state.reverse = true;
   break;
  case CSI_m_PRI_FONT: /* ANSI X3.64-1979 (SCO-ish?)
  * Select primary font, don't display control chars if
  * defined, don't set bit 8 on output.
  */

   vc->vc_translate = set_translate(vc->state.Gx_charset[vc->state.charset], vc);
   vc->vc_disp_ctrl = 0;
   vc->vc_toggle_meta = 0;
   break;
  case CSI_m_ALT_FONT1: /* ANSI X3.64-1979 (SCO-ish?)
  * Select first alternate font, lets chars < 32 be
  * displayed as ROM chars.
  */

   vc->vc_translate = set_translate(IBMPC_MAP, vc);
   vc->vc_disp_ctrl = 1;
   vc->vc_toggle_meta = 0;
   break;
  case CSI_m_ALT_FONT2: /* ANSI X3.64-1979 (SCO-ish?)
  * Select second alternate font, toggle high bit
  * before displaying as ROM char.
  */

   vc->vc_translate = set_translate(IBMPC_MAP, vc);
   vc->vc_disp_ctrl = 1;
   vc->vc_toggle_meta = 1;
   break;
  case CSI_m_NORMAL_INTENSITY:
   vc->state.intensity = VCI_NORMAL;
   break;
  case CSI_m_NO_ITALIC:
   vc->state.italic = false;
   break;
  case CSI_m_NO_UNDERLINE:
   vc->state.underline = false;
   break;
  case CSI_m_NO_BLINK:
   vc->state.blink = false;
   break;
  case CSI_m_NO_REVERSE:
   vc->state.reverse = false;
   break;
  case CSI_m_FG_COLOR:
   i = vc_t416_color(vc, i, rgb_foreground);
   break;
  case CSI_m_BG_COLOR:
   i = vc_t416_color(vc, i, rgb_background);
   break;
  case CSI_m_DEFAULT_FG_COLOR:
   vc->state.color = (vc->vc_def_color & 0x0f) |
    (vc->state.color & 0xf0);
   break;
  case CSI_m_DEFAULT_BG_COLOR:
   vc->state.color = (vc->vc_def_color & 0xf0) |
    (vc->state.color & 0x0f);
   break;
  case CSI_m_BRIGHT_FG_COLOR_BEG ... CSI_m_BRIGHT_FG_COLOR_END:
   vc->state.intensity = VCI_BOLD;
   vc->vc_par[i] -= CSI_m_BRIGHT_FG_COLOR_OFF;
   fallthrough;
  case CSI_m_FG_COLOR_BEG ... CSI_m_FG_COLOR_END:
   vc->vc_par[i] -= CSI_m_FG_COLOR_BEG;
   vc->state.color = color_table[vc->vc_par[i]] |
    (vc->state.color & 0xf0);
   break;
  case CSI_m_BRIGHT_BG_COLOR_BEG ... CSI_m_BRIGHT_BG_COLOR_END:
   vc->vc_par[i] -= CSI_m_BRIGHT_BG_COLOR_OFF;
   fallthrough;
  case CSI_m_BG_COLOR_BEG ... CSI_m_BG_COLOR_END:
   vc->vc_par[i] -= CSI_m_BG_COLOR_BEG;
   vc->state.color = (color_table[vc->vc_par[i]] << 4) |
    (vc->state.color & 0x0f);
   break;
  }
 update_attr(vc);
}

static void respond_string(const char *p, size_t len, struct tty_port *port)
{
 tty_insert_flip_string(port, p, len);
 tty_flip_buffer_push(port);
}

static void cursor_report(struct vc_data *vc, struct tty_struct *tty)
{
 char buf[40];
 int len;

 len = sprintf(buf, "\033[%d;%dR", vc->state.y +
   (vc->vc_decom ? vc->vc_top + 1 : 1),
   vc->state.x + 1);
 respond_string(buf, len, tty->port);
}

static inline void status_report(struct tty_struct *tty)
{
 static const char teminal_ok[] = "\033[0n";

 respond_string(teminal_ok, strlen(teminal_ok), tty->port);
}

static inline void respond_ID(struct tty_struct *tty)
{
 /* terminal answer to an ESC-Z or csi0c query. */
 static const char vt102_id[] = "\033[?6c";

 respond_string(vt102_id, strlen(vt102_id), tty->port);
}

void mouse_report(struct tty_struct *tty, int butt, int mrx, int mry)
{
 char buf[8];
 int len;

 len = sprintf(buf, "\033[M%c%c%c", (char)(' ' + butt),
   (char)('!' + mrx), (char)('!' + mry));
 respond_string(buf, len, tty->port);
}

/* invoked via ioctl(TIOCLINUX) and through set_selection_user */
int mouse_reporting(void)
{
 return vc_cons[fg_console].d->vc_report_mouse;
}

/* invoked via ioctl(TIOCLINUX) */
static int get_bracketed_paste(struct tty_struct *tty)
{
 struct vc_data *vc = tty->driver_data;

 return vc->vc_bracketed_paste;
}

enum {
 CSI_DEC_hl_CURSOR_KEYS = 1, /* CKM: cursor keys send ^[Ox/^[[x */
 CSI_DEC_hl_132_COLUMNS = 3, /* COLM: 80/132 mode switch */
 CSI_DEC_hl_REVERSE_VIDEO = 5, /* SCNM */
 CSI_DEC_hl_ORIGIN_MODE = 6, /* OM: origin relative/absolute */
 CSI_DEC_hl_AUTOWRAP = 7, /* AWM */
 CSI_DEC_hl_AUTOREPEAT = 8, /* ARM */
 CSI_DEC_hl_MOUSE_X10 = 9,
 CSI_DEC_hl_SHOW_CURSOR = 25, /* TCEM */
 CSI_DEC_hl_MOUSE_VT200 = 1000,
 CSI_DEC_hl_BRACKETED_PASTE = 2004,
};

/* console_lock is held */
static void csi_DEC_hl(struct vc_data *vc, bool on_off)
{
 unsigned int i;

 for (i = 0; i <= vc->vc_npar; i++)
  switch (vc->vc_par[i]) {
  case CSI_DEC_hl_CURSOR_KEYS:
   if (on_off)
    set_kbd(vc, decckm);
   else
    clr_kbd(vc, decckm);
   break;
  case CSI_DEC_hl_132_COLUMNS: /* unimplemented */
#if 0
   vc_resize(deccolm ? 132 : 80, vc->vc_rows);
   /* this alone does not suffice; some user mode
   utility has to change the hardware regs */

#endif
   break;
  case CSI_DEC_hl_REVERSE_VIDEO:
   if (vc->vc_decscnm != on_off) {
    vc->vc_decscnm = on_off;
    invert_screen(vc, 0, vc->vc_screenbuf_size,
           false);
    update_attr(vc);
   }
   break;
  case CSI_DEC_hl_ORIGIN_MODE:
   vc->vc_decom = on_off;
   gotoxay(vc, 0, 0);
   break;
  case CSI_DEC_hl_AUTOWRAP:
   vc->vc_decawm = on_off;
   break;
  case CSI_DEC_hl_AUTOREPEAT:
   if (on_off)
    set_kbd(vc, decarm);
   else
    clr_kbd(vc, decarm);
   break;
  case CSI_DEC_hl_MOUSE_X10:
   vc->vc_report_mouse = on_off ? 1 : 0;
   break;
  case CSI_DEC_hl_SHOW_CURSOR:
   vc->vc_deccm = on_off;
   break;
  case CSI_DEC_hl_MOUSE_VT200:
   vc->vc_report_mouse = on_off ? 2 : 0;
   break;
  case CSI_DEC_hl_BRACKETED_PASTE:
   vc->vc_bracketed_paste = on_off;
   break;
  }
}

enum {
 CSI_hl_DISPLAY_CTRL = 3, /* handle ansi control chars */
 CSI_hl_INSERT  = 4, /* IRM: insert/replace */
 CSI_hl_AUTO_NL  = 20, /* LNM: Enter == CrLf/Lf */
};

/* console_lock is held */
static void csi_hl(struct vc_data *vc, bool on_off)
{
 unsigned int i;

 for (i = 0; i <= vc->vc_npar; i++)
  switch (vc->vc_par[i]) { /* ANSI modes set/reset */
  case CSI_hl_DISPLAY_CTRL:
   vc->vc_disp_ctrl = on_off;
   break;
  case CSI_hl_INSERT:
   vc->vc_decim = on_off;
   break;
  case CSI_hl_AUTO_NL:
   if (on_off)
    set_kbd(vc, lnm);
   else
    clr_kbd(vc, lnm);
   break;
  }
}

enum CSI_right_square_bracket {
 CSI_RSB_COLOR_FOR_UNDERLINE  = 1,
 CSI_RSB_COLOR_FOR_HALF_BRIGHT  = 2,
 CSI_RSB_MAKE_CUR_COLOR_DEFAULT  = 8,
 CSI_RSB_BLANKING_INTERVAL  = 9,
 CSI_RSB_BELL_FREQUENCY   = 10,
 CSI_RSB_BELL_DURATION   = 11,
 CSI_RSB_BRING_CONSOLE_TO_FRONT  = 12,
 CSI_RSB_UNBLANK    = 13,
 CSI_RSB_VESA_OFF_INTERVAL  = 14,
 CSI_RSB_BRING_PREV_CONSOLE_TO_FRONT = 15,
 CSI_RSB_CURSOR_BLINK_INTERVAL  = 16,
};

/*
 * csi_RSB - csi+] (Right Square Bracket) handler
 *
 * These are linux console private sequences.
 *
 * console_lock is held
 */

static void csi_RSB(struct vc_data *vc)
{
 switch (vc->vc_par[0]) {
 case CSI_RSB_COLOR_FOR_UNDERLINE:
  if (vc->vc_can_do_color && vc->vc_par[1] < 16) {
   vc->vc_ulcolor = color_table[vc->vc_par[1]];
   if (vc->state.underline)
    update_attr(vc);
  }
  break;
 case CSI_RSB_COLOR_FOR_HALF_BRIGHT:
  if (vc->vc_can_do_color && vc->vc_par[1] < 16) {
   vc->vc_halfcolor = color_table[vc->vc_par[1]];
   if (vc->state.intensity == VCI_HALF_BRIGHT)
    update_attr(vc);
  }
  break;
 case CSI_RSB_MAKE_CUR_COLOR_DEFAULT:
  vc->vc_def_color = vc->vc_attr;
  if (vc->vc_hi_font_mask == 0x100)
   vc->vc_def_color >>= 1;
  default_attr(vc);
  update_attr(vc);
  break;
 case CSI_RSB_BLANKING_INTERVAL:
  blankinterval = min(vc->vc_par[1], 60U) * 60;
  poke_blanked_console();
  break;
 case CSI_RSB_BELL_FREQUENCY:
  if (vc->vc_npar >= 1)
   vc->vc_bell_pitch = vc->vc_par[1];
  else
   vc->vc_bell_pitch = DEFAULT_BELL_PITCH;
  break;
 case CSI_RSB_BELL_DURATION:
  if (vc->vc_npar >= 1)
   vc->vc_bell_duration = (vc->vc_par[1] < 2000) ?
    msecs_to_jiffies(vc->vc_par[1]) : 0;
  else
   vc->vc_bell_duration = DEFAULT_BELL_DURATION;
  break;
 case CSI_RSB_BRING_CONSOLE_TO_FRONT:
  if (vc->vc_par[1] >= 1 && vc_cons_allocated(vc->vc_par[1] - 1))
   set_console(vc->vc_par[1] - 1);
  break;
 case CSI_RSB_UNBLANK:
  poke_blanked_console();
  break;
 case CSI_RSB_VESA_OFF_INTERVAL:
  vesa_off_interval = min(vc->vc_par[1], 60U) * 60 * HZ;
  break;
 case CSI_RSB_BRING_PREV_CONSOLE_TO_FRONT:
  set_console(last_console);
  break;
 case CSI_RSB_CURSOR_BLINK_INTERVAL:
  if (vc->vc_npar >= 1 && vc->vc_par[1] >= 50 &&
    vc->vc_par[1] <= USHRT_MAX)
   vc->vc_cur_blink_ms = vc->vc_par[1];
  else
   vc->vc_cur_blink_ms = DEFAULT_CURSOR_BLINK_MS;
  break;
 }
}

/* console_lock is held */
static void csi_at(struct vc_data *vc, unsigned int nr)
{
 nr = clamp(nr, 1, vc->vc_cols - vc->state.x);
 insert_char(vc, nr);
}

/* console_lock is held */
static void csi_L(struct vc_data *vc)
{
 unsigned int nr = clamp(vc->vc_par[0], 1, vc->vc_rows - vc->state.y);

 con_scroll(vc, vc->state.y, vc->vc_bottom, SM_DOWN, nr);
 vc->vc_need_wrap = 0;
}

/* console_lock is held */
static void csi_P(struct vc_data *vc)
{
 unsigned int nr = clamp(vc->vc_par[0], 1, vc->vc_cols - vc->state.x);

 delete_char(vc, nr);
}

/* console_lock is held */
static void csi_M(struct vc_data *vc)
{
 unsigned int nr = clamp(vc->vc_par[0], 1, vc->vc_rows - vc->state.y);

 con_scroll(vc, vc->state.y, vc->vc_bottom, SM_UP, nr);
 vc->vc_need_wrap = 0;
}

/* console_lock is held (except via vc_init->reset_terminal */
static void save_cur(struct vc_data *vc)
{
 memcpy(&vc->saved_state, &vc->state, sizeof(vc->state));
}

/* console_lock is held */
static void restore_cur(struct vc_data *vc)
{
 memcpy(&vc->state, &vc->saved_state, sizeof(vc->state));

 gotoxy(vc, vc->state.x, vc->state.y);
 vc->vc_translate = set_translate(vc->state.Gx_charset[vc->state.charset],
   vc);
 update_attr(vc);
 vc->vc_need_wrap = 0;
}

/**
 * enum vc_ctl_state - control characters state of a vt
 *
 * @ESnormal: initial state, no control characters parsed
 * @ESesc: ESC parsed
 * @ESsquare: CSI parsed -- modifiers/parameters/ctrl chars expected
 * @ESgetpars: CSI parsed -- parameters/ctrl chars expected
 * @ESfunckey: CSI [ parsed
 * @EShash: ESC # parsed
 * @ESsetG0: ESC ( parsed
 * @ESsetG1: ESC ) parsed
 * @ESpercent: ESC % parsed
 * @EScsiignore: CSI [0x20-0x3f] parsed
 * @ESnonstd: OSC parsed
 * @ESpalette: OSC P parsed
 * @ESosc: OSC [0-9] parsed
 * @ESANSI_first: first state for ignoring ansi control sequences
 * @ESapc: ESC _ parsed
 * @ESpm: ESC ^ parsed
 * @ESdcs: ESC P parsed
 * @ESANSI_last: last state for ignoring ansi control sequences
 */

enum vc_ctl_state {
 ESnormal,
 ESesc,
 ESsquare,
 ESgetpars,
 ESfunckey,
 EShash,
 ESsetG0,
 ESsetG1,
 ESpercent,
 EScsiignore,
 ESnonstd,
 ESpalette,
 ESosc,
 ESANSI_first = ESosc,
 ESapc,
 ESpm,
 ESdcs,
 ESANSI_last = ESdcs,
};

/* console_lock is held (except via vc_init()) */
static void reset_terminal(struct vc_data *vc, int do_clear)
{
 unsigned int i;

 vc->vc_top  = 0;
 vc->vc_bottom  = vc->vc_rows;
 vc->vc_state  = ESnormal;
 vc->vc_priv  = EPecma;
 vc->vc_translate = set_translate(LAT1_MAP, vc);
 vc->state.Gx_charset[0] = LAT1_MAP;
 vc->state.Gx_charset[1] = GRAF_MAP;
 vc->state.charset = 0;
 vc->vc_need_wrap = 0;
 vc->vc_report_mouse = 0;
 vc->vc_bracketed_paste = 0;
 vc->vc_utf              = default_utf8;
 vc->vc_utf_count = 0;

 vc->vc_disp_ctrl = 0;
 vc->vc_toggle_meta = 0;

 vc->vc_decscnm  = 0;
 vc->vc_decom  = 0;
 vc->vc_decawm  = 1;
 vc->vc_deccm  = global_cursor_default;
 vc->vc_decim  = 0;

 vt_reset_keyboard(vc->vc_num);

 vc->vc_cursor_type = cur_default;
 vc->vc_complement_mask = vc->vc_s_complement_mask;

 default_attr(vc);
 update_attr(vc);

 bitmap_zero(vc->vc_tab_stop, VC_TABSTOPS_COUNT);
 for (i = 0; i < VC_TABSTOPS_COUNT; i += 8)
  set_bit(i, vc->vc_tab_stop);

 vc->vc_bell_pitch = DEFAULT_BELL_PITCH;
 vc->vc_bell_duration = DEFAULT_BELL_DURATION;
 vc->vc_cur_blink_ms = DEFAULT_CURSOR_BLINK_MS;

 gotoxy(vc, 0, 0);
 save_cur(vc);
 if (do_clear)
     csi_J(vc, CSI_J_VISIBLE);
}

static void vc_setGx(struct vc_data *vc, unsigned int which, u8 c)
{
 unsigned char *charset = &vc->state.Gx_charset[which];

 switch (c) {
 case '0':
  *charset = GRAF_MAP;
  break;
 case 'B':
  *charset = LAT1_MAP;
  break;
 case 'U':
  *charset = IBMPC_MAP;
  break;
 case 'K':
  *charset = USER_MAP;
  break;
 }

 if (vc->state.charset == which)
  vc->vc_translate = set_translate(*charset, vc);
}

static bool ansi_control_string(enum vc_ctl_state state)
{
 return state >= ESANSI_first && state <= ESANSI_last;
}

enum {
 ASCII_NULL  = 0,
 ASCII_BELL  = 7,
 ASCII_BACKSPACE  = 8,
 ASCII_IGNORE_FIRST = ASCII_BACKSPACE,
 ASCII_HTAB  = 9,
 ASCII_LINEFEED  = 10,
 ASCII_VTAB  = 11,
 ASCII_FORMFEED  = 12,
 ASCII_CAR_RET  = 13,
 ASCII_IGNORE_LAST = ASCII_CAR_RET,
 ASCII_SHIFTOUT  = 14,
 ASCII_SHIFTIN  = 15,
 ASCII_CANCEL  = 24,
 ASCII_SUBSTITUTE = 26,
 ASCII_ESCAPE  = 27,
 ASCII_CSI_IGNORE_FIRST = ' '/* 0x2x, 0x3a and 0x3c - 0x3f */
 ASCII_CSI_IGNORE_LAST = '?',
 ASCII_DEL  = 127,
 ASCII_EXT_CSI  = 128 + ASCII_ESCAPE,
};

/*
 * Handle ascii characters in control sequences and change states accordingly.
 * E.g. ESC sets the state of vc to ESesc.
 *
 * Returns: true if @c handled.
 */

static bool handle_ascii(struct tty_struct *tty, struct vc_data *vc, u8 c)
{
 switch (c) {
 case ASCII_NULL:
  return true;
 case ASCII_BELL:
  if (ansi_control_string(vc->vc_state))
   vc->vc_state = ESnormal;
  else if (vc->vc_bell_duration)
   kd_mksound(vc->vc_bell_pitch, vc->vc_bell_duration);
  return true;
 case ASCII_BACKSPACE:
  bs(vc);
  return true;
 case ASCII_HTAB:
  vc->vc_pos -= (vc->state.x << 1);

  vc->state.x = find_next_bit(vc->vc_tab_stop,
    min(vc->vc_cols - 1, VC_TABSTOPS_COUNT),
    vc->state.x + 1);
  if (vc->state.x >= VC_TABSTOPS_COUNT)
   vc->state.x = vc->vc_cols - 1;

  vc->vc_pos += (vc->state.x << 1);
  notify_write(vc, '\t');
  return true;
 case ASCII_LINEFEED:
 case ASCII_VTAB:
 case ASCII_FORMFEED:
  lf(vc);
  if (!is_kbd(vc, lnm))
   return true;
  fallthrough;
 case ASCII_CAR_RET:
  cr(vc);
  return true;
 case ASCII_SHIFTOUT:
  vc->state.charset = 1;
  vc->vc_translate = set_translate(vc->state.Gx_charset[1], vc);
  vc->vc_disp_ctrl = 1;
  return true;
 case ASCII_SHIFTIN:
  vc->state.charset = 0;
  vc->vc_translate = set_translate(vc->state.Gx_charset[0], vc);
  vc->vc_disp_ctrl = 0;
  return true;
 case ASCII_CANCEL:
 case ASCII_SUBSTITUTE:
  vc->vc_state = ESnormal;
  return true;
 case ASCII_ESCAPE:
  vc->vc_state = ESesc;
  return true;
 case ASCII_DEL:
  del(vc);
  return true;
 case ASCII_EXT_CSI:
  vc->vc_state = ESsquare;
  return true;
 }

 return false;
}

/*
 * Handle a character (@c) following an ESC (when @vc is in the ESesc state).
 * E.g. previous ESC with @c == '[' here yields the ESsquare state (that is:
 * CSI).
 */

static void handle_esc(struct tty_struct *tty, struct vc_data *vc, u8 c)
{
 vc->vc_state = ESnormal;
 switch (c) {
 case '[':
  vc->vc_state = ESsquare;
  break;
 case ']':
  vc->vc_state = ESnonstd;
  break;
 case '_':
  vc->vc_state = ESapc;
  break;
 case '^':
  vc->vc_state = ESpm;
  break;
 case '%':
  vc->vc_state = ESpercent;
  break;
 case 'E':
  cr(vc);
  lf(vc);
  break;
 case 'M':
  ri(vc);
  break;
 case 'D':
  lf(vc);
  break;
 case 'H':
  if (vc->state.x < VC_TABSTOPS_COUNT)
   set_bit(vc->state.x, vc->vc_tab_stop);
  break;
 case 'P':
  vc->vc_state = ESdcs;
  break;
 case 'Z':
  respond_ID(tty);
  break;
 case '7':
  save_cur(vc);
  break;
 case '8':
  restore_cur(vc);
  break;
 case '(':
  vc->vc_state = ESsetG0;
  break;
 case ')':
  vc->vc_state = ESsetG1;
  break;
 case '#':
  vc->vc_state = EShash;
  break;
 case 'c':
  reset_terminal(vc, 1);
  break;
 case '>':  /* Numeric keypad */
  clr_kbd(vc, kbdapplic);
  break;
 case '=':  /* Appl. keypad */
  set_kbd(vc, kbdapplic);
  break;
 }
}

/*
 * Handle special DEC control sequences ("ESC [ ? parameters char"). Parameters
 * are in @vc->vc_par and the char is in @c here.
 */

static void csi_DEC(struct tty_struct *tty, struct vc_data *vc, u8 c)
{
 switch (c) {
 case 'h':
  csi_DEC_hl(vc, true);
  break;
 case 'l':
  csi_DEC_hl(vc, false);
  break;
 case 'c':
  if (vc->vc_par[0])
   vc->vc_cursor_type = CUR_MAKE(vc->vc_par[0],
            vc->vc_par[1],
            vc->vc_par[2]);
  else
   vc->vc_cursor_type = cur_default;
  break;
 case 'm':
  clear_selection();
  if (vc->vc_par[0])
   vc->vc_complement_mask = vc->vc_par[0] << 8 | vc->vc_par[1];
  else
   vc->vc_complement_mask = vc->vc_s_complement_mask;
  break;
 case 'n':
  if (vc->vc_par[0] == 5)
   status_report(tty);
  else if (vc->vc_par[0] == 6)
   cursor_report(vc, tty);
  break;
 }
}

/*
 * Handle Control Sequence Introducer control characters. That is
 * "ESC [ parameters char". Parameters are in @vc->vc_par and the char is in
 * @c here.
 */

static void csi_ECMA(struct tty_struct *tty, struct vc_data *vc, u8 c)
{
 switch (c) {
 case 'G':
 case '`':
  if (vc->vc_par[0])
   vc->vc_par[0]--;
  gotoxy(vc, vc->vc_par[0], vc->state.y);
  break;
 case 'A':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, vc->state.x, vc->state.y - vc->vc_par[0]);
  break;
 case 'B':
 case 'e':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, vc->state.x, vc->state.y + vc->vc_par[0]);
  break;
 case 'C':
 case 'a':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, vc->state.x + vc->vc_par[0], vc->state.y);
  break;
 case 'D':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, vc->state.x - vc->vc_par[0], vc->state.y);
  break;
 case 'E':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, 0, vc->state.y + vc->vc_par[0]);
  break;
 case 'F':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  gotoxy(vc, 0, vc->state.y - vc->vc_par[0]);
  break;
 case 'd':
  if (vc->vc_par[0])
   vc->vc_par[0]--;
  gotoxay(vc, vc->state.x ,vc->vc_par[0]);
  break;
 case 'H':
 case 'f':
  if (vc->vc_par[0])
   vc->vc_par[0]--;
  if (vc->vc_par[1])
   vc->vc_par[1]--;
  gotoxay(vc, vc->vc_par[1], vc->vc_par[0]);
  break;
 case 'J':
  csi_J(vc, vc->vc_par[0]);
  break;
 case 'K':
  csi_K(vc);
  break;
 case 'L':
  csi_L(vc);
  break;
 case 'M':
  csi_M(vc);
  break;
 case 'P':
  csi_P(vc);
  break;
 case 'c':
  if (!vc->vc_par[0])
   respond_ID(tty);
  break;
 case 'g':
  if (!vc->vc_par[0] && vc->state.x < VC_TABSTOPS_COUNT)
   set_bit(vc->state.x, vc->vc_tab_stop);
  else if (vc->vc_par[0] == 3)
   bitmap_zero(vc->vc_tab_stop, VC_TABSTOPS_COUNT);
  break;
 case 'h':
  csi_hl(vc, true);
  break;
 case 'l':
  csi_hl(vc, false);
  break;
 case 'm':
  csi_m(vc);
  break;
 case 'n':
  if (vc->vc_par[0] == 5)
   status_report(tty);
  else if (vc->vc_par[0] == 6)
   cursor_report(vc, tty);
  break;
 case 'q'/* DECLL - but only 3 leds */
  /* map 0,1,2,3 to 0,1,2,4 */
  if (vc->vc_par[0] < 4)
   vt_set_led_state(vc->vc_num,
        (vc->vc_par[0] < 3) ? vc->vc_par[0] : 4);
  break;
 case 'r':
  if (!vc->vc_par[0])
   vc->vc_par[0]++;
  if (!vc->vc_par[1])
   vc->vc_par[1] = vc->vc_rows;
  /* Minimum allowed region is 2 lines */
  if (vc->vc_par[0] < vc->vc_par[1] &&
      vc->vc_par[1] <= vc->vc_rows) {
   vc->vc_top = vc->vc_par[0] - 1;
   vc->vc_bottom = vc->vc_par[1];
   gotoxay(vc, 0, 0);
  }
  break;
 case 's':
  save_cur(vc);
  break;
 case 'u':
  restore_cur(vc);
  break;
 case 'X':
  csi_X(vc);
  break;
 case '@':
  csi_at(vc, vc->vc_par[0]);
  break;
 case ']':
  csi_RSB(vc);
  break;
 }

}

static void vc_reset_params(struct vc_data *vc)
{
 memset(vc->vc_par, 0, sizeof(vc->vc_par));
 vc->vc_npar = 0;
}

/* console_lock is held */
static void do_con_trol(struct tty_struct *tty, struct vc_data *vc, u8 c)
{
 /*
 *  Control characters can be used in the _middle_
 *  of an escape sequence, aside from ANSI control strings.
 */

 if (ansi_control_string(vc->vc_state) && c >= ASCII_IGNORE_FIRST &&
     c <= ASCII_IGNORE_LAST)
  return;

 if (handle_ascii(tty, vc, c))
  return;

 switch(vc->vc_state) {
 case ESesc: /* ESC */
  handle_esc(tty, vc, c);
  return;
 case ESnonstd: /* ESC ] aka OSC */
  switch (c) {
  case 'P'/* palette escape sequence */
   vc_reset_params(vc);
   vc->vc_state = ESpalette;
   return;
  case 'R'/* reset palette */
   reset_palette(vc);
   break;
  case '0' ... '9':
   vc->vc_state = ESosc;
   return;
  }
  vc->vc_state = ESnormal;
  return;
 case ESpalette: /* ESC ] P aka OSC P */
  if (isxdigit(c)) {
   vc->vc_par[vc->vc_npar++] = hex_to_bin(c);
   if (vc->vc_npar == 7) {
--> --------------------

--> maximum size reached

--> --------------------

Messung V0.5
C=96 H=97 G=96

¤ Dauer der Verarbeitung: 0.35 Sekunden  ¤

*© Formatika GbR, Deutschland






Entwurf

Ziele

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Ergonomie der
Schnittstellen

Diese beiden folgenden Angebotsgruppen bietet das Unternehmen

Angebot

Hier finden Sie eine Liste der Produkte des Unternehmens