shithub: purgatorio

ref: a8083462e62459b2ae8a243dc4ba88416eba03b1

View raw version
#include "lib9.h"
#include "draw.h"
#include "kernel.h"
#include "interp.h"

int	_drawdebug;

enum {
	CHECKLOCKING = 0
};

/*
 * Attach, or possibly reattach, to window.
 * If reattaching, maintain value of screen pointer.
 */
int
gengetwindow(Display *d, char *winname, Image **winp, Screen **scrp, int ref)
{
	int n, fd;
	char buf[64+1];
	Image *image;

	fd = libopen(winname, OREAD);
	if(fd<0 || (n=libread(fd, buf, sizeof buf-1))<=0){
		*winp = d->image;
		assert(*winp && (*winp)->chan != 0);
		return 1;
	}
	libclose(fd);
	buf[n] = '\0';
	if(*winp != nil){
		_freeimage1(*winp);
		freeimage((*scrp)->image);
		freescreen(*scrp);
		*scrp = nil;
	}
	image = namedimage(d, buf);
	if(image == 0){
		*winp = nil;
		return -1;
	}
	assert(image->chan != 0);

	*scrp = allocscreen(image, d->white, 0);
	if(*scrp == nil){
		*winp = nil;
		return -1;
	}

	*winp = _allocwindow(*winp, *scrp, insetrect(image->r, Borderwidth), ref, DWhite);
	if(*winp == nil)
		return -1;
	assert((*winp)->chan != 0);
	return 1;
}

#define	NINFO	12*12

Display*
initdisplay(char *dev, char *win, void(*error)(Display*, char*))
{
	char buf[128], info[NINFO+1], *t;
	int datafd, ctlfd, reffd;
	Display *disp;
	Image *image;
	Dir *dir;
	void *q;
	ulong chan;

	fmtinstall('P', Pfmt);
	fmtinstall('R', Rfmt);
	if(dev == 0)
		dev = "/dev";
	if(win == 0)
		win = "/dev";
	if(strlen(dev)>sizeof buf-25 || strlen(win)>sizeof buf-25){
		kwerrstr("initdisplay: directory name too long");
		return nil;
	}
	t = strdup(win);
	if(t == nil)
		return nil;

	q = libqlalloc();
	if(q == nil)
		return nil;

	sprint(buf, "%s/draw/new", dev);
	ctlfd = libopen(buf, ORDWR);
	if(ctlfd < 0){
		if(libbind("#i", dev, MBEFORE) < 0){
    Error1:
			libqlfree(q);
			free(t);
			kwerrstr("initdisplay: %s: %r", buf);
			return 0;
		}
		ctlfd = libopen(buf, ORDWR);
	}
	if(ctlfd < 0)
		goto Error1;
	if(libread(ctlfd, info, sizeof info) < NINFO){
    Error2:
		libclose(ctlfd);
		goto Error1;
	}

	if((chan=strtochan(info+2*12)) == 0){
		kwerrstr("bad channel in %s", buf);
		goto Error2;
	}

	sprint(buf, "%s/draw/%d/data", dev, atoi(info+0*12));
	datafd = libopen(buf, ORDWR);
	if(datafd < 0)
		goto Error2;
	sprint(buf, "%s/draw/%d/refresh", dev, atoi(info+0*12));
	reffd = libopen(buf, OREAD);
	if(reffd < 0){
    Error3:
		libclose(datafd);
		goto Error2;
	}
	strcpy(buf, "allocation failed");
	disp = malloc(sizeof(Display));
	if(disp == 0){
    Error4:
		libclose(reffd);
		goto Error3;
	}
	image = malloc(sizeof(Image));
	if(image == 0){
    Error5:
		free(disp);
		goto Error4;
	}
	memset(image, 0, sizeof(Image));
	memset(disp, 0, sizeof(Display));
	image->display = disp;
	image->id = 0;
	image->chan = chan;
	image->depth = chantodepth(chan);
	image->repl = atoi(info+3*12);
	image->r.min.x = atoi(info+4*12);
	image->r.min.y = atoi(info+5*12);
	image->r.max.x = atoi(info+6*12);
	image->r.max.y = atoi(info+7*12);
	image->clipr.min.x = atoi(info+8*12);
	image->clipr.min.y = atoi(info+9*12);
	image->clipr.max.x = atoi(info+10*12);
	image->clipr.max.y = atoi(info+11*12);
	disp->dirno = atoi(info+0*12);
	disp->datachan = libfdtochan(datafd, ORDWR);
	disp->refchan = libfdtochan(reffd, OREAD);
	disp->ctlchan = libfdtochan(ctlfd, ORDWR);
	if(disp->datachan == nil || disp->refchan == nil || disp->ctlchan == nil)
		goto Error4;
	disp->bufsize = Displaybufsize;	/* TO DO: iounit(datafd) */
	if(disp->bufsize <= 0)
		disp->bufsize = Displaybufsize;
	if(disp->bufsize < 512){
		kwerrstr("iounit %d too small", disp->bufsize);
		goto Error5;
	}
	/* TO DO: allocate buffer */

	libclose(datafd);
	libclose(reffd);
	disp->image = image;
	disp->bufp = disp->buf;
	disp->error = error;
	disp->chan = image->chan;
	disp->depth = image->depth;
	disp->windir = t;
	disp->devdir = strdup(dev);
	disp->qlock = q;
	libqlock(q);
	disp->white = allocimage(disp, Rect(0, 0, 1, 1), GREY1, 1, DWhite);
	disp->black = allocimage(disp, Rect(0, 0, 1, 1), GREY1, 1, DBlack);
	disp->opaque = allocimage(disp, Rect(0, 0, 1, 1), GREY1, 1, DWhite);
	disp->transparent = allocimage(disp, Rect(0, 0, 1, 1), GREY1, 1, DBlack);
	if(disp->white == nil || disp->black == nil || disp->opaque == nil || disp->transparent == nil){
		free(image);
		free(disp->devdir);
		free(disp->white);
		free(disp->black);
		libclose(ctlfd);
		goto Error5;
	}
	if((dir = libdirfstat(ctlfd))!=nil && dir->type=='i'){
		disp->local = 1;
		disp->dataqid = dir->qid.path;
	}
	free(dir);
	libclose(ctlfd);

	if(CHECKLOCKING)
		disp->local = 0;	/* force display locking even for local access */

	assert(disp->chan != 0 && image->chan != 0);
	return disp;
}

