home *** CD-ROM | disk | FTP | other *** search
/ InfoMagic Source Code 1993 July / THE_SOURCE_CODE_CD_ROM.iso / gnu / lucid / lemacs-19.6 / src / term.c < prev    next >
Encoding:
C/C++ Source or Header  |  1992-12-21  |  38.8 KB  |  1,501 lines

  1. /* Terminal control module for terminals described by TERMCAP
  2.    Copyright (C) 1985, 1986, 1987, 1992 Free Software Foundation, Inc.
  3.  
  4. This file is part of GNU Emacs.
  5.  
  6. GNU Emacs is free software; you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation; either version 2, or (at your option)
  9. any later version.
  10.  
  11. GNU Emacs is distributed in the hope that it will be useful,
  12. but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  14. GNU General Public License for more details.
  15.  
  16. You should have received a copy of the GNU General Public License
  17. along with GNU Emacs; see the file COPYING.  If not, write to
  18. the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
  19.  
  20.  
  21. #include <stdio.h>
  22. #include <ctype.h>
  23. #include "config.h"
  24. #include "lisp.h"
  25. #include "termhooks.h"
  26. #include "termchar.h"
  27. #include "termopts.h"
  28. #include "cm.h"
  29.  
  30. #include "dispextern.h"
  31. #include "screen.h"
  32. #include "disptab.h"
  33.  
  34. #ifndef tputs
  35. extern int tputs (char *, int, int (*) (char));
  36. extern int tgetent (char *, char *);
  37. extern int tgetflag (char *);
  38. extern int tgetnum (char *);
  39. #endif
  40.  
  41. #define max(a, b) ((a) > (b) ? (a) : (b))
  42. #define min(a, b) ((a) < (b) ? (a) : (b))
  43.  
  44. #define CHECK_HOOK(HOOK) (! SCREEN_IS_TERMCAP(updating_screen    \
  45.                           ? updating_screen    \
  46.                           : selected_screen)\
  47.               && HOOK != 0)
  48.  
  49. #define OUTPUT(a) tputs (a, SCREEN_HEIGHT (selected_screen) - curY, cmputc)
  50. #define OUTPUT1(a) tputs (a, 1, cmputc)
  51. #define OUTPUTL(a, lines) tputs (a, lines, cmputc)
  52. #define OUTPUT_IF(a) { if (a) tputs (a, SCREEN_HEIGHT (selected_screen) - curY, cmputc); }
  53. #define OUTPUT1_IF(a) { if (a) tputs (a, 1, cmputc); }
  54.  
  55. /* Terminal charateristics that higher levels want to look at.
  56.    These are all extern'd in termchar.h */
  57.  
  58. #ifndef MULTI_SCREEN
  59. int screen_width;        /* Number of usable columns */
  60. int screen_height;        /* Number of lines */
  61. #endif
  62.  
  63. int must_write_spaces;        /* Nonzero means spaces in the text
  64.                    must actually be output; can't just skip
  65.                    over some columns to leave them blank.  */
  66. int min_padding_speed;    /* Speed below which no padding necessary */
  67.  
  68. int line_ins_del_ok;        /* Terminal can insert and delete lines */
  69. int char_ins_del_ok;        /* Terminal can insert and delete chars */
  70. int scroll_region_ok;        /* Terminal supports setting the
  71.                    scroll window */
  72. int memory_below_screen;    /* Terminal remembers lines
  73.                    scrolled off bottom */
  74. int term_does_underline;    /* Terminal can do underlining. */
  75. int fast_clear_end_of_line;    /* Terminal has a `ce' string */
  76.  
  77. int dont_calculate_costs;    /* Nonzero means don't bother computing */
  78.                 /* various cost tables; we won't use them.  */
  79.  
  80. /* Nonzero means no need to redraw the entire screen on resuming
  81.    a suspended Emacs.  This is useful on terminals with multiple pages,
  82.    where one page is used for Emacs and another for all else. */
  83. int no_redraw_on_reenter;
  84.  
  85. /* Hook functions that you can set to snap out the functions in this file.
  86.    These are all extern'd in termhooks.h  */
  87.  
  88. void (*cursor_to_hook) ();
  89. void (*raw_cursor_to_hook) ();
  90.  
  91. void (*clear_to_end_hook) ();
  92. void (*clear_screen_hook) ();
  93. void (*clear_end_of_line_hook) ();
  94.  
  95. void (*ins_del_lines_hook) ();
  96.  
  97. void (*change_line_highlight_hook) ();
  98. void (*reassert_line_highlight_hook) ();
  99.  
  100. void (*insert_glyphs_hook) ();
  101. void (*write_glyphs_hook) ();
  102. void (*delete_glyphs_hook) ();
  103.  
  104. void (*ring_bell_hook) ();
  105.  
  106. void (*reset_terminal_modes_hook) ();
  107. void (*set_terminal_modes_hook) ();
  108. void (*update_begin_hook) ();
  109. void (*update_end_hook) ();
  110. void (*set_terminal_window_hook) ();
  111.  
  112. Lisp_Object (*read_socket_hook) ();
  113.  
  114. /* Strings, numbers and flags taken from the termcap entry.  */
  115.  
  116. static char *TS_ins_line;        /* termcap "al" */
  117. static char *TS_ins_multi_lines;    /* "AL" (one parameter, # lines to insert) */
  118. static char *TS_bell;            /* "bl" */
  119. static char *TS_clr_to_bottom;        /* "cd" */
  120. static char *TS_clr_line;        /* "ce", clear to end of line */
  121. static char *TS_clr_screen;        /* "cl" */
  122. static char *TS_set_scroll_region;    /* "cs" (2 params, first line and last line) */
  123. static char *TS_set_scroll_region_1;   /* "cS" (4 params: total lines,
  124.                    lines above scroll region, lines below it,
  125.                    total lines again) */
  126. static char *TS_del_char;        /* "dc" */
  127. static char *TS_del_multi_chars;    /* "DC" (one parameter, # chars to delete) */
  128. static char *TS_del_line;        /* "dl" */
  129. static char *TS_del_multi_lines;    /* "DL" (one parameter, # lines to delete) */
  130. static char *TS_delete_mode;        /* "dm", enter character-delete mode */
  131. static char *TS_end_delete_mode;    /* "ed", leave character-delete mode */
  132. static char *TS_end_insert_mode;    /* "ei", leave character-insert mode */
  133. static char *TS_ins_char;        /* "ic" */
  134. static char *TS_ins_multi_chars;    /* "IC" (one parameter, # chars to insert) */
  135. static char *TS_insert_mode;        /* "im", enter character-insert mode */
  136. static char *TS_pad_inserted_char;    /* "ip".  Just padding, no commands.  */
  137. static char *TS_end_keypad_mode;    /* "ke" */
  138. static char *TS_keypad_mode;        /* "ks" */
  139. static char *TS_pad_char;        /* "pc", char to use as padding */
  140. static char *TS_repeat;        /* "rp" (2 params, # times to repeat
  141.                    and character to be repeated) */
  142. static char *TS_end_standout_mode;    /* "se" */
  143. static char *TS_fwd_scroll;        /* "sf" */
  144. static char *TS_standout_mode;        /* "so" */
  145. static char *TS_rev_scroll;        /* "sr" */
  146. static char *TS_end_termcap_modes;    /* "te" */
  147. static char *TS_termcap_modes;        /* "ti" */
  148. static char *TS_visible_bell;        /* "vb" */
  149. static char *TS_end_visual_mode;    /* "ve" */
  150. static char *TS_visual_mode;        /* "vi" */
  151. static char *TS_set_window;        /* "wi" (4 params, start and end of window,
  152.                    each as vpos and hpos) */
  153. static char *TS_underline_mode;    /* "us" */
  154. static char *TS_end_underline_mode;    /* "ue" */
  155.  
  156. static int TF_hazeltine;    /* termcap hz flag. */
  157. static int TF_insmode_motion;    /* termcap mi flag: can move while in insert mode. */
  158. static int TF_standout_motion;    /* termcap mi flag: can move while in standout mode. */
  159. static int TF_underscore;    /* termcap ul flag: _ underlines if overstruck on
  160.                nonblank position.  Must clear before writing _.  */
  161. static int TF_teleray;        /* termcap xt flag: many weird consequences.
  162.                For t1061. */
  163.  
  164. static int TF_xs;        /* Nonzero for "xs".  If set together with
  165.                TN_standout_width == 0, it means don't bother
  166.                to write any end-standout cookies.  */
  167.  
  168. /* These are present if the terminal uses the magic cookie model. */
  169.  
  170. static int TN_standout_width;    /* termcap "sg" number: width occupied by standout
  171.                markers */
  172. static int TN_underline_width;    /* termcap "ug" number: width occupied by underline
  173.                markers. */
  174.  
  175.  
  176. static int RPov;    /* # chars to start a TS_repeat */
  177.  
  178. static int delete_in_insert_mode;    /* delete mode == insert mode */
  179.  
  180. static int se_is_so;    /* 1 if same string both enters and leaves
  181.                standout mode */
  182.  
  183.  
  184.  
  185. /* Number of chars of space used for standout marker at beginning of line,
  186.    or'd with 0100.  Zero if no standout marker at all.
  187.  
  188.    Used IFF TN_standout_width >= 0. */
  189. static char *chars_wasted;
  190. static char *copybuf;
  191.  
  192. #if 0
  193. /* nonzero means supposed to write text in standout mode.  */
  194. static int standout_requested;
  195. #endif
  196.  
  197. static int insert_mode;        /* Nonzero when in insert mode.  */
  198. static int standout_mode;    /* Nonzero when in standout mode.  */
  199. static int underline_mode;    /* Nonzero when in underline mode.  */
  200.  
  201. /* Size of window specified by higher levels.
  202.    This is the number of lines, from the top of screen downwards,
  203.    which can participate in insert-line/delete-line operations.
  204.  
  205.    Effectively it excludes the bottom screen_height - specified_window_size
  206.    lines from those operations.  */
  207. int specified_window;
  208.  
  209. /* Run of text the cursor is in. */
  210. static int cursor_run;
  211.  
  212. /* How many glyphs are left in cursor_run of glyphs. */
  213. static int left_in_this_run;
  214.  
  215. /* Screen currently being redisplayed; 0 if not currently redisplaying.
  216.    (Direct output does not count).  */
  217. SCREEN_PTR updating_screen;
  218.  
  219. char *tparam ();
  220.  
  221.  
  222. int 
  223. initial_screen_is_tty ()
  224. {
  225. #ifdef HAVE_X_WINDOWS
  226.   return (inhibit_window_system || !egetenv ("DISPLAY"));
  227. #else
  228.   return 1;
  229. #endif
  230. }
  231.  
  232.  
  233.  
  234.  
  235. static void
  236. set_scroll_region (start, stop)
  237.      int start, stop;
  238. {
  239.   char *buf;
  240.   if (TS_set_scroll_region)
  241.     {
  242.       buf = tparam (TS_set_scroll_region, 0, 0, start, stop - 1);
  243.     }
  244.   else if (TS_set_scroll_region_1)
  245.     {
  246.       buf = tparam (TS_set_scroll_region_1, 0, 0,
  247.             SCREEN_HEIGHT (selected_screen), start,
  248.             SCREEN_HEIGHT (selected_screen) - stop,
  249.             SCREEN_HEIGHT (selected_screen));
  250.     }
  251.   else
  252.     {
  253.       buf = tparam (TS_set_window, 0, 0, start, 0,
  254.             stop, SCREEN_WIDTH (selected_screen));
  255.     }
  256.   OUTPUT (buf);
  257.   xfree (buf);
  258.   losecursor ();
  259. }
  260.  
  261. void
  262. set_terminal_window (size)
  263.      int size;
  264. {
  265.   if (CHECK_HOOK (set_terminal_window_hook))
  266.     {
  267.       (*set_terminal_window_hook) (size);
  268.       return;
  269.     }
  270.   specified_window = size ? size : SCREEN_HEIGHT (selected_screen);
  271.   if (!scroll_region_ok)
  272.     return;
  273.   set_scroll_region (0, specified_window);
  274. }
  275.  
  276. static void
  277. turn_on_insert ()
  278. {
  279.   if (!insert_mode)
  280.     OUTPUT (TS_insert_mode);
  281.   insert_mode = 1;
  282. }
  283.  
  284. static void
  285. turn_off_insert ()
  286. {
  287.   if (insert_mode)
  288.     OUTPUT (TS_end_insert_mode);
  289.   insert_mode = 0;
  290. }
  291.  
  292. /* Handle standout mode for terminals in which TN_standout_width >= 0.
  293.    On these terminals, standout is controlled by markers that
  294.    live inside the screen memory.  TN_standout_width is the width
  295.    that the marker occupies in memory.  Standout runs from the marker
  296.    to the end of the line on some terminals, or to the next
  297.    turn-off-standout marker (TS_end_standout_mode) string
  298.    on other terminals.
  299.  
  300.    Write a standout marker or end-standout marker at hpos on the line
  301.    at vertical position vpos.  */
  302.  
  303. static void
  304. write_standout_marker (flag, hpos, vpos)
  305.      int flag, hpos, vpos;
  306. {
  307.   if (flag || (TS_end_standout_mode && !TF_teleray && !se_is_so
  308.            && !(TF_xs && TN_standout_width == 0)))
  309.     {
  310.       cmgoto (vpos, hpos);
  311.       cmplus (TN_standout_width);
  312.       OUTPUT (flag ? TS_standout_mode : TS_end_standout_mode);
  313.       chars_wasted[curY] = TN_standout_width | 0100;
  314.     }
  315. }
  316.  
  317. #define write_standout_begin_marker(hpos, vpos) \
  318.     write_standout_marker (1, hpos, vpos)
  319. #define write_standout_end_marker(hpos, vpos)   \
  320.     write_standout_marker (0, hpos, vpos)
  321.  
  322. /* Set standout mode to the state it should be in for
  323.    empty space inside windows.  What this is,
  324.    depends on the user option inverse-video.  */
  325.  
  326. static void
  327. turn_off_highlight ()
  328. {
  329.   if (TN_standout_width < 0)
  330.     {
  331.       if (standout_mode)
  332.     OUTPUT_IF (TS_end_standout_mode);
  333.     }
  334.   else
  335.     {
  336.       write_standout_end_marker (curX, curY);
  337.     }
  338.   standout_mode = 0;
  339. }
  340.  
  341. static void
  342. turn_on_highlight ()
  343. {
  344.   if (TN_standout_width < 0)
  345.     {
  346.       if (!standout_mode)
  347.     OUTPUT_IF (TS_standout_mode);
  348.     }
  349.   else
  350.     {
  351.       write_standout_begin_marker (curX, curY);
  352.     }
  353.   standout_mode = 1;
  354. }
  355.  
  356. static void
  357. write_underline_marker (flag, hpos, vpos)
  358.      int flag, hpos, vpos;
  359. {
  360.   cmgoto (vpos, hpos);
  361.   cmplus (TN_standout_width);
  362.   OUTPUT (flag ? TS_underline_mode : TS_end_underline_mode);
  363.   chars_wasted[curY] = TN_underline_width | 0100;
  364. }
  365.  
  366. #define write_underline_begin_marker(hpos, vpos)  \
  367.         write_underline_marker (1, (hpos), (vpos))
  368.  
  369. #define write_underline_end_marker(hpos, vpos)    \
  370.         write_underline_marker (0, (hpos), (vpos))
  371.  
  372. static void
  373. turn_on_underline ()
  374. {
  375.   if (TN_underline_width < 0)
  376.     {
  377.       if (!underline_mode)
  378.     OUTPUT_IF (TS_underline_mode);
  379.     }
  380.   else
  381.     {
  382.       write_underline_begin_marker (curX, curY);
  383.     }
  384.   underline_mode = 1;
  385. }
  386.  
  387. static void
  388. turn_off_underline ()
  389. {
  390.   if (TN_underline_width < 0)
  391.     {
  392.       if (underline_mode)
  393.     OUTPUT_IF (TS_end_underline_mode);
  394.     }
  395.   else
  396.     {
  397.       write_underline_end_marker (curX, curY);
  398.     }
  399.   underline_mode = 0;
  400. }
  401.  
  402. static void
  403. enter_background_mode ()
  404. {
  405.   turn_off_underline ();
  406.   if (inverse_video)
  407.     turn_on_highlight ();
  408.   else
  409.     turn_off_highlight ();
  410. }
  411.  
  412. static void
  413. standout_as_desired (faceptr)
  414.      struct face *faceptr;
  415. {
  416.   if (faceptr->hilited && ! faceptr->underline)
  417.     {
  418.       turn_on_highlight ();
  419.       return;
  420.     }
  421.  
  422.   turn_off_highlight ();
  423.  
  424.   if (faceptr->underline)
  425.     turn_on_underline ();
  426.   else
  427.     turn_off_underline ();
  428. }
  429.  
  430. void
  431. ring_bell (sound)
  432.      Lisp_Object sound;
  433. {
  434.   if (CHECK_HOOK (ring_bell_hook))
  435.     {
  436.       /* Lucid sound change */
  437.       (*ring_bell_hook) (sound);   /* the window-system bell hook */
  438.       return;
  439.     }
  440.   OUTPUT (TS_visible_bell && visible_bell ? TS_visible_bell : TS_bell);
  441. }
  442.  
  443. void
  444. set_terminal_modes ()
  445. {
  446.   if (! initial_screen_is_tty ())
  447.     return;
  448.   if (CHECK_HOOK (set_terminal_modes_hook))
  449.     {
  450.       (*set_terminal_modes_hook) ();
  451.       return;
  452.     }
  453.   OUTPUT_IF (TS_termcap_modes);
  454.   OUTPUT_IF (TS_visual_mode);
  455.   OUTPUT_IF (TS_keypad_mode);
  456.   losecursor ();
  457. }
  458.  
  459. void
  460. reset_terminal_modes ()
  461. {
  462.   if (CHECK_HOOK (reset_terminal_modes_hook))
  463.     {
  464.       (*reset_terminal_modes_hook) ();
  465.       return;
  466.     }
  467.   if (TN_standout_width < 0)
  468.     turn_off_highlight ();
  469.   turn_off_insert ();
  470.   OUTPUT_IF (TS_end_keypad_mode);
  471.   OUTPUT_IF (TS_end_visual_mode);
  472.   OUTPUT_IF (TS_end_termcap_modes);
  473.   /* Output raw CR so kernel can track the cursor hpos.
  474.      But on magic-cookie terminals this can erase a end-standout marker and
  475.      cause the rest of the screen to be in standout, so move down first.  */
  476.   if (TN_standout_width >= 0)
  477.     cmputc ('\n');
  478.   cmputc ('\r');
  479. }
  480.  
  481. void
  482. update_begin (s)
  483.      SCREEN_PTR s;
  484. {
  485.   updating_screen = s;
  486.   if (CHECK_HOOK (update_begin_hook))
  487.     (*update_begin_hook) (s);
  488. }
  489.  
  490. void
  491. update_end (s)
  492.      SCREEN_PTR s;
  493. {
  494.   if (CHECK_HOOK (update_end_hook))
  495.     {
  496.       (*update_end_hook) (s);
  497.       updating_screen = 0;
  498.       return;
  499.     }
  500.   turn_off_insert ();
  501.   enter_background_mode ();
  502.   updating_screen = 0;
  503. }
  504.  
  505. /* Move to absolute position, specified origin 0 */
  506.  
  507. void
  508. cursor_to (row, col)
  509.      int row, col;
  510. {
  511.   register struct run *face_list
  512.     = selected_screen->current_glyphs->face_list[row];
  513.   int run_len, this_run;
  514.  
  515.   if (CHECK_HOOK (cursor_to_hook))
  516.     {
  517.       (*cursor_to_hook) (row, col);
  518.       return;
  519.     }
  520.  
  521.   if (face_list [(face_list[0].type == window) ? 1 : 0 ].length == 0)
  522.     {
  523.       if (col != 0)
  524.     abort ();
  525.       cursor_run = 0;
  526.     }
  527.   else if (col == selected_screen->current_glyphs->used[row])
  528.     {
  529.       cursor_run = selected_screen->current_glyphs->nruns[row] - 1;
  530.     }
  531.   else
  532.     {
  533.       this_run = 0;
  534.       run_len = face_list[this_run].length;
  535.       while (col >= run_len)
  536.     {
  537.       run_len += face_list[++this_run].length;
  538.     }
  539.       cursor_run = this_run;
  540.       left_in_this_run = run_len - col;
  541.     }
  542.  
  543.   col += chars_wasted[row] & 077;
  544.   if (curY == row && curX == col)
  545.     return;
  546.   if (!TF_standout_motion)
  547.     enter_background_mode ();
  548.   if (!TF_insmode_motion)
  549.     turn_off_insert ();
  550.   cmgoto (row, col);
  551. }
  552.  
  553. /* Similar but don't take any account of the wasted characters.  */
  554.  
  555. static void
  556. raw_cursor_to (row, col)
  557.      int row, col;
  558. {
  559.   register struct run *face_list = selected_screen->current_glyphs->face_list[row];
  560.   int run_len;
  561.   int this_run = 0;
  562.  
  563.   if (CHECK_HOOK (raw_cursor_to_hook))
  564.     {
  565.       (*raw_cursor_to_hook) (row, col);
  566.       return;
  567.     }
  568.  
  569.   if (curY == row && curX == col)
  570.     return;
  571.  
  572.   if (face_list[this_run].length == 0)
  573.     {
  574.       if (col != 0)
  575.     abort ();
  576.       cursor_run = 0;
  577.     }
  578.   else if (col == selected_screen->current_glyphs->used[row])
  579.     {
  580.       cursor_run = selected_screen->current_glyphs->nruns[row] - 1;
  581.     }
  582.   else
  583.     {
  584.       this_run = 0;
  585.       run_len = face_list[this_run].length;
  586.       while (col >= run_len)
  587.     {
  588.       run_len += face_list[++this_run].length;
  589.     }
  590.       cursor_run = this_run;
  591.       left_in_this_run = run_len - col;
  592.     }
  593.  
  594.   if (!TF_standout_motion)
  595.     enter_background_mode ();
  596.   if (!TF_insmode_motion)
  597.     turn_off_insert ();
  598.   cmgoto (row, col);
  599. }
  600.  
  601. static void
  602. write_raw_glyph (g)
  603.      GLYPH g;
  604. {
  605.   register int tlen = GLYPH_TABLE_LENGTH;
  606.   register Lisp_Object *tbase = GLYPH_TABLE_BASE;
  607.  
  608.   while (GLYPH_ALIAS_P (tbase, tlen, (int) g))
  609.     g = GLYPH_ALIAS (tbase, g);
  610.  
  611.   if (GLYPH_SIMPLE_P (tbase, tlen, (int) g))
  612.     {
  613.       putc (g & 0xff, stdout);
  614.       if (ferror (stdout))
  615.     clearerr (stdout);
  616.       if (termscript)
  617.     putc (g & 0xff, termscript);
  618.     }
  619.   else
  620.     {
  621.       fwrite (GLYPH_STRING (tbase, g), 1, GLYPH_LENGTH (tbase, g),
  622.           stdout);
  623.       if (ferror (stdout))
  624.     clearerr (stdout);
  625.       if (termscript)
  626.     fwrite (GLYPH_STRING (tbase, g), 1, GLYPH_LENGTH (tbase, g),
  627.         termscript);
  628.     }
  629. }
  630.  
  631. /* Clear to end of line, but do not clear any standout marker.
  632.    Assumes that the cursor is positioned at a character of real text,
  633.    which implies it cannot be before a standout marker
  634.    unless the marker has zero width.
  635.  
  636.    Note that the cursor may be moved.
  637.  
  638.    Clear from cursor to end of line.
  639.    Assume that the line is already clear starting at column first_unused_hpos.
  640.    If the cursor is at a standout marker, erase the marker.
  641.  
  642.    Note that the cursor may be moved, on terminals lacking a `ce' string.  */
  643.  
  644. static void
  645. clear_end_of_line_raw (first_unused_hpos)
  646.      int first_unused_hpos;
  647. {
  648.   register int i;
  649.  
  650.   if (CHECK_HOOK (clear_end_of_line_hook))
  651.     {
  652.       (*clear_end_of_line_hook) (first_unused_hpos);
  653.       return;
  654.     }
  655.  
  656.   first_unused_hpos += chars_wasted[curY] & 077;
  657.   if (curX >= first_unused_hpos)
  658.     return;
  659.   /* Notice if we are erasing a magic cookie */
  660.   if (curX == 0)
  661.     chars_wasted[curY] = 0;
  662.   enter_background_mode ();
  663.   if (TS_clr_line)
  664.     {
  665.       OUTPUT1 (TS_clr_line);
  666.     }
  667.   else     /* have to do it the hard way */
  668.     {
  669.       turn_off_insert ();
  670.  
  671.       if (AutoWrap && curY == SCREEN_HEIGHT (selected_screen) - 1
  672.       && first_unused_hpos == SCREEN_WIDTH (selected_screen))
  673.     first_unused_hpos--;
  674.  
  675.       for (i = curX; i < first_unused_hpos; i++)
  676.     {
  677.       if (termscript)
  678.         fputc (' ', termscript);
  679.       putchar (' ');
  680.     }
  681.       cmplus (first_unused_hpos - curX);
  682.     }
  683. }
  684.  
  685. void
  686. clear_end_of_line (first_unused_hpos)
  687.      int first_unused_hpos;
  688. {
  689.   if (SCREEN_IS_TERMCAP (selected_screen)
  690.       && TN_standout_width == 0
  691.       && curX == 0 && chars_wasted[curY] != 0)
  692.     write_raw_glyph (SPACEGLYPH);
  693.  
  694.   clear_end_of_line_raw (first_unused_hpos);
  695. }
  696.  
  697. /* clear from cursor to end of screen */
  698.  
  699. static void
  700. clear_to_end ()
  701. {
  702.   register int i;
  703.  
  704.   if (CHECK_HOOK (clear_to_end_hook))
  705.     {
  706.       (*clear_to_end_hook) ();
  707.       return;
  708.     }
  709.   if (TS_clr_to_bottom)
  710.     {
  711.       enter_background_mode ();
  712.       OUTPUT (TS_clr_to_bottom);
  713.       memset (chars_wasted + curY, 0, SCREEN_HEIGHT (selected_screen) - curY);
  714.     }
  715.   else
  716.     {
  717.       for (i = curY; i < SCREEN_HEIGHT (selected_screen); i++)
  718.     {
  719.       cursor_to (i, 0);
  720.       clear_end_of_line_raw (SCREEN_WIDTH (selected_screen));
  721.     }
  722.     }
  723. }
  724.  
  725. /* Clear entire screen */
  726.  
  727. void
  728. clear_screen ()
  729. {
  730.   if (CHECK_HOOK (clear_screen_hook))
  731.     {
  732.       (*clear_screen_hook) ();
  733.       return;
  734.     }
  735.  
  736.   if (TS_clr_screen)
  737.     {
  738.       enter_background_mode ();
  739.       OUTPUT (TS_clr_screen);
  740.       memset (chars_wasted, 0, SCREEN_HEIGHT (selected_screen));
  741.       cmat (0, 0);
  742.     }
  743.   else
  744.     {
  745.       cursor_to (0, 0);
  746.       clear_to_end ();
  747.     }
  748. }
  749.  
  750. void
  751. write_glyphs (hpos, vpos, len)
  752.      register int hpos, vpos, len;
  753. {
  754.   register struct run *face_list;
  755.   register int this_run = 0;
  756.   register int run_len;
  757.  
  758.   if (CHECK_HOOK (write_glyphs_hook))
  759.     {
  760.       (*write_glyphs_hook) (hpos, vpos, len);
  761.       return;
  762.     }
  763.  
  764.   if (len == 0)
  765.     return;
  766.  
  767.   face_list = selected_screen->current_glyphs->face_list[vpos];
  768.   run_len = face_list[this_run].length;
  769.  
  770.   /* Don't dare write in last column of bottom line, if AutoWrap,
  771.      since that would scroll the whole screen on some terminals.  */
  772.   if (AutoWrap
  773.       && curY + 1 == SCREEN_HEIGHT (selected_screen)
  774.       && (curX + len - (chars_wasted[curY] & 077)
  775.       == SCREEN_WIDTH (selected_screen)))
  776.     len --;
  777.  
  778. #if 0
  779.   /* Paranoia */
  780.   while (hpos >= run_len)
  781.     {
  782.       run_len += face_list[++this_run].length;
  783.     }
  784.  
  785.   if (this_run != cursor_run)
  786.     abort ();
  787. #endif
  788.  
  789.   standout_as_desired (face_list[cursor_run].faceptr);
  790.   turn_off_insert ();
  791.   cmplus (len);
  792.   while (--len >= 0)
  793.     {
  794.       write_raw_glyph (selected_screen->current_glyphs->glyphs[vpos][hpos++]);
  795.       if (--left_in_this_run == 0 && len > 0)
  796.     {
  797.       ++cursor_run;
  798.       standout_as_desired (face_list[cursor_run].faceptr);
  799.       left_in_this_run = face_list[cursor_run].length;
  800.     }
  801.     }
  802. }
  803.  
  804. void
  805. insert_spaceglyphs (hpos, vpos, len)
  806.      register int hpos, vpos, len;
  807. {
  808.   register struct run *face_list
  809.     = selected_screen->current_glyphs->face_list[vpos];
  810.  
  811.   if (CHECK_HOOK (insert_glyphs_hook))
  812.     {
  813.       (*insert_glyphs_hook) (hpos, vpos, len);
  814.       return;
  815.     }
  816.  
  817.   if (TS_ins_multi_chars)
  818.     {
  819.       char *buf;
  820.  
  821.       buf = tparam (TS_ins_multi_chars, 0, 0, len);
  822.       OUTPUT1 (buf);
  823.       xfree (buf);
  824.       return;
  825.     }
  826.  
  827.   standout_as_desired (face_list[cursor_run].faceptr);
  828.   turn_on_insert ();
  829.   cmplus (len);
  830.   while (--len >= 0)
  831.     {
  832.       OUTPUT1_IF (TS_ins_char);
  833.       write_raw_glyph (SPACEGLYPH);
  834.       OUTPUT1_IF (TS_pad_inserted_char);
  835.  
  836.       if (--left_in_this_run == 0 && len >= 1)
  837.     {
  838.       ++cursor_run;
  839.       standout_as_desired (face_list[cursor_run].faceptr);
  840.       left_in_this_run = face_list[cursor_run].length;
  841.     }
  842.     }
  843. }
  844.  
  845. void
  846. insert_glyphs (hpos, vpos, len)
  847.      register int hpos, vpos, len;
  848. {
  849.   register GLYPH **glyphs = selected_screen->current_glyphs->glyphs;
  850.   register struct run *face_list
  851.     = selected_screen->current_glyphs->face_list[vpos];
  852.   char *buf;
  853.  
  854.   if (CHECK_HOOK (insert_glyphs_hook))
  855.     {
  856.       (*insert_glyphs_hook) (hpos, vpos, len);
  857.       return;
  858.     }
  859.  
  860.   standout_as_desired (face_list[cursor_run].faceptr);
  861.  
  862.   if (TS_ins_multi_chars)
  863.     {
  864.       buf = tparam (TS_ins_multi_chars, 0, 0, len);
  865.       OUTPUT1 (buf);
  866.       xfree (buf);
  867.       write_glyphs (curX, curY, len);
  868.       return;
  869.     }
  870.  
  871.   turn_on_insert ();
  872.   cmplus (len);
  873.   while (--len >= 0)
  874.     {
  875.       OUTPUT1_IF (TS_ins_char);
  876.       write_raw_glyph (glyphs[vpos][hpos++]);
  877.       OUTPUT1_IF (TS_pad_inserted_char);
  878.  
  879.       if (--left_in_this_run == 0 && len >= 1)
  880.     {
  881.       standout_as_desired (face_list[++cursor_run].faceptr);
  882.       left_in_this_run = face_list[cursor_run].length;
  883.     }
  884.     }
  885. }
  886.  
  887. void
  888. delete_glyphs (n)
  889.      register int n;
  890. {
  891.   char *buf;
  892.   register int i;
  893.  
  894.   if (CHECK_HOOK (delete_glyphs_hook))
  895.     {
  896.       (*delete_glyphs_hook) (n);
  897.       return;
  898.     }
  899.  
  900.   if (delete_in_insert_mode)
  901.     {
  902.       turn_on_insert ();
  903.     }
  904.   else
  905.     {
  906.       turn_off_insert ();
  907.       OUTPUT_IF (TS_delete_mode);
  908.     }
  909.  
  910.   if (TS_del_multi_chars)
  911.     {
  912.       buf = tparam (TS_del_multi_chars, 0, 0, n);
  913.       OUTPUT1 (buf);
  914.       xfree (buf);
  915.     }
  916.   else
  917.     for (i = 0; i < n; i++)
  918.       OUTPUT1 (TS_del_char);
  919.   if (!delete_in_insert_mode)
  920.     OUTPUT_IF (TS_end_delete_mode);
  921. }
  922.  
  923. /* Insert N lines at vpos VPOS.  If N is negative, delete -N lines.  */
  924.  
  925. void
  926. ins_del_lines (vpos, n)
  927.      int vpos, n;
  928. {
  929.   char *multi = n > 0 ? TS_ins_multi_lines : TS_del_multi_lines;
  930.   char *single = n > 0 ? TS_ins_line : TS_del_line;
  931.   char *scroll = n > 0 ? TS_rev_scroll : TS_fwd_scroll;
  932.  
  933.   register int i = n > 0 ? n : -n;
  934.   register char *buf;
  935.  
  936.   if (CHECK_HOOK (ins_del_lines_hook))
  937.     {
  938.       (*ins_del_lines_hook) (vpos, n);
  939.       return;
  940.     }
  941.  
  942.   /* If the lines below the insertion are being pushed
  943.      into the end of the window, this is the same as clearing;
  944.      and we know the lines are already clear, since the matching
  945.      deletion has already been done.  So can ignore this.  */
  946.   /* If the lines below the deletion are blank lines coming
  947.      out of the end of the window, don't bother,
  948.      as there will be a matching inslines later that will flush them. */
  949.   if (scroll_region_ok && vpos + i >= specified_window)
  950.     return;
  951.   if (!memory_below_screen && vpos + i >= SCREEN_HEIGHT (selected_screen))
  952.     return;
  953.  
  954.   if (multi)
  955.     {
  956.       raw_cursor_to (vpos, 0);
  957.       enter_background_mode ();
  958.       buf = tparam (multi, 0, 0, i);
  959.       OUTPUT (buf);
  960.       xfree (buf);
  961.     }
  962.   else if (single)
  963.     {
  964.       raw_cursor_to (vpos, 0);
  965.       enter_background_mode ();
  966.       while (--i >= 0)
  967.     OUTPUT (single);
  968.       if (TF_teleray)
  969.     curX = 0;
  970.     }
  971.   else
  972.     {
  973.       set_scroll_region (vpos, specified_window);
  974.       if (n < 0)
  975.     raw_cursor_to (specified_window - 1, 0);
  976.       else
  977.     raw_cursor_to (vpos, 0);
  978.       enter_background_mode ();
  979.       while (--i >= 0)
  980.     OUTPUTL (scroll, specified_window - vpos);
  981.       set_scroll_region (0, specified_window);
  982.     }
  983.  
  984.   if (TN_standout_width >= 0)
  985.     {
  986.       if (n < 0)
  987.     {
  988.       memcpy (&chars_wasted[curY], &chars_wasted[curY - n],
  989.           SCREEN_HEIGHT (selected_screen) - curY + n);
  990.       memset (&chars_wasted[SCREEN_HEIGHT (selected_screen) + n], 0, - n);
  991.     }
  992.       else
  993.     {
  994.       memcpy (©buf[curY], &chars_wasted[curY],
  995.           SCREEN_HEIGHT (selected_screen) - curY - n);
  996.       memcpy (&chars_wasted[curY + n], ©buf[curY],
  997.           SCREEN_HEIGHT (selected_screen) - curY - n);
  998.       memset (&chars_wasted[curY], 0, n);
  999.     }
  1000.     }
  1001.   if (!scroll_region_ok && memory_below_screen && n < 0)
  1002.     {
  1003.       cursor_to (SCREEN_HEIGHT (selected_screen) + n, 0);
  1004.       clear_to_end ();
  1005.     }
  1006. }
  1007.  
  1008. /* Compute cost of sending "str", in characters,
  1009.    not counting any line-dependent padding.  */
  1010.  
  1011. int
  1012. string_cost (str)
  1013.      char *str;
  1014. {
  1015.   cost = 0;
  1016.   if (str)
  1017.     tputs (str, 0, evalcost);
  1018.   return cost;
  1019. }
  1020.  
  1021. /* Compute cost of sending "str", in characters,
  1022.    counting any line-dependent padding at one line.  */
  1023.  
  1024. static int
  1025. string_cost_one_line (str)
  1026.      char *str;
  1027. {
  1028.   cost = 0;
  1029.   if (str)
  1030.     tputs (str, 1, evalcost);
  1031.   return cost;
  1032. }
  1033.  
  1034. /* Compute per line amount of line-dependent padding,
  1035.    in tenths of characters.  */
  1036.  
  1037. int
  1038. per_line_cost (str)
  1039.      register char *str;
  1040. {
  1041.   cost = 0;
  1042.   if (str)
  1043.     tputs (str, 0, evalcost);
  1044.   cost = - cost;
  1045.   if (str)
  1046.     tputs (str, 10, evalcost);
  1047.   return cost;
  1048. }
  1049.  
  1050. #ifndef old
  1051. /* char_ins_del_cost[n] is cost of inserting N characters.
  1052.    char_ins_del_cost[-n] is cost of deleting N characters. */
  1053.  
  1054. int *char_ins_del_vector;
  1055.  
  1056. #define char_ins_del_cost(s) (&char_ins_del_vector[SCREEN_WIDTH ((s))])
  1057. #endif
  1058.  
  1059. /* ARGSUSED */
  1060. static void
  1061. calculate_ins_del_char_costs (screen)
  1062.      SCREEN_PTR screen;
  1063. {
  1064.   int ins_startup_cost, del_startup_cost;
  1065.   int ins_cost_per_char, del_cost_per_char;
  1066.   register int i;
  1067.   register int *p;
  1068.  
  1069.   if (TS_ins_multi_chars)
  1070.     {
  1071.       ins_cost_per_char = 0;
  1072.       ins_startup_cost = string_cost_one_line (TS_ins_multi_chars);
  1073.     }
  1074.   else if (TS_ins_char || TS_pad_inserted_char
  1075.        || (TS_insert_mode && TS_end_insert_mode))
  1076.     {
  1077.       ins_startup_cost = (30 * (string_cost (TS_insert_mode)
  1078.                 + string_cost (TS_end_insert_mode))) / 100;
  1079.       ins_cost_per_char = (string_cost_one_line (TS_ins_char)
  1080.                + string_cost_one_line (TS_pad_inserted_char));
  1081.     }
  1082.   else
  1083.     {
  1084.       ins_startup_cost = 9999;
  1085.       ins_cost_per_char = 0;
  1086.     }
  1087.  
  1088.   if (TS_del_multi_chars)
  1089.     {
  1090.       del_cost_per_char = 0;
  1091.       del_startup_cost = string_cost_one_line (TS_del_multi_chars);
  1092.     }
  1093.   else if (TS_del_char)
  1094.     {
  1095.       del_startup_cost = (string_cost (TS_delete_mode)
  1096.               + string_cost (TS_end_delete_mode));
  1097.       if (delete_in_insert_mode)
  1098.     del_startup_cost /= 2;
  1099.       del_cost_per_char = string_cost_one_line (TS_del_char);
  1100.     }
  1101.   else
  1102.     {
  1103.       del_startup_cost = 9999;
  1104.       del_cost_per_char = 0;
  1105.     }
  1106.  
  1107.   /* Delete costs are at negative offsets */
  1108.   p = &char_ins_del_cost (screen)[0];
  1109.   for (i = SCREEN_WIDTH (selected_screen); --i >= 0;)
  1110.     *--p = (del_startup_cost += del_cost_per_char);
  1111.  
  1112.   /* Doing nothing is free */
  1113.   p = &char_ins_del_cost (screen)[0];
  1114.   *p++ = 0;
  1115.  
  1116.   /* Insert costs are at positive offsets */
  1117.   for (i = SCREEN_WIDTH (screen); --i >= 0;)
  1118.     *p++ = (ins_startup_cost += ins_cost_per_char);
  1119. }
  1120.  
  1121. extern void do_line_insertion_deletion_costs (SCREEN_PTR, 
  1122.                           char *, char *, char *, char *, 
  1123.                           char *, char *, int);
  1124.  
  1125. void
  1126. calculate_costs (screen)
  1127.      SCREEN_PTR screen;
  1128. {
  1129.   register char *s = TS_set_scroll_region ?
  1130.                        TS_set_scroll_region
  1131.              : TS_set_scroll_region_1;
  1132.  
  1133.   if (dont_calculate_costs)
  1134.     return;
  1135.  
  1136. #ifdef HAVE_X_WINDOWS
  1137.   if (SCREEN_IS_X (screen))
  1138.     {
  1139.       do_line_insertion_deletion_costs (screen, 0, ".5*", 0, ".5*",
  1140.                     0, 0, 1 /*x_screen_planes*/);
  1141.       return;
  1142.     }
  1143. #endif
  1144.  
  1145.   /* These variables are only used for terminal stuff.  They are allocated
  1146.      once for the terminal screen of X-windows emacs, but not used afterwards.
  1147.  
  1148.      char_ins_del_vector (i.e., char_ins_del_cost) isn't used because
  1149.      X turns off char_ins_del_ok.
  1150.  
  1151.      chars_wasted and copybuf are only used here in term.c in cases where
  1152.      the term hook isn't called. */
  1153.  
  1154.   if (chars_wasted != 0)
  1155.     chars_wasted = (char *) xrealloc (chars_wasted, SCREEN_HEIGHT (screen));
  1156.   else
  1157.     chars_wasted = (char *) xmalloc (SCREEN_HEIGHT (screen));
  1158.  
  1159.   if (copybuf != 0)
  1160.     copybuf = (char *) xrealloc (copybuf, SCREEN_HEIGHT (screen));
  1161.   else
  1162.     copybuf = (char *) xmalloc (SCREEN_HEIGHT (screen));
  1163.  
  1164.   if (char_ins_del_vector != 0)
  1165.     char_ins_del_vector
  1166.       = (int *) xrealloc (char_ins_del_vector,
  1167.               (sizeof (int)
  1168.                + 2 * SCREEN_WIDTH (screen) * sizeof (int)));
  1169.   else
  1170.     char_ins_del_vector
  1171.       = (int *) xmalloc (sizeof (int)
  1172.              + 2 * SCREEN_WIDTH (screen) * sizeof (int));
  1173.  
  1174.   memset (chars_wasted, 0, SCREEN_HEIGHT (screen));
  1175.   memset (copybuf, 0, SCREEN_HEIGHT (screen));
  1176.   memset (char_ins_del_vector, 0,
  1177.       (sizeof (int) + 2 * SCREEN_WIDTH (screen) * sizeof (int)));
  1178.  
  1179.   if (s && (!TS_ins_line && !TS_del_line))
  1180.     do_line_insertion_deletion_costs (screen,
  1181.                       TS_rev_scroll, TS_ins_multi_lines,
  1182.                       TS_fwd_scroll, TS_del_multi_lines,
  1183.                       s, s, 1);
  1184.   else
  1185.     do_line_insertion_deletion_costs (screen,
  1186.                       TS_ins_line, TS_ins_multi_lines,
  1187.                       TS_del_line, TS_del_multi_lines,
  1188.                       0, 0, 1);
  1189.  
  1190.   calculate_ins_del_char_costs (screen);
  1191.  
  1192.   /* Don't use TS_repeat if its padding is worse than sending the chars */
  1193.   if (TS_repeat && per_line_cost (TS_repeat) * baud_rate < 9000)
  1194.     RPov = string_cost (TS_repeat);
  1195.   else
  1196.     RPov = SCREEN_WIDTH (screen) * 2;
  1197.  
  1198.   cmcostinit ();        /* set up cursor motion costs */
  1199. }
  1200.  
  1201. /* VARARGS 1 */
  1202. void
  1203. fatal (str, arg1, arg2)
  1204.      char *str;
  1205.      int arg1, arg2;
  1206. {
  1207.   fprintf (stderr, "emacs: ");
  1208.   fprintf (stderr, str, arg1, arg2);
  1209.   fflush (stderr);
  1210.   exit (1);
  1211. }
  1212.  
  1213. extern int tabs_safe_p (void);
  1214. extern void init_baud_rate (void);
  1215.  
  1216. void
  1217. term_init (terminal_type)
  1218.      char *terminal_type;
  1219. {
  1220.   char *combuf;
  1221.   char *fill;
  1222.   char tbuf[2044];
  1223.   register char *p;
  1224.   int status;
  1225.  
  1226.   extern char *tgetstr ();
  1227.  
  1228.   if (! initial_screen_is_tty ())
  1229.     return;
  1230.  
  1231.   Wcm_clear ();
  1232.   dont_calculate_costs = 0;
  1233.  
  1234.   status = tgetent (tbuf, terminal_type);
  1235.   if (status < 0)
  1236.     fatal ("Cannot open termcap database file.\n", 0, 0);
  1237.   if (status == 0)
  1238.     fatal ("Terminal type %s is not defined.\n", (int)terminal_type, 0);
  1239.  
  1240. #ifdef TERMINFO
  1241.   combuf = (char *) xmalloc (2044);
  1242. #else
  1243.   combuf = (char *) xmalloc (strlen (tbuf));
  1244. #endif /* not TERMINFO */
  1245.   if (combuf == 0)
  1246.     abort ();
  1247.   fill = combuf;
  1248.  
  1249.   TS_ins_line = tgetstr ("al", &fill);
  1250.   TS_ins_multi_lines = tgetstr ("AL", &fill);
  1251.   TS_bell = tgetstr ("bl", &fill);
  1252.   BackTab = tgetstr ("bt", &fill);
  1253.   TS_clr_to_bottom = tgetstr ("cd", &fill);
  1254.   TS_clr_line = tgetstr ("ce", &fill);
  1255.   TS_clr_screen = tgetstr ("cl", &fill);
  1256.   ColPosition = tgetstr ("ch", &fill);
  1257.   AbsPosition = tgetstr ("cm", &fill);
  1258.   CR = tgetstr ("cr", &fill);
  1259.   TS_set_scroll_region = tgetstr ("cs", &fill);
  1260.   TS_set_scroll_region_1 = tgetstr ("cS", &fill);
  1261.   RowPosition = tgetstr ("cv", &fill);
  1262.   TS_del_char = tgetstr ("dc", &fill);
  1263.   TS_del_multi_chars = tgetstr ("DC", &fill);
  1264.   TS_del_line = tgetstr ("dl", &fill);
  1265.   TS_del_multi_lines = tgetstr ("DL", &fill);
  1266.   TS_delete_mode = tgetstr ("dm", &fill);
  1267.   TS_end_delete_mode = tgetstr ("ed", &fill);
  1268.   TS_end_insert_mode = tgetstr ("ei", &fill);
  1269.   Home = tgetstr ("ho", &fill);
  1270.   TS_ins_char = tgetstr ("ic", &fill);
  1271.   TS_ins_multi_chars = tgetstr ("IC", &fill);
  1272.   TS_insert_mode = tgetstr ("im", &fill);
  1273.   TS_pad_inserted_char = tgetstr ("ip", &fill);
  1274.   TS_end_keypad_mode = tgetstr ("ke", &fill);
  1275.   TS_keypad_mode = tgetstr ("ks", &fill);
  1276.   LastLine = tgetstr ("ll", &fill);
  1277.   Right = tgetstr ("nd", &fill);
  1278.   Down = tgetstr ("do", &fill);
  1279.   if (!Down)
  1280.     Down = tgetstr ("nl", &fill); /* Obsolete name for "do" */
  1281. #ifdef VMS
  1282.   /* VMS puts a carriage return before each linefeed,
  1283.      so it is not safe to use linefeeds.  */
  1284.   if (Down && Down[0] == '\n' && Down[1] == '\0')
  1285.     Down = 0;
  1286. #endif /* VMS */
  1287.   if (tgetflag ("bs"))
  1288.     Left = "\b";          /* can't possibly be longer! */
  1289.   else                  /* (Actually, "bs" is obsolete...) */
  1290.     Left = tgetstr ("le", &fill);
  1291.   if (!Left)
  1292.     Left = tgetstr ("bc", &fill); /* Obsolete name for "le" */
  1293.   TS_pad_char = tgetstr ("pc", &fill);
  1294.   TS_repeat = tgetstr ("rp", &fill);
  1295.   TS_end_standout_mode = tgetstr ("se", &fill);
  1296.   TS_fwd_scroll = tgetstr ("sf", &fill);
  1297.   TS_standout_mode = tgetstr ("so", &fill);
  1298.   TS_rev_scroll = tgetstr ("sr", &fill);
  1299.   Wcm.cm_tab = tgetstr ("ta", &fill);
  1300.   TS_end_termcap_modes = tgetstr ("te", &fill);
  1301.   TS_termcap_modes = tgetstr ("ti", &fill);
  1302.   Up = tgetstr ("up", &fill);
  1303.   TS_visible_bell = tgetstr ("vb", &fill);
  1304.   TS_end_visual_mode = tgetstr ("ve", &fill);
  1305.   TS_visual_mode = tgetstr ("vs", &fill);
  1306.   TS_set_window = tgetstr ("wi", &fill);
  1307.   MultiUp = tgetstr ("UP", &fill);
  1308.   MultiDown = tgetstr ("DO", &fill);
  1309.   MultiLeft = tgetstr ("LE", &fill);
  1310.   MultiRight = tgetstr ("RI", &fill);
  1311.  
  1312.   TS_underline_mode = tgetstr ("us", &fill);
  1313.   TS_end_underline_mode = tgetstr ("ue", &fill);
  1314.   TN_underline_width = tgetnum ("ug");
  1315.  
  1316.   AutoWrap = tgetflag ("am");
  1317.   memory_below_screen = tgetflag ("db");
  1318.   TF_hazeltine = tgetflag ("hz");
  1319.   must_write_spaces = tgetflag ("in");
  1320.   meta_key = tgetflag ("km") || tgetflag ("MT");
  1321.   TF_insmode_motion = tgetflag ("mi");
  1322.   TF_standout_motion = tgetflag ("ms");
  1323.   TF_underscore = tgetflag ("ul");
  1324.   MagicWrap = tgetflag ("xn");
  1325.   TF_xs = tgetflag ("xs");
  1326.   TF_teleray = tgetflag ("xt");
  1327.  
  1328.   /* Get screen size from system, or else from termcap.  */
  1329.   get_screen_size (&SCREEN_WIDTH (selected_screen),
  1330.            &SCREEN_HEIGHT (selected_screen));
  1331.   if (SCREEN_WIDTH (selected_screen) <= 0)
  1332.     SCREEN_WIDTH (selected_screen) = tgetnum ("co");
  1333.   if (SCREEN_HEIGHT (selected_screen) <= 0)
  1334.     SCREEN_HEIGHT (selected_screen) = tgetnum ("li");
  1335.  
  1336.   min_padding_speed = tgetnum ("pb");
  1337.   TN_standout_width = tgetnum ("sg");
  1338.   TabWidth = tgetnum ("tw");
  1339.  
  1340. #ifdef VMS
  1341.   /* These capabilities commonly use ^J.
  1342.      I don't know why, but sending them on VMS does not work;
  1343.      it causes following spaces to be lost, sometimes.
  1344.      For now, the simplest fix is to avoid using these capabilities ever.  */
  1345.   if (Down && Down[0] == '\n')
  1346.     Down = 0;
  1347. #endif /* VMS */
  1348.  
  1349.   if (!TS_bell)
  1350.     TS_bell = "\07";
  1351.  
  1352.   if (!TS_fwd_scroll)
  1353.     TS_fwd_scroll = Down;
  1354.  
  1355.   PC = TS_pad_char ? *TS_pad_char : 0;
  1356.  
  1357.   if (TabWidth < 0)
  1358.     TabWidth = 8;
  1359.   
  1360. /* Turned off since /etc/termcap seems to have :ta= for most terminals
  1361.    and newer termcap doc does not seem to say there is a default.
  1362.   if (!Wcm.cm_tab)
  1363.     Wcm.cm_tab = "\t";
  1364. */
  1365.  
  1366.   if (TS_standout_mode == 0)
  1367.     {
  1368.       TN_standout_width = tgetnum ("ug");
  1369.       TS_end_standout_mode = tgetstr ("ue", &fill);
  1370.       TS_standout_mode = tgetstr ("us", &fill);
  1371.     }
  1372.  
  1373.   if (TF_teleray)
  1374.     {
  1375.       Wcm.cm_tab = 0;
  1376.       /* Teleray: most programs want a space in front of TS_standout_mode,
  1377.        but Emacs can do without it (and give one extra column).  */
  1378.       TS_standout_mode = "\033RD";
  1379.       TN_standout_width = 1;
  1380.       /* But that means we cannot rely on ^M to go to column zero! */
  1381.       CR = 0;
  1382.       /* LF can't be trusted either -- can alter hpos */
  1383.       /* if move at column 0 thru a line with TS_standout_mode */
  1384.       Down = 0;
  1385.     }
  1386.  
  1387.   /* Special handling for certain terminal types known to need it */
  1388.  
  1389.   if (!strcmp (terminal_type, "supdup"))
  1390.     {
  1391.       memory_below_screen = 1;
  1392.       Wcm.cm_losewrap = 1;
  1393.     }
  1394.   if (!strncmp (terminal_type, "c10", 3)
  1395.       || !strcmp (terminal_type, "perq"))
  1396.     {
  1397.       /* Supply a makeshift :wi string.
  1398.      This string is not valid in general since it works only
  1399.      for windows starting at the upper left corner;
  1400.      but that is all Emacs uses.
  1401.  
  1402.      This string works only if the screen is using
  1403.      the top of the video memory, because addressing is memory-relative.
  1404.      So first check the :ti string to see if that is true.
  1405.  
  1406.      It would be simpler if the :wi string could go in the termcap
  1407.      entry, but it can't because it is not fully valid.
  1408.      If it were in the termcap entry, it would confuse other programs.  */
  1409.       if (!TS_set_window)
  1410.     {
  1411.       p = TS_termcap_modes;
  1412.       while (*p && strcmp (p, "\033v  "))
  1413.         p++;
  1414.       if (*p)
  1415.         TS_set_window = "\033v%C %C %C %C ";
  1416.     }
  1417.       /* Termcap entry often fails to have :in: flag */
  1418.       must_write_spaces = 1;
  1419.       /* :ti string typically fails to have \E^G! in it */
  1420.       /* This limits scope of insert-char to one line.  */
  1421.       strcpy (fill, TS_termcap_modes);
  1422.       strcat (fill, "\033\007!");
  1423.       TS_termcap_modes = fill;
  1424.       fill += strlen (fill) + 1;
  1425.       p = combuf;
  1426.       /* Change all %+ parameters to %C, to handle
  1427.      values above 96 correctly for the C100.  */
  1428.       while (p != fill)
  1429.     {
  1430.       if (p[0] == '%' && p[1] == '+')
  1431.         p[1] = 'C';
  1432.       p++;
  1433.     }
  1434.     }
  1435.  
  1436.   ScreenRows = SCREEN_HEIGHT (selected_screen);
  1437.   ScreenCols = SCREEN_WIDTH (selected_screen);
  1438.   specified_window = SCREEN_HEIGHT (selected_screen);
  1439.  
  1440.   if (Wcm_init () == -1)    /* can't do cursor motion */
  1441.     {
  1442.  
  1443.       /* #### */
  1444.       fatal ("Sorry, this Emacs only works under X for now\n\
  1445. Check that your $DISPLAY environment variable is properly set.\n");
  1446.  
  1447. #ifdef VMS
  1448.       fatal ("Terminal type \"%s\" is not powerful enough to run Emacs.\n\
  1449. It lacks the ability to position the cursor.\n\
  1450. If that is not the actual type of terminal you have, use either the\n\
  1451. DCL command `SET TERMINAL/DEVICE= ...' for DEC-compatible terminals,\n\
  1452. or `define EMACS_TERM \"terminal type\"' for non-DEC terminals.\n",
  1453.          (int)terminal_type);
  1454. #else
  1455.       fatal ("Terminal type \"%s\" is not powerful enough to run Emacs.\n\
  1456. It lacks the ability to position the cursor.\n\
  1457. If that is not the actual type of terminal you have,\n\
  1458. use the C-shell command `setenv TERM ...' to specify the correct type.\n\
  1459. It may be necessary to do `unsetenv TERMCAP' as well.\n",
  1460.          (int)terminal_type, 0);
  1461. #endif
  1462.     }
  1463.   if (SCREEN_HEIGHT (selected_screen) <= 0
  1464.       || SCREEN_WIDTH (selected_screen) <= 0)
  1465.     fatal ("The screen size has not been specified.", 0, 0);
  1466.  
  1467.   delete_in_insert_mode
  1468.     = TS_delete_mode && TS_insert_mode
  1469.       && !strcmp (TS_delete_mode, TS_insert_mode);
  1470.  
  1471.   se_is_so = TS_standout_mode && TS_end_standout_mode
  1472.     && !strcmp (TS_standout_mode, TS_end_standout_mode);
  1473.  
  1474.   /* Remove width of standout marker from usable width of line */
  1475.   if (TN_standout_width > 0)
  1476.     SCREEN_WIDTH (selected_screen) -= TN_standout_width;
  1477.  
  1478.   UseTabs = tabs_safe_p () && TabWidth == 8;
  1479.  
  1480.   scroll_region_ok
  1481.     = (Wcm.cm_abs
  1482.        && (TS_set_window || TS_set_scroll_region || TS_set_scroll_region_1));
  1483.  
  1484.   line_ins_del_ok = (((TS_ins_line || TS_ins_multi_lines)
  1485.               && (TS_del_line || TS_del_multi_lines))
  1486.              || (scroll_region_ok && TS_fwd_scroll && TS_rev_scroll));
  1487.  
  1488.   char_ins_del_ok = ((TS_ins_char || TS_insert_mode
  1489.               || TS_pad_inserted_char || TS_ins_multi_chars)
  1490.              && (TS_del_char || TS_del_multi_chars));
  1491.  
  1492.   fast_clear_end_of_line = TS_clr_line != 0;
  1493.  
  1494.   init_baud_rate ();
  1495.   if (read_socket_hook)        /* Baudrate is somewhat */
  1496.                 /* meaningless in this case */
  1497.     baud_rate = 9600;
  1498.  
  1499.   calculate_costs (selected_screen);
  1500. }
  1501.