shithub: riscv

ref: ec1976eb0521d668696c6b5946c045bdca6245b0
dir: /sys/src/games/mines/ghost.c/

View raw version
#include <u.h>
#include <libc.h>
#include <draw.h>
#include <event.h>
#include <mp.h>
#include <cursor.h>
#include "dat.h"
#include "fns.h"

enum { TURBO = 0 };

static Cursor bunny = {
	.set = {0x0,0x32,0x60,0x76,0x40,0x6e,0x3,0xec,0xf,0xfc,0x1f,0xf8,0x27,0x9c,0x3,0xc,0x33,0xcc,0x73,0xce,0x67,0x9e,0x7f,0xfe,0x7f,0xfe,0x7f,0xfe,0x6e,0x76},
	.clr = {0xf8,0xcd,0x90,0x89,0xa3,0x91,0xcc,0x12,0x90,0x2,0x20,0x6,0x58,0x62,0x7c,0xf2,0x4c,0x33,0x8c,0x31,0x98,0x61,0x80,0x1,0x80,0x1,0x80,0x1,0x91,0x89}
};

enum {
	MEmpty,
	MMine,
	MUnknown,
	NState,
};

typedef struct Ghost Ghost;
typedef struct CList CList;
typedef struct CGroup CGroup;

struct Ghost {
	QLock;
	Rendez;
	int active, wait;
	int pid;
	
	enum { NOTARG, LTARG, RTARG } targettype;
	Point target;
	int generation;
	
	int nmines;
	uchar *f;
	
	int w, h;
	
	int cursor;
	Point moves[3];
	int nmoves;
	Point cpos;
};

struct CList {
	int mines, pts;
	int pt[8];
	int next;
};

struct CGroup {
	CList *cl;
	int ncl;
	mpint **cnt;
	int max;
	int cur;
};

static Ghost ghost;

static uchar ***
neighbours(void)
{
	int x, y, p;
	uchar ***n;

	n = calloc(sizeof(void*), ghost.w);
	for(x = 0; x < ghost.w; x++){
		n[x] = calloc(sizeof(void*), ghost.h);
		for(y = 0; y < ghost.h; y++)
			n[x][y] = calloc(sizeof(uchar), NState);
	}
	
	for(y = 0; y < ghost.h; y++)
		for(x = 0; x < ghost.w; x++){
			p = ghost.f[x + y * ghost.w];
			if(x > 0 && y > 0) n[x-1][y-1][p]++;
			if(y > 0) n[x][y-1][p]++;
			if(x < ghost.w-1 && y > 0) n[x+1][y-1][p]++;
			if(x > 0) n[x-1][y][p]++;
			if(x < ghost.w-1) n[x+1][y][p]++;
			if(x > 0 && y < ghost.h-1) n[x-1][y+1][p]++;
			if(y < ghost.h-1) n[x][y+1][p]++;
			if(x < ghost.w-1 && y < ghost.h-1) n[x+1][y+1][p]++;
		}
	return n;
}

static void
freeneighbours(uchar ***n)
{
	int x, y;

	for(x = 0; x < ghost.w; x++){
		for(y = 0; y < ghost.h; y++)
			free(n[x][y]);
		free(n[x]);
	}
	free(n);
}

static void
printcl(CList *cl, int ncl)
{
	int i, j;
	
	for(i = 0; i < ncl; i++){
		print("%d: ", cl[i].mines);
		for(j = 0; j < cl[i].pts; j++)
			print("%d ", cl[i].pt[j]);
		print("\n");
	}
	print("--\n");
}

