1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239 |
- /* Copyright (c) 1992, 1995 John E. Davis
- * All rights reserved.
- *
- * You may distribute under the terms of either the GNU General Public
- * License or the Perl Artistic License.
- */
- /*
- * SLTT_TRANSP_ACS_PATCH (#define/#undef):
- *
- * The problem: some terminals (e.g. QNX/qansi*) map the whole upper half of
- * the ASCII table to the lower half, when alt-char-set is activated with
- * the smacs/as string-sequence. This means, that if 0 <= ch < 128 written
- * to the terminal, it will be translated to (ch+128) automatically by the
- * terminal: so not only the line-drawing characters can be written, when
- * the alt-char-set is activated. It implicitly means, that space, NL, CR,
- * etc. characters (exactly: anything besides the "standard" line drawing
- * characters) can not be written directly to the terminal, when the
- * alt-char-set is activated, because writing these characters doesn't cause
- * an implicit/temporary switching-back to the standard char-set!
- *
- * The original code in SLang assumes that space, NL, CR, etc. can be
- * printed when alt-char-set is activated. If SLTT_TRANSP_ACS_PATCH is
- * defined, the modified code will not use this assumption.
- * [Remark: the patch-code is not the most exact solution, but works...]
- */
- /*#define SLTT_TRANSP_ACS_PATCH 1*/
- /*
- * QNX_QANSI_SLANG_COMPAT_ACS (#define/#undef):
- *
- * A more OS/terminal-specific solution for the problem mentioned above
- * (->SLTT_TRANSP_ACS_PATCH).
- *
- * If QNX_QANSI_SLANG_COMPAT_ACS is defined, the default smacs/sa, rmacs/ae,
- * acsc/ac [and sgr/sa, if it would be used!] command sequences will be
- * replaced internally with the "old style" (pre-QNX 4.23) sequences in case
- * of QNX/qansi terminals. Using these optional command sequences the terminal
- * remains compatible with the original SLang code (without using the
- * workaround-code enabled by defining SLTT_TRANSP_ACS_PATCH).
- */
- /*#define QNX_QANSI_SLANG_COMPAT_ACS 1*/
- /* auto-configuration */
- #ifdef SLTT_TRANSP_ACS_PATCH
- # if defined(__QNX__) && defined(QNX_QANSI_SLANG_COMPAT_ACS)
- # undef SLTT_TRANSP_ACS_PATCH
- # endif
- #else
- # if defined(__QNX__) && !defined(QNX_QANSI_SLANG_COMPAT_ACS)
- # define QNX_QANSI_SLANG_COMPAT_ACS 1
- # endif
- #endif
- #include "sl-feat.h"
- #include "config.h"
- #include <stdio.h>
- #include <string.h>
- #ifdef SCO_FLAVOR
- # include <sys/types.h>
- # include <sys/timeb.h> /* for struct timeb, used in time.h */
- #endif
- #include <time.h>
- #include <ctype.h>
- #ifndef VMS
- # include <sys/time.h>
- # ifdef __QNX__
- # include <sys/select.h>
- # endif
- # include <sys/types.h>
- #endif
- #ifdef __BEOS__
- /* Prototype for select */
- # include <net/socket.h>
- #endif
- #ifdef HAVE_TERMIOS_H
- # include <termios.h>
- #endif
- #ifdef VMS
- # include <unixlib.h>
- # include <unixio.h>
- # include <dvidef.h>
- # include <descrip.h>
- # include <lib$routines.h>
- # include <starlet.h>
- #else
- # if !defined(sun)
- # include <sys/ioctl.h>
- # endif
- #endif
- #ifdef SYSV
- # include <sys/termio.h>
- # include <sys/stream.h>
- # include <sys/ptem.h>
- # include <sys/tty.h>
- #endif
- #if defined (_AIX) && !defined (FD_SET)
- # include <sys/select.h> /* for FD_ISSET, FD_SET, FD_ZERO */
- #endif
- #include <errno.h>
- #include "_slang.h"
- #ifdef HAVE_STDLIB_H
- # include <stdlib.h>
- #endif
- #ifdef HAVE_UNISTD_H
- # include <unistd.h>
- #endif
- #if defined(__DECC) && defined(VMS)
- /* These get prototypes for write an sleep */
- # include <unixio.h>
- #endif
- #include <signal.h>
- /* Colors: These definitions are used for the display. However, the
- * application only uses object handles which get mapped to this
- * internal representation. The mapping is performed by the Color_Map
- * structure below. */
- #define CHAR_MASK 0x000000FF
- #define FG_MASK 0x0000FF00
- #define BG_MASK 0x00FF0000
- #define ATTR_MASK 0x1F000000
- #define BGALL_MASK 0x0FFF0000
- /* The 0x10000000 bit represents the alternate character set. BGALL_MASK does
- * not include this attribute.
- */
- #define GET_FG(color) ((color & FG_MASK) >> 8)
- #define GET_BG(color) ((color & BG_MASK) >> 16)
- #define MAKE_COLOR(fg, bg) (((fg) | ((bg) << 8)) << 8)
- int SLtt_Screen_Cols;
- int SLtt_Screen_Rows;
- int SLtt_Term_Cannot_Insert;
- int SLtt_Term_Cannot_Scroll;
- int SLtt_Use_Ansi_Colors;
- int SLtt_Blink_Mode = 1;
- int SLtt_Use_Blink_For_ACS = 0;
- int SLtt_Newline_Ok = 0;
- int SLtt_Has_Alt_Charset = 0;
- int SLtt_Force_Keypad_Init = 0;
- /* -1 means unknown */
- int SLtt_Has_Status_Line = -1; /* hs */
- static int Automatic_Margins;
- /* static int No_Move_In_Standout; */
- static int Worthless_Highlight;
- #define HP_GLITCH_CODE
- #ifdef HP_GLITCH_CODE
- /* This glitch is exclusive to HP term. Basically it means that to clear
- * attributes, one has to erase to the end of the line.
- */
- static int Has_HP_Glitch;
- #endif
- static char *Reset_Color_String;
- static int Linux_Console;
- /* It is crucial that JMAX_COLORS must be less than 128 since the high bit
- * is used to indicate a character from the ACS (alt char set). The exception
- * to this rule is if SLtt_Use_Blink_For_ACS is true. This means that of
- * the highbit is set, we interpret that as a blink character. This is
- * exploited by DOSemu.
- */
- #define JMAX_COLORS 256
- #define JNORMAL_COLOR 0
- typedef struct
- {
- SLtt_Char_Type fgbg;
- SLtt_Char_Type mono;
- char *custom_esc;
- } Ansi_Color_Type;
- #define RGB1(r, g, b) ((r) | ((g) << 1) | ((b) << 2))
- #define RGB(r, g, b, br, bg, bb) ((RGB1(r, g, b) << 8) | (RGB1(br, bg, bb) << 16))
- static Ansi_Color_Type Ansi_Color_Map[JMAX_COLORS] =
- {
- {RGB(1, 1, 1, 0, 0, 0), 0x00000000, NULL}, /* white/black */
- {RGB(0, 1, 0, 0, 0, 0), SLTT_REV_MASK, NULL}, /* green/black */
- {RGB(1, 0, 1, 0, 0, 0), SLTT_REV_MASK, NULL}, /* magenta/black */
- {RGB(0, 1, 1, 0, 0, 0), SLTT_REV_MASK, NULL}, /* cyan/black */
- {RGB(1, 0, 0, 0, 0, 0), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 0, 0, 1), SLTT_REV_MASK, NULL},
- {RGB(1, 0, 0, 0, 0, 1), SLTT_REV_MASK, NULL},
- {RGB(1, 0, 0, 0, 1, 0), SLTT_REV_MASK, NULL},
- {RGB(0, 0, 1, 1, 0, 0), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 1, 0, 0), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 1, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(1, 1, 0, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(1, 0, 1, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(0, 0, 0, 0, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 1, 1, 1), SLTT_REV_MASK, NULL},
- {RGB(0, 1, 0, 1, 1, 1), SLTT_REV_MASK, NULL}
- };
- /* This is the string to use to use when outputting color information.
- */
- #ifdef M_UNIX
- /* work around for sco console bug that can't handle combined sequences */
- static char *Color_Escape_Sequence = "\033[3%dm\033[4%dm";
- #else
- /* Believe it or not, this is what is in the linux terminfo database. It
- * produces the same escape sequence but it is much more CPU intensive.
- * Why not just encode it as "\033[3%p1%dm\033[4%p2%dm" ???
- */
- /* static char *Color_Escape_Sequence = "\033[%p1%{30}%+%dm\033[%p2%{40}%+%dm"; */
- static char *Color_Escape_Sequence = "\033[3%d;4%dm";
- #endif
- char *SLtt_Graphics_Char_Pairs; /* ac termcap string -- def is vt100 */
-
- /* 1 if terminal lacks the ability to do into insert mode or into delete
- mode. Currently controlled by S-Lang but later perhaps termcap. */
- static char *UnderLine_Vid_Str;
- static char *Blink_Vid_Str;
- static char *Bold_Vid_Str;
- static char *Ins_Mode_Str; /* = "\033[4h"; */ /* ins mode (im) */
- static char *Eins_Mode_Str; /* = "\033[4l"; */ /* end ins mode (ei) */
- static char *Scroll_R_Str; /* = "\033[%d;%dr"; */ /* scroll region */
- static char *Cls_Str; /* = "\033[2J\033[H"; */ /* cl termcap STR for ansi terminals */
- static char *Rev_Vid_Str; /* = "\033[7m"; */ /* mr,so termcap string */
- static char *Norm_Vid_Str; /* = "\033[m"; */ /* me,se termcap string */
- static char *Del_Eol_Str; /* = "\033[K"; */ /* ce */
- static char *Del_Char_Str; /* = "\033[P"; */ /* dc */
- static char *Del_N_Lines_Str; /* = "\033[%dM"; */ /* DL */
- static char *Add_N_Lines_Str; /* = "\033[%dL"; */ /* AL */
- static char *Rev_Scroll_Str;
- static char *Curs_Up_Str;
- static char *Curs_F_Str; /* RI termcap string */
- static char *Cursor_Visible_Str; /* ve termcap string */
- static char *Cursor_Invisible_Str; /* vi termcap string */
- static char *Start_Alt_Chars_Str; /* as */
- static char *End_Alt_Chars_Str; /* ae */
- static char *Enable_Alt_Char_Set; /* eA */
- static char *Term_Init_Str;
- static char *Keypad_Init_Str;
- static char *Term_Reset_Str;
- static char *Keypad_Reset_Str;
- /* status line functions */
- static char *Disable_Status_line_Str; /* ds */
- static char *Return_From_Status_Line_Str; /* fs */
- static char *Goto_Status_Line_Str; /* ts */
- static int Num_Status_Line_Columns; /* ws */
- static int Status_Line_Esc_Ok; /* es */
- /* static int Len_Curs_F_Str = 5; */
- /* cm string has %i%d since termcap numbers columns from 0 */
- /* char *CURS_POS_STR = "\033[%d;%df"; ansi-- hor and vert pos */
- static char *Curs_Pos_Str; /* = "\033[%i%d;%dH";*/ /* cm termcap string */
- /* scrolling region */
- static int Scroll_r1 = 0, Scroll_r2 = 23;
- static int Cursor_r, Cursor_c; /* 0 based */
- /* current attributes --- initialized to impossible value */
- static SLtt_Char_Type Current_Fgbg = 0xFFFFFFFFU;
- static int Cursor_Set; /* 1 if cursor position known, 0
- * if not. -1 if only row is known
- */
- #define MAX_OUTPUT_BUFFER_SIZE 4096
- static unsigned char Output_Buffer[MAX_OUTPUT_BUFFER_SIZE];
- static unsigned char *Output_Bufferp = Output_Buffer;
- unsigned long SLtt_Num_Chars_Output;
- #ifdef SLTT_TRANSP_ACS_PATCH
- static int SLtt_ACS_Active = 0;
- #endif
- static int sl_usleep (unsigned long usecs)
- {
- #ifndef VMS
- struct timeval tv;
- tv.tv_sec = usecs / 1000000;
- tv.tv_usec = usecs % 1000000;
- return select(0, NULL, NULL, NULL, &tv);
- #else
- return 0;
- #endif
- }
- int SLtt_flush_output (void)
- {
- int nwrite = 0;
- unsigned int total;
- int n = (int) (Output_Bufferp - Output_Buffer);
-
- SLtt_Num_Chars_Output += n;
-
- total = 0;
- while (n > 0)
- {
- nwrite = write (fileno(stdout), (char *) Output_Buffer + total, n);
- if (nwrite == -1)
- {
- nwrite = 0;
- #ifdef EAGAIN
- if (errno == EAGAIN)
- {
- sl_usleep (100000); /* 1/10 sec */
- continue;
- }
- #endif
- #ifdef EWOULDBLOCK
- if (errno == EWOULDBLOCK)
- {
- sl_usleep (100000);
- continue;
- }
- #endif
- #ifdef EINTR
- if (errno == EINTR) continue;
- #endif
- break;
- }
- n -= nwrite;
- total += nwrite;
- }
- Output_Bufferp = Output_Buffer;
- return n;
- }
- int SLtt_Baud_Rate;
- static void tt_write(char *str, unsigned int n)
- {
- static unsigned long last_time;
- static int total;
- unsigned long now;
- unsigned int ndiff;
-
- if ((str == NULL) || (n == 0)) return;
- total += n;
-
- while (1)
- {
- ndiff = MAX_OUTPUT_BUFFER_SIZE - (int) (Output_Bufferp - Output_Buffer);
- if (ndiff < n)
- {
- SLMEMCPY ((char *) Output_Bufferp, (char *) str, ndiff);
- Output_Bufferp += ndiff;
- SLtt_flush_output ();
- n -= ndiff;
- str += ndiff;
- }
- else
- {
- SLMEMCPY ((char *) Output_Bufferp, str, n);
- Output_Bufferp += n;
- break;
- }
- }
-
- if (((SLtt_Baud_Rate > 150) && (SLtt_Baud_Rate <= 9600))
- && (10 * total > SLtt_Baud_Rate))
- {
- total = 0;
- if ((now = (unsigned long) time(NULL)) - last_time <= 1)
- {
- SLtt_flush_output ();
- sleep((unsigned) 1);
- }
- last_time = now;
- }
- }
- void SLtt_write_string (char *str)
- {
- if (str != NULL) tt_write(str, strlen(str));
- }
- void SLtt_putchar (char ch)
- {
- #ifdef SLTT_TRANSP_ACS_PATCH
- int restore_acs = 0;
- #endif
-
- SLtt_normal_video ();
- if (Cursor_Set == 1)
- {
- if (ch >= ' ') Cursor_c++;
- #ifndef SLTT_TRANSP_ACS_PATCH
- else if (ch == '\b') Cursor_c--;
- #else
- if (ch <= ' ' && SLtt_ACS_Active)
- {
- SLtt_set_alt_char_set (0);
- restore_acs = 1;
- }
- if (ch == '\b') Cursor_c--;
- #endif
- else if (ch == '\r') Cursor_c = 0;
- else Cursor_Set = 0;
-
- if ((Cursor_c + 1 == SLtt_Screen_Cols)
- && Automatic_Margins) Cursor_Set = 0;
- }
-
- if (Output_Bufferp < Output_Buffer + MAX_OUTPUT_BUFFER_SIZE)
- {
- *Output_Bufferp++ = (unsigned char) ch;
- }
- else tt_write (&ch, 1);
-
- #ifdef SLTT_TRANSP_ACS_PATCH
- if (restore_acs)
- {
- SLtt_set_alt_char_set (1);
- }
- #endif
- }
- /* this is supposed to be fast--- also handles
- termcap: %d, &i, %., %+, %r strings as well as terminfo stuff */
- static unsigned int tt_sprintf(char *buf, char *fmt, int x, int y)
- {
- register unsigned char *f = (unsigned char *) fmt, *b, ch;
- int offset = 0, tinfo = 0;
- int stack[10];
- int i = 0, z;
- stack[0] = y; stack[1] = x; i = 2;
-
- b = (unsigned char *) buf;
- if (fmt != NULL) while ((ch = *f++) != 0)
- {
- if (ch != '%') *b++ = ch;
- else
- {
- ch = *f++;
- if (tinfo)
- {
- if ((ch <= '3') && (ch >= '0'))
- {
- /* map it to termcap. Since this is terminfo,
- * it must be one of:
- * %2d, %3d, %02d, %03d
- *
- * I am assuming that a terminal that understands
- * %2d form will also understand the %02d form. These
- * only differ by a space padding the field.
- */
-
- /* skip the 'd'-- hope it is there */
- if (ch == '0')
- {
- ch = *f;
- f += 2;
- }
- else f++;
- }
- }
-
- switch (ch)
- {
- case 'p':
- tinfo = 1;
- ch = *f++;
- if (ch == '1') stack[i++] = x; else stack[i++] = y;
- break;
-
- case '\'': /* 'x' */
- stack[i++] = *f++;
- f++;
- break;
-
- case '{': /* literal constant, e.g. {30} */
- z = 0;
- while (((ch = *f) <= '9') && (ch >= '0'))
- {
- z = z * 10 + (ch - '0');
- f++;
- }
- stack[i++] = z;
- if (ch == '}') f++;
- break;
-
- case 'd':
- case '2':
- case '3':
- z = stack[--i];
- z += offset;
- if (z >= 100)
- {
- *b++ = z / 100 + '0';
- z = z % 100;
- goto ten;
- }
- else if (ch == 3)
- {
- *b++ = '0';
- ch = '2';
- }
-
- if (z >= 10)
- {
- ten:
- *b++ = z / 10 + '0';
- z = z % 10;
- }
- else if (ch == 2) *b++ = '0';
-
- *b++ = z + '0';
- break;
-
- case 'i':
- offset = 1;
- break;
-
- case '+':
- if (tinfo)
- {
- z = stack[--i];
- stack[i-1] += z;
- }
- else
- {
- ch = *f++;
- if ((unsigned char) ch == 128) ch = 0;
- ch = ch + (unsigned char) stack[--i];
- if (ch == '\n') ch++;
- *b++ = ch;
- }
- break;
-
- case 'r':
- stack[0] = x;
- stack[1] = y;
- break;
-
- case '.':
- case 'c':
- ch = (unsigned char) stack[--i];
- if (ch == '\n') ch++;
- *b++ = ch;
- break;
-
- default:
- *b++ = ch;
- }
- }
- }
- *b = 0;
- return((unsigned int) (b - (unsigned char *) buf));
- }
- static void tt_printf(char *fmt, int x, int y)
- {
- char buf[256];
- unsigned int n;
- if (fmt == NULL) return;
- n = tt_sprintf(buf, fmt, x, y);
- tt_write(buf, n);
- }
- void SLtt_set_scroll_region (int r1, int r2)
- {
- Scroll_r1 = r1;
- Scroll_r2 = r2;
- tt_printf (Scroll_R_Str, Scroll_r1, Scroll_r2);
- Cursor_Set = 0;
- }
- void SLtt_reset_scroll_region (void)
- {
- SLtt_set_scroll_region(0, SLtt_Screen_Rows - 1);
- }
- int SLtt_set_cursor_visibility (int show)
- {
- if ((Cursor_Visible_Str == NULL) || (Cursor_Invisible_Str == NULL))
- return -1;
-
- SLtt_write_string (show ? Cursor_Visible_Str : Cursor_Invisible_Str);
- return 0;
- }
-
- /* the goto_rc function moves to row relative to scrolling region */
- void SLtt_goto_rc(int r, int c)
- {
- char *s = NULL;
- int n;
- char buf[6];
- #ifdef SLTT_TRANSP_ACS_PATCH
- int check_alt_acs = 0;
- #endif
-
- if (c < 0)
- {
- c = -c - 1;
- Cursor_Set = 0;
- }
-
- /* if (No_Move_In_Standout && Current_Fgbg) SLtt_normal_video (); */
- r += Scroll_r1;
-
- if ((Cursor_Set > 0) || ((Cursor_Set < 0) && !Automatic_Margins))
- {
- n = r - Cursor_r;
- if ((n == -1) && (Cursor_Set > 0) && (Cursor_c == c)
- && (Curs_Up_Str != NULL))
- {
- s = Curs_Up_Str;
- }
- else if ((n >= 0) && (n <= 4))
- {
- if ((n == 0) && (Cursor_Set == 1)
- && ((c > 1) || (c == Cursor_c)))
- {
- if (Cursor_c == c) return;
- if (Cursor_c == c + 1)
- {
- s = buf;
- *s++ = '\b'; *s = 0;
- s = buf;
- #ifdef SLTT_TRANSP_ACS_PATCH
- check_alt_acs = 1;
- #endif
- }
- }
- else if (c == 0)
- {
- s = buf;
- if ((Cursor_Set != 1) || (Cursor_c != 0)) *s++ = '\r';
- while (n--) *s++ = '\n';
- #ifdef VMS
- /* Need to add this after \n to start a new record. Sheesh. */
- *s++ = '\r';
- #endif
- *s = 0;
- s = buf;
- #ifdef SLTT_TRANSP_ACS_PATCH
- check_alt_acs = 1;
- #endif
- }
- /* Will fail on VMS */
- #ifndef VMS
- else if (SLtt_Newline_Ok && (Cursor_Set == 1) &&
- (Cursor_c >= c) && (c + 3 > Cursor_c))
- {
- s = buf;
- while (n--) *s++ = '\n';
- n = Cursor_c - c;
- while (n--) *s++ = '\b';
- *s = 0;
- s = buf;
- #ifdef SLTT_TRANSP_ACS_PATCH
- check_alt_acs = 1;
- #endif
- }
- #endif
- }
- }
- #ifndef SLTT_TRANSP_ACS_PATCH
- if (s != NULL) SLtt_write_string(s);
- #else
- if (s != NULL)
- {
- if (check_alt_acs && SLtt_ACS_Active)
- {
- SLtt_set_alt_char_set (0);
- SLtt_write_string(s);
- SLtt_set_alt_char_set (1);
- }
- else SLtt_write_string(s);
- }
- #endif
- else tt_printf(Curs_Pos_Str, r, c);
- Cursor_c = c; Cursor_r = r;
- Cursor_Set = 1;
- }
- void SLtt_begin_insert (void)
- {
- SLtt_write_string(Ins_Mode_Str);
- }
- void SLtt_end_insert (void)
- {
- SLtt_write_string(Eins_Mode_Str);
- }
- void SLtt_delete_char (void)
- {
- SLtt_normal_video ();
- SLtt_write_string(Del_Char_Str);
- }
- void SLtt_erase_line (void)
- {
- SLtt_write_string("\r");
- Cursor_Set = 1; Cursor_c = 0;
- SLtt_del_eol();
- }
- void SLtt_delete_nlines (int n)
- {
- int r1, curs;
- char buf[132];
- #ifdef SLTT_TRANSP_ACS_PATCH
- int restore_acs = 0;
- #endif
-
- if (n <= 0) return;
- SLtt_normal_video ();
- if (Del_N_Lines_Str != NULL) tt_printf(Del_N_Lines_Str,n, 0);
- else
- /* get a new terminal */
- {
- r1 = Scroll_r1;
- curs = Cursor_r;
- SLtt_set_scroll_region(curs, Scroll_r2);
- SLtt_goto_rc(Scroll_r2 - Scroll_r1, 0);
- #ifdef SLTT_TRANSP_ACS_PATCH
- if (SLtt_ACS_Active)
- {
- SLtt_set_alt_char_set (0);
- restore_acs = 1;
- }
- #endif
- SLMEMSET(buf, '\n', (unsigned int) n);
- tt_write(buf, (unsigned int) n);
- #ifdef SLTT_TRANSP_ACS_PATCH
- if (restore_acs) SLtt_set_alt_char_set (1);
- #endif
- /* while (n--) tt_putchar('\n'); */
- SLtt_set_scroll_region(r1, Scroll_r2);
- SLtt_goto_rc(curs, 0);
- }
- }
- void SLtt_cls (void)
- {
- SLtt_normal_video();
- SLtt_reset_scroll_region ();
- SLtt_write_string(Cls_Str);
- }
- void SLtt_reverse_index (int n)
- {
- if (!n) return;
-
- SLtt_normal_video();
- if (Add_N_Lines_Str != NULL) tt_printf(Add_N_Lines_Str,n, 0);
- else
- {
- while(n--) SLtt_write_string(Rev_Scroll_Str);
- }
- }
- int SLtt_Ignore_Beep = 1;
- static char *Visible_Bell_Str;
- void SLtt_beep (void)
- {
- if (SLtt_Ignore_Beep & 0x1) SLtt_putchar('\007');
-
- if (SLtt_Ignore_Beep & 0x2)
- {
- if (Visible_Bell_Str != NULL) SLtt_write_string (Visible_Bell_Str);
- #ifdef __linux__
- else if (Linux_Console)
- {
- SLtt_write_string ("\033[?5h");
- SLtt_flush_output ();
- sl_usleep (50000);
- SLtt_write_string ("\033[?5l");
- }
- #endif
- }
- SLtt_flush_output ();
- }
- void SLtt_del_eol (void)
- {
- if (Current_Fgbg != 0xFFFFFFFFU) SLtt_normal_video ();
- SLtt_write_string(Del_Eol_Str);
- }
- typedef const struct
- {
- char *name;
- SLtt_Char_Type color;
- }
- Color_Def_Type;
- #define MAX_COLOR_NAMES 17
- static Color_Def_Type Color_Defs [MAX_COLOR_NAMES] =
- {
- {"black", SLSMG_COLOR_BLACK},
- {"red", SLSMG_COLOR_RED},
- {"green", SLSMG_COLOR_GREEN},
- {"brown", SLSMG_COLOR_BROWN},
- {"blue", SLSMG_COLOR_BLUE},
- {"magenta", SLSMG_COLOR_MAGENTA},
- {"cyan", SLSMG_COLOR_CYAN},
- {"lightgray", SLSMG_COLOR_LGRAY},
- {"gray", SLSMG_COLOR_GRAY},
- {"brightred", SLSMG_COLOR_BRIGHT_RED},
- {"brightgreen", SLSMG_COLOR_BRIGHT_GREEN},
- {"yellow", SLSMG_COLOR_BRIGHT_BROWN},
- {"brightblue", SLSMG_COLOR_BRIGHT_BLUE},
- {"brightmagenta", SLSMG_COLOR_BRIGHT_CYAN},
- {"brightcyan", SLSMG_COLOR_BRIGHT_MAGENTA},
- {"white", SLSMG_COLOR_BRIGHT_WHITE},
- {"default", 25}
- };
- void SLtt_set_mono (int obj, char *what, SLtt_Char_Type mask)
- {
- (void) what;
- if ((obj < 0) || (obj >= JMAX_COLORS))
- {
- return;
- }
- Ansi_Color_Map[obj].mono = mask & ATTR_MASK;
- }
- static char *check_color_for_digit_form (char *color)
- {
- unsigned int i, ich;
- char *s = color;
-
- i = 0;
- while ((ich = (int) *s) != 0)
- {
- if ((ich < '0') || (ich > '9'))
- return color;
-
- i = i * 10 + (ich - '0');
- s++;
- }
-
- if (i < MAX_COLOR_NAMES)
- color = Color_Defs[i].name;
-
- return color;
- }
- static int get_default_colors (char **fgp, char **bgp)
- {
- static char fg_buf[16], bg_buf[16], *bg, *fg;
- static int already_parsed;
- char *p, *pmax;
-
- if (already_parsed == -1)
- return -1;
- if (already_parsed)
- {
- *fgp = fg;
- *bgp = bg;
- return 0;
- }
-
- already_parsed = -1;
-
- bg = getenv ("COLORFGBG");
- if (bg == NULL)
- {
- bg = getenv ("DEFAULT_COLORS");
- if (bg == NULL)
- return -1;
- }
-
- p = fg_buf;
- pmax = p + (sizeof (fg_buf) - 1);
-
- while ((*bg != 0) && (*bg != ';'))
- {
- if (p < pmax) *p++ = *bg;
- bg++;
- }
- *p = 0;
-
- if (*bg) bg++;
-
- p = bg_buf;
- pmax = p + (sizeof (bg_buf) - 1);
-
- /* Mark suggested allowing for extra spplication specific stuff following
- * the background color. That is what the check for the semi-colon is for.
- */
- while ((*bg != 0) && (*bg != ';'))
- {
- if (p < pmax) *p++ = *bg;
- bg++;
- }
- *p = 0;
- if (!strcmp (fg_buf, "default") || !strcmp(bg_buf, "default"))
- {
- *fgp = *bgp = fg = bg = "default";
- }
- else
- {
- *fgp = fg = check_color_for_digit_form (fg_buf);
- *bgp = bg = check_color_for_digit_form (bg_buf);
- }
- already_parsed = 1;
- return 0;
- }
- static unsigned char FgBg_Stats[JMAX_COLORS];
- static int Color_0_Modified = 0;
- void SLtt_set_color_object (int obj, SLtt_Char_Type attr)
- {
- char *cust_esc;
-
- if ((obj < 0) || (obj >= JMAX_COLORS)) return;
-
- cust_esc = Ansi_Color_Map[obj].custom_esc;
- if (cust_esc != NULL)
- {
- SLFREE (cust_esc);
- FgBg_Stats[(Ansi_Color_Map[obj].fgbg >> 8) & 0x7F] -= 1;
- Ansi_Color_Map[obj].custom_esc = NULL;
- }
-
- Ansi_Color_Map[obj].fgbg = attr;
- if (obj == 0) Color_0_Modified = 1;
- }
- SLtt_Char_Type SLtt_get_color_object (int obj)
- {
- if ((obj < 0) || (obj >= JMAX_COLORS)) return 0;
- return Ansi_Color_Map[obj].fgbg;
- }
- void SLtt_add_color_attribute (int obj, SLtt_Char_Type attr)
- {
- if ((obj < 0) || (obj >= JMAX_COLORS)) return;
-
- Ansi_Color_Map[obj].fgbg |= (attr & ATTR_MASK);
- if (obj == 0) Color_0_Modified = 1;
- }
- static SLtt_Char_Type fb_to_fgbg (SLtt_Char_Type f, SLtt_Char_Type b)
- {
- SLtt_Char_Type attr = 0;
- if ((f & 0xF0) == 0)
- {
- if (f & 0x8) attr = SLTT_BOLD_MASK;
- f &= 0x7;
- }
- else f = 9;
-
- if ((b & 0xF0) == 0)
- {
- if (b & 0x8) attr |= SLTT_BLINK_MASK;
- b &= 0x7;
- }
- else b = 9;
-
- return ((f << 8) | (b << 16) | attr);
- }
- static int make_color_fgbg (char *fg, char *bg, SLtt_Char_Type *fgbg)
- {
- SLtt_Char_Type f = 0xFFFFFFFFU, b = 0xFFFFFFFFU;
- char *dfg, *dbg;
- unsigned int i;
-
- if ((fg != NULL) && (*fg == 0)) fg = NULL;
- if ((bg != NULL) && (*bg == 0)) bg = NULL;
-
- if ((fg == NULL) || (bg == NULL))
- {
- if (-1 == get_default_colors (&dfg, &dbg))
- return -1;
-
- if (fg == NULL) fg = dfg;
- if (bg == NULL) bg = dbg;
- }
-
- for (i = 0; i < MAX_COLOR_NAMES; i++)
- {
- if (strcmp(fg, Color_Defs[i].name)) continue;
- f = Color_Defs[i].color;
- break;
- }
- for (i = 0; i < MAX_COLOR_NAMES; i++)
- {
- if (strcmp(bg, Color_Defs[i].name)) continue;
- b = Color_Defs[i].color;
- break;
- }
-
- if ((f == 0xFFFFFFFFU) || (b == 0xFFFFFFFFU))
- return -1;
-
- *fgbg = fb_to_fgbg (f, b);
- return 0;
- }
- void SLtt_set_color (int obj, char *what, char *fg, char *bg)
- {
- SLtt_Char_Type fgbg;
- (void) what;
- if ((obj < 0) || (obj >= JMAX_COLORS))
- return;
- if (-1 != make_color_fgbg (fg, bg, &fgbg))
- SLtt_set_color_object (obj, fgbg);
- }
- void SLtt_set_color_fgbg (int obj, SLtt_Char_Type f, SLtt_Char_Type b)
- {
- SLtt_set_color_object (obj, fb_to_fgbg (f, b));
- }
- void SLtt_set_color_esc (int obj, char *esc)
- {
- char *cust_esc;
- SLtt_Char_Type fgbg = 0;
- int i;
-
- if ((obj < 0) || (obj >= JMAX_COLORS))
- {
- return;
- }
-
- cust_esc = Ansi_Color_Map[obj].custom_esc;
- if (cust_esc != NULL)
- {
- SLFREE (cust_esc);
- FgBg_Stats[(Ansi_Color_Map[obj].fgbg >> 8) & 0x7F] -= 1;
- }
-
- cust_esc = (char *) SLMALLOC (strlen(esc) + 1);
- if (cust_esc != NULL) strcpy (cust_esc, esc);
-
- Ansi_Color_Map[obj].custom_esc = cust_esc;
- if (cust_esc == NULL) fgbg = 0;
- else
- {
- /* The whole point of this is to generate a unique fgbg */
- for (i = 0; i < JMAX_COLORS; i++)
- {
- if (FgBg_Stats[i] == 0) fgbg = i;
-
- if (obj == i) continue;
- if ((Ansi_Color_Map[i].custom_esc) == NULL) continue;
- if (!strcmp (Ansi_Color_Map[i].custom_esc, cust_esc))
- {
- fgbg = (Ansi_Color_Map[i].fgbg >> 8) & 0x7F;
- break;
- }
- }
- FgBg_Stats[fgbg] += 1;
- }
-
- fgbg |= 0x80;
- Ansi_Color_Map[obj].fgbg = (fgbg | (fgbg << 8)) << 8;
- if (obj == 0) Color_0_Modified = 1;
- }
- void SLtt_set_alt_char_set (int i)
- {
- #ifndef SLTT_TRANSP_ACS_PATCH
- static int last_i;
- #else
- #define last_i SLtt_ACS_Active
- #endif
- if (SLtt_Has_Alt_Charset == 0) return;
- if (i == last_i) return;
- SLtt_write_string (i ? Start_Alt_Chars_Str : End_Alt_Chars_Str );
- /* if (i) Current_Fgbg |= SLTT_ALTC_MASK;
- else Current_Fgbg &= ~SLTT_ALTC_MASK; */
- last_i = i;
- #ifdef SLTT_TRANSP_ACS_PATCH
- #undef last_i
- #endif
- }
- static void write_attributes (SLtt_Char_Type fgbg)
- {
- int bg0, fg0;
- if (Worthless_Highlight) return;
- if (fgbg == Current_Fgbg) return;
-
- /* Before spitting out colors, fix attributes */
- if ((fgbg & ATTR_MASK) != (Current_Fgbg & ATTR_MASK))
- {
- if (Current_Fgbg & ATTR_MASK)
- {
- SLtt_write_string(Norm_Vid_Str);
- /* In case normal video turns off ALL attributes: */
- if (fgbg & SLTT_ALTC_MASK)
- Current_Fgbg &= ~SLTT_ALTC_MASK;
- SLtt_set_alt_char_set (0);
- }
-
- if ((fgbg & SLTT_ALTC_MASK)
- != (Current_Fgbg & SLTT_ALTC_MASK))
- {
- SLtt_set_alt_char_set ((int) (fgbg & SLTT_ALTC_MASK));
- }
-
- if (fgbg & SLTT_ULINE_MASK) SLtt_write_string (UnderLine_Vid_Str);
- if (fgbg & SLTT_BOLD_MASK) SLtt_bold_video ();
- if (fgbg & SLTT_REV_MASK) SLtt_write_string (Rev_Vid_Str);
- if (fgbg & SLTT_BLINK_MASK)
- {
- /* Someday Linux will have a blink mode that set high intensity
- * background. Lets be prepared.
- */
- if (SLtt_Blink_Mode) SLtt_write_string (Blink_Vid_Str);
- }
- }
-
- if (SLtt_Use_Ansi_Colors)
- {
- fg0 = (int) GET_FG(fgbg);
- bg0 = (int) GET_BG(fgbg);
- tt_printf(Color_Escape_Sequence, fg0, bg0);
- }
- Current_Fgbg = fgbg;
- }
- static int Video_Initialized;
- void SLtt_reverse_video (int color)
- {
- SLtt_Char_Type fgbg;
- char *esc;
-
- if (Worthless_Highlight) return;
- if ((color < 0) || (color >= JMAX_COLORS)) return;
-
- if (Video_Initialized == 0)
- {
- if (color == JNORMAL_COLOR)
- {
- SLtt_write_string (Norm_Vid_Str);
- }
- else SLtt_write_string (Rev_Vid_Str);
- Current_Fgbg = 0xFFFFFFFFU;
- return;
- }
-
- if (SLtt_Use_Ansi_Colors)
- {
- fgbg = Ansi_Color_Map[color].fgbg;
- if ((esc = Ansi_Color_Map[color].custom_esc) != NULL)
- {
- if (fgbg != Current_Fgbg)
- {
- Current_Fgbg = fgbg;
- SLtt_write_string (esc);
- return;
- }
- }
- }
- else fgbg = Ansi_Color_Map[color].mono;
- if (fgbg == Current_Fgbg) return;
- write_attributes (fgbg);
- }
- void SLtt_normal_video (void)
- {
- SLtt_reverse_video(JNORMAL_COLOR);
- }
- void SLtt_narrow_width (void)
- {
- SLtt_write_string("\033[?3l");
- }
- void SLtt_wide_width (void)
- {
- SLtt_write_string("\033[?3h");
- }
- /* Highest bit represents the character set. */
- #define COLOR_MASK 0x7F00
- #define COLOR_OF(x) (((unsigned int)(x) & COLOR_MASK) >> 8)
- /*
- #define COLOR_EQS(a, b) \
- (Ansi_Color_Map[COLOR_OF(a)].fgbg == Ansi_Color_Map[COLOR_OF(b)].fgbg)
- */
- #define COLOR_EQS(a, b) \
- (SLtt_Use_Ansi_Colors \
- ? (Ansi_Color_Map[COLOR_OF(a)].fgbg == Ansi_Color_Map[COLOR_OF(b)].fgbg)\
- : (Ansi_Color_Map[COLOR_OF(a)].mono == Ansi_Color_Map[COLOR_OF(b)].mono))
- #define CHAR_EQS(a, b) (((a) == (b))\
- || ((((a) & ~COLOR_MASK) == ((b) & ~COLOR_MASK))\
- && COLOR_EQS((a), (b))))
- /* The whole point of this routine is to prevent writing to the last column
- * and last row on terminals with automatic margins.
- */
- static void write_string_with_care (char *str)
- {
- unsigned int len;
-
- if (str == NULL) return;
-
- len = strlen (str);
- if (Automatic_Margins && (Cursor_r + 1 == SLtt_Screen_Rows))
- {
- if (len + (unsigned int) Cursor_c >= (unsigned int) SLtt_Screen_Cols)
- {
- /* For now, just do not write there. Later, something more
- * sophisticated will be implemented.
- */
- if (SLtt_Screen_Cols > Cursor_c)
- len = SLtt_Screen_Cols - Cursor_c - 1;
- else len = 0;
- }
- }
- tt_write (str, len);
- }
- static void send_attr_str (unsigned short *s)
- {
- unsigned char out[256], ch, *p;
- register SLtt_Char_Type attr;
- register unsigned short sh;
- int color, last_color = -1;
-
- p = out;
- while (0 != (sh = *s++))
- {
- ch = sh & 0xFF;
- color = ((int) sh & 0xFF00) >> 8;
- #ifdef SLTT_TRANSP_ACS_PATCH
- if (ch <= ' ' && (color & 0x80)) color &= ~0x80;
- #endif
- if (color != last_color)
- {
- if (SLtt_Use_Ansi_Colors) attr = Ansi_Color_Map[color & 0x7F].fgbg;
- else attr = Ansi_Color_Map[color & 0x7F].mono;
-
- /* sh => color */
- if (color & 0x80) /* alternate char set */
- {
- if (SLtt_Use_Blink_For_ACS)
- {
- if (SLtt_Blink_Mode) attr |= SLTT_BLINK_MASK;
- }
- else attr |= SLTT_ALTC_MASK;
- }
-
-
- if (attr != Current_Fgbg)
- {
- #ifndef SLTT_TRANSP_ACS_PATCH
- if ((ch != ' ') ||
- /* it is a space so only consider it different if it
- * has different attributes.
- */
- (attr & BGALL_MASK) != (Current_Fgbg & BGALL_MASK))
- #endif
- {
- if (p != out)
- {
- *p = 0;
- write_string_with_care ((char *) out);
- Cursor_c += (int) (p - out);
- p = out;
- }
- if (SLtt_Use_Ansi_Colors && (NULL != Ansi_Color_Map[color & 0x7F].custom_esc))
- {
- SLtt_write_string (Ansi_Color_Map[color & 0x7F].custom_esc);
- /* Just in case the custom escape sequence screwed up
- * the alt character set state...
- */
- if ((attr & SLTT_ALTC_MASK) != (Current_Fgbg & SLTT_ALTC_MASK))
- SLtt_set_alt_char_set ((int) (attr & SLTT_ALTC_MASK));
- Current_Fgbg = attr;
- }
- else write_attributes (attr);
-
- last_color = color;
- }
- }
- }
- *p++ = ch;
- }
- *p = 0;
- if (p != out) write_string_with_care ((char *) out);
- Cursor_c += (int) (p - out);
- }
- static void forward_cursor (unsigned int n, int row)
- {
- char buf[30];
-
-
- if (n <= 4)
- {
- SLtt_normal_video ();
- SLMEMSET (buf, ' ', n);
- buf[n] = 0;
- write_string_with_care (buf);
- Cursor_c += n;
- }
- else if (Curs_F_Str != NULL)
- {
- Cursor_c += n;
- n = tt_sprintf(buf, Curs_F_Str, (int) n, 0);
- tt_write(buf, n);
- }
- else SLtt_goto_rc (row, (int) (Cursor_c + n));
- }
- #define SPACE_CHAR (0x20 | (JNORMAL_COLOR << 8))
- void SLtt_smart_puts(unsigned short *neww, unsigned short *oldd, int len, int row)
- {
- register unsigned short *p, *q, *qmax, *pmax, *buf;
- unsigned short buffer[256];
- unsigned int n_spaces;
- unsigned short *space_match, *last_buffered_match;
- #ifdef HP_GLITCH_CODE
- int handle_hp_glitch = 0;
- #endif
-
- q = oldd; p = neww;
- qmax = oldd + len;
- pmax = p + len;
-
- /* Find out where to begin --- while they match, we are ok */
- while (1)
- {
- if (q == qmax) return;
- #if SLANG_HAS_KANJI_SUPPORT
- if (*p & 0x80)
- { /* new is kanji */
- if ((*q & 0x80) && ((q + 1) < qmax))
- { /* old is also kanji */
- if (((0xFF & *q) != (0xFF & *p))
- || ((0xFF & q[1]) != (0xFF & p[1])))
- break; /* both kanji, but not match */
-
- else
- { /* kanji match ! */
- if (!COLOR_EQS(*q, *p)) break;
- q++; p++;
- if (!COLOR_EQS(*q, *p)) break;
- /* really match! */
- q++; p++;
- continue;
- }
- }
- else break; /* old is not kanji */
- }
- else
- { /* new is not kanji */
- if (*q & 0x80) break; /* old is kanji */
- }
- #endif
- if (!CHAR_EQS(*q, *p)) break;
- q++; p++;
- }
- /*position the cursor */
- SLtt_goto_rc (row, (int) (p - neww));
- #ifdef HP_GLITCH_CODE
- if (Has_HP_Glitch)
- {
- unsigned short *qq = q;
- while (qq < qmax)
- {
- if (*qq & 0xFF00)
- {
- SLtt_normal_video ();
- SLtt_del_eol ();
- qmax = q;
- handle_hp_glitch = 1;
- break;
- }
- qq++;
- }
- }
- #endif
- /* Find where the last non-blank character on old/new screen is */
-
- while (qmax > q)
- {
- qmax--;
- if (!CHAR_EQS(*qmax, SPACE_CHAR))
- {
- qmax++;
- break;
- }
- }
-
- while (pmax > p)
- {
- pmax--;
- if (!CHAR_EQS(*pmax, SPACE_CHAR))
- {
- pmax++;
- break;
- }
- }
-
- last_buffered_match = buf = buffer; /* buffer is empty */
- #ifdef HP_GLITCH_CODE
- if (handle_hp_glitch)
- {
- while (p < pmax)
- {
- *buf++ = *p++;
- }
- }
- #endif
- /* loop using overwrite then skip algorithm until done */
- while (1)
- {
- /* while they do not match and we do not hit a space, buffer them up */
- n_spaces = 0;
- while (p < pmax)
- {
- if (CHAR_EQS(*q,SPACE_CHAR) && CHAR_EQS(*p, SPACE_CHAR))
- {
- /* If *q is not a space, we would have to overwrite it.
- * However, if *q is a space, then while *p is also one,
- * we only need to skip over the blank field.
- */
- space_match = p;
- p++; q++;
- while ((p < pmax)
- && CHAR_EQS(*q,SPACE_CHAR)
- && CHAR_EQS(*p, SPACE_CHAR))
- {
- p++;
- q++;
- }
- n_spaces = (unsigned int) (p - space_match);
- break;
- }
- #if SLANG_HAS_KANJI_SUPPORT
- if ((*p & 0x80) && ((p + 1) < pmax))
- { /* new is kanji */
- if (*q & 0x80)
- { /* old is also kanji */
- if (((0xFF & *q) != (0xFF & *p))
- || ((0xFF & q[1]) != (0xFF & p[1])))
- {
- /* both kanji, but not match */
- *buf++ = *p++;
- *buf++ = *p++;
- q += 2;
- continue;
- }
- else
- { /* kanji match ? */
- if (!COLOR_EQS(*q, *p) || !COLOR_EQS(*(q+1), *(p+1)))
- {
- /* code is match ,but color is diff */
- *buf++ = *p++;
- *buf++ = *p++;
- continue;
- }
- /* really match ! */
- break;
- }
- }
- else
- { /* old is not kanji */
- *buf++ = *p++;
- *buf++ = *p++;
- q += 2;
- continue;
- }
- }
- else
- { /* new is not kanji */
- if (*q & 0x80)
- { /* old is kanji */
- *buf++ = *p++;
- q++;
- continue;
- }
- }
- #endif
- if (CHAR_EQS(*q, *p)) break;
- *buf++ = *p++;
- q++;
- }
- *buf = 0;
-
- if (buf != buffer) send_attr_str (buffer);
- buf = buffer;
-
- if (n_spaces && (p < pmax))
- {
- forward_cursor (n_spaces, row);
- }
-
- /* Now we overwrote what we could and cursor is placed at position
- * of a possible match of new and old. If this is the case, skip
- * some more.
- */
- #if !SLANG_HAS_KANJI_SUPPORT
- while ((p < pmax) && CHAR_EQS(*p, *q))
- {
- *buf++ = *p++;
- q++;
- }
- #else
- /* Kanji */
- while (p < pmax)
- {
- if ((*p & 0x80) && ((p + 1) < pmax))
- { /* new is kanji */
- if (*q & 0x80)
- { /* old is also kanji */
- if (((0xFF & *q) == (0xFF & *p))
- && ((0xFF & q[1]) == (0xFF & p[1])))
- {
- /* kanji match ? */
- if (!COLOR_EQS(*q, *p)
- || !COLOR_EQS(q[1], p[1]))
- break;
-
- *buf++ = *p++;
- q++;
- if (p >= pmax)
- {
- *buf++ = SPACE_CHAR;
- p++;
- break;
- }
- else
- {
- *buf++ = *p++;
- q++;
- continue;
- }
- }
- else break; /* both kanji, but not match */
- }
- else break; /* old is not kanji */
- }
- else
- { /* new is not kanji */
- if (*q & 0x80) break; /* old is kanji */
- if (!CHAR_EQS(*q, *p)) break;
- *buf++ = *p++;
- q++;
- }
- }
- #endif
- last_buffered_match = buf;
- if (p >= pmax) break;
-
- /* jump to new position is it is greater than 5 otherwise
- * let it sit in the buffer and output it later.
- */
- if ((int) (buf - buffer) >= 5)
- {
- forward_cursor ((unsigned int) (buf - buffer), row);
- last_buffered_match = buf = buffer;
- }
- }
- if (buf != buffer)
- {
- if (q < qmax)
- {
- if ((buf == last_buffered_match)
- && ((int) (buf - buffer) >= 5))
- {
- forward_cursor ((unsigned int) (buf - buffer), row);
- }
- else
- {
- *buf = 0;
- send_attr_str (buffer);
- }
- }
- }
- if (q < qmax) SLtt_del_eol ();
- if (Automatic_Margins && (Cursor_c + 1 >= SLtt_Screen_Cols)) Cursor_Set = 0;
- }
- static void get_color_info (void)
- {
- char *fg, *bg;
-
- SLtt_Use_Ansi_Colors = (NULL != getenv ("COLORTERM"));
-
- if (-1 == get_default_colors (&fg, &bg))
- return;
-
- /* Check to see if application has already set them. */
- if (Color_0_Modified)
- return;
-
- SLtt_set_color (0, NULL, fg, bg);
- SLtt_set_color (1, NULL, bg, fg);
- }
- /* termcap stuff */
- #ifdef __unix__
- #ifndef USE_TERMCAP
- static char *Tbuf;
- static char *Tstr_Buf;
- #define tgetstr SLtt_tigetstr
- #define tgetent SLtt_tigetent
- #define TGETNUM(x) SLtt_tigetnum((x), &Tbuf)
- #define TGETFLAG(x) SLtt_tigetflag((x), &Tbuf)
- #else
- extern char *tgetstr(char *, char **);
- extern int tgetent(char *, char *);
- extern int tgetnum(char *);
- extern int tgetflag(char *);
- static char Tstr_Buf[1024];
- static char Tbuf[4096];
- #define TGETNUM tgetnum
- #define TGETFLAG tgetflag
- #endif
- static char *my_tgetstr(char *what, char **p)
- {
- register char *w, *w1;
- char *wsave;
- what = tgetstr(what, p);
- if (what != NULL)
- {
- /* Check for AIX brain-damage */
- if (*what == '@')
- return NULL;
-
- /* lose pad info --- with today's technology, term is a loser if
- it is really needed */
- while ((*what == '.') ||
- ((*what >= '0') && (*what <= '9'))) what++;
- if (*what == '*') what++;
-
- /* lose terminfo padding--- looks like $<...> */
- w = what;
- while (*w) if ((*w++ == '$') && (*w == '<'))
- {
- w1 = w - 1;
- while (*w && (*w != '>')) w++;
- if (*w == 0) break;
- w++;
- wsave = w1;
- while ((*w1++ = *w++) != 0);
- w = wsave;
- }
- if (*what == 0) what = NULL;
- }
- return(what);
- }
- char *SLtt_tgetstr (char *s)
- {
- #ifdef USE_TERMCAP
- static
- #endif
- char *p = Tstr_Buf;
- return my_tgetstr (s, &p);
- }
- int SLtt_tgetnum (char *s)
- {
- return TGETNUM (s);
- }
- int SLtt_tgetflag (char *s)
- {
- return TGETFLAG (s);
- }
- static int Vt100_Like = 0;
- void SLtt_get_terminfo (void)
- {
- char *term, *t, ch;
- int is_xterm;
- int almost_vtxxx;
-
- get_color_info ();
-
- if (NULL == (term = (char *) getenv("TERM")))
- {
- SLang_exit_error("TERM environment variable needs set.");
- }
-
- Linux_Console = (!strncmp (term, "linux", 5)
- #ifdef linux
- || !strncmp(term, "con", 3)
- #endif
- );
-
- t = term;
-
- if (strcmp(t, "vt52") && (*t++ == 'v') && (*t++ == 't')
- && (ch = *t, (ch >= '1') && (ch <= '9'))) Vt100_Like = 1;
- is_xterm = !strncmp (term, "xterm", 5);
- almost_vtxxx = (Vt100_Like
- || Linux_Console
- || is_xterm
- || !strcmp (term, "screen"));
-
- #ifndef USE_TERMCAP
- if (NULL == (Tbuf = tgetent (term)))
- {
- char err_buf[512];
- if (almost_vtxxx) /* Special cases. */
- {
- int vt102 = 1;
- if (!strcmp (term, "vt100")) vt102 = 0;
- SLtt_set_term_vtxxx (&vt102);
- return;
- }
- g_snprintf (err_buf, sizeof (err_buf), "Unknown terminal: %s\n\
- Check the TERM environment variable.\n\
- Also make sure that the terminal is defined in the terminfo database.\n\
- Alternatively, set the TERMCAP environment variable to the desired\n\
- termcap entry.", term);
- SLang_exit_error(err_buf);
- }
- Tstr_Buf = Tbuf;
- #else /* USE_TERMCAP */
- if (1 != tgetent(Tbuf, term)) SLang_exit_error("Unknown terminal.");
- #endif /* NOT USE_TERMCAP */
-
- if ((NULL == (Cls_Str = SLtt_tgetstr("cl")))
- || (NULL == (Curs_Pos_Str = SLtt_tgetstr("cm"))))
- {
- SLang_exit_error("Terminal not powerful enough for SLang.");
- }
-
- if ((NULL == (Ins_Mode_Str = SLtt_tgetstr("im")))
- || ( NULL == (Eins_Mode_Str = SLtt_tgetstr("ei")))
- || ( NULL == (Del_Char_Str = SLtt_tgetstr("dc"))))
- SLtt_Term_Cannot_Insert = 1;
-
- Visible_Bell_Str = SLtt_tgetstr ("vb");
- Curs_Up_Str = SLtt_tgetstr ("up");
- Rev_Scroll_Str = SLtt_tgetstr("sr");
- Del_N_Lines_Str = SLtt_tgetstr("DL");
- Add_N_Lines_Str = SLtt_tgetstr("AL");
-
- /* Actually these are used to initialize terminals that use cursor
- * addressing. Hard to believe.
- */
- Term_Init_Str = SLtt_tgetstr ("ti");
- Term_Reset_Str = SLtt_tgetstr ("te");
- /* If I do this for vtxxx terminals, arrow keys start sending ESC O A,
- * which I do not want. This is mainly for HP terminals.
- */
- if ((almost_vtxxx == 0) || SLtt_Force_Keypad_Init)
- {
- Keypad_Init_Str = SLtt_tgetstr ("ks");
- Keypad_Reset_Str = SLtt_tgetstr ("ke");
- }
- /* Make up for defective termcap/terminfo databases */
- if ((Vt100_Like && (term[2] != '1'))
- || Linux_Console
- || is_xterm
- )
- {
- if (Del_N_Lines_Str == NULL) Del_N_Lines_Str = "\033[%dM";
- if (Add_N_Lines_Str == NULL) Add_N_Lines_Str = "\033[%dL";
- }
-
- Scroll_R_Str = SLtt_tgetstr("cs");
-
- SLtt_get_screen_size ();
- if ((Scroll_R_Str == NULL)
- || (((NULL == Del_N_Lines_Str) || (NULL == Add_N_Lines_Str))
- && (NULL == Rev_Scroll_Str)))
- {
- if (is_xterm
- || Linux_Console
- )
- {
- /* Defective termcap mode!!!! */
- SLtt_set_term_vtxxx (NULL);
- }
- else SLtt_Term_Cannot_Scroll = 1;
- }
-
- Del_Eol_Str = SLtt_tgetstr("ce");
- Rev_Vid_Str = SLtt_tgetstr("mr");
- if (Rev_Vid_Str == NULL) Rev_Vid_Str = SLtt_tgetstr("so");
-
- Bold_Vid_Str = SLtt_tgetstr("md");
- /* Although xterm cannot blink, it does display the blinking characters
- * as bold ones. Some Rxvt will display the background as high intensity.
- */
- if ((NULL == (Blink_Vid_Str = SLtt_tgetstr("mb")))
- && is_xterm)
- Blink_Vid_Str = "\033[5m";
-
- UnderLine_Vid_Str = SLtt_tgetstr("us");
-
- Start_Alt_Chars_Str = SLtt_tgetstr ("as"); /* smacs */
- End_Alt_Chars_Str = SLtt_tgetstr ("ae"); /* rmacs */
- Enable_Alt_Char_Set = SLtt_tgetstr ("eA"); /* enacs */
- SLtt_Graphics_Char_Pairs = SLtt_tgetstr ("ac");
- #ifndef NCURSES_BRAIN_DAMAGE_CONTROL
- # define NCURSES_BRAIN_DAMAGE_CONTROL 0
- #endif
-
- #if NCURSES_BRAIN_DAMAGE_CONTROL
- if (Linux_Console)
- {
- # if 0
- char *lgcp = "l\332m\300k\277j\331u\264t\303v\301w\302q\304x\263n\053o\176s\137`\004a\260f\370g\361~\011,\020+\021.\031-\030h\261i\0250\333";
- SLtt_Graphics_Char_Pairs = lgcp;
- Start_Alt_Chars_Str = "\033(B\033)U\016";
- End_Alt_Chars_Str = "\033(B\033)0\017";
- Enable_Alt_Char_Set = NULL;
- # else
- char *lgcp = "`\004a\261f\370g\361h\260j\331k\277l\332m\300n\305o\302q\304r\362s_t\303u\264v\301w\302x\263y\371z\372{\373|\374}\375~";
-
- SLtt_Graphics_Char_Pairs = lgcp;
- Start_Alt_Chars_Str = "\033[11m";
- End_Alt_Chars_Str = "\033[10m";
- Enable_Alt_Char_Set = NULL;
- # endif
- }
- #endif
-
- if (NULL == SLtt_Graphics_Char_Pairs)
- {
- /* make up for defective termcap/terminfo */
- if (Vt100_Like)
- {
- Start_Alt_Chars_Str = "\016";
- End_Alt_Chars_Str = "\017";
- Enable_Alt_Char_Set = "\033)0";
- }
- }
-
- /* aixterm added by willi */
- if (is_xterm || !strncmp (term, "aixterm", 7))
- {
- Start_Alt_Chars_Str = "\016";
- End_Alt_Chars_Str = "\017";
- Enable_Alt_Char_Set = "\033(B\033)0";
- }
-
- if ((SLtt_Graphics_Char_Pairs == NULL) &&
- ((Start_Alt_Chars_Str == NULL) || (End_Alt_Chars_Str == NULL)))
- {
- SLtt_Has_Alt_Charset = 0;
- Enable_Alt_Char_Set = NULL;
- }
- else SLtt_Has_Alt_Charset = 1;
-
- /* status line capabilities */
- if ((SLtt_Has_Status_Line == -1)
- && (0 != (SLtt_Has_Status_Line = TGETFLAG ("hs"))))
- {
- Disable_Status_line_Str = SLtt_tgetstr ("ds");
- Return_From_Status_Line_Str = SLtt_tgetstr ("fs");
- Goto_Status_Line_Str = SLtt_tgetstr ("ts");
- Status_Line_Esc_Ok = TGETFLAG("es");
- Num_Status_Line_Columns = TGETNUM("ws");
- if (Num_Status_Line_Columns < 0) Num_Status_Line_Columns = 0;
- }
-
- if (NULL == (Norm_Vid_Str = SLtt_tgetstr("me")))
- {
- Norm_Vid_Str = SLtt_tgetstr("se");
- }
-
- Cursor_Invisible_Str = SLtt_tgetstr("vi");
- Cursor_Visible_Str = SLtt_tgetstr("ve");
-
- Curs_F_Str = SLtt_tgetstr("RI");
-
- #if 0
- if (NULL != Curs_F_Str)
- {
- Len_Curs_F_Str = strlen(Curs_F_Str);
- }
- else Len_Curs_F_Str = strlen(Curs_Pos_Str);
- #endif
-
- Automatic_Margins = TGETFLAG ("am");
- /* No_Move_In_Standout = !TGETFLAG ("ms"); */
- #ifdef HP_GLITCH_CODE
- Has_HP_Glitch = TGETFLAG ("xs");
- #else
- Worthless_Highlight = TGETFLAG ("xs");
- #endif
- if (Worthless_Highlight == 0)
- { /* Magic cookie glitch */
- Worthless_Highlight = (TGETNUM ("sg") > 0);
- }
- if (Worthless_Highlight)
- SLtt_Has_Alt_Charset = 0;
-
- /* Check for color information in the termcap. A program should not
- * rely on this information being accurate.
- */
- if (SLtt_Use_Ansi_Colors == 0)
- {
- Reset_Color_String = SLtt_tgetstr ("op");
-
- SLtt_Use_Ansi_Colors = ((NULL != Reset_Color_String)
- || (NULL != SLtt_tgetstr ("Sf"))
- || (NULL != SLtt_tgetstr ("Sb"))
- || (NULL != SLtt_tgetstr ("AF"))
- || (NULL != SLtt_tgetstr ("AB"))
- || (-1 != SLtt_tgetnum ("Co"))
- || (-1 != SLtt_tgetnum ("pa")));
-
- }
-
- #if defined(__QNX__) && defined(QNX_QANSI_SLANG_COMPAT_ACS)
- /*
- * Override the alt-char-set handling string in case of a
- * QNX/qansi terminal: use the "old style" strings in order
- * to be compatible with S-Lang without the SLTT_TRANSP_ACS_PATCH
- * code...
- */
- if (SLtt_Has_Alt_Charset &&
- strncmp(term, "qansi", 5) == 0 &&
- Start_Alt_Chars_Str[0] != '\016')
- {
- Start_Alt_Chars_Str = "\016"; /* smacs/as (^N) */
- End_Alt_Chars_Str = "\017"; /* rmacs/ae (^O) */
- SLtt_Graphics_Char_Pairs = /* acsc/ac */
- "``aaffggjjkkllmmnnooppqqrrssttuuvvwwxxyyzz{{||}}~~O\141";
-
- /*
- * it would be required to modify the sgr/sa entry also, if it
- * would be used (->embedded as/ae sequences)...
- */
- }
- #endif /* __QNX__ && QNX_QANSI_SLANG_COMPAT_ACS */
- }
- #endif
- /* Unix */
- /* specific to vtxxx only */
- void SLtt_enable_cursor_keys (void)
- {
- #ifdef __unix__
- if (Vt100_Like)
- #endif
- SLtt_write_string("\033=\033[?1l");
- }
- #ifdef VMS
- void SLtt_get_terminfo ()
- {
- int zero = 0;
-
- get_color_info ();
- SLtt_set_term_vtxxx(&zero);
- Start_Alt_Chars_Str = "\016";
- End_Alt_Chars_Str = "\017";
- SLtt_Has_Alt_Charset = 1;
- SLtt_Graphics_Char_Pairs = "aaffgghhjjkkllmmnnooqqssttuuvvwwxx";
- Enable_Alt_Char_Set = "\033(B\033)0";
- SLtt_get_screen_size ();
- }
- #endif
- /* This sets term for vt102 terminals it parameter vt100 is 0. If vt100
- * is non-zero, set terminal appropriate for a only vt100
- * (no add line capability). */
-
- void SLtt_set_term_vtxxx(int *vt100)
- {
- Norm_Vid_Str = "\033[m";
-
- Scroll_R_Str = "\033[%i%d;%dr";
- Cls_Str = "\033[2J\033[H";
- Rev_Vid_Str = "\033[7m";
- Bold_Vid_Str = "\033[1m";
- Blink_Vid_Str = "\033[5m";
- UnderLine_Vid_Str = "\033[4m";
- Del_Eol_Str = "\033[K";
- Rev_Scroll_Str = "\033M";
- Curs_F_Str = "\033[%dC";
- /* Len_Curs_F_Str = 5; */
- Curs_Pos_Str = "\033[%i%d;%dH";
- if ((vt100 == NULL) || (*vt100 == 0))
- {
- Ins_Mode_Str = "\033[4h";
- Eins_Mode_Str = "\033[4l";
- Del_Char_Str = "\033[P";
- Del_N_Lines_Str = "\033[%dM";
- Add_N_Lines_Str = "\033[%dL";
- SLtt_Term_Cannot_Insert = 0;
- }
- else
- {
- Del_N_Lines_Str = NULL;
- Add_N_Lines_Str = NULL;
- SLtt_Term_Cannot_Insert = 1;
- }
- SLtt_Term_Cannot_Scroll = 0;
- /* No_Move_In_Standout = 0; */
- }
- void SLtt_init_video (void)
- {
- /* send_string_to_term("\033[?6h"); */
- /* relative origin mode */
- SLtt_write_string (Term_Init_Str);
- SLtt_write_string (Keypad_Init_Str);
- SLtt_reset_scroll_region();
- SLtt_end_insert();
- SLtt_write_string (Enable_Alt_Char_Set);
- Video_Initialized = 1;
- }
- void SLtt_reset_video (void)
- {
- SLtt_goto_rc (SLtt_Screen_Rows - 1, 0);
- Cursor_Set = 0;
- SLtt_normal_video (); /* MSKermit requires this */
- SLtt_write_string(Norm_Vid_Str);
- Current_Fgbg = 0xFFFFFFFFU;
- SLtt_set_alt_char_set (0);
- if (SLtt_Use_Ansi_Colors)
- {
- if (Reset_Color_String == NULL)
- {
- SLtt_Char_Type attr;
- if (-1 != make_color_fgbg (NULL, NULL, &attr))
- write_attributes (attr);
- else SLtt_write_string ("\033[0m\033[m");
- }
- else SLtt_write_string (Reset_Color_String);
- Current_Fgbg = 0xFFFFFFFFU;
- }
- SLtt_erase_line ();
- SLtt_write_string (Keypad_Reset_Str);
- SLtt_write_string (Term_Reset_Str);
- SLtt_flush_output ();
- Video_Initialized = 0;
- }
- void SLtt_bold_video (void)
- {
- SLtt_write_string (Bold_Vid_Str);
- }
- int SLtt_set_mouse_mode (int mode, int force)
- {
- char *term;
-
- if (force == 0)
- {
- if (NULL == (term = (char *) getenv("TERM"))) return -1;
- if (strncmp ("xterm", term, 5))
- return -1;
- }
- if (mode)
- SLtt_write_string ("\033[?9h");
- else
- SLtt_write_string ("\033[?9l");
-
- return 0;
- }
- void SLtt_disable_status_line (void)
- {
- if (SLtt_Has_Status_Line > 0)
- SLtt_write_string (Disable_Status_line_Str);
- }
- int SLtt_write_to_status_line (char *s, int col)
- {
- if ((SLtt_Has_Status_Line <= 0)
- || (Goto_Status_Line_Str == NULL)
- || (Return_From_Status_Line_Str == NULL))
- return -1;
-
- tt_printf (Goto_Status_Line_Str, col, 0);
- SLtt_write_string (s);
- SLtt_write_string (Return_From_Status_Line_Str);
- return 0;
- }
-
- void SLtt_get_screen_size (void)
- {
- #ifdef VMS
- int status, code;
- unsigned short chan;
- $DESCRIPTOR(dev_dsc, "SYS$INPUT:");
- #endif
- #ifdef __os2__
- VIOMODEINFO vioModeInfo;
- #endif
- int r = 0, c = 0;
-
- #if defined(TIOCGWINSZ) && !defined(SCO_FLAVOR)
- struct winsize wind_struct;
-
- do
- {
- if ((ioctl(1,TIOCGWINSZ,&wind_struct) == 0)
- || (ioctl(0, TIOCGWINSZ, &wind_struct) == 0)
- || (ioctl(2, TIOCGWINSZ, &wind_struct) == 0))
- {
- c = (int) wind_struct.ws_col;
- r = (int) wind_struct.ws_row;
- break;
- }
- }
- while (errno == EINTR);
-
- #endif
- #ifdef VMS
- status = sys$assign(&dev_dsc,&chan,0,0,0);
- if (status & 1)
- {
- code = DVI$_DEVBUFSIZ;
- status = lib$getdvi(&code, &chan,0, &c, 0,0);
- if (!(status & 1))
- c = 80;
- code = DVI$_TT_PAGE;
- status = lib$getdvi(&code, &chan,0, &r, 0,0);
- if (!(status & 1))
- r = 24;
- sys$dassgn(chan);
- }
- #endif
- #ifdef __os2__
- vioModeInfo.cb = sizeof(vioModeInfo);
- VioGetMode (&vioModeInfo, 0);
- c = vioModeInfo.col;
- r = vioModeInfo.row;
- #endif
-
- if (r <= 0)
- {
- char *s = getenv ("LINES");
- if (s != NULL) r = atoi (s);
- }
-
- if (c <= 0)
- {
- char *s = getenv ("COLUMNS");
- if (s != NULL) c = atoi (s);
- }
-
- if ((r <= 0) || (r > 200)) r = 24;
- if ((c <= 0) || (c > 250)) c = 80;
- SLtt_Screen_Rows = r;
- SLtt_Screen_Cols = c;
- }
|