/*
 * Call with d unlocked.
 * Note that disp->defaultfont and defaultsubfont are not freed here.
 */
void
closedisplay(Display *disp)
{
	int fd;
	char buf[128];

	if(disp == nil)
		return;
	libqlock(disp->qlock);
	if(disp->oldlabel[0]){
		snprint(buf, sizeof buf, "%s/label", disp->windir);
		fd = libopen(buf, OWRITE);
		if(fd >= 0){
			libwrite(fd, disp->oldlabel, strlen(disp->oldlabel));
			libclose(fd);
		}
	}

	free(disp->devdir);
	free(disp->windir);
	freeimage(disp->white);
	freeimage(disp->black);
	freeimage(disp->opaque);
	freeimage(disp->transparent);
	free(disp->image);
	libchanclose(disp->datachan);
	libchanclose(disp->refchan);
	libchanclose(disp->ctlchan);
	/* should cause refresh slave to shut down */
	libqunlock(disp->qlock);
	libqlfree(disp->qlock);
	free(disp);
}

int
lockdisplay(Display *disp)
{
	if(disp->local)
		return 0;
	if(libqlowner(disp->qlock) != currun()){
		libqlock(disp->qlock);
		return 1;
	}
	return 0;
}

void
unlockdisplay(Display *disp)
{
	if(disp->local)
		return;
	libqunlock(disp->qlock);
}

/* use static buffer to avoid stack bloat */
int
_drawprint(int fd, char *fmt, ...)
{
	int n;
	va_list arg;
	char buf[128];
//	static QLock l;

//	qlock(&l);
	va_start(arg, fmt);
	vseprint(buf, buf+sizeof buf, fmt, arg);
	va_end(arg);
	n = libwrite(fd, buf, strlen(buf));
//	qunlock(&l);
	return n;
}

#ifdef YYY
void
drawerror(Display *d, char *s)
{
	char err[ERRMAX];

	if(d->error)
		d->error(d, s);
	else{
		err[0] = 0;
		errstr(err, sizeof err);
		_drawprint(2, "draw: %s: %s\n", s, err);
		exits(s);
	}
}

static
int
doflush(Display *d)
{
	int n;

	n = d->bufp-d->buf;
	if(n <= 0)
		return 1;

	if(kchanio(d->datachan, d->buf, n, OWRITE) != n){
		if(_drawdebug)
			_drawprint(2, "flushimage fail: d=%p: %r\n", d); /**/
		d->bufp = d->buf;	/* might as well; chance of continuing */
		return -1;
	}
	d->bufp = d->buf;
	return 1;
}

int
flushimage(Display *d, int visible)
{
	if(visible)
		*d->bufp++ = 'v';	/* one byte always reserved for this */
	return doflush(d);
}

uchar*
bufimage(Display *d, int n)
{
	uchar *p;

	if(n<0 || n>Displaybufsize){
		kwerrstr("bad count in bufimage");
		return 0;
	}
	if(d->bufp+n > d->buf+Displaybufsize)
		if(doflush(d) < 0)
			return 0;
	p = d->bufp;
	d->bufp += n;
	return p;
}

#endif