static void
mklists(CList **clp, int *nclp)
{
	CList *c, *cl;
	int ncl, x, y;
	uchar ***nei;
	
	cl = nil;
	ncl = 0;
	nei = neighbours();
	for(y = 0; y < ghost.h; y++)
		for(x = 0; x < ghost.w; x++)
			if(MineField[x][y].Picture <= Empty8 && nei[x][y][MUnknown] > 0){
				cl = realloc(cl, (ncl + 1) * sizeof(CList));
				c = &cl[ncl++];
				memset(c, 0, sizeof(CList));
				c->mines = MineField[x][y].Picture - nei[x][y][MMine];
				if(x > 0 && y > 0 && ghost.f[(x-1)+(y-1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x-1)+(y-1)*ghost.w;
				if(y > 0 && ghost.f[(x)+(y-1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x)+(y-1)*ghost.w;
				if(x < ghost.w-1 && y > 0 && ghost.f[(x+1)+(y-1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x+1)+(y-1)*ghost.w;
				if(x > 0 && ghost.f[(x-1)+(y)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x-1)+(y)*ghost.w;
				if(x < ghost.w-1 && ghost.f[(x+1)+(y)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x+1)+(y)*ghost.w;
				if(x > 0 && y < ghost.h-1 && ghost.f[(x-1)+(y+1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x-1)+(y+1)*ghost.w;
				if(y < ghost.h-1 && ghost.f[(x)+(y+1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x)+(y+1)*ghost.w;
				if(x < ghost.w-1 && y < ghost.h-1 && ghost.f[(x+1)+(y+1)*ghost.w] == MUnknown)
					c->pt[c->pts++] = (x+1)+(y+1)*ghost.w;
			}
	freeneighbours(nei);
	*clp = cl;
	*nclp = ncl;
}

static int
ismember(CList *c, int p)
{
	int i;
	
	for(i = 0; i < c->pts; i++)
		if(c->pt[i] == p)
			return 1;
	return 0;
}

static void
splitknown(CList **clp, int *nclp, int i, int *lastq)
{
	int j;
	CList *cl;
	int ncl;
	
	cl = *clp;
	ncl = *nclp;
	cl = realloc(cl, sizeof(CList) * (ncl + cl[i].pts - 1));
	memset(cl + ncl, 0, sizeof(CList) * (cl[i].pts - 1));
	for(j = 1; j < cl[i].pts; j++){
		cl[ncl - 1 + j].mines = cl[i].pts == cl[i].mines;
		cl[ncl - 1 + j].pts = 1;
		cl[ncl - 1 + j].pt[0] = cl[i].pt[j];
		cl[*lastq].next = ncl - 1 + j;
		*lastq = ncl - 1 + j;
	}
	cl[i].mines = cl[i].pts == cl[i].mines;
	*nclp += cl[i].pts - 1;
	cl[i].pts = 1;
	*clp = cl;
}

static int
merge(CList **clp, int *nclp, int start, int split)
{
	int i, j, k, l, p;
	CList *cl, *q, *r;
	int qi, lastq;
	int zero;

	cl = *clp;
	qi = -1;
	lastq = -1;
	zero = 0;
	for(i = 0; i < *nclp; i++)
		if(cl[i].pts == 0 || start >= 0 && start != i)
			cl[i].next = -2;
		else{
			cl[i].next = -1;
			if(lastq >= 0)
				cl[lastq].next = i;
			else
				qi = i;
			lastq = i;
		}
	while(qi >= 0){
		q = &cl[qi];
		for(i = 0; i < *nclp; i++){
			r = &cl[i];
			if(r == q || r->pts < q->pts) continue;
			for(j = k = 0; j < q->pts; j++, k++){
				p = q->pt[j];
				if(r->pt[r->pts - 1] < p)
					goto next;
				while(r->pt[k] < p)
					k++;
				if(r->pt[k] > p)
					goto next;
			}
			for(j = k = l = 0; j < q->pts; j++, k++){
				p = q->pt[j];
				while(r->pt[k] < p)
					r->pt[l++] = r->pt[k++];
			}
			while(k < r->pts)
				r->pt[l++] = r->pt[k++];
			r->pts = l;
			r->mines -= q->mines;
			if(r->mines < 0 || r->mines > r->pts){
				*clp = cl;
				return -1;
			}
			if(r->pts == 0 && r->mines == 0)
				zero++;
			if(r->next == -2 && r->pts > 0){
				r->next = -1;
				cl[lastq].next = i;
				lastq = i;
			}
			if(split && r->pts > 1 && (r->pts == r->mines || r->mines == 0)){
				splitknown(&cl, nclp, i, &lastq);
				q = &cl[qi];
			}
		next: ;
		}
		qi = q->next;
		q->next = -2;
	}
	if(zero != 0){
		for(i = 0, j = 0; i < *nclp; i++)
			if(cl[i].mines != 0 || cl[i].pts != 0)
				cl[j++] = cl[i];
		*nclp = j;
	}
	*clp = cl;
	return 0;
}

static void
countref(CList *cl, int ncl, uchar *ref)
{
	int i, j;
	
	memset(ref, 0, ghost.w * ghost.h);
	for(i = 0; i < ncl; i++)
		for(j = 0; j < cl[i].pts; j++)
			ref[cl[i].pt[j]]++;
}

static int *
clgroup(CList *cl, int ncl, uchar *ref)
{
	int i, j, k, og, ng;
	int *g;
	
	g = calloc(ncl, sizeof(int));
	for(i = 0; i < ncl; i++)
		g[i] = i;
start:
	for(i = 0; i < ncl; i++)
		for(j = 0; j < cl[i].pts; j++){
			if(ref[cl[i].pt[j]] == 1) continue;
			for(k = 0; k < ncl; k++)
				if(g[i] != g[k] && ismember(&cl[k], cl[i].pt[j]))
					break;
			if(k == ncl) continue;
			if(g[i] < g[k]){
				og = g[i];
				ng = g[k];
			}else{
				og = g[k];
				ng = g[i];
			}
			for(k = 0; k < ncl; k++)
				if(g[k] == og)
					g[k] = ng;
			goto start;
		}
	return g;
}

static int
groupmin(int *g, int ncl)
{
	int i, j, og, ng;
	u32int *bs;

	bs = calloc(sizeof(u32int), ncl + 31 >> 5);
	for(i = 0; i < ncl; i++)
		bs[g[i]>>5] |= 1<<(g[i] & 31);
	for(i = 0; i < ncl; i++){
		for(j = 0; j < g[i]; j++)
			if((bs[j>>5] & 1<<(j & 31)) == 0)
				break;
		if(j == g[i]) continue;
		og = g[i];
		ng = j;
		for(j = 0; j < ncl; j++)
			if(g[j] == og)
				g[j] = ng;
		bs[og>>5] &= ~(1<<(og & 31));
		bs[ng>>5] |= 1<<(ng & 31);
	}
	og = 0;
	for(i = 0; i < ncl; i++)
		if(g[i] + 1 > og)
			og = g[i] + 1;
	free(bs);
	return og;
}

static int
mkgroups(CList *cl, int ncl, uchar *ref, CGroup **gp)
{
	CGroup *g, *p;
	int i, j, k, ng;
	int *gr;
	
	gr = clgroup(cl, ncl, ref);
	ng = groupmin(gr, ncl);
	g = calloc(sizeof(CGroup), ng);
	for(i = 0; i < ng; i++){
		p = &g[i];
		for(j = 0; j < ncl; j++)
			if(gr[j] == i)
				p->ncl++;
		p->cl = calloc(sizeof(CList), p->ncl);
		for(j = 0, k = 0; j < ncl; j++)
			if(gr[j] == i)
				p->cl[k++] = cl[j];
	}
	free(gr);
	*gp = g;
	return ng;
}

static void
freegroups(CGroup *g, int ng)
{
	int i, j;
	
	for(i = 0; i < ng; i++){
		for(j = 0; j <= g[i].max; j++)
			mpfree(g[i].cnt[j]);
		free(g[i].cl);
		free(g[i].cnt);
	}
	free(g);
}

static void
binom(mpint *acc, mpint *temp, int n, int k)
{
	int i;
	
	for(i = 0; i < k; i++){
		itomp(n - i, temp);
		mpmul(acc, temp, acc);
	}
	for(i = 2; i <= k; i++){
		itomp(i, temp);
		mpdiv(acc, temp, acc, nil);
	}
}

static void countcomb(CList *cl, int ncl, mpint **);

static void
cltry(CList *cl, int ncl, int p, int v, mpint **cnt)
{
	CList *cm;
	int ncm;
	
	cm = calloc(ncl + 1, sizeof(CList));
	memcpy(cm, cl, ncl * sizeof(CList));
	memset(&cm[ncl], 0, sizeof(CList));
	cm[ncl].mines = v;
	cm[ncl].pts = 1;
	cm[ncl].pt[0] = p;
	ncm = ncl + 1;
	if(merge(&cm, &ncm, ncl, 1) < 0){
		free(cm);
		return;
	}
	countcomb(cm, ncm, cnt);
	free(cm);
}

static void
countcomb(CList *cl, int ncl, mpint **cnt)
{
	uchar *ref;
	int p, i, j, nm;
	mpint *rc, *c0;
	
	ref = calloc(ghost.w, ghost.h);
	countref(cl, ncl, ref);
	for(i = 0; i < ncl; i++)
		for(j = 0; j < cl[i].pts; j++){
			p = cl[i].pt[j];
			if(ref[p] > 1){
				cltry(cl, ncl, p, 0, cnt);
				cltry(cl, ncl, p, 1, cnt);
				free(ref);
				return;
			}
		}
	rc = itomp(1, nil);
	c0 = mpnew(0);
	nm = 0;
	for(i = 0; i < ncl; i++){
		nm += cl[i].mines;
		binom(rc, c0, cl[i].pts, cl[i].mines);
	}
	if(mpcmp(rc, mpzero) != 0)
		if(cnt[nm] == nil)
			cnt[nm] = rc;
		else{
			mpadd(cnt[nm], rc, cnt[nm]);
			mpfree(rc);
		}
	else
		mpfree(rc);
	mpfree(c0);
	free(ref);
}

static mpint *
totcomb(int tot, int nf, CGroup *g, int ng)
{
	int i, j, s, carry;
	mpint *r, *v, *t, *m;
	
	s = 0;
	for(i = 0; i < ng; i++){
		for(j = 0; j <= g[i].max; j++)
			if(g[i].cnt[j] != nil)
				break;
		if(j > g[i].max)
			return mpnew(0);
		g[i].cur = j;
		s += j;
	}
	if(s > tot)
		return mpnew(0);
	r = mpnew(0);
	v = mpnew(0);
	t = mpnew(0);
	do{
		itomp(1, v);
		s = 0;
		for(i = 0; i < ng; i++){
			m = g[i].cnt[g[i].cur];
			mpmul(v, m, v);
			s += g[i].cur;
		}
		if(s <= tot){
			binom(v, t, nf, tot - s);
			mpadd(r, v, r);
		}
		for(i = 0; i < ng; i++){
			carry = 0;
			do
				if(++g[i].cur > g[i].max){
					g[i].cur = 0;
					carry = 1;
				}
			while(g[i].cnt[g[i].cur] == nil);
			if(!carry) break;
		}
	}while(i < ng);
	mpfree(v);
	mpfree(t);
	return r;
}

static int
freefields(uchar *ref)
{
	int nf, p;

	nf = 0;
	for(p = 0; p < ghost.w * ghost.h; p++)
		if(ref[p] == 0 && ghost.f[p] == MUnknown)
			nf++;
	return nf;
}

static int
basecounts(CGroup *g, int ng)
{
	int i, j;
	int allmax;
	CGroup *p;

	allmax = 0;
	for(i = 0; i < ng; i++){
		p = &g[i];
		p->max = 0;
		for(j = 0; j < p->ncl; j++)
			p->max += p->cl[j].mines;
		p->cnt = calloc(p->max + 1, sizeof(mpint *));
		countcomb(p->cl, p->ncl, p->cnt);
		if(p->max > allmax) allmax = p->max;
	}
	return allmax;
}

static void
fieldcombs(CList *cl, int ncl, int *xp, int *yp)
{
	uchar *ref;
	int i, k, p;
	int nf, ng;
	int allmax;
	int ref1;
	mpint *min, *v, **alt, **backup;
	CList *l;
	CGroup *g, *gp;

	ref = calloc(ghost.w, ghost.h);
	countref(cl, ncl, ref);
	nf = freefields(ref);
	ng = mkgroups(cl, ncl, ref, &g);
	allmax = basecounts(g, ng);
	alt = calloc(allmax + 1, sizeof(mpint *));
	
	*xp = -1;
	*yp = -1;
	min = mpnew(0);
	
	for(gp = g; gp < g + ng; gp++)
		for(l = gp->cl; l < gp->cl + gp->ncl; l++){
			ref1 = 0;
			for(i = 0; i < l->pts; i++){
				p = l->pt[i];
				if(ref[p] == 0xff)
					continue;
				if(ref[p] == 1 && ref1++ != 0)
					continue;
				ref[p] = 0xff;
				
				cltry(gp->cl, gp->ncl, p, 1, alt);
				backup = gp->cnt;
				gp->cnt = alt;
				v = totcomb(ghost.nmines, nf, g, ng);
				if(*xp < 0 || mpcmp(v, min) < 0){
					mpassign(v, min);
					*xp = p % ghost.w;
					*yp = p / ghost.w;
				}
				
				gp->cnt = backup;
				for(k = 0; k <= gp->max; k++)
					mpfree(alt[k]);
				memset(alt, 0, (gp->max + 1) * sizeof(mpint *));
				mpfree(v);
			}
		}
	if(nf > 0){
		v = totcomb(ghost.nmines - 1, nf - 1, g, ng);
		if(*xp < 0 || mpcmp(v, min) < 0){
			i = nrand(nf);
			for(p = 0; p < ghost.w * ghost.h; p++)
				if(ref[p] == 0 && ghost.f[p] == MUnknown && i-- == 0)
					break;
			mpassign(v, min);
			*xp = p % ghost.w;
			*yp = p / ghost.w;
		}
		mpfree(v);
	}
	mpfree(min);
	free(alt);
	free(ref);
	freegroups(g, ng);
}

static void
ghostprep(CList **clp, int *nclp)
{
	int x, y;

	ghost.w = MaxX;
	ghost.h = MaxY;
	ghost.nmines = MinesRemain;
	ghost.f = calloc(ghost.w, ghost.h);
	for(x = 0; x < ghost.w; x++){
		for(y = 0; y < ghost.h; y++)
			switch(MineField[x][y].Picture){
			case Empty0:
			case Empty1:
			case Empty2:
			case Empty3:
			case Empty4:
			case Empty5:
			case Empty6:
			case Empty7:
			case Empty8:
				ghost.f[x + y * ghost.w] = MEmpty;
				break;
			case Mark:
				ghost.f[x + y * ghost.w] = MMine;
				break;
			default:
				ghost.f[x + y * ghost.w] = MUnknown;
			}
	}
	mklists(clp, nclp);
}

static int
ghostfind(CList *cl, int ncl, Point *targp)
{
	int i, j, x, y;
	Point p, pd;
	int d, min;
	Point targ;
	int rt;
	
	merge(&cl, &ncl, -1, 0);
	min = 0;
	rt = NOTARG;
	targ = ghost.cpos;
	for(i = 0; i < ncl; i++)
		if(cl[i].mines == 0 || cl[i].mines == cl[i].pts)
			for(j = 0; j < cl[i].pts; j++){
				p = Pt(cl[i].pt[j]%ghost.w, cl[i].pt[j]/ghost.w);
				pd = subpt(addpt(addpt(mulpt(p, 16), Pt(12+8, 57+8)), Origin), targ);
				d = pd.x * pd.x + pd.y * pd.y;
				if(rt == NOTARG || d < min){
					rt = cl[i].mines == cl[i].pts ? RTARG : LTARG;
					*targp = p;
					min = d;
				}
				ghost.f[cl[i].pt[j]] = cl[i].mines == cl[i].pts ? MMine : MEmpty;
			}
	if(rt == NOTARG){
		fieldcombs(cl, ncl, &x, &y);
		if(x >= 0){
			rt = LTARG;
			*targp = Pt(x, y);
		}
	}
	free(ghost.f);
	free(cl);
	return rt;
}

static void
killchild(void)
{
	postnote(PNPROC, ghost.pid, "kill");
}

void
GhostInit(void)
{
	CList *cl;
	int ncl;
	int rt;
	Point p;
	int rc;
	int g;

	ghost.Rendez.l = &ghost;
	if(rc = rfork(RFPROC|RFMEM), rc == 0){
		qlock(&ghost);
		for(;;){
			while(!ghost.active || ghost.targettype != NOTARG)
				rsleep(&ghost);
			g = ghost.generation;
			ghostprep(&cl, &ncl);
			qunlock(&ghost);
			rt = ghostfind(cl, ncl, &p);
			qlock(&ghost);
			if(ghost.generation == g){
				ghost.targettype = rt;
				ghost.target = p;
			}
		}
	}else{
		ghost.pid = rc;
		atexit(killchild);
	}
}

int
GhostCheck(Point p)
{
	int i;
	
	for(i = 0; i < ghost.nmoves; i++)
		if(eqpt(ghost.moves[i], p))
			return 1;
	return 0;
}

static void
move(Point p)
{
	if(ghost.nmoves == nelem(ghost.moves))
		memmove(ghost.moves + 1, ghost.moves, sizeof(ghost.moves) - sizeof(Point));
	else
		memmove(ghost.moves + 1, ghost.moves, ghost.nmoves++ * sizeof(Point));
	ghost.moves[0] = p;
	ghost.cpos = p;
	emoveto(p);
}

static void
approach(Point p)
{
	Point q;
	double d;

	q = subpt(p, ghost.cpos);
	d = hypot(q.x, q.y);
	if(d == 0)
		return;
	d = 2 / d * 2 * (1 + d / (400 + d));
	move(addpt(ghost.cpos, Pt(ceil(q.x * d), ceil(q.y * d))));
}

void
GhostMode(void)
{
	Point p;

	if(!ghost.cursor){
		esetcursor(&bunny);
		ghost.cursor = 1;
	}
	if(ghost.wait > 0){
		if(Status == Oops)
			move(addpt(ghost.cpos, Pt(nrand(20) - 10, nrand(20) - 10)));
		ghost.wait--;
		return;
	}
	if(Status != Game){
		ghost.active = 0;
		p = Pt(Origin.x + MaxX * 8 + 12, Origin.y + 28);
		if(TURBO || ptinrect(ghost.cpos, insetrect(Rpt(p, p), -4))){
			InitMineField();
			qlock(&ghost);
			ghost.active = 0;
			ghost.targettype = NOTARG;
			ghost.generation++;
			qunlock(&ghost);
			eresized(0);
		}else
			approach(p);
		return;
	}
	qlock(&ghost);
	if(!ghost.active){
		ghost.active = 1;
		rwakeup(&ghost);
	}
	if(ghost.targettype != NOTARG){
		p = addpt(addpt(mulpt(ghost.target, 16), Pt(12+8, 57+8)), Origin);
		if(TURBO || ptinrect(ghost.cpos, insetrect(Rpt(p, p), -4))){
			switch(ghost.targettype){
			case LTARG: LeftClick(ghost.target); break;
			case RTARG: RightClick(ghost.target); break;
			}
			ghost.targettype = NOTARG;
			rwakeup(&ghost);
			qunlock(&ghost);
			if(Status != Game && !TURBO) ghost.wait = 100;
			DrawButton(Status);
			flushimage(display, 1);
			return;
		}else{
			qunlock(&ghost);
			approach(p);
		}
	}else
		qunlock(&ghost);
}

void
GhostReset(int deltarg)
{
	if(ghost.cursor){
		esetcursor(nil);
		ghost.cursor = 0;
	}
	ghost.cpos = LastMouse.xy;
	qlock(&ghost);
	ghost.active = 0;
	ghost.wait = 0;
	ghost.targettype = NOTARG;
	if(deltarg)
		ghost.generation++;
	qunlock(&ghost);
}