| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 | /* * (C)opyright MMVI Anselm R. Garbe <garbeam at gmail dot com> * See LICENSE file for license details. */#include <stdarg.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <sys/types.h>#include <sys/wait.h>#include <unistd.h>#include <X11/Xatom.h>#include "util.h"static char *shell = NULL;voiderror(char *errstr, ...) {	va_list ap;	va_start(ap, errstr);	vfprintf(stderr, errstr, ap);	va_end(ap);	exit(1);}static voidbad_malloc(unsigned int size){	fprintf(stderr, "fatal: could not malloc() %d bytes\n",			(int) size);	exit(1);}void *emallocz(unsigned int size){	void *res = calloc(1, size);	if(!res)		bad_malloc(size);	return res;}void *emalloc(unsigned int size){	void *res = malloc(size);	if(!res)		bad_malloc(size);	return res;}void *erealloc(void *ptr, unsigned int size){	void *res = realloc(ptr, size);	if(!res)		bad_malloc(size);	return res;}char *estrdup(const char *str){	void *res = strdup(str);	if(!res)		bad_malloc(strlen(str));	return res;}voidfailed_assert(char *a, char *file, int line){	fprintf(stderr, "Assertion \"%s\" failed at %s:%d\n", a, file, line);	abort();}voidswap(void **p1, void **p2){	void *tmp = *p1;	*p1 = *p2;	*p2 = tmp;}voidspawn(Display *dpy, const char *cmd){	if(!shell && !(shell = getenv("SHELL")))		shell = "/bin/sh";	if(!cmd)		return;	if(fork() == 0) {		if(fork() == 0) {			if(dpy)				close(ConnectionNumber(dpy));			setsid();			fprintf(stderr, "gridwm: execlp %s %s -c %s", shell, shell, cmd);			execlp(shell, shell, "-c", cmd, NULL);			fprintf(stderr, "gridwm: execlp %s", cmd);			perror(" failed");		}		exit (0);	}	wait(0);}voidpipe_spawn(char *buf, unsigned int len, Display *dpy, const char *cmd){	unsigned int l, n;	int pfd[2];	if(!shell && !(shell = getenv("SHELL")))		shell = "/bin/sh";	if(!cmd)		return;	if(pipe(pfd) == -1) {		perror("pipe");		exit(1);	}	if(fork() == 0) {		if(dpy)			close(ConnectionNumber(dpy));		setsid();		dup2(pfd[1], STDOUT_FILENO);		close(pfd[0]);		close(pfd[1]);		execlp(shell, shell, "-c", cmd, NULL);		fprintf(stderr, "gridwm: execlp %s", cmd);		perror(" failed");	}	else {		n = 0;		close(pfd[1]);		while(l > n) {			if((l = read(pfd[0], buf + n, len - n)) < 1)				break;			n += l;		}		close(pfd[0]);		buf[n - 1] = 0;	}	wait(0);}unsigned char *getselection(unsigned long offset, unsigned long *len, unsigned long *remain){	Display *dpy;	Atom xa_clip_string;	Window w;	XEvent ev;	Atom typeret;	int format;	unsigned char *data;	unsigned char *result = NULL;	dpy = XOpenDisplay(0);	if(!dpy)		return NULL;	xa_clip_string = XInternAtom(dpy, "_SEL_STRING", False);	w = XCreateSimpleWindow(dpy, DefaultRootWindow(dpy), 10, 10, 200, 200,			1, CopyFromParent, CopyFromParent);	XConvertSelection(dpy, XA_PRIMARY, XA_STRING, xa_clip_string,			w, CurrentTime);	XFlush(dpy);	XNextEvent(dpy, &ev);	if(ev.type == SelectionNotify && ev.xselection.property != None) {		XGetWindowProperty(dpy, w, ev.xselection.property, offset, 4096L, False,				AnyPropertyType, &typeret, &format, len, remain, &data);		if(*len) {			result = emalloc(sizeof(unsigned char) * *len);			memcpy(result, data, *len);		}		XDeleteProperty(dpy, w, ev.xselection.property);	}	XDestroyWindow(dpy, w);	XCloseDisplay(dpy);	return result;}
 |