From 1a61e0c697ba9407456e223c2af7e36a623d68a6 Mon Sep 17 00:00:00 2001 From: WanderingPenwing Date: Mon, 29 Jul 2024 12:57:21 +0200 Subject: [PATCH] cleaned up --- FAQ | 253 ---- LEGACY | 17 - TODO | 28 - st | Bin 96768 -> 0 bytes st-scrollback-0.8.4.diff | 351 ----- st.c.orig | 2678 -------------------------------------- st.c.rej | 13 - x.o | Bin 75720 -> 75304 bytes 8 files changed, 3340 deletions(-) delete mode 100644 FAQ delete mode 100644 LEGACY delete mode 100644 TODO delete mode 100755 st delete mode 100644 st-scrollback-0.8.4.diff delete mode 100644 st.c.orig delete mode 100644 st.c.rej diff --git a/FAQ b/FAQ deleted file mode 100644 index 6287a27..0000000 --- a/FAQ +++ /dev/null @@ -1,253 +0,0 @@ -## Why does st not handle utmp entries? - -Use the excellent tool of [utmp](https://git.suckless.org/utmp/) for this task. - - -## Some _random program_ complains that st is unknown/not recognised/unsupported/whatever! - -It means that st doesn’t have any terminfo entry on your system. Chances are -you did not `make install`. If you just want to test it without installing it, -you can manually run `tic -sx st.info`. - - -## Nothing works, and nothing is said about an unknown terminal! - -* Some programs just assume they’re running in xterm i.e. they don’t rely on - terminfo. What you see is the current state of the “xterm compliance”. -* Some programs don’t complain about the lacking st description and default to - another terminal. In that case see the question about terminfo. - - -## How do I scroll back up? - -* Using a terminal multiplexer. - * `st -e tmux` using C-b [ - * `st -e screen` using C-a ESC -* Using the excellent tool of [scroll](https://git.suckless.org/scroll/). -* Using the scrollback [patch](https://st.suckless.org/patches/scrollback/). - - -## I would like to have utmp and/or scroll functionality by default - -You can add the absolute path of both programs in your config.h file. You only -have to modify the value of utmp and scroll variables. - - -## Why doesn't the Del key work in some programs? - -Taken from the terminfo manpage: - - If the terminal has a keypad that transmits codes when the keys - are pressed, this information can be given. Note that it is not - possible to handle terminals where the keypad only works in - local (this applies, for example, to the unshifted HP 2621 keys). - If the keypad can be set to transmit or not transmit, give these - codes as smkx and rmkx. Otherwise the keypad is assumed to - always transmit. - -In the st case smkx=E[?1hE= and rmkx=E[?1lE>, so it is mandatory that -applications which want to test against keypad keys send these -sequences. - -But buggy applications (like bash and irssi, for example) don't do this. A fast -solution for them is to use the following command: - - $ printf '\033[?1h\033=' >/dev/tty - -or - $ tput smkx - -In the case of bash, readline is used. Readline has a different note in its -manpage about this issue: - - enable-keypad (Off) - When set to On, readline will try to enable the - application keypad when it is called. Some systems - need this to enable arrow keys. - -Adding this option to your .inputrc will fix the keypad problem for all -applications using readline. - -If you are using zsh, then read the zsh FAQ -: - - It should be noted that the O / [ confusion can occur with other keys - such as Home and End. Some systems let you query the key sequences - sent by these keys from the system's terminal database, terminfo. - Unfortunately, the key sequences given there typically apply to the - mode that is not the one zsh uses by default (it's the "application" - mode rather than the "raw" mode). Explaining the use of terminfo is - outside of the scope of this FAQ, but if you wish to use the key - sequences given there you can tell the line editor to turn on - "application" mode when it starts and turn it off when it stops: - - function zle-line-init () { echoti smkx } - function zle-line-finish () { echoti rmkx } - zle -N zle-line-init - zle -N zle-line-finish - -Putting these lines into your .zshrc will fix the problems. - - -## How can I use meta in 8bit mode? - -St supports meta in 8bit mode, but the default terminfo entry doesn't -use this capability. If you want it, you have to use the 'st-meta' value -in TERM. - - -## I cannot compile st in OpenBSD - -OpenBSD lacks librt, despite it being mandatory in POSIX -. -If you want to compile st for OpenBSD you have to remove -lrt from config.mk, and -st will compile without any loss of functionality, because all the functions are -included in libc on this platform. - - -## The Backspace Case - -St is emulating the Linux way of handling backspace being delete and delete being -backspace. - -This is an issue that was discussed in suckless mailing list -. Here is why some old grumpy -terminal users wants its backspace to be how he feels it: - - Well, I am going to comment why I want to change the behaviour - of this key. When ASCII was defined in 1968, communication - with computers was done using punched cards, or hardcopy - terminals (basically a typewriter machine connected with the - computer using a serial port). ASCII defines DELETE as 7F, - because, in punched-card terms, it means all the holes of the - card punched; it is thus a kind of 'physical delete'. In the - same way, the BACKSPACE key was a non-destructive backspace, - as on a typewriter. So, if you wanted to delete a character, - you had to BACKSPACE and then DELETE. Another use of BACKSPACE - was to type accented characters, for example 'a BACKSPACE `'. - The VT100 had no BACKSPACE key; it was generated using the - CONTROL key as another control character (CONTROL key sets to - 0 b7 b6 b5, so it converts H (code 0x48) into BACKSPACE (code - 0x08)), but it had a DELETE key in a similar position where - the BACKSPACE key is located today on common PC keyboards. - All the terminal emulators emulated the difference between - these keys correctly: the backspace key generated a BACKSPACE - (^H) and delete key generated a DELETE (^?). - - But a problem arose when Linus Torvalds wrote Linux. Unlike - earlier terminals, the Linux virtual terminal (the terminal - emulator integrated in the kernel) returned a DELETE when - backspace was pressed, due to the VT100 having a DELETE key in - the same position. This created a lot of problems (see [1] - and [2]). Since Linux has become the king, a lot of terminal - emulators today generate a DELETE when the backspace key is - pressed in order to avoid problems with Linux. The result is - that the only way of generating a BACKSPACE on these systems - is by using CONTROL + H. (I also think that emacs had an - important point here because the CONTROL + H prefix is used - in emacs in some commands (help commands).) - - From point of view of the kernel, you can change the key - for deleting a previous character with stty erase. When you - connect a real terminal into a machine you describe the type - of terminal, so getty configures the correct value of stty - erase for this terminal. In the case of terminal emulators, - however, you don't have any getty that can set the correct - value of stty erase, so you always get the default value. - For this reason, it is necessary to add 'stty erase ^H' to your - profile if you have changed the value of the backspace key. - Of course, another solution is for st itself to modify the - value of stty erase. I usually have the inverse problem: - when I connect to non-Unix machines, I have to press CONTROL + - h to get a BACKSPACE. The inverse problem occurs when a user - connects to my Unix machines from a different system with a - correct backspace key. - - [1] http://www.ibb.net/~anne/keyboard.html - [2] http://www.tldp.org/HOWTO/Keyboard-and-Console-HOWTO-5.html - - -## But I really want the old grumpy behaviour of my terminal - -Apply [1]. - -[1] https://st.suckless.org/patches/delkey - - -## Why do images not work in st using the w3m image hack? - -w3mimg uses a hack that draws an image on top of the terminal emulator Drawable -window. The hack relies on the terminal to use a single buffer to draw its -contents directly. - -st uses double-buffered drawing so the image is quickly replaced and may show a -short flicker effect. - -Below is a patch example to change st double-buffering to a single Drawable -buffer. - -diff --git a/x.c b/x.c ---- a/x.c -+++ b/x.c -@@ -732,10 +732,6 @@ xresize(int col, int row) - win.tw = col * win.cw; - win.th = row * win.ch; - -- XFreePixmap(xw.dpy, xw.buf); -- xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, -- DefaultDepth(xw.dpy, xw.scr)); -- XftDrawChange(xw.draw, xw.buf); - xclear(0, 0, win.w, win.h); - - /* resize to new width */ -@@ -1148,8 +1144,7 @@ xinit(int cols, int rows) - gcvalues.graphics_exposures = False; - dc.gc = XCreateGC(xw.dpy, parent, GCGraphicsExposures, - &gcvalues); -- xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, -- DefaultDepth(xw.dpy, xw.scr)); -+ xw.buf = xw.win; - XSetForeground(xw.dpy, dc.gc, dc.col[defaultbg].pixel); - XFillRectangle(xw.dpy, xw.buf, dc.gc, 0, 0, win.w, win.h); - -@@ -1632,8 +1627,6 @@ xdrawline(Line line, int x1, int y1, int x2) - void - xfinishdraw(void) - { -- XCopyArea(xw.dpy, xw.buf, xw.win, dc.gc, 0, 0, win.w, -- win.h, 0, 0); - XSetForeground(xw.dpy, dc.gc, - dc.col[IS_SET(MODE_REVERSE)? - defaultfg : defaultbg].pixel); - - -## BadLength X error in Xft when trying to render emoji - -Xft makes st crash when rendering color emojis with the following error: - -"X Error of failed request: BadLength (poly request too large or internal Xlib length error)" - Major opcode of failed request: 139 (RENDER) - Minor opcode of failed request: 20 (RenderAddGlyphs) - Serial number of failed request: 1595 - Current serial number in output stream: 1818" - -This is a known bug in Xft (not st) which happens on some platforms and -combination of particular fonts and fontconfig settings. - -See also: -https://gitlab.freedesktop.org/xorg/lib/libxft/issues/6 -https://bugs.freedesktop.org/show_bug.cgi?id=107534 -https://bugzilla.redhat.com/show_bug.cgi?id=1498269 - -The solution is to remove color emoji fonts or disable this in the fontconfig -XML configuration. As an ugly workaround (which may work only on newer -fontconfig versions (FC_COLOR)), the following code can be used to mask color -fonts: - - FcPatternAddBool(fcpattern, FC_COLOR, FcFalse); - -Please don't bother reporting this bug to st, but notify the upstream Xft -developers about fixing this bug. - -As of 2022-09-05 this now seems to be finally fixed in libXft 2.3.5: -https://gitlab.freedesktop.org/xorg/lib/libxft/-/blob/libXft-2.3.5/NEWS diff --git a/LEGACY b/LEGACY deleted file mode 100644 index bf28b1e..0000000 --- a/LEGACY +++ /dev/null @@ -1,17 +0,0 @@ -A STATEMENT ON LEGACY SUPPORT - -In the terminal world there is much cruft that comes from old and unsup‐ -ported terminals that inherit incompatible modes and escape sequences -which noone is able to know, except when he/she comes from that time and -developed a graphical vt100 emulator at that time. - -One goal of st is to only support what is really needed. When you en‐ -counter a sequence which you really need, implement it. But while you -are at it, do not add the other cruft you might encounter while sneek‐ -ing at other terminal emulators. History has bloated them and there is -no real evidence that most of the sequences are used today. - - -Christoph Lohmann <20h@r-36.net> -2012-09-13T07:00:36.081271045+02:00 - diff --git a/TODO b/TODO deleted file mode 100644 index 5f74cd5..0000000 --- a/TODO +++ /dev/null @@ -1,28 +0,0 @@ -vt emulation ------------- - -* double-height support - -code & interface ----------------- - -* add a simple way to do multiplexing - -drawing -------- -* add diacritics support to xdraws() - * switch to a suckless font drawing library -* make the font cache simpler -* add better support for brightening of the upper colors - -bugs ----- - -* fix shift up/down (shift selection in emacs) -* remove DEC test sequence when appropriate - -misc ----- - - $ grep -nE 'XXX|TODO' st.c - diff --git a/st b/st deleted file mode 100755 index 7c3b572ed06603ec5f4ab0dc9817a46ca6304dbe..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 96768 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 diff --git a/st-scrollback-0.8.4.diff b/st-scrollback-0.8.4.diff deleted file mode 100644 index 88e0c79..0000000 --- a/st-scrollback-0.8.4.diff +++ /dev/null @@ -1,351 +0,0 @@ -diff --git a/config.def.h b/config.def.h -index 6f05dce..93cbcc0 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -199,6 +199,8 @@ static Shortcut shortcuts[] = { - { TERMMOD, XK_Y, selpaste, {.i = 0} }, - { ShiftMask, XK_Insert, selpaste, {.i = 0} }, - { TERMMOD, XK_Num_Lock, numlock, {.i = 0} }, -+ { ShiftMask, XK_Page_Up, kscrollup, {.i = -1} }, -+ { ShiftMask, XK_Page_Down, kscrolldown, {.i = -1} }, - }; - - /* -diff --git a/st.c b/st.c -index 76b7e0d..edec064 100644 ---- a/st.c -+++ b/st.c -@@ -35,6 +35,7 @@ - #define ESC_ARG_SIZ 16 - #define STR_BUF_SIZ ESC_BUF_SIZ - #define STR_ARG_SIZ ESC_ARG_SIZ -+#define HISTSIZE 2000 - - /* macros */ - #define IS_SET(flag) ((term.mode & (flag)) != 0) -@@ -42,6 +43,9 @@ - #define ISCONTROLC1(c) (BETWEEN(c, 0x80, 0x9f)) - #define ISCONTROL(c) (ISCONTROLC0(c) || ISCONTROLC1(c)) - #define ISDELIM(u) (u && wcschr(worddelimiters, u)) -+#define TLINE(y) ((y) < term.scr ? term.hist[((y) + term.histi - \ -+ term.scr + HISTSIZE + 1) % HISTSIZE] : \ -+ term.line[(y) - term.scr]) - - enum term_mode { - MODE_WRAP = 1 << 0, -@@ -115,6 +119,9 @@ typedef struct { - int col; /* nb col */ - Line *line; /* screen */ - Line *alt; /* alternate screen */ -+ Line hist[HISTSIZE]; /* history buffer */ -+ int histi; /* history index */ -+ int scr; /* scroll back */ - int *dirty; /* dirtyness of lines */ - TCursor c; /* cursor */ - int ocx; /* old cursor col */ -@@ -184,8 +191,8 @@ static void tnewline(int); - static void tputtab(int); - static void tputc(Rune); - static void treset(void); --static void tscrollup(int, int); --static void tscrolldown(int, int); -+static void tscrollup(int, int, int); -+static void tscrolldown(int, int, int); - static void tsetattr(int *, int); - static void tsetchar(Rune, Glyph *, int, int); - static void tsetdirt(int, int); -@@ -414,10 +421,10 @@ tlinelen(int y) - { - int i = term.col; - -- if (term.line[y][i - 1].mode & ATTR_WRAP) -+ if (TLINE(y)[i - 1].mode & ATTR_WRAP) - return i; - -- while (i > 0 && term.line[y][i - 1].u == ' ') -+ while (i > 0 && TLINE(y)[i - 1].u == ' ') - --i; - - return i; -@@ -526,7 +533,7 @@ selsnap(int *x, int *y, int direction) - * Snap around if the word wraps around at the end or - * beginning of a line. - */ -- prevgp = &term.line[*y][*x]; -+ prevgp = &TLINE(*y)[*x]; - prevdelim = ISDELIM(prevgp->u); - for (;;) { - newx = *x + direction; -@@ -541,14 +548,14 @@ selsnap(int *x, int *y, int direction) - yt = *y, xt = *x; - else - yt = newy, xt = newx; -- if (!(term.line[yt][xt].mode & ATTR_WRAP)) -+ if (!(TLINE(yt)[xt].mode & ATTR_WRAP)) - break; - } - - if (newx >= tlinelen(newy)) - break; - -- gp = &term.line[newy][newx]; -+ gp = &TLINE(newy)[newx]; - delim = ISDELIM(gp->u); - if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim - || (delim && gp->u != prevgp->u))) -@@ -569,14 +576,14 @@ selsnap(int *x, int *y, int direction) - *x = (direction < 0) ? 0 : term.col - 1; - if (direction < 0) { - for (; *y > 0; *y += direction) { -- if (!(term.line[*y-1][term.col-1].mode -+ if (!(TLINE(*y-1)[term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } else if (direction > 0) { - for (; *y < term.row-1; *y += direction) { -- if (!(term.line[*y][term.col-1].mode -+ if (!(TLINE(*y)[term.col-1].mode - & ATTR_WRAP)) { - break; - } -@@ -607,13 +614,13 @@ getsel(void) - } - - if (sel.type == SEL_RECTANGULAR) { -- gp = &term.line[y][sel.nb.x]; -+ gp = &TLINE(y)[sel.nb.x]; - lastx = sel.ne.x; - } else { -- gp = &term.line[y][sel.nb.y == y ? sel.nb.x : 0]; -+ gp = &TLINE(y)[sel.nb.y == y ? sel.nb.x : 0]; - lastx = (sel.ne.y == y) ? sel.ne.x : term.col-1; - } -- last = &term.line[y][MIN(lastx, linelen-1)]; -+ last = &TLINE(y)[MIN(lastx, linelen-1)]; - while (last >= gp && last->u == ' ') - --last; - -@@ -848,6 +855,9 @@ void - ttywrite(const char *s, size_t n, int may_echo) - { - const char *next; -+ Arg arg = (Arg) { .i = term.scr }; -+ -+ kscrolldown(&arg); - - if (may_echo && IS_SET(MODE_ECHO)) - twrite(s, n, 1); -@@ -1059,13 +1069,53 @@ tswapscreen(void) - } - - void --tscrolldown(int orig, int n) -+kscrolldown(const Arg* a) -+{ -+ int n = a->i; -+ -+ if (n < 0) -+ n = term.row + n; -+ -+ if (n > term.scr) -+ n = term.scr; -+ -+ if (term.scr > 0) { -+ term.scr -= n; -+ selscroll(0, -n); -+ tfulldirt(); -+ } -+} -+ -+void -+kscrollup(const Arg* a) -+{ -+ int n = a->i; -+ -+ if (n < 0) -+ n = term.row + n; -+ -+ if (term.scr <= HISTSIZE-n) { -+ term.scr += n; -+ selscroll(0, n); -+ tfulldirt(); -+ } -+} -+ -+void -+tscrolldown(int orig, int n, int copyhist) - { - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - -+ if (copyhist) { -+ term.histi = (term.histi - 1 + HISTSIZE) % HISTSIZE; -+ temp = term.hist[term.histi]; -+ term.hist[term.histi] = term.line[term.bot]; -+ term.line[term.bot] = temp; -+ } -+ - tsetdirt(orig, term.bot-n); - tclearregion(0, term.bot-n+1, term.col-1, term.bot); - -@@ -1075,17 +1125,28 @@ tscrolldown(int orig, int n) - term.line[i-n] = temp; - } - -- selscroll(orig, n); -+ if (term.scr == 0) -+ selscroll(orig, n); - } - - void --tscrollup(int orig, int n) -+tscrollup(int orig, int n, int copyhist) - { - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - -+ if (copyhist) { -+ term.histi = (term.histi + 1) % HISTSIZE; -+ temp = term.hist[term.histi]; -+ term.hist[term.histi] = term.line[orig]; -+ term.line[orig] = temp; -+ } -+ -+ if (term.scr > 0 && term.scr < HISTSIZE) -+ term.scr = MIN(term.scr + n, HISTSIZE-1); -+ - tclearregion(0, orig, term.col-1, orig+n-1); - tsetdirt(orig+n, term.bot); - -@@ -1095,7 +1156,8 @@ tscrollup(int orig, int n) - term.line[i+n] = temp; - } - -- selscroll(orig, -n); -+ if (term.scr == 0) -+ selscroll(orig, -n); - } - - void -@@ -1124,7 +1186,7 @@ tnewline(int first_col) - int y = term.c.y; - - if (y == term.bot) { -- tscrollup(term.top, 1); -+ tscrollup(term.top, 1, 1); - } else { - y++; - } -@@ -1289,14 +1351,14 @@ void - tinsertblankline(int n) - { - if (BETWEEN(term.c.y, term.top, term.bot)) -- tscrolldown(term.c.y, n); -+ tscrolldown(term.c.y, n, 0); - } - - void - tdeleteline(int n) - { - if (BETWEEN(term.c.y, term.top, term.bot)) -- tscrollup(term.c.y, n); -+ tscrollup(term.c.y, n, 0); - } - - int32_t -@@ -1733,11 +1795,11 @@ csihandle(void) - break; - case 'S': /* SU -- Scroll line up */ - DEFAULT(csiescseq.arg[0], 1); -- tscrollup(term.top, csiescseq.arg[0]); -+ tscrollup(term.top, csiescseq.arg[0], 0); - break; - case 'T': /* SD -- Scroll line down */ - DEFAULT(csiescseq.arg[0], 1); -- tscrolldown(term.top, csiescseq.arg[0]); -+ tscrolldown(term.top, csiescseq.arg[0], 0); - break; - case 'L': /* IL -- Insert blank lines */ - DEFAULT(csiescseq.arg[0], 1); -@@ -2241,7 +2303,7 @@ eschandle(uchar ascii) - return 0; - case 'D': /* IND -- Linefeed */ - if (term.c.y == term.bot) { -- tscrollup(term.top, 1); -+ tscrollup(term.top, 1, 1); - } else { - tmoveto(term.c.x, term.c.y+1); - } -@@ -2254,7 +2316,7 @@ eschandle(uchar ascii) - break; - case 'M': /* RI -- Reverse index */ - if (term.c.y == term.top) { -- tscrolldown(term.top, 1); -+ tscrolldown(term.top, 1, 1); - } else { - tmoveto(term.c.x, term.c.y-1); - } -@@ -2464,7 +2526,7 @@ twrite(const char *buf, int buflen, int show_ctrl) - void - tresize(int col, int row) - { -- int i; -+ int i, j; - int minrow = MIN(row, term.row); - int mincol = MIN(col, term.col); - int *bp; -@@ -2501,6 +2563,14 @@ tresize(int col, int row) - term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty)); - term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs)); - -+ for (i = 0; i < HISTSIZE; i++) { -+ term.hist[i] = xrealloc(term.hist[i], col * sizeof(Glyph)); -+ for (j = mincol; j < col; j++) { -+ term.hist[i][j] = term.c.attr; -+ term.hist[i][j].u = ' '; -+ } -+ } -+ - /* resize each row to new width, zero-pad if needed */ - for (i = 0; i < minrow; i++) { - term.line[i] = xrealloc(term.line[i], col * sizeof(Glyph)); -@@ -2559,7 +2629,7 @@ drawregion(int x1, int y1, int x2, int y2) - continue; - - term.dirty[y] = 0; -- xdrawline(term.line[y], x1, y, x2); -+ xdrawline(TLINE(y), x1, y, x2); - } - } - -@@ -2580,8 +2650,9 @@ draw(void) - cx--; - - drawregion(0, 0, term.col, term.row); -- xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], -- term.ocx, term.ocy, term.line[term.ocy][term.ocx]); -+ if (term.scr == 0) -+ xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], -+ term.ocx, term.ocy, term.line[term.ocy][term.ocx]); - term.ocx = cx; - term.ocy = term.c.y; - xfinishdraw(); -diff --git a/st.h b/st.h -index 3d351b6..f44e1d3 100644 ---- a/st.h -+++ b/st.h -@@ -81,6 +81,8 @@ void die(const char *, ...); - void redraw(void); - void draw(void); - -+void kscrolldown(const Arg *); -+void kscrollup(const Arg *); - void printscreen(const Arg *); - void printsel(const Arg *); - void sendbreak(const Arg *); diff --git a/st.c.orig b/st.c.orig deleted file mode 100644 index 57c6e96..0000000 --- a/st.c.orig +++ /dev/null @@ -1,2678 +0,0 @@ -/* See LICENSE for license details. */ -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "st.h" -#include "win.h" - -#if defined(__linux) - #include -#elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__) - #include -#elif defined(__FreeBSD__) || defined(__DragonFly__) - #include -#endif - -/* Arbitrary sizes */ -#define UTF_INVALID 0xFFFD -#define UTF_SIZ 4 -#define ESC_BUF_SIZ (128*UTF_SIZ) -#define ESC_ARG_SIZ 16 -#define STR_BUF_SIZ ESC_BUF_SIZ -#define STR_ARG_SIZ ESC_ARG_SIZ - -/* macros */ -#define IS_SET(flag) ((term.mode & (flag)) != 0) -#define ISCONTROLC0(c) (BETWEEN(c, 0, 0x1f) || (c) == 0x7f) -#define ISCONTROLC1(c) (BETWEEN(c, 0x80, 0x9f)) -#define ISCONTROL(c) (ISCONTROLC0(c) || ISCONTROLC1(c)) -#define ISDELIM(u) (u && wcschr(worddelimiters, u)) - -enum term_mode { - MODE_WRAP = 1 << 0, - MODE_INSERT = 1 << 1, - MODE_ALTSCREEN = 1 << 2, - MODE_CRLF = 1 << 3, - MODE_ECHO = 1 << 4, - MODE_PRINT = 1 << 5, - MODE_UTF8 = 1 << 6, -}; - -enum cursor_movement { - CURSOR_SAVE, - CURSOR_LOAD -}; - -enum cursor_state { - CURSOR_DEFAULT = 0, - CURSOR_WRAPNEXT = 1, - CURSOR_ORIGIN = 2 -}; - -enum charset { - CS_GRAPHIC0, - CS_GRAPHIC1, - CS_UK, - CS_USA, - CS_MULTI, - CS_GER, - CS_FIN -}; - -enum escape_state { - ESC_START = 1, - ESC_CSI = 2, - ESC_STR = 4, /* DCS, OSC, PM, APC */ - ESC_ALTCHARSET = 8, - ESC_STR_END = 16, /* a final string was encountered */ - ESC_TEST = 32, /* Enter in test mode */ - ESC_UTF8 = 64, -}; - -typedef struct { - Glyph attr; /* current char attributes */ - int x; - int y; - char state; -} TCursor; - -typedef struct { - int mode; - int type; - int snap; - /* - * Selection variables: - * nb – normalized coordinates of the beginning of the selection - * ne – normalized coordinates of the end of the selection - * ob – original coordinates of the beginning of the selection - * oe – original coordinates of the end of the selection - */ - struct { - int x, y; - } nb, ne, ob, oe; - - int alt; -} Selection; - -/* Internal representation of the screen */ -typedef struct { - int row; /* nb row */ - int col; /* nb col */ - Line *line; /* screen */ - Line *alt; /* alternate screen */ - int *dirty; /* dirtyness of lines */ - TCursor c; /* cursor */ - int ocx; /* old cursor col */ - int ocy; /* old cursor row */ - int top; /* top scroll limit */ - int bot; /* bottom scroll limit */ - int mode; /* terminal mode flags */ - int esc; /* escape state flags */ - char trantbl[4]; /* charset table translation */ - int charset; /* current charset */ - int icharset; /* selected charset for sequence */ - int *tabs; - Rune lastc; /* last printed char outside of sequence, 0 if control */ -} Term; - -/* CSI Escape sequence structs */ -/* ESC '[' [[ [] [;]] []] */ -typedef struct { - char buf[ESC_BUF_SIZ]; /* raw string */ - size_t len; /* raw string length */ - char priv; - int arg[ESC_ARG_SIZ]; - int narg; /* nb of args */ - char mode[2]; -} CSIEscape; - -/* STR Escape sequence structs */ -/* ESC type [[ [] [;]] ] ESC '\' */ -typedef struct { - char type; /* ESC type ... */ - char *buf; /* allocated raw string */ - size_t siz; /* allocation size */ - size_t len; /* raw string length */ - char *args[STR_ARG_SIZ]; - int narg; /* nb of args */ -} STREscape; - -static void execsh(char *, char **); -static void stty(char **); -static void sigchld(int); -static void ttywriteraw(const char *, size_t); - -static void csidump(void); -static void csihandle(void); -static void csiparse(void); -static void csireset(void); -static void osc_color_response(int, int, int); -static int eschandle(uchar); -static void strdump(void); -static void strhandle(void); -static void strparse(void); -static void strreset(void); - -static void tprinter(char *, size_t); -static void tdumpsel(void); -static void tdumpline(int); -static void tdump(void); -static void tclearregion(int, int, int, int); -static void tcursor(int); -static void tdeletechar(int); -static void tdeleteline(int); -static void tinsertblank(int); -static void tinsertblankline(int); -static int tlinelen(int); -static void tmoveto(int, int); -static void tmoveato(int, int); -static void tnewline(int); -static void tputtab(int); -static void tputc(Rune); -static void treset(void); -static void tscrollup(int, int); -static void tscrolldown(int, int); -static void tsetattr(const int *, int); -static void tsetchar(Rune, const Glyph *, int, int); -static void tsetdirt(int, int); -static void tsetscroll(int, int); -static void tswapscreen(void); -static void tsetmode(int, int, const int *, int); -static int twrite(const char *, int, int); -static void tfulldirt(void); -static void tcontrolcode(uchar ); -static void tdectest(char ); -static void tdefutf8(char); -static int32_t tdefcolor(const int *, int *, int); -static void tdeftran(char); -static void tstrsequence(uchar); - -static void drawregion(int, int, int, int); - -static void selnormalize(void); -static void selscroll(int, int); -static void selsnap(int *, int *, int); - -static size_t utf8decode(const char *, Rune *, size_t); -static Rune utf8decodebyte(char, size_t *); -static char utf8encodebyte(Rune, size_t); -static size_t utf8validate(Rune *, size_t); - -static char *base64dec(const char *); -static char base64dec_getc(const char **); - -static ssize_t xwrite(int, const char *, size_t); - -/* Globals */ -static Term term; -static Selection sel; -static CSIEscape csiescseq; -static STREscape strescseq; -static int iofd = 1; -static int cmdfd; -static pid_t pid; - -static const uchar utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; -static const uchar utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; -static const Rune utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000}; -static const Rune utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; - -ssize_t -xwrite(int fd, const char *s, size_t len) -{ - size_t aux = len; - ssize_t r; - - while (len > 0) { - r = write(fd, s, len); - if (r < 0) - return r; - len -= r; - s += r; - } - - return aux; -} - -void * -xmalloc(size_t len) -{ - void *p; - - if (!(p = malloc(len))) - die("malloc: %s\n", strerror(errno)); - - return p; -} - -void * -xrealloc(void *p, size_t len) -{ - if ((p = realloc(p, len)) == NULL) - die("realloc: %s\n", strerror(errno)); - - return p; -} - -char * -xstrdup(const char *s) -{ - char *p; - - if ((p = strdup(s)) == NULL) - die("strdup: %s\n", strerror(errno)); - - return p; -} - -size_t -utf8decode(const char *c, Rune *u, size_t clen) -{ - size_t i, j, len, type; - Rune udecoded; - - *u = UTF_INVALID; - if (!clen) - return 0; - udecoded = utf8decodebyte(c[0], &len); - if (!BETWEEN(len, 1, UTF_SIZ)) - return 1; - for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { - udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type); - if (type != 0) - return j; - } - if (j < len) - return 0; - *u = udecoded; - utf8validate(u, len); - - return len; -} - -Rune -utf8decodebyte(char c, size_t *i) -{ - for (*i = 0; *i < LEN(utfmask); ++(*i)) - if (((uchar)c & utfmask[*i]) == utfbyte[*i]) - return (uchar)c & ~utfmask[*i]; - - return 0; -} - -size_t -utf8encode(Rune u, char *c) -{ - size_t len, i; - - len = utf8validate(&u, 0); - if (len > UTF_SIZ) - return 0; - - for (i = len - 1; i != 0; --i) { - c[i] = utf8encodebyte(u, 0); - u >>= 6; - } - c[0] = utf8encodebyte(u, len); - - return len; -} - -char -utf8encodebyte(Rune u, size_t i) -{ - return utfbyte[i] | (u & ~utfmask[i]); -} - -size_t -utf8validate(Rune *u, size_t i) -{ - if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF)) - *u = UTF_INVALID; - for (i = 1; *u > utfmax[i]; ++i) - ; - - return i; -} - -char -base64dec_getc(const char **src) -{ - while (**src && !isprint((unsigned char)**src)) - (*src)++; - return **src ? *((*src)++) : '='; /* emulate padding if string ends */ -} - -char * -base64dec(const char *src) -{ - size_t in_len = strlen(src); - char *result, *dst; - static const char base64_digits[256] = { - [43] = 62, 0, 0, 0, 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, - 0, 0, 0, -1, 0, 0, 0, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, - 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 0, 0, 0, 0, - 0, 0, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, - 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51 - }; - - if (in_len % 4) - in_len += 4 - (in_len % 4); - result = dst = xmalloc(in_len / 4 * 3 + 1); - while (*src) { - int a = base64_digits[(unsigned char) base64dec_getc(&src)]; - int b = base64_digits[(unsigned char) base64dec_getc(&src)]; - int c = base64_digits[(unsigned char) base64dec_getc(&src)]; - int d = base64_digits[(unsigned char) base64dec_getc(&src)]; - - /* invalid input. 'a' can be -1, e.g. if src is "\n" (c-str) */ - if (a == -1 || b == -1) - break; - - *dst++ = (a << 2) | ((b & 0x30) >> 4); - if (c == -1) - break; - *dst++ = ((b & 0x0f) << 4) | ((c & 0x3c) >> 2); - if (d == -1) - break; - *dst++ = ((c & 0x03) << 6) | d; - } - *dst = '\0'; - return result; -} - -void -selinit(void) -{ - sel.mode = SEL_IDLE; - sel.snap = 0; - sel.ob.x = -1; -} - -int -tlinelen(int y) -{ - int i = term.col; - - if (term.line[y][i - 1].mode & ATTR_WRAP) - return i; - - while (i > 0 && term.line[y][i - 1].u == ' ') - --i; - - return i; -} - -void -selstart(int col, int row, int snap) -{ - selclear(); - sel.mode = SEL_EMPTY; - sel.type = SEL_REGULAR; - sel.alt = IS_SET(MODE_ALTSCREEN); - sel.snap = snap; - sel.oe.x = sel.ob.x = col; - sel.oe.y = sel.ob.y = row; - selnormalize(); - - if (sel.snap != 0) - sel.mode = SEL_READY; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -selextend(int col, int row, int type, int done) -{ - int oldey, oldex, oldsby, oldsey, oldtype; - - if (sel.mode == SEL_IDLE) - return; - if (done && sel.mode == SEL_EMPTY) { - selclear(); - return; - } - - oldey = sel.oe.y; - oldex = sel.oe.x; - oldsby = sel.nb.y; - oldsey = sel.ne.y; - oldtype = sel.type; - - sel.oe.x = col; - sel.oe.y = row; - selnormalize(); - sel.type = type; - - if (oldey != sel.oe.y || oldex != sel.oe.x || oldtype != sel.type || sel.mode == SEL_EMPTY) - tsetdirt(MIN(sel.nb.y, oldsby), MAX(sel.ne.y, oldsey)); - - sel.mode = done ? SEL_IDLE : SEL_READY; -} - -void -selnormalize(void) -{ - int i; - - if (sel.type == SEL_REGULAR && sel.ob.y != sel.oe.y) { - sel.nb.x = sel.ob.y < sel.oe.y ? sel.ob.x : sel.oe.x; - sel.ne.x = sel.ob.y < sel.oe.y ? sel.oe.x : sel.ob.x; - } else { - sel.nb.x = MIN(sel.ob.x, sel.oe.x); - sel.ne.x = MAX(sel.ob.x, sel.oe.x); - } - sel.nb.y = MIN(sel.ob.y, sel.oe.y); - sel.ne.y = MAX(sel.ob.y, sel.oe.y); - - selsnap(&sel.nb.x, &sel.nb.y, -1); - selsnap(&sel.ne.x, &sel.ne.y, +1); - - /* expand selection over line breaks */ - if (sel.type == SEL_RECTANGULAR) - return; - i = tlinelen(sel.nb.y); - if (i < sel.nb.x) - sel.nb.x = i; - if (tlinelen(sel.ne.y) <= sel.ne.x) - sel.ne.x = term.col - 1; -} - -int -selected(int x, int y) -{ - if (sel.mode == SEL_EMPTY || sel.ob.x == -1 || - sel.alt != IS_SET(MODE_ALTSCREEN)) - return 0; - - if (sel.type == SEL_RECTANGULAR) - return BETWEEN(y, sel.nb.y, sel.ne.y) - && BETWEEN(x, sel.nb.x, sel.ne.x); - - return BETWEEN(y, sel.nb.y, sel.ne.y) - && (y != sel.nb.y || x >= sel.nb.x) - && (y != sel.ne.y || x <= sel.ne.x); -} - -void -selsnap(int *x, int *y, int direction) -{ - int newx, newy, xt, yt; - int delim, prevdelim; - const Glyph *gp, *prevgp; - - switch (sel.snap) { - case SNAP_WORD: - /* - * Snap around if the word wraps around at the end or - * beginning of a line. - */ - prevgp = &term.line[*y][*x]; - prevdelim = ISDELIM(prevgp->u); - for (;;) { - newx = *x + direction; - newy = *y; - if (!BETWEEN(newx, 0, term.col - 1)) { - newy += direction; - newx = (newx + term.col) % term.col; - if (!BETWEEN(newy, 0, term.row - 1)) - break; - - if (direction > 0) - yt = *y, xt = *x; - else - yt = newy, xt = newx; - if (!(term.line[yt][xt].mode & ATTR_WRAP)) - break; - } - - if (newx >= tlinelen(newy)) - break; - - gp = &term.line[newy][newx]; - delim = ISDELIM(gp->u); - if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim - || (delim && gp->u != prevgp->u))) - break; - - *x = newx; - *y = newy; - prevgp = gp; - prevdelim = delim; - } - break; - case SNAP_LINE: - /* - * Snap around if the the previous line or the current one - * has set ATTR_WRAP at its end. Then the whole next or - * previous line will be selected. - */ - *x = (direction < 0) ? 0 : term.col - 1; - if (direction < 0) { - for (; *y > 0; *y += direction) { - if (!(term.line[*y-1][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } else if (direction > 0) { - for (; *y < term.row-1; *y += direction) { - if (!(term.line[*y][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } - break; - } -} - -char * -getsel(void) -{ - char *str, *ptr; - int y, bufsize, lastx, linelen; - const Glyph *gp, *last; - - if (sel.ob.x == -1) - return NULL; - - bufsize = (term.col+1) * (sel.ne.y-sel.nb.y+1) * UTF_SIZ; - ptr = str = xmalloc(bufsize); - - /* append every set & selected glyph to the selection */ - for (y = sel.nb.y; y <= sel.ne.y; y++) { - if ((linelen = tlinelen(y)) == 0) { - *ptr++ = '\n'; - continue; - } - - if (sel.type == SEL_RECTANGULAR) { - gp = &term.line[y][sel.nb.x]; - lastx = sel.ne.x; - } else { - gp = &term.line[y][sel.nb.y == y ? sel.nb.x : 0]; - lastx = (sel.ne.y == y) ? sel.ne.x : term.col-1; - } - last = &term.line[y][MIN(lastx, linelen-1)]; - while (last >= gp && last->u == ' ') - --last; - - for ( ; gp <= last; ++gp) { - if (gp->mode & ATTR_WDUMMY) - continue; - - ptr += utf8encode(gp->u, ptr); - } - - /* - * Copy and pasting of line endings is inconsistent - * in the inconsistent terminal and GUI world. - * The best solution seems like to produce '\n' when - * something is copied from st and convert '\n' to - * '\r', when something to be pasted is received by - * st. - * FIXME: Fix the computer world. - */ - if ((y < sel.ne.y || lastx >= linelen) && - (!(last->mode & ATTR_WRAP) || sel.type == SEL_RECTANGULAR)) - *ptr++ = '\n'; - } - *ptr = 0; - return str; -} - -void -selclear(void) -{ - if (sel.ob.x == -1) - return; - sel.mode = SEL_IDLE; - sel.ob.x = -1; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -die(const char *errstr, ...) -{ - va_list ap; - - va_start(ap, errstr); - vfprintf(stderr, errstr, ap); - va_end(ap); - exit(1); -} - -void -execsh(char *cmd, char **args) -{ - char *sh, *prog, *arg; - const struct passwd *pw; - - errno = 0; - if ((pw = getpwuid(getuid())) == NULL) { - if (errno) - die("getpwuid: %s\n", strerror(errno)); - else - die("who are you?\n"); - } - - if ((sh = getenv("SHELL")) == NULL) - sh = (pw->pw_shell[0]) ? pw->pw_shell : cmd; - - if (args) { - prog = args[0]; - arg = NULL; - } else if (scroll) { - prog = scroll; - arg = utmp ? utmp : sh; - } else if (utmp) { - prog = utmp; - arg = NULL; - } else { - prog = sh; - arg = NULL; - } - DEFAULT(args, ((char *[]) {prog, arg, NULL})); - - unsetenv("COLUMNS"); - unsetenv("LINES"); - unsetenv("TERMCAP"); - setenv("LOGNAME", pw->pw_name, 1); - setenv("USER", pw->pw_name, 1); - setenv("SHELL", sh, 1); - setenv("HOME", pw->pw_dir, 1); - setenv("TERM", termname, 1); - - signal(SIGCHLD, SIG_DFL); - signal(SIGHUP, SIG_DFL); - signal(SIGINT, SIG_DFL); - signal(SIGQUIT, SIG_DFL); - signal(SIGTERM, SIG_DFL); - signal(SIGALRM, SIG_DFL); - - execvp(prog, args); - _exit(1); -} - -void -sigchld(int a) -{ - int stat; - pid_t p; - - if ((p = waitpid(pid, &stat, WNOHANG)) < 0) - die("waiting for pid %hd failed: %s\n", pid, strerror(errno)); - - if (pid != p) - return; - - if (WIFEXITED(stat) && WEXITSTATUS(stat)) - die("child exited with status %d\n", WEXITSTATUS(stat)); - else if (WIFSIGNALED(stat)) - die("child terminated due to signal %d\n", WTERMSIG(stat)); - _exit(0); -} - -void -stty(char **args) -{ - char cmd[_POSIX_ARG_MAX], **p, *q, *s; - size_t n, siz; - - if ((n = strlen(stty_args)) > sizeof(cmd)-1) - die("incorrect stty parameters\n"); - memcpy(cmd, stty_args, n); - q = cmd + n; - siz = sizeof(cmd) - n; - for (p = args; p && (s = *p); ++p) { - if ((n = strlen(s)) > siz-1) - die("stty parameter length too long\n"); - *q++ = ' '; - memcpy(q, s, n); - q += n; - siz -= n + 1; - } - *q = '\0'; - if (system(cmd) != 0) - perror("Couldn't call stty"); -} - -int -ttynew(const char *line, char *cmd, const char *out, char **args) -{ - int m, s; - - if (out) { - term.mode |= MODE_PRINT; - iofd = (!strcmp(out, "-")) ? - 1 : open(out, O_WRONLY | O_CREAT, 0666); - if (iofd < 0) { - fprintf(stderr, "Error opening %s:%s\n", - out, strerror(errno)); - } - } - - if (line) { - if ((cmdfd = open(line, O_RDWR)) < 0) - die("open line '%s' failed: %s\n", - line, strerror(errno)); - dup2(cmdfd, 0); - stty(args); - return cmdfd; - } - - /* seems to work fine on linux, openbsd and freebsd */ - if (openpty(&m, &s, NULL, NULL, NULL) < 0) - die("openpty failed: %s\n", strerror(errno)); - - switch (pid = fork()) { - case -1: - die("fork failed: %s\n", strerror(errno)); - break; - case 0: - close(iofd); - close(m); - setsid(); /* create a new process group */ - dup2(s, 0); - dup2(s, 1); - dup2(s, 2); - if (ioctl(s, TIOCSCTTY, NULL) < 0) - die("ioctl TIOCSCTTY failed: %s\n", strerror(errno)); - if (s > 2) - close(s); -#ifdef __OpenBSD__ - if (pledge("stdio getpw proc exec", NULL) == -1) - die("pledge\n"); -#endif - execsh(cmd, args); - break; - default: -#ifdef __OpenBSD__ - if (pledge("stdio rpath tty proc", NULL) == -1) - die("pledge\n"); -#endif - close(s); - cmdfd = m; - signal(SIGCHLD, sigchld); - break; - } - return cmdfd; -} - -size_t -ttyread(void) -{ - static char buf[BUFSIZ]; - static int buflen = 0; - int ret, written; - - /* append read bytes to unprocessed bytes */ - ret = read(cmdfd, buf+buflen, LEN(buf)-buflen); - - switch (ret) { - case 0: - exit(0); - case -1: - die("couldn't read from shell: %s\n", strerror(errno)); - default: - buflen += ret; - written = twrite(buf, buflen, 0); - buflen -= written; - /* keep any incomplete UTF-8 byte sequence for the next call */ - if (buflen > 0) - memmove(buf, buf + written, buflen); - return ret; - } -} - -void -ttywrite(const char *s, size_t n, int may_echo) -{ - const char *next; - - if (may_echo && IS_SET(MODE_ECHO)) - twrite(s, n, 1); - - if (!IS_SET(MODE_CRLF)) { - ttywriteraw(s, n); - return; - } - - /* This is similar to how the kernel handles ONLCR for ttys */ - while (n > 0) { - if (*s == '\r') { - next = s + 1; - ttywriteraw("\r\n", 2); - } else { - next = memchr(s, '\r', n); - DEFAULT(next, s + n); - ttywriteraw(s, next - s); - } - n -= next - s; - s = next; - } -} - -void -ttywriteraw(const char *s, size_t n) -{ - fd_set wfd, rfd; - ssize_t r; - size_t lim = 256; - - /* - * Remember that we are using a pty, which might be a modem line. - * Writing too much will clog the line. That's why we are doing this - * dance. - * FIXME: Migrate the world to Plan 9. - */ - while (n > 0) { - FD_ZERO(&wfd); - FD_ZERO(&rfd); - FD_SET(cmdfd, &wfd); - FD_SET(cmdfd, &rfd); - - /* Check if we can write. */ - if (pselect(cmdfd+1, &rfd, &wfd, NULL, NULL, NULL) < 0) { - if (errno == EINTR) - continue; - die("select failed: %s\n", strerror(errno)); - } - if (FD_ISSET(cmdfd, &wfd)) { - /* - * Only write the bytes written by ttywrite() or the - * default of 256. This seems to be a reasonable value - * for a serial line. Bigger values might clog the I/O. - */ - if ((r = write(cmdfd, s, (n < lim)? n : lim)) < 0) - goto write_error; - if (r < n) { - /* - * We weren't able to write out everything. - * This means the buffer is getting full - * again. Empty it. - */ - if (n < lim) - lim = ttyread(); - n -= r; - s += r; - } else { - /* All bytes have been written. */ - break; - } - } - if (FD_ISSET(cmdfd, &rfd)) - lim = ttyread(); - } - return; - -write_error: - die("write error on tty: %s\n", strerror(errno)); -} - -void -ttyresize(int tw, int th) -{ - struct winsize w; - - w.ws_row = term.row; - w.ws_col = term.col; - w.ws_xpixel = tw; - w.ws_ypixel = th; - if (ioctl(cmdfd, TIOCSWINSZ, &w) < 0) - fprintf(stderr, "Couldn't set window size: %s\n", strerror(errno)); -} - -void -ttyhangup(void) -{ - /* Send SIGHUP to shell */ - kill(pid, SIGHUP); -} - -int -tattrset(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) - return 1; - } - } - - return 0; -} - -void -tsetdirt(int top, int bot) -{ - int i; - - LIMIT(top, 0, term.row-1); - LIMIT(bot, 0, term.row-1); - - for (i = top; i <= bot; i++) - term.dirty[i] = 1; -} - -void -tsetdirtattr(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) { - tsetdirt(i, i); - break; - } - } - } -} - -void -tfulldirt(void) -{ - tsetdirt(0, term.row-1); -} - -void -tcursor(int mode) -{ - static TCursor c[2]; - int alt = IS_SET(MODE_ALTSCREEN); - - if (mode == CURSOR_SAVE) { - c[alt] = term.c; - } else if (mode == CURSOR_LOAD) { - term.c = c[alt]; - tmoveto(c[alt].x, c[alt].y); - } -} - -void -treset(void) -{ - uint i; - - term.c = (TCursor){{ - .mode = ATTR_NULL, - .fg = defaultfg, - .bg = defaultbg - }, .x = 0, .y = 0, .state = CURSOR_DEFAULT}; - - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - for (i = tabspaces; i < term.col; i += tabspaces) - term.tabs[i] = 1; - term.top = 0; - term.bot = term.row - 1; - term.mode = MODE_WRAP|MODE_UTF8; - memset(term.trantbl, CS_USA, sizeof(term.trantbl)); - term.charset = 0; - - for (i = 0; i < 2; i++) { - tmoveto(0, 0); - tcursor(CURSOR_SAVE); - tclearregion(0, 0, term.col-1, term.row-1); - tswapscreen(); - } -} - -void -tnew(int col, int row) -{ - term = (Term){ .c = { .attr = { .fg = defaultfg, .bg = defaultbg } } }; - tresize(col, row); - treset(); -} - -void -tswapscreen(void) -{ - Line *tmp = term.line; - - term.line = term.alt; - term.alt = tmp; - term.mode ^= MODE_ALTSCREEN; - tfulldirt(); -} - -void -tscrolldown(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tsetdirt(orig, term.bot-n); - tclearregion(0, term.bot-n+1, term.col-1, term.bot); - - for (i = term.bot; i >= orig+n; i--) { - temp = term.line[i]; - term.line[i] = term.line[i-n]; - term.line[i-n] = temp; - } - - selscroll(orig, n); -} - -void -tscrollup(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tclearregion(0, orig, term.col-1, orig+n-1); - tsetdirt(orig+n, term.bot); - - for (i = orig; i <= term.bot-n; i++) { - temp = term.line[i]; - term.line[i] = term.line[i+n]; - term.line[i+n] = temp; - } - - selscroll(orig, -n); -} - -void -selscroll(int orig, int n) -{ - if (sel.ob.x == -1 || sel.alt != IS_SET(MODE_ALTSCREEN)) - return; - - if (BETWEEN(sel.nb.y, orig, term.bot) != BETWEEN(sel.ne.y, orig, term.bot)) { - selclear(); - } else if (BETWEEN(sel.nb.y, orig, term.bot)) { - sel.ob.y += n; - sel.oe.y += n; - if (sel.ob.y < term.top || sel.ob.y > term.bot || - sel.oe.y < term.top || sel.oe.y > term.bot) { - selclear(); - } else { - selnormalize(); - } - } -} - -void -tnewline(int first_col) -{ - int y = term.c.y; - - if (y == term.bot) { - tscrollup(term.top, 1); - } else { - y++; - } - tmoveto(first_col ? 0 : term.c.x, y); -} - -void -csiparse(void) -{ - char *p = csiescseq.buf, *np; - long int v; - int sep = ';'; /* colon or semi-colon, but not both */ - - csiescseq.narg = 0; - if (*p == '?') { - csiescseq.priv = 1; - p++; - } - - csiescseq.buf[csiescseq.len] = '\0'; - while (p < csiescseq.buf+csiescseq.len) { - np = NULL; - v = strtol(p, &np, 10); - if (np == p) - v = 0; - if (v == LONG_MAX || v == LONG_MIN) - v = -1; - csiescseq.arg[csiescseq.narg++] = v; - p = np; - if (sep == ';' && *p == ':') - sep = ':'; /* allow override to colon once */ - if (*p != sep || csiescseq.narg == ESC_ARG_SIZ) - break; - p++; - } - csiescseq.mode[0] = *p++; - csiescseq.mode[1] = (p < csiescseq.buf+csiescseq.len) ? *p : '\0'; -} - -/* for absolute user moves, when decom is set */ -void -tmoveato(int x, int y) -{ - tmoveto(x, y + ((term.c.state & CURSOR_ORIGIN) ? term.top: 0)); -} - -void -tmoveto(int x, int y) -{ - int miny, maxy; - - if (term.c.state & CURSOR_ORIGIN) { - miny = term.top; - maxy = term.bot; - } else { - miny = 0; - maxy = term.row - 1; - } - term.c.state &= ~CURSOR_WRAPNEXT; - term.c.x = LIMIT(x, 0, term.col-1); - term.c.y = LIMIT(y, miny, maxy); -} - -void -tsetchar(Rune u, const Glyph *attr, int x, int y) -{ - static const char *vt100_0[62] = { /* 0x41 - 0x7e */ - "↑", "↓", "→", "←", "█", "▚", "☃", /* A - G */ - 0, 0, 0, 0, 0, 0, 0, 0, /* H - O */ - 0, 0, 0, 0, 0, 0, 0, 0, /* P - W */ - 0, 0, 0, 0, 0, 0, 0, " ", /* X - _ */ - "◆", "▒", "␉", "␌", "␍", "␊", "°", "±", /* ` - g */ - "␤", "␋", "┘", "┐", "┌", "└", "┼", "⎺", /* h - o */ - "⎻", "─", "⎼", "⎽", "├", "┤", "┴", "┬", /* p - w */ - "│", "≤", "≥", "π", "≠", "£", "·", /* x - ~ */ - }; - - /* - * The table is proudly stolen from rxvt. - */ - if (term.trantbl[term.charset] == CS_GRAPHIC0 && - BETWEEN(u, 0x41, 0x7e) && vt100_0[u - 0x41]) - utf8decode(vt100_0[u - 0x41], &u, UTF_SIZ); - - if (term.line[y][x].mode & ATTR_WIDE) { - if (x+1 < term.col) { - term.line[y][x+1].u = ' '; - term.line[y][x+1].mode &= ~ATTR_WDUMMY; - } - } else if (term.line[y][x].mode & ATTR_WDUMMY) { - term.line[y][x-1].u = ' '; - term.line[y][x-1].mode &= ~ATTR_WIDE; - } - - term.dirty[y] = 1; - term.line[y][x] = *attr; - term.line[y][x].u = u; -} - -void -tclearregion(int x1, int y1, int x2, int y2) -{ - int x, y, temp; - Glyph *gp; - - if (x1 > x2) - temp = x1, x1 = x2, x2 = temp; - if (y1 > y2) - temp = y1, y1 = y2, y2 = temp; - - LIMIT(x1, 0, term.col-1); - LIMIT(x2, 0, term.col-1); - LIMIT(y1, 0, term.row-1); - LIMIT(y2, 0, term.row-1); - - for (y = y1; y <= y2; y++) { - term.dirty[y] = 1; - for (x = x1; x <= x2; x++) { - gp = &term.line[y][x]; - if (selected(x, y)) - selclear(); - gp->fg = term.c.attr.fg; - gp->bg = term.c.attr.bg; - gp->mode = 0; - gp->u = ' '; - } - } -} - -void -tdeletechar(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x; - src = term.c.x + n; - size = term.col - src; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(term.col-n, term.c.y, term.col-1, term.c.y); -} - -void -tinsertblank(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x + n; - src = term.c.x; - size = term.col - dst; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(src, term.c.y, dst - 1, term.c.y); -} - -void -tinsertblankline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrolldown(term.c.y, n); -} - -void -tdeleteline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrollup(term.c.y, n); -} - -int32_t -tdefcolor(const int *attr, int *npar, int l) -{ - int32_t idx = -1; - uint r, g, b; - - switch (attr[*npar + 1]) { - case 2: /* direct color in RGB space */ - if (*npar + 4 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - r = attr[*npar + 2]; - g = attr[*npar + 3]; - b = attr[*npar + 4]; - *npar += 4; - if (!BETWEEN(r, 0, 255) || !BETWEEN(g, 0, 255) || !BETWEEN(b, 0, 255)) - fprintf(stderr, "erresc: bad rgb color (%u,%u,%u)\n", - r, g, b); - else - idx = TRUECOLOR(r, g, b); - break; - case 5: /* indexed color */ - if (*npar + 2 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - *npar += 2; - if (!BETWEEN(attr[*npar], 0, 255)) - fprintf(stderr, "erresc: bad fgcolor %d\n", attr[*npar]); - else - idx = attr[*npar]; - break; - case 0: /* implemented defined (only foreground) */ - case 1: /* transparent */ - case 3: /* direct color in CMY space */ - case 4: /* direct color in CMYK space */ - default: - fprintf(stderr, - "erresc(38): gfx attr %d unknown\n", attr[*npar]); - break; - } - - return idx; -} - -void -tsetattr(const int *attr, int l) -{ - int i; - int32_t idx; - - for (i = 0; i < l; i++) { - switch (attr[i]) { - case 0: - term.c.attr.mode &= ~( - ATTR_BOLD | - ATTR_FAINT | - ATTR_ITALIC | - ATTR_UNDERLINE | - ATTR_BLINK | - ATTR_REVERSE | - ATTR_INVISIBLE | - ATTR_STRUCK ); - term.c.attr.fg = defaultfg; - term.c.attr.bg = defaultbg; - break; - case 1: - term.c.attr.mode |= ATTR_BOLD; - break; - case 2: - term.c.attr.mode |= ATTR_FAINT; - break; - case 3: - term.c.attr.mode |= ATTR_ITALIC; - break; - case 4: - term.c.attr.mode |= ATTR_UNDERLINE; - break; - case 5: /* slow blink */ - /* FALLTHROUGH */ - case 6: /* rapid blink */ - term.c.attr.mode |= ATTR_BLINK; - break; - case 7: - term.c.attr.mode |= ATTR_REVERSE; - break; - case 8: - term.c.attr.mode |= ATTR_INVISIBLE; - break; - case 9: - term.c.attr.mode |= ATTR_STRUCK; - break; - case 22: - term.c.attr.mode &= ~(ATTR_BOLD | ATTR_FAINT); - break; - case 23: - term.c.attr.mode &= ~ATTR_ITALIC; - break; - case 24: - term.c.attr.mode &= ~ATTR_UNDERLINE; - break; - case 25: - term.c.attr.mode &= ~ATTR_BLINK; - break; - case 27: - term.c.attr.mode &= ~ATTR_REVERSE; - break; - case 28: - term.c.attr.mode &= ~ATTR_INVISIBLE; - break; - case 29: - term.c.attr.mode &= ~ATTR_STRUCK; - break; - case 38: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.fg = idx; - break; - case 39: - term.c.attr.fg = defaultfg; - break; - case 48: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.bg = idx; - break; - case 49: - term.c.attr.bg = defaultbg; - break; - default: - if (BETWEEN(attr[i], 30, 37)) { - term.c.attr.fg = attr[i] - 30; - } else if (BETWEEN(attr[i], 40, 47)) { - term.c.attr.bg = attr[i] - 40; - } else if (BETWEEN(attr[i], 90, 97)) { - term.c.attr.fg = attr[i] - 90 + 8; - } else if (BETWEEN(attr[i], 100, 107)) { - term.c.attr.bg = attr[i] - 100 + 8; - } else { - fprintf(stderr, - "erresc(default): gfx attr %d unknown\n", - attr[i]); - csidump(); - } - break; - } - } -} - -void -tsetscroll(int t, int b) -{ - int temp; - - LIMIT(t, 0, term.row-1); - LIMIT(b, 0, term.row-1); - if (t > b) { - temp = t; - t = b; - b = temp; - } - term.top = t; - term.bot = b; -} - -void -tsetmode(int priv, int set, const int *args, int narg) -{ - int alt; const int *lim; - - for (lim = args + narg; args < lim; ++args) { - if (priv) { - switch (*args) { - case 1: /* DECCKM -- Cursor key */ - xsetmode(set, MODE_APPCURSOR); - break; - case 5: /* DECSCNM -- Reverse video */ - xsetmode(set, MODE_REVERSE); - break; - case 6: /* DECOM -- Origin */ - MODBIT(term.c.state, set, CURSOR_ORIGIN); - tmoveato(0, 0); - break; - case 7: /* DECAWM -- Auto wrap */ - MODBIT(term.mode, set, MODE_WRAP); - break; - case 0: /* Error (IGNORED) */ - case 2: /* DECANM -- ANSI/VT52 (IGNORED) */ - case 3: /* DECCOLM -- Column (IGNORED) */ - case 4: /* DECSCLM -- Scroll (IGNORED) */ - case 8: /* DECARM -- Auto repeat (IGNORED) */ - case 18: /* DECPFF -- Printer feed (IGNORED) */ - case 19: /* DECPEX -- Printer extent (IGNORED) */ - case 42: /* DECNRCM -- National characters (IGNORED) */ - case 12: /* att610 -- Start blinking cursor (IGNORED) */ - break; - case 25: /* DECTCEM -- Text Cursor Enable Mode */ - xsetmode(!set, MODE_HIDE); - break; - case 9: /* X10 mouse compatibility mode */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEX10); - break; - case 1000: /* 1000: report button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEBTN); - break; - case 1002: /* 1002: report motion on button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMOTION); - break; - case 1003: /* 1003: enable all mouse motions */ - xsetpointermotion(set); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMANY); - break; - case 1004: /* 1004: send focus events to tty */ - xsetmode(set, MODE_FOCUS); - break; - case 1006: /* 1006: extended reporting mode */ - xsetmode(set, MODE_MOUSESGR); - break; - case 1034: - xsetmode(set, MODE_8BIT); - break; - case 1049: /* swap screen & set/restore cursor as xterm */ - if (!allowaltscreen) - break; - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - /* FALLTHROUGH */ - case 47: /* swap screen */ - case 1047: - if (!allowaltscreen) - break; - alt = IS_SET(MODE_ALTSCREEN); - if (alt) { - tclearregion(0, 0, term.col-1, - term.row-1); - } - if (set ^ alt) /* set is always 1 or 0 */ - tswapscreen(); - if (*args != 1049) - break; - /* FALLTHROUGH */ - case 1048: - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - break; - case 2004: /* 2004: bracketed paste mode */ - xsetmode(set, MODE_BRCKTPASTE); - break; - /* Not implemented mouse modes. See comments there. */ - case 1001: /* mouse highlight mode; can hang the - terminal by design when implemented. */ - case 1005: /* UTF-8 mouse mode; will confuse - applications not supporting UTF-8 - and luit. */ - case 1015: /* urxvt mangled mouse mode; incompatible - and can be mistaken for other control - codes. */ - break; - default: - fprintf(stderr, - "erresc: unknown private set/reset mode %d\n", - *args); - break; - } - } else { - switch (*args) { - case 0: /* Error (IGNORED) */ - break; - case 2: - xsetmode(set, MODE_KBDLOCK); - break; - case 4: /* IRM -- Insertion-replacement */ - MODBIT(term.mode, set, MODE_INSERT); - break; - case 12: /* SRM -- Send/Receive */ - MODBIT(term.mode, !set, MODE_ECHO); - break; - case 20: /* LNM -- Linefeed/new line */ - MODBIT(term.mode, set, MODE_CRLF); - break; - default: - fprintf(stderr, - "erresc: unknown set/reset mode %d\n", - *args); - break; - } - } - } -} - -void -csihandle(void) -{ - char buf[40]; - int len; - - switch (csiescseq.mode[0]) { - default: - unknown: - fprintf(stderr, "erresc: unknown csi "); - csidump(); - /* die(""); */ - break; - case '@': /* ICH -- Insert blank char */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblank(csiescseq.arg[0]); - break; - case 'A': /* CUU -- Cursor Up */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y-csiescseq.arg[0]); - break; - case 'B': /* CUD -- Cursor Down */ - case 'e': /* VPR --Cursor Down */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y+csiescseq.arg[0]); - break; - case 'i': /* MC -- Media Copy */ - switch (csiescseq.arg[0]) { - case 0: - tdump(); - break; - case 1: - tdumpline(term.c.y); - break; - case 2: - tdumpsel(); - break; - case 4: - term.mode &= ~MODE_PRINT; - break; - case 5: - term.mode |= MODE_PRINT; - break; - } - break; - case 'c': /* DA -- Device Attributes */ - if (csiescseq.arg[0] == 0) - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'b': /* REP -- if last char is printable print it more times */ - LIMIT(csiescseq.arg[0], 1, 65535); - if (term.lastc) - while (csiescseq.arg[0]-- > 0) - tputc(term.lastc); - break; - case 'C': /* CUF -- Cursor Forward */ - case 'a': /* HPR -- Cursor Forward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x+csiescseq.arg[0], term.c.y); - break; - case 'D': /* CUB -- Cursor Backward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x-csiescseq.arg[0], term.c.y); - break; - case 'E': /* CNL -- Cursor Down and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y+csiescseq.arg[0]); - break; - case 'F': /* CPL -- Cursor Up and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y-csiescseq.arg[0]); - break; - case 'g': /* TBC -- Tabulation clear */ - switch (csiescseq.arg[0]) { - case 0: /* clear current tab stop */ - term.tabs[term.c.x] = 0; - break; - case 3: /* clear all the tabs */ - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - break; - default: - goto unknown; - } - break; - case 'G': /* CHA -- Move to */ - case '`': /* HPA */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(csiescseq.arg[0]-1, term.c.y); - break; - case 'H': /* CUP -- Move to */ - case 'f': /* HVP */ - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], 1); - tmoveato(csiescseq.arg[1]-1, csiescseq.arg[0]-1); - break; - case 'I': /* CHT -- Cursor Forward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(csiescseq.arg[0]); - break; - case 'J': /* ED -- Clear screen */ - switch (csiescseq.arg[0]) { - case 0: /* below */ - tclearregion(term.c.x, term.c.y, term.col-1, term.c.y); - if (term.c.y < term.row-1) { - tclearregion(0, term.c.y+1, term.col-1, - term.row-1); - } - break; - case 1: /* above */ - if (term.c.y > 1) - tclearregion(0, 0, term.col-1, term.c.y-1); - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, 0, term.col-1, term.row-1); - break; - default: - goto unknown; - } - break; - case 'K': /* EL -- Clear line */ - switch (csiescseq.arg[0]) { - case 0: /* right */ - tclearregion(term.c.x, term.c.y, term.col-1, - term.c.y); - break; - case 1: /* left */ - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, term.c.y, term.col-1, term.c.y); - break; - } - break; - case 'S': /* SU -- Scroll line up */ - if (csiescseq.priv) break; - DEFAULT(csiescseq.arg[0], 1); - tscrollup(term.top, csiescseq.arg[0]); - break; - case 'T': /* SD -- Scroll line down */ - DEFAULT(csiescseq.arg[0], 1); - tscrolldown(term.top, csiescseq.arg[0]); - break; - case 'L': /* IL -- Insert blank lines */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblankline(csiescseq.arg[0]); - break; - case 'l': /* RM -- Reset Mode */ - tsetmode(csiescseq.priv, 0, csiescseq.arg, csiescseq.narg); - break; - case 'M': /* DL -- Delete lines */ - DEFAULT(csiescseq.arg[0], 1); - tdeleteline(csiescseq.arg[0]); - break; - case 'X': /* ECH -- Erase char */ - DEFAULT(csiescseq.arg[0], 1); - tclearregion(term.c.x, term.c.y, - term.c.x + csiescseq.arg[0] - 1, term.c.y); - break; - case 'P': /* DCH -- Delete char */ - DEFAULT(csiescseq.arg[0], 1); - tdeletechar(csiescseq.arg[0]); - break; - case 'Z': /* CBT -- Cursor Backward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(-csiescseq.arg[0]); - break; - case 'd': /* VPA -- Move to */ - DEFAULT(csiescseq.arg[0], 1); - tmoveato(term.c.x, csiescseq.arg[0]-1); - break; - case 'h': /* SM -- Set terminal mode */ - tsetmode(csiescseq.priv, 1, csiescseq.arg, csiescseq.narg); - break; - case 'm': /* SGR -- Terminal attribute (color) */ - tsetattr(csiescseq.arg, csiescseq.narg); - break; - case 'n': /* DSR -- Device Status Report */ - switch (csiescseq.arg[0]) { - case 5: /* Status Report "OK" `0n` */ - ttywrite("\033[0n", sizeof("\033[0n") - 1, 0); - break; - case 6: /* Report Cursor Position (CPR) ";R" */ - len = snprintf(buf, sizeof(buf), "\033[%i;%iR", - term.c.y+1, term.c.x+1); - ttywrite(buf, len, 0); - break; - default: - goto unknown; - } - break; - case 'r': /* DECSTBM -- Set Scrolling Region */ - if (csiescseq.priv) { - goto unknown; - } else { - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], term.row); - tsetscroll(csiescseq.arg[0]-1, csiescseq.arg[1]-1); - tmoveato(0, 0); - } - break; - case 's': /* DECSC -- Save cursor position (ANSI.SYS) */ - tcursor(CURSOR_SAVE); - break; - case 'u': /* DECRC -- Restore cursor position (ANSI.SYS) */ - tcursor(CURSOR_LOAD); - break; - case ' ': - switch (csiescseq.mode[1]) { - case 'q': /* DECSCUSR -- Set Cursor Style */ - if (xsetcursor(csiescseq.arg[0])) - goto unknown; - break; - default: - goto unknown; - } - break; - } -} - -void -csidump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC["); - for (i = 0; i < csiescseq.len; i++) { - c = csiescseq.buf[i] & 0xff; - if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - putc('\n', stderr); -} - -void -csireset(void) -{ - memset(&csiescseq, 0, sizeof(csiescseq)); -} - -void -osc_color_response(int num, int index, int is_osc4) -{ - int n; - char buf[32]; - unsigned char r, g, b; - - if (xgetcolor(is_osc4 ? num : index, &r, &g, &b)) { - fprintf(stderr, "erresc: failed to fetch %s color %d\n", - is_osc4 ? "osc4" : "osc", - is_osc4 ? num : index); - return; - } - - n = snprintf(buf, sizeof buf, "\033]%s%d;rgb:%02x%02x/%02x%02x/%02x%02x\007", - is_osc4 ? "4;" : "", num, r, r, g, g, b, b); - if (n < 0 || n >= sizeof(buf)) { - fprintf(stderr, "error: %s while printing %s response\n", - n < 0 ? "snprintf failed" : "truncation occurred", - is_osc4 ? "osc4" : "osc"); - } else { - ttywrite(buf, n, 1); - } -} - -void -strhandle(void) -{ - char *p = NULL, *dec; - int j, narg, par; - const struct { int idx; char *str; } osc_table[] = { - { defaultfg, "foreground" }, - { defaultbg, "background" }, - { defaultcs, "cursor" } - }; - - term.esc &= ~(ESC_STR_END|ESC_STR); - strparse(); - par = (narg = strescseq.narg) ? atoi(strescseq.args[0]) : 0; - - switch (strescseq.type) { - case ']': /* OSC -- Operating System Command */ - switch (par) { - case 0: - if (narg > 1) { - xsettitle(strescseq.args[1]); - xseticontitle(strescseq.args[1]); - } - return; - case 1: - if (narg > 1) - xseticontitle(strescseq.args[1]); - return; - case 2: - if (narg > 1) - xsettitle(strescseq.args[1]); - return; - case 52: - if (narg > 2 && allowwindowops) { - dec = base64dec(strescseq.args[2]); - if (dec) { - xsetsel(dec); - xclipcopy(); - } else { - fprintf(stderr, "erresc: invalid base64\n"); - } - } - return; - case 10: - case 11: - case 12: - if (narg < 2) - break; - p = strescseq.args[1]; - if ((j = par - 10) < 0 || j >= LEN(osc_table)) - break; /* shouldn't be possible */ - - if (!strcmp(p, "?")) { - osc_color_response(par, osc_table[j].idx, 0); - } else if (xsetcolorname(osc_table[j].idx, p)) { - fprintf(stderr, "erresc: invalid %s color: %s\n", - osc_table[j].str, p); - } else { - tfulldirt(); - } - return; - case 4: /* color set */ - if (narg < 3) - break; - p = strescseq.args[2]; - /* FALLTHROUGH */ - case 104: /* color reset */ - j = (narg > 1) ? atoi(strescseq.args[1]) : -1; - - if (p && !strcmp(p, "?")) { - osc_color_response(j, 0, 1); - } else if (xsetcolorname(j, p)) { - if (par == 104 && narg <= 1) { - xloadcols(); - return; /* color reset without parameter */ - } - fprintf(stderr, "erresc: invalid color j=%d, p=%s\n", - j, p ? p : "(null)"); - } else { - /* - * TODO if defaultbg color is changed, borders - * are dirty - */ - tfulldirt(); - } - return; - } - break; - case 'k': /* old title set compatibility */ - xsettitle(strescseq.args[0]); - return; - case 'P': /* DCS -- Device Control String */ - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - return; - } - - fprintf(stderr, "erresc: unknown str "); - strdump(); -} - -void -strparse(void) -{ - int c; - char *p = strescseq.buf; - - strescseq.narg = 0; - strescseq.buf[strescseq.len] = '\0'; - - if (*p == '\0') - return; - - while (strescseq.narg < STR_ARG_SIZ) { - strescseq.args[strescseq.narg++] = p; - while ((c = *p) != ';' && c != '\0') - ++p; - if (c == '\0') - return; - *p++ = '\0'; - } -} - -void -strdump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC%c", strescseq.type); - for (i = 0; i < strescseq.len; i++) { - c = strescseq.buf[i] & 0xff; - if (c == '\0') { - putc('\n', stderr); - return; - } else if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - fprintf(stderr, "ESC\\\n"); -} - -void -strreset(void) -{ - strescseq = (STREscape){ - .buf = xrealloc(strescseq.buf, STR_BUF_SIZ), - .siz = STR_BUF_SIZ, - }; -} - -void -sendbreak(const Arg *arg) -{ - if (tcsendbreak(cmdfd, 0)) - perror("Error sending break"); -} - -void -tprinter(char *s, size_t len) -{ - if (iofd != -1 && xwrite(iofd, s, len) < 0) { - perror("Error writing to output file"); - close(iofd); - iofd = -1; - } -} - -void -toggleprinter(const Arg *arg) -{ - term.mode ^= MODE_PRINT; -} - -void -printscreen(const Arg *arg) -{ - tdump(); -} - -void -printsel(const Arg *arg) -{ - tdumpsel(); -} - -void -tdumpsel(void) -{ - char *ptr; - - if ((ptr = getsel())) { - tprinter(ptr, strlen(ptr)); - free(ptr); - } -} - -void -tdumpline(int n) -{ - char buf[UTF_SIZ]; - const Glyph *bp, *end; - - bp = &term.line[n][0]; - end = &bp[MIN(tlinelen(n), term.col) - 1]; - if (bp != end || bp->u != ' ') { - for ( ; bp <= end; ++bp) - tprinter(buf, utf8encode(bp->u, buf)); - } - tprinter("\n", 1); -} - -void -tdump(void) -{ - int i; - - for (i = 0; i < term.row; ++i) - tdumpline(i); -} - -void -tputtab(int n) -{ - uint x = term.c.x; - - if (n > 0) { - while (x < term.col && n--) - for (++x; x < term.col && !term.tabs[x]; ++x) - /* nothing */ ; - } else if (n < 0) { - while (x > 0 && n++) - for (--x; x > 0 && !term.tabs[x]; --x) - /* nothing */ ; - } - term.c.x = LIMIT(x, 0, term.col-1); -} - -void -tdefutf8(char ascii) -{ - if (ascii == 'G') - term.mode |= MODE_UTF8; - else if (ascii == '@') - term.mode &= ~MODE_UTF8; -} - -void -tdeftran(char ascii) -{ - static char cs[] = "0B"; - static int vcs[] = {CS_GRAPHIC0, CS_USA}; - char *p; - - if ((p = strchr(cs, ascii)) == NULL) { - fprintf(stderr, "esc unhandled charset: ESC ( %c\n", ascii); - } else { - term.trantbl[term.icharset] = vcs[p - cs]; - } -} - -void -tdectest(char c) -{ - int x, y; - - if (c == '8') { /* DEC screen alignment test. */ - for (x = 0; x < term.col; ++x) { - for (y = 0; y < term.row; ++y) - tsetchar('E', &term.c.attr, x, y); - } - } -} - -void -tstrsequence(uchar c) -{ - switch (c) { - case 0x90: /* DCS -- Device Control String */ - c = 'P'; - break; - case 0x9f: /* APC -- Application Program Command */ - c = '_'; - break; - case 0x9e: /* PM -- Privacy Message */ - c = '^'; - break; - case 0x9d: /* OSC -- Operating System Command */ - c = ']'; - break; - } - strreset(); - strescseq.type = c; - term.esc |= ESC_STR; -} - -void -tcontrolcode(uchar ascii) -{ - switch (ascii) { - case '\t': /* HT */ - tputtab(1); - return; - case '\b': /* BS */ - tmoveto(term.c.x-1, term.c.y); - return; - case '\r': /* CR */ - tmoveto(0, term.c.y); - return; - case '\f': /* LF */ - case '\v': /* VT */ - case '\n': /* LF */ - /* go to first col if the mode is set */ - tnewline(IS_SET(MODE_CRLF)); - return; - case '\a': /* BEL */ - if (term.esc & ESC_STR_END) { - /* backwards compatibility to xterm */ - strhandle(); - } else { - xbell(); - } - break; - case '\033': /* ESC */ - csireset(); - term.esc &= ~(ESC_CSI|ESC_ALTCHARSET|ESC_TEST); - term.esc |= ESC_START; - return; - case '\016': /* SO (LS1 -- Locking shift 1) */ - case '\017': /* SI (LS0 -- Locking shift 0) */ - term.charset = 1 - (ascii - '\016'); - return; - case '\032': /* SUB */ - tsetchar('?', &term.c.attr, term.c.x, term.c.y); - /* FALLTHROUGH */ - case '\030': /* CAN */ - csireset(); - break; - case '\005': /* ENQ (IGNORED) */ - case '\000': /* NUL (IGNORED) */ - case '\021': /* XON (IGNORED) */ - case '\023': /* XOFF (IGNORED) */ - case 0177: /* DEL (IGNORED) */ - return; - case 0x80: /* TODO: PAD */ - case 0x81: /* TODO: HOP */ - case 0x82: /* TODO: BPH */ - case 0x83: /* TODO: NBH */ - case 0x84: /* TODO: IND */ - break; - case 0x85: /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 0x86: /* TODO: SSA */ - case 0x87: /* TODO: ESA */ - break; - case 0x88: /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 0x89: /* TODO: HTJ */ - case 0x8a: /* TODO: VTS */ - case 0x8b: /* TODO: PLD */ - case 0x8c: /* TODO: PLU */ - case 0x8d: /* TODO: RI */ - case 0x8e: /* TODO: SS2 */ - case 0x8f: /* TODO: SS3 */ - case 0x91: /* TODO: PU1 */ - case 0x92: /* TODO: PU2 */ - case 0x93: /* TODO: STS */ - case 0x94: /* TODO: CCH */ - case 0x95: /* TODO: MW */ - case 0x96: /* TODO: SPA */ - case 0x97: /* TODO: EPA */ - case 0x98: /* TODO: SOS */ - case 0x99: /* TODO: SGCI */ - break; - case 0x9a: /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 0x9b: /* TODO: CSI */ - case 0x9c: /* TODO: ST */ - break; - case 0x90: /* DCS -- Device Control String */ - case 0x9d: /* OSC -- Operating System Command */ - case 0x9e: /* PM -- Privacy Message */ - case 0x9f: /* APC -- Application Program Command */ - tstrsequence(ascii); - return; - } - /* only CAN, SUB, \a and C1 chars interrupt a sequence */ - term.esc &= ~(ESC_STR_END|ESC_STR); -} - -/* - * returns 1 when the sequence is finished and it hasn't to read - * more characters for this sequence, otherwise 0 - */ -int -eschandle(uchar ascii) -{ - switch (ascii) { - case '[': - term.esc |= ESC_CSI; - return 0; - case '#': - term.esc |= ESC_TEST; - return 0; - case '%': - term.esc |= ESC_UTF8; - return 0; - case 'P': /* DCS -- Device Control String */ - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - case ']': /* OSC -- Operating System Command */ - case 'k': /* old title set compatibility */ - tstrsequence(ascii); - return 0; - case 'n': /* LS2 -- Locking shift 2 */ - case 'o': /* LS3 -- Locking shift 3 */ - term.charset = 2 + (ascii - 'n'); - break; - case '(': /* GZD4 -- set primary charset G0 */ - case ')': /* G1D4 -- set secondary charset G1 */ - case '*': /* G2D4 -- set tertiary charset G2 */ - case '+': /* G3D4 -- set quaternary charset G3 */ - term.icharset = ascii - '('; - term.esc |= ESC_ALTCHARSET; - return 0; - case 'D': /* IND -- Linefeed */ - if (term.c.y == term.bot) { - tscrollup(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y+1); - } - break; - case 'E': /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 'H': /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 'M': /* RI -- Reverse index */ - if (term.c.y == term.top) { - tscrolldown(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y-1); - } - break; - case 'Z': /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'c': /* RIS -- Reset to initial state */ - treset(); - resettitle(); - xloadcols(); - xsetmode(0, MODE_HIDE); - break; - case '=': /* DECPAM -- Application keypad */ - xsetmode(1, MODE_APPKEYPAD); - break; - case '>': /* DECPNM -- Normal keypad */ - xsetmode(0, MODE_APPKEYPAD); - break; - case '7': /* DECSC -- Save Cursor */ - tcursor(CURSOR_SAVE); - break; - case '8': /* DECRC -- Restore Cursor */ - tcursor(CURSOR_LOAD); - break; - case '\\': /* ST -- String Terminator */ - if (term.esc & ESC_STR_END) - strhandle(); - break; - default: - fprintf(stderr, "erresc: unknown sequence ESC 0x%02X '%c'\n", - (uchar) ascii, isprint(ascii)? ascii:'.'); - break; - } - return 1; -} - -void -tputc(Rune u) -{ - char c[UTF_SIZ]; - int control; - int width, len; - Glyph *gp; - - control = ISCONTROL(u); - if (u < 127 || !IS_SET(MODE_UTF8)) { - c[0] = u; - width = len = 1; - } else { - len = utf8encode(u, c); - if (!control && (width = wcwidth(u)) == -1) - width = 1; - } - - if (IS_SET(MODE_PRINT)) - tprinter(c, len); - - /* - * STR sequence must be checked before anything else - * because it uses all following characters until it - * receives a ESC, a SUB, a ST or any other C1 control - * character. - */ - if (term.esc & ESC_STR) { - if (u == '\a' || u == 030 || u == 032 || u == 033 || - ISCONTROLC1(u)) { - term.esc &= ~(ESC_START|ESC_STR); - term.esc |= ESC_STR_END; - goto check_control_code; - } - - if (strescseq.len+len >= strescseq.siz) { - /* - * Here is a bug in terminals. If the user never sends - * some code to stop the str or esc command, then st - * will stop responding. But this is better than - * silently failing with unknown characters. At least - * then users will report back. - * - * In the case users ever get fixed, here is the code: - */ - /* - * term.esc = 0; - * strhandle(); - */ - if (strescseq.siz > (SIZE_MAX - UTF_SIZ) / 2) - return; - strescseq.siz *= 2; - strescseq.buf = xrealloc(strescseq.buf, strescseq.siz); - } - - memmove(&strescseq.buf[strescseq.len], c, len); - strescseq.len += len; - return; - } - -check_control_code: - /* - * Actions of control codes must be performed as soon they arrive - * because they can be embedded inside a control sequence, and - * they must not cause conflicts with sequences. - */ - if (control) { - /* in UTF-8 mode ignore handling C1 control characters */ - if (IS_SET(MODE_UTF8) && ISCONTROLC1(u)) - return; - tcontrolcode(u); - /* - * control codes are not shown ever - */ - if (!term.esc) - term.lastc = 0; - return; - } else if (term.esc & ESC_START) { - if (term.esc & ESC_CSI) { - csiescseq.buf[csiescseq.len++] = u; - if (BETWEEN(u, 0x40, 0x7E) - || csiescseq.len >= \ - sizeof(csiescseq.buf)-1) { - term.esc = 0; - csiparse(); - csihandle(); - } - return; - } else if (term.esc & ESC_UTF8) { - tdefutf8(u); - } else if (term.esc & ESC_ALTCHARSET) { - tdeftran(u); - } else if (term.esc & ESC_TEST) { - tdectest(u); - } else { - if (!eschandle(u)) - return; - /* sequence already finished */ - } - term.esc = 0; - /* - * All characters which form part of a sequence are not - * printed - */ - return; - } - if (selected(term.c.x, term.c.y)) - selclear(); - - gp = &term.line[term.c.y][term.c.x]; - if (IS_SET(MODE_WRAP) && (term.c.state & CURSOR_WRAPNEXT)) { - gp->mode |= ATTR_WRAP; - tnewline(1); - gp = &term.line[term.c.y][term.c.x]; - } - - if (IS_SET(MODE_INSERT) && term.c.x+width < term.col) { - memmove(gp+width, gp, (term.col - term.c.x - width) * sizeof(Glyph)); - gp->mode &= ~ATTR_WIDE; - } - - if (term.c.x+width > term.col) { - if (IS_SET(MODE_WRAP)) - tnewline(1); - else - tmoveto(term.col - width, term.c.y); - gp = &term.line[term.c.y][term.c.x]; - } - - tsetchar(u, &term.c.attr, term.c.x, term.c.y); - term.lastc = u; - - if (width == 2) { - gp->mode |= ATTR_WIDE; - if (term.c.x+1 < term.col) { - if (gp[1].mode == ATTR_WIDE && term.c.x+2 < term.col) { - gp[2].u = ' '; - gp[2].mode &= ~ATTR_WDUMMY; - } - gp[1].u = '\0'; - gp[1].mode = ATTR_WDUMMY; - } - } - if (term.c.x+width < term.col) { - tmoveto(term.c.x+width, term.c.y); - } else { - term.c.state |= CURSOR_WRAPNEXT; - } -} - -int -twrite(const char *buf, int buflen, int show_ctrl) -{ - int charsize; - Rune u; - int n; - - for (n = 0; n < buflen; n += charsize) { - if (IS_SET(MODE_UTF8)) { - /* process a complete utf8 char */ - charsize = utf8decode(buf + n, &u, buflen - n); - if (charsize == 0) - break; - } else { - u = buf[n] & 0xFF; - charsize = 1; - } - if (show_ctrl && ISCONTROL(u)) { - if (u & 0x80) { - u &= 0x7f; - tputc('^'); - tputc('['); - } else if (u != '\n' && u != '\r' && u != '\t') { - u ^= 0x40; - tputc('^'); - } - } - tputc(u); - } - return n; -} - -void -tresize(int col, int row) -{ - int i; - int minrow = MIN(row, term.row); - int mincol = MIN(col, term.col); - int *bp; - TCursor c; - - if (col < 1 || row < 1) { - fprintf(stderr, - "tresize: error resizing to %dx%d\n", col, row); - return; - } - - /* - * slide screen to keep cursor where we expect it - - * tscrollup would work here, but we can optimize to - * memmove because we're freeing the earlier lines - */ - for (i = 0; i <= term.c.y - row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - /* ensure that both src and dst are not NULL */ - if (i > 0) { - memmove(term.line, term.line + i, row * sizeof(Line)); - memmove(term.alt, term.alt + i, row * sizeof(Line)); - } - for (i += row; i < term.row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - - /* resize to new height */ - term.line = xrealloc(term.line, row * sizeof(Line)); - term.alt = xrealloc(term.alt, row * sizeof(Line)); - term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty)); - term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs)); - - /* resize each row to new width, zero-pad if needed */ - for (i = 0; i < minrow; i++) { - term.line[i] = xrealloc(term.line[i], col * sizeof(Glyph)); - term.alt[i] = xrealloc(term.alt[i], col * sizeof(Glyph)); - } - - /* allocate any new rows */ - for (/* i = minrow */; i < row; i++) { - term.line[i] = xmalloc(col * sizeof(Glyph)); - term.alt[i] = xmalloc(col * sizeof(Glyph)); - } - if (col > term.col) { - bp = term.tabs + term.col; - - memset(bp, 0, sizeof(*term.tabs) * (col - term.col)); - while (--bp > term.tabs && !*bp) - /* nothing */ ; - for (bp += tabspaces; bp < term.tabs + col; bp += tabspaces) - *bp = 1; - } - /* update terminal size */ - term.col = col; - term.row = row; - /* reset scrolling region */ - tsetscroll(0, row-1); - /* make use of the LIMIT in tmoveto */ - tmoveto(term.c.x, term.c.y); - /* Clearing both screens (it makes dirty all lines) */ - c = term.c; - for (i = 0; i < 2; i++) { - if (mincol < col && 0 < minrow) { - tclearregion(mincol, 0, col - 1, minrow - 1); - } - if (0 < col && minrow < row) { - tclearregion(0, minrow, col - 1, row - 1); - } - tswapscreen(); - tcursor(CURSOR_LOAD); - } - term.c = c; -} - -void -resettitle(void) -{ - xsettitle(NULL); -} - -void -drawregion(int x1, int y1, int x2, int y2) -{ - int y; - - for (y = y1; y < y2; y++) { - if (!term.dirty[y]) - continue; - - term.dirty[y] = 0; - xdrawline(term.line[y], x1, y, x2); - } -} - -void -draw(void) -{ - int cx = term.c.x, ocx = term.ocx, ocy = term.ocy; - - if (!xstartdraw()) - return; - - /* adjust cursor position */ - LIMIT(term.ocx, 0, term.col-1); - LIMIT(term.ocy, 0, term.row-1); - if (term.line[term.ocy][term.ocx].mode & ATTR_WDUMMY) - term.ocx--; - if (term.line[term.c.y][cx].mode & ATTR_WDUMMY) - cx--; - - drawregion(0, 0, term.col, term.row); - xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], - term.ocx, term.ocy, term.line[term.ocy][term.ocx]); - term.ocx = cx; - term.ocy = term.c.y; - xfinishdraw(); - if (ocx != term.ocx || ocy != term.ocy) - xximspot(term.ocx, term.ocy); -} - -void -redraw(void) -{ - tfulldirt(); - draw(); -} diff --git a/st.c.rej b/st.c.rej deleted file mode 100644 index 406fd0e..0000000 --- a/st.c.rej +++ /dev/null @@ -1,13 +0,0 @@ ---- st.c -+++ st.c -@@ -191,8 +198,8 @@ static void tnewline(int); - static void tputtab(int); - static void tputc(Rune); - static void treset(void); --static void tscrollup(int, int); --static void tscrolldown(int, int); -+static void tscrollup(int, int, int); -+static void tscrolldown(int, int, int); - static void tsetattr(int *, int); - static void tsetchar(Rune, Glyph *, int, int); - static void tsetdirt(int, int); diff --git a/x.o b/x.o index b6428f34d219190cf47da0440606793b24432a98..ffc889fda57173e4360c3a2f81ed4eb94c3567f7 100644 GIT binary patch delta 19105 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 delta 19292 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 Z|9DcK{Fnm!#Cw5PA&`yNi}51r{{c?J?a}}M