diff options
Diffstat (limited to 'x.c')
| -rw-r--r-- | x.c | 423 | 
1 files changed, 347 insertions, 76 deletions
| @@ -14,11 +14,13 @@  #include <X11/keysym.h>  #include <X11/Xft/Xft.h>  #include <X11/XKBlib.h> +#include <X11/Xresource.h>  char *argv0;  #include "arg.h"  #include "st.h"  #include "win.h" +#include "hb.h"  /* types used in config.h */  typedef struct { @@ -45,6 +47,19 @@ typedef struct {  	signed char appcursor; /* application cursor */  } Key; +/* Xresources preferences */ +enum resource_type { +	STRING = 0, +	INTEGER = 1, +	FLOAT = 2 +}; + +typedef struct { +	char *name; +	enum resource_type type; +	void *dst; +} ResourcePref; +  /* X modifiers */  #define XK_ANY_MOD    UINT_MAX  #define XK_NO_MOD     0 @@ -55,6 +70,7 @@ static void clipcopy(const Arg *);  static void clippaste(const Arg *);  static void numlock(const Arg *);  static void selpaste(const Arg *); +static void changealpha(const Arg *);  static void zoom(const Arg *);  static void zoomabs(const Arg *);  static void zoomreset(const Arg *); @@ -105,6 +121,7 @@ typedef struct {  	XSetWindowAttributes attrs;  	int scr;  	int isfixed; /* is fixed geometry? */ +	int depth; /* bit depth */  	int l, t; /* left and top offset */  	int gm; /* geometry mask */  } XWindow; @@ -156,6 +173,8 @@ static void xresize(int, int);  static void xhints(void);  static int xloadcolor(int, const char *, Color *);  static int xloadfont(Font *, FcPattern *); +static int xloadsparefont(FcPattern *, int); +static void xloadsparefonts(void);  static void xloadfonts(const char *, double);  static void xunloadfont(Font *);  static void xunloadfonts(void); @@ -163,6 +182,7 @@ static void xsetenv(void);  static void xseturgency(int);  static int evcol(XEvent *);  static int evrow(XEvent *); +static float clamp(float, float, float);  static void expose(XEvent *);  static void visibility(XEvent *); @@ -243,6 +263,7 @@ static char *usedfont = NULL;  static double usedfontsize = 0;  static double defaultfontsize = 0; +static char *opt_alpha = NULL;  static char *opt_class = NULL;  static char **opt_cmd  = NULL;  static char *opt_embed = NULL; @@ -252,6 +273,9 @@ static char *opt_line  = NULL;  static char *opt_name  = NULL;  static char *opt_title = NULL; +static int focused = 0; + +static int oldbutton = 3; /* button event on startup: 3 = release */  static uint buttons; /* bit field of pressed buttons */  void @@ -293,6 +317,18 @@ numlock(const Arg *dummy)  }  void +changealpha(const Arg *arg) +{ +    if((alpha > 0 && arg->f < 0) || (alpha < 1 && arg->f > 0)) +        alpha += arg->f; +    alpha = clamp(alpha, 0.0, 1.0); +    alphaUnfocus = clamp(alpha-alphaOffset, 0.0, 1.0); + +    xloadcols(); +    redraw(); +} + +void  zoom(const Arg *arg)  {  	Arg larg; @@ -306,6 +342,7 @@ zoomabs(const Arg *arg)  {  	xunloadfonts();  	xloadfonts(usedfont, arg->f); +	xloadsparefonts();  	cresize(0, 0);  	redraw();  	xhints(); @@ -344,6 +381,15 @@ evrow(XEvent *e)  	return y / win.ch;  } +float +clamp(float value, float lower, float upper) { +    if(value < lower) +        return lower; +    if(value > upper) +        return upper; +    return value; +} +  void  mousesel(XEvent *e, int done)  { @@ -364,59 +410,51 @@ mousesel(XEvent *e, int done)  void  mousereport(XEvent *e)  { -	int len, btn, code; -	int x = evcol(e), y = evrow(e); -	int state = e->xbutton.state; +    int code; +	int len, x = evcol(e), y = evrow(e), +	    button = e->xbutton.button, state = e->xbutton.state;  	char buf[40];  	static int ox, oy; -	if (e->type == MotionNotify) { +	/* from urxvt */ +	if (e->xbutton.type == MotionNotify) {  		if (x == ox && y == oy)  			return;  		if (!IS_SET(MODE_MOUSEMOTION) && !IS_SET(MODE_MOUSEMANY))  			return; -		/* MODE_MOUSEMOTION: no reporting if no button is pressed */ -		if (IS_SET(MODE_MOUSEMOTION) && buttons == 0) +		/* MOUSE_MOTION: no reporting if no button is pressed */ +		if (IS_SET(MODE_MOUSEMOTION) && oldbutton == 3)  			return; -		/* Set btn to lowest-numbered pressed button, or 12 if no -		 * buttons are pressed. */ -		for (btn = 1; btn <= 11 && !(buttons & (1<<(btn-1))); btn++) -			; -		code = 32; + +		button = oldbutton + 32; +		ox = x; +		oy = y;  	} else { -		btn = e->xbutton.button; -		/* Only buttons 1 through 11 can be encoded */ -		if (btn < 1 || btn > 11) -			return; -		if (e->type == ButtonRelease) { +		if (!IS_SET(MODE_MOUSESGR) && e->xbutton.type == ButtonRelease) { +			button = 3; +		} else { +			button -= Button1; +			if (button >= 3) +				button += 64 - 3; +		} +		if (e->xbutton.type == ButtonPress) { +			oldbutton = button; +			ox = x; +			oy = y; +		} else if (e->xbutton.type == ButtonRelease) { +			oldbutton = 3;  			/* MODE_MOUSEX10: no button release reporting */  			if (IS_SET(MODE_MOUSEX10))  				return; -			/* Don't send release events for the scroll wheel */ -			if (btn == 4 || btn == 5) +			if (button == 64 || button == 65)  				return;  		} -		code = 0;  	} -	ox = x; -	oy = y; - -	/* Encode btn into code. If no button is pressed for a motion event in -	 * MODE_MOUSEMANY, then encode it as a release. */ -	if ((!IS_SET(MODE_MOUSESGR) && e->type == ButtonRelease) || btn == 12) -		code += 3; -	else if (btn >= 8) -		code += 128 + btn - 8; -	else if (btn >= 4) -		code += 64 + btn - 4; -	else -		code += btn - 1; -  	if (!IS_SET(MODE_MOUSEX10)) { -		code += ((state & ShiftMask  ) ?  4 : 0) -		      + ((state & Mod1Mask   ) ?  8 : 0) /* meta key: alt */ -		      + ((state & ControlMask) ? 16 : 0); +		button += ((state & ShiftMask  ) ? 4  : 0) +			+ ((state & Mod4Mask   ) ? 8  : 0) +			+ ((state & ControlMask) ? 16 : 0);  	}  	if (IS_SET(MODE_MOUSESGR)) { @@ -752,7 +790,7 @@ xresize(int col, int row)  	XFreePixmap(xw.dpy, xw.buf);  	xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, -			DefaultDepth(xw.dpy, xw.scr)); +			xw.depth);  	XftDrawChange(xw.draw, xw.buf);  	xclear(0, 0, win.w, win.h); @@ -791,27 +829,39 @@ xloadcolor(int i, const char *name, Color *ncolor)  }  void +xloadalpha(void) +{ +	float const usedAlpha = focused ? alpha : alphaUnfocus; +	if (opt_alpha) alpha = strtof(opt_alpha, NULL); +	dc.col[defaultbg].color.alpha = (unsigned short)(0xffff * usedAlpha); +	dc.col[defaultbg].pixel &= 0x00FFFFFF; +	dc.col[defaultbg].pixel |= (unsigned char)(0xff * usedAlpha) << 24; +} + +void  xloadcols(void)  {  	int i;  	static int loaded;  	Color *cp; -	if (loaded) { -		for (cp = dc.col; cp < &dc.col[dc.collen]; ++cp) -			XftColorFree(xw.dpy, xw.vis, xw.cmap, cp); -	} else { -		dc.collen = MAX(LEN(colorname), 256); +	if (!loaded) { +		dc.collen = 1 + (defaultbg = MAX(LEN(colorname), 256));  		dc.col = xmalloc(dc.collen * sizeof(Color));  	} -	for (i = 0; i < dc.collen; i++) +	for (i = 0; i+1 < dc.collen; i++)  		if (!xloadcolor(i, NULL, &dc.col[i])) {  			if (colorname[i])  				die("could not allocate color '%s'\n", colorname[i]);  			else  				die("could not allocate color %d\n", i);  		} + +	if (dc.collen) // cannot die, as the color is already loaded. +		xloadcolor(background, NULL, &dc.col[defaultbg]); + +	xloadalpha();  	loaded = 1;  } @@ -859,8 +909,8 @@ xclear(int x1, int y1, int x2, int y2)  void  xhints(void)  { -	XClassHint class = {opt_name ? opt_name : termname, -	                    opt_class ? opt_class : termname}; +	XClassHint class = {opt_name ? opt_name : "st", +	                    opt_class ? opt_class : "St"};  	XWMHints wm = {.flags = InputHint, .input = 1};  	XSizeHints *sizeh; @@ -1050,6 +1100,101 @@ xloadfonts(const char *fontstr, double fontsize)  	FcPatternDestroy(pattern);  } +int +xloadsparefont(FcPattern *pattern, int flags) +{ +	FcPattern *match; +	FcResult result; + +	match = FcFontMatch(NULL, pattern, &result); +	if (!match) { +		return 1; +	} + +	if (!(frc[frclen].font = XftFontOpenPattern(xw.dpy, match))) { +		FcPatternDestroy(match); +		return 1; +	} + +	frc[frclen].flags = flags; +	/* Believe U+0000 glyph will present in each default font */ +	frc[frclen].unicodep = 0; +	frclen++; + +	return 0; +} + +void +xloadsparefonts(void) +{ +	FcPattern *pattern; +	double sizeshift, fontval; +	int fc; +	char **fp; + +	if (frclen != 0) +		die("can't embed spare fonts. cache isn't empty"); + +	/* Calculate count of spare fonts */ +	fc = sizeof(font2) / sizeof(*font2); +	if (fc == 0) +		return; + +	/* Allocate memory for cache entries. */ +	if (frccap < 4 * fc) { +		frccap += 4 * fc - frccap; +		frc = xrealloc(frc, frccap * sizeof(Fontcache)); +	} + +	for (fp = font2; fp - font2 < fc; ++fp) { + +		if (**fp == '-') +			pattern = XftXlfdParse(*fp, False, False); +		else +			pattern = FcNameParse((FcChar8 *)*fp); + +		if (!pattern) +			die("can't open spare font %s\n", *fp); + +		if (defaultfontsize > 0) { +			sizeshift = usedfontsize - defaultfontsize; +			if (sizeshift != 0 && +					FcPatternGetDouble(pattern, FC_PIXEL_SIZE, 0, &fontval) == +					FcResultMatch) { +				fontval += sizeshift; +				FcPatternDel(pattern, FC_PIXEL_SIZE); +				FcPatternDel(pattern, FC_SIZE); +				FcPatternAddDouble(pattern, FC_PIXEL_SIZE, fontval); +			} +		} + +		FcPatternAddBool(pattern, FC_SCALABLE, 1); + +		FcConfigSubstitute(NULL, pattern, FcMatchPattern); +		XftDefaultSubstitute(xw.dpy, xw.scr, pattern); + +		if (xloadsparefont(pattern, FRC_NORMAL)) +			die("can't open spare font %s\n", *fp); + +		FcPatternDel(pattern, FC_SLANT); +		FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ITALIC); +		if (xloadsparefont(pattern, FRC_ITALIC)) +			die("can't open spare font %s\n", *fp); + +		FcPatternDel(pattern, FC_WEIGHT); +		FcPatternAddInteger(pattern, FC_WEIGHT, FC_WEIGHT_BOLD); +		if (xloadsparefont(pattern, FRC_ITALICBOLD)) +			die("can't open spare font %s\n", *fp); + +		FcPatternDel(pattern, FC_SLANT); +		FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ROMAN); +		if (xloadsparefont(pattern, FRC_BOLD)) +			die("can't open spare font %s\n", *fp); + +		FcPatternDestroy(pattern); +	} +} +  void  xunloadfont(Font *f)  { @@ -1062,6 +1207,9 @@ xunloadfont(Font *f)  void  xunloadfonts(void)  { +	/* Clear Harfbuzz font cache. */ +	hbunloadfonts(); +  	/* Free the loaded fonts in the font cache.  */  	while (frclen > 0)  		XftFontClose(xw.dpy, frc[--frclen].font); @@ -1134,11 +1282,21 @@ xinit(int cols, int rows)  	Window parent;  	pid_t thispid = getpid();  	XColor xmousefg, xmousebg; +	XWindowAttributes attr; +	XVisualInfo vis; -	if (!(xw.dpy = XOpenDisplay(NULL))) -		die("can't open display\n");  	xw.scr = XDefaultScreen(xw.dpy); -	xw.vis = XDefaultVisual(xw.dpy, xw.scr); + +	if (!(opt_embed && (parent = strtol(opt_embed, NULL, 0)))) { +		parent = XRootWindow(xw.dpy, xw.scr); +		xw.depth = 32; +	} else { +		XGetWindowAttributes(xw.dpy, parent, &attr); +		xw.depth = attr.depth; +	} + +	XMatchVisualInfo(xw.dpy, xw.scr, xw.depth, TrueColor, &vis); +	xw.vis = vis.visual;  	/* font */  	if (!FcInit()) @@ -1147,8 +1305,11 @@ xinit(int cols, int rows)  	usedfont = (opt_font == NULL)? font : opt_font;  	xloadfonts(usedfont, 0); +	/* spare fonts */ +	xloadsparefonts(); +  	/* colors */ -	xw.cmap = XDefaultColormap(xw.dpy, xw.scr); +	xw.cmap = XCreateColormap(xw.dpy, parent, xw.vis, None);  	xloadcols();  	/* adjust fixed window geometry */ @@ -1168,19 +1329,15 @@ xinit(int cols, int rows)  		| ButtonMotionMask | ButtonPressMask | ButtonReleaseMask;  	xw.attrs.colormap = xw.cmap; -	if (!(opt_embed && (parent = strtol(opt_embed, NULL, 0)))) -		parent = XRootWindow(xw.dpy, xw.scr);  	xw.win = XCreateWindow(xw.dpy, parent, xw.l, xw.t, -			win.w, win.h, 0, XDefaultDepth(xw.dpy, xw.scr), InputOutput, +			win.w, win.h, 0, xw.depth, InputOutput,  			xw.vis, CWBackPixel | CWBorderPixel | CWBitGravity  			| CWEventMask | CWColormap, &xw.attrs);  	memset(&gcvalues, 0, sizeof(gcvalues));  	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 = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, xw.depth); +	dc.gc = XCreateGC(xw.dpy, xw.buf, GCGraphicsExposures, &gcvalues);  	XSetForeground(xw.dpy, dc.gc, dc.col[defaultbg].pixel);  	XFillRectangle(xw.dpy, xw.buf, dc.gc, 0, 0, win.w, win.h); @@ -1237,6 +1394,8 @@ xinit(int cols, int rows)  	xsel.xtarget = XInternAtom(xw.dpy, "UTF8_STRING", 0);  	if (xsel.xtarget == None)  		xsel.xtarget = XA_STRING; + +	boxdraw_xinit(xw.dpy, xw.cmap, xw.draw, xw.vis);  }  int @@ -1261,7 +1420,7 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x  		mode = glyphs[i].mode;  		/* Skip dummy wide-character spacing. */ -		if (mode == ATTR_WDUMMY) +		if (mode & ATTR_WDUMMY)  			continue;  		/* Determine font for glyph if different from previous glyph. */ @@ -1283,8 +1442,13 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x  			yp = winy + font->ascent;  		} -		/* Lookup character index with default font. */ -		glyphidx = XftCharIndex(xw.dpy, font->match, rune); +		if (mode & ATTR_BOXDRAW) { +			/* minor shoehorning: boxdraw uses only this ushort */ +			glyphidx = boxdrawindex(&glyphs[i]); +		} else { +			/* Lookup character index with default font. */ +			glyphidx = XftCharIndex(xw.dpy, font->match, rune); +		}  		if (glyphidx) {  			specs[numspecs].font = font->match;  			specs[numspecs].glyph = glyphidx; @@ -1368,6 +1532,9 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x  		numspecs++;  	} +	/* Harfbuzz transformation for ligatures. */ +	hbtransform(specs, glyphs, len, x, y); +  	return numspecs;  } @@ -1488,13 +1655,17 @@ xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, i  	r.width = width;  	XftDrawSetClipRectangles(xw.draw, winx, winy, &r, 1); -	/* Render the glyphs. */ -	XftDrawGlyphFontSpec(xw.draw, fg, specs, len); +	if (base.mode & ATTR_BOXDRAW) { +		drawboxes(winx, winy, width / len, win.ch, fg, bg, specs, len); +	} else { +		/* Render the glyphs. */ +		XftDrawGlyphFontSpec(xw.draw, fg, specs, len); +	}  	/* Render underline and strikethrough. */  	if (base.mode & ATTR_UNDERLINE) { -		XftDrawRect(xw.draw, fg, winx, winy + dc.font.ascent * chscale + 1, -				width, 1); +		XftDrawRect(xw.draw, fg, winx, winy + dc.font.ascent + 1, +                width, 1);  	}  	if (base.mode & ATTR_STRUCK) { @@ -1517,14 +1688,17 @@ xdrawglyph(Glyph g, int x, int y)  }  void -xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og) +xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og, Line line, int len)  {  	Color drawcol;  	/* remove the old cursor */  	if (selected(ox, oy))  		og.mode ^= ATTR_REVERSE; -	xdrawglyph(og, ox, oy); + +	/* Redraw the line where cursor was previously. +	 * It will restore the ligatures broken by the cursor. */ +	xdrawline(line, 0, oy, len);  	if (IS_SET(MODE_HIDE))  		return; @@ -1532,7 +1706,7 @@ xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og)  	/*  	 * Select the right color for the right mode.  	 */ -	g.mode &= ATTR_BOLD|ATTR_ITALIC|ATTR_UNDERLINE|ATTR_STRUCK|ATTR_WIDE; +	g.mode &= ATTR_BOLD|ATTR_ITALIC|ATTR_UNDERLINE|ATTR_STRUCK|ATTR_WIDE|ATTR_BOXDRAW;  	if (IS_SET(MODE_REVERSE)) {  		g.mode |= ATTR_REVERSE; @@ -1777,12 +1951,22 @@ focus(XEvent *ev)  		xseturgency(0);  		if (IS_SET(MODE_FOCUS))  			ttywrite("\033[I", 3, 0); +		if (!focused) { +			focused = 1; +			xloadcols(); +			tfulldirt(); +		}  	} else {  		if (xw.ime.xic)  			XUnsetICFocus(xw.ime.xic);  		win.mode &= ~MODE_FOCUSED;  		if (IS_SET(MODE_FOCUS))  			ttywrite("\033[O", 3, 0); +		if (focused) { +			focused = 0; +			xloadcols(); +			tfulldirt(); +		}  	}  } @@ -1834,8 +2018,10 @@ kpress(XEvent *ev)  {  	XKeyEvent *e = &ev->xkey;  	KeySym ksym; -	char buf[64], *customkey; -	int len; +	char *buf = NULL, *customkey; +	int len = 0; +	int buf_size = 64; +	int critical = - 1;  	Rune c;  	Status status;  	Shortcut *bp; @@ -1843,27 +2029,44 @@ kpress(XEvent *ev)  	if (IS_SET(MODE_KBDLOCK))  		return; -	if (xw.ime.xic) -		len = XmbLookupString(xw.ime.xic, e, buf, sizeof buf, &ksym, &status); -	else -		len = XLookupString(e, buf, sizeof buf, &ksym, NULL); +reallocbuf: +	if (critical > 0) +		goto cleanup; +	if (buf) +		free(buf); + +	buf = xmalloc((buf_size) * sizeof(char)); +	critical += 1; + +	if (xw.ime.xic) { +		len = XmbLookupString(xw.ime.xic, e, buf, buf_size, &ksym, &status); +		if (status == XBufferOverflow) { +			buf_size = len; +			goto reallocbuf; +		} +	} else { +		// Not sure how to fix this and if it is fixable +		// but at least it does write something into the buffer +		// so it is not as critical +		len = XLookupString(e, buf, buf_size, &ksym, NULL); +	}  	/* 1. shortcuts */  	for (bp = shortcuts; bp < shortcuts + LEN(shortcuts); bp++) {  		if (ksym == bp->keysym && match(bp->mod, e->state)) {  			bp->func(&(bp->arg)); -			return; +			goto cleanup;  		}  	}  	/* 2. custom keys from config.h */  	if ((customkey = kmap(ksym, e->state))) {  		ttywrite(customkey, strlen(customkey), 1); -		return; +		goto cleanup;  	}  	/* 3. composed string from input method */  	if (len == 0) -		return; +		goto cleanup;  	if (len == 1 && e->state & Mod1Mask) {  		if (IS_SET(MODE_8BIT)) {  			if (*buf < 0177) { @@ -1876,7 +2079,11 @@ kpress(XEvent *ev)  			len = 2;  		}  	} -	ttywrite(buf, len, 1); +	if (len <= buf_size) +		ttywrite(buf, len, 1); +cleanup: +	if (buf) +		free(buf);  }  void @@ -2011,6 +2218,59 @@ run(void)  	}  } +int +resource_load(XrmDatabase db, char *name, enum resource_type rtype, void *dst) +{ +	char **sdst = dst; +	int *idst = dst; +	float *fdst = dst; + +	char fullname[256]; +	char fullclass[256]; +	char *type; +	XrmValue ret; + +	snprintf(fullname, sizeof(fullname), "%s.%s", +			opt_name ? opt_name : "st", name); +	snprintf(fullclass, sizeof(fullclass), "%s.%s", +			opt_class ? opt_class : "St", name); +	fullname[sizeof(fullname) - 1] = fullclass[sizeof(fullclass) - 1] = '\0'; + +	XrmGetResource(db, fullname, fullclass, &type, &ret); +	if (ret.addr == NULL || strncmp("String", type, 64)) +		return 1; + +	switch (rtype) { +	case STRING: +		*sdst = ret.addr; +		break; +	case INTEGER: +		*idst = strtoul(ret.addr, NULL, 10); +		break; +	case FLOAT: +		*fdst = strtof(ret.addr, NULL); +		break; +	} +	return 0; +} + +void +config_init(void) +{ +	char *resm; +	XrmDatabase db; +	ResourcePref *p; + +	XrmInitialize(); +	resm = XResourceManagerString(xw.dpy); +	if (!resm) +		return; + +	db = XrmGetStringDatabase(resm); +	for (p = resources; p < resources + LEN(resources); p++) +		resource_load(db, p->name, p->type, p->dst); +} +  void  usage(void)  { @@ -2035,6 +2295,9 @@ main(int argc, char *argv[])  	case 'a':  		allowaltscreen = 0;  		break; +	case 'A': +		opt_alpha = EARGF(usage()); +		break;  	case 'c':  		opt_class = EARGF(usage());  		break; @@ -2084,8 +2347,15 @@ run:  	setlocale(LC_CTYPE, "");  	XSetLocaleModifiers(""); + +	if(!(xw.dpy = XOpenDisplay(NULL))) +		die("Can't open display\n"); + +	config_init();  	cols = MAX(cols, 1);  	rows = MAX(rows, 1); +	defaultbg = MAX(LEN(colorname), 256); +	alphaUnfocus = alpha-alphaOffset;  	tnew(cols, rows);  	xinit(cols, rows);  	xsetenv(); @@ -2094,3 +2364,4 @@ run:  	return 0;  } + | 
