shithub: riscv

ref: ec19c5697f3caf52e6be57049169e2eb048299e6
dir: /sys/src/cmd/reform/pm.c/

View raw version
#include <u.h>
#include <libc.h>
#include <fcall.h>
#include <thread.h>
#include <9p.h>
#include <draw.h>
#include <memdraw.h>

enum
{
	Mhz = 1000*1000,
	Pwmsrcclk = 25*Mhz,
	Kbdlightmax = 8,

	Scharge = 0,
	Sovervolted,
	Scooldown,
	Sundervolted,
	Smissing,
	Sfullycharged,
	Spowersave,

	Psomoff = 1,

	Light = 1,
	Temp,
	Battery,
	Kbdoled,
	Pmctl,

	KbdoledW = 126,
	KbdoledH = 32,

	Lcd = 0,
	Kbd,

	PWMSAR = 0x0c/4,
	PWMPR = 0x10/4,

	TMUTMR = 0x00/4,
		TMR_ME = 1<<31,
		TMR_ALPF_SHIFT = 26,
		TMR_MSITE_SHIFT = 13,
	TMUTSR = 0x04/4,
		TSR_MIE = 1<<30,
		TSR_ORL = 1<<29,
		TSR_ORH = 1<<28,
	TMUTMTMIR = 0x08/4,
	TMUTIER = 0x20/4,
	TMUTIDR = 0x24/4,
		TIDR_MASK = 0xe0000000,
	TMUTISCR = 0x28/4,
	TMUTICSCR = 0x2c/4,
	TMUTTCFGR = 0x80/4,
	TMUTSCFGR = 0x84/4,
	TMUTRITSR0 = 0x100/4,
	TMUTRITSR1 = 0x110/4,
	TMUTRITSR2 = 0x120/4,
	TMUTTR0CR = 0xf10/4,
	TMUTTR1CR = 0xf14/4,
	TMUTTR2CR = 0xf18/4,
	TMUTTR3CR = 0xf1c/4,
		CR_CAL_PTR_SHIFT = 16,

	SPIx_RXDATA = 0x00/4,
	SPIx_TXDATA = 0x04/4,
	SPIx_CONREG = 0x08/4,
		CON_BURST_LENGTH = 1<<20,
		CON_PRE_DIVIDER = 1<<12,
		CON_POST_DIVIDER = 1<<8,
		CON_CHAN_MASTER = 1<<4,
		CON_XCH = 1<<2,
		CON_EN = 1<<0,
	SPIx_CONFIGREG = 0x0c/4,
		CONFIG_SS_CTL_NCSS = 1<<8,
		CONFIG_SCLK_PHA_1 = 1<<0,
	SPIx_STATREG = 0x18/4,
		STAT_RR = 1<<3,
};

static char *uid = "pm";
static Reqqueue *lpcreq;
static u32int *pwm2, *tmu, *spi2;
static int kbdlight = 0;
static int kbdhidfd = -1;
static Memimage *kbdoled, *image;
static u8int kbdoledraw[4+KbdoledW*KbdoledH/8] = {'W', 'B', 'I', 'T', 0};

static void
wr(u32int *base, int reg, u32int v)
{
	//fprint(2, "[0%x] ← 0x%ux\n", reg*4, v);
	if(base != nil)
		base[reg] = v;
}

static u32int
rd(u32int *base, int reg)
{
	return base != nil ? base[reg] : -1;
}

static char *
readall(int f)
{
	int bufsz, sz, n;
	char *s;

	bufsz = 2047;
	s = nil;
	for(sz = 0;; sz += n){
		if(bufsz-sz < 2048){
			bufsz *= 2;
			s = realloc(s, bufsz);
		}
		if((n = readn(f, s+sz, bufsz-sz-1)) < 1)
			break;
	}
	if(n < 0 || sz < 1){
		if(n == 0)
			werrstr("empty");
		free(s);
		return nil;
	}
	s[sz] = 0;

	return s;
}

static int
openkbdhid(void)
{
	char path[32], *s, *k, *e;
	int f;

	if(kbdhidfd < 0 && (f = open("/dev/usb/ctl", OREAD)) >= 0){
		if((s = readall(f)) != nil &&
			(k = strstr(s, "MNT 'Reform Keyboard'")) != nil &&
			(e = strchr(k+22, ' ')) != nil){
			*e = 0;
			snprint(path, sizeof(path), "/dev/hidU%sctl", k+22);
			if((kbdhidfd = open(path, OWRITE)) >= 0 && write(kbdhidfd, "rawon", 5) != 5){
				close(kbdhidfd);
				kbdhidfd = -1;
			}
		}
		free(s);
		close(f);
	}

	return kbdhidfd < 0;
}

static int
loadkbdoled(char *data, int offset, int size)
{
	int x, y, i, k, v, bpl, used;
	static Rectangle r;
	char hdr[5*12+1];
	u8int *p, q;
	ulong chan;

	if(openkbdhid() != 0)
		return -1;
	if(size == 0)
		return write(kbdhidfd, "WCLR", 4);

	used = 0;
	if(offset == 0){
		if(size < 60){
			werrstr("invalid header");
			return -1;
		}
		memmove(hdr, data, 60);
		hdr[11] = 0;
		hdr[60] = 0;
		if((chan = strtochan(data)) == 0){
			werrstr("bad channel string %s", (char*)data);
			return -1;
		}
		r.min.x = atoi(data+1*12);
		r.min.y = atoi(data+2*12);
		r.max.x = atoi(data+3*12);
		r.max.y = atoi(data+4*12);
		if(badrect(r)){
			werrstr("bad rect");
			return -1;
		}
		data += 60;
		size -= 60;
		used += 60;
		if(image == nil || chan != image->chan || !eqrect(r, image->r)){
			freememimage(image);
			if((image = allocmemimage(r, chan)) == nil)
				return -1;
		}
		r.max.y = r.min.y;
	}
	if(image == nil){
		werrstr("no header");
		return -1;
	}
	bpl = bytesperline(image->r, image->depth);
	i = size / bpl;
	if(i < 1)
		return used;
	v = loadmemimage(image, Rect(r.min.x, r.max.y, r.max.x, r.max.y+i), (uchar*)data, size);
	if(v <= 0){
		werrstr("loadmemimage: failed");
		return -1;
	}
	r.max.y += i;
	used += v;
	if(r.max.y < image->r.max.y)
		return used;

	memimagedraw(kbdoled, kbdoled->r, image, image->r.min, nil, ZP, S);
	bpl = bytesperline(kbdoled->r, kbdoled->depth);

	for(y = 0, i = 4; y < KbdoledH; y += 8){
		for(x = v = 0; x < KbdoledW; x++, v = (v+1)&7){
			SET(p);
			if(v == 0)
				p = byteaddr(kbdoled, Pt(x,y));
			for(k = q = 0; k < 8; k++)
				q |= ((p[bpl*k] >> (7-v)) & 1) << k;
			kbdoledraw[i++] = q;
		}
	}

	if(write(kbdhidfd, kbdoledraw, sizeof(kbdoledraw)) != sizeof(kbdoledraw))
		return -1;

	return used;
}

static int
setlight(int k, int p)
{
	u32int v;

	if(p < 0)
		p = 0;
	if(p > 100)
		p = 100;

	if(k == Lcd){
		v = Pwmsrcclk / rd(pwm2, PWMSAR);
		wr(pwm2, PWMPR, (Pwmsrcclk/(v*p/100))-2);
		return 0;
	}else if(k == Kbd && openkbdhid() == 0){
		v = Kbdlightmax*p/100;
		if(fprint(kbdhidfd, "LITE%c", '0'+v) > 0){
			kbdlight = v;
			return 0;
		}
		close(kbdhidfd);
		kbdhidfd = -1;
		kbdlight = 0;
	}

	return -1;
}

static int
getlight(int k)
{
	u32int m, v;

	SET(m, v);
	if(k == Lcd){
		m = Pwmsrcclk / rd(pwm2, PWMSAR);
		v = Pwmsrcclk / (rd(pwm2, PWMPR)+2);
	}else if(k == Kbd){
		m = Kbdlightmax;
		v = kbdlight;
	}
	return v*100/m;
}

static int
getcputemp(void)
{
	u32int s;
	int i, c;

	/* enable: all sites, ALPF 11=0.125 */
	wr(tmu, TMUTMR, TMR_ME | 3<<TMR_ALPF_SHIFT | 0<<TMR_MSITE_SHIFT);
	sleep(50);

	s = rd(tmu, TMUTSR);
	if(s & TSR_MIE){
		werrstr("monitoring interval exceeded");
		return -1;
	}
	if(s & (TSR_ORL|TSR_ORH)){
		werrstr("out of range");
		return -1;
	}

	for(i = 0; (c = rd(tmu, TMUTRITSR0)) >= 0 && i < 10; i++)
		sleep(10);
	wr(tmu, TMUTMR, 0);

	return c & 0xff;
}

static void
tmuinit(void)
{
	/* without proper calibration data sensing is useless */
	static u8int cfg[4][12] = {
		{0x23, 0x29, 0x2f, 0x35, 0x3d, 0x43, 0x4b, 0x51, 0x57, 0x5f, 0x67, 0x6f},
		{0x1b, 0x23, 0x2b, 0x33, 0x3b, 0x43, 0x4b, 0x55, 0x5d, 0x67, 0x70, 0},
		{0x17, 0x23, 0x2d, 0x37, 0x41, 0x4b, 0x57, 0x63, 0x6f, 0},
		{0x15, 0x21, 0x2d, 0x39, 0x45, 0x53, 0x5f, 0x71, 0},
	};
	int i, j;

	wr(tmu, TMUTMR, 0); /* disable */
	wr(tmu, TMUTIER, 0); /* disable all interrupts */
	wr(tmu, TMUTMTMIR, 0xf); /* no monitoring interval */

	/* configure default ranges */
	wr(tmu, TMUTTR0CR, 11<<CR_CAL_PTR_SHIFT | 0);
	wr(tmu, TMUTTR1CR, 10<<CR_CAL_PTR_SHIFT | 38);
	wr(tmu, TMUTTR2CR, 8<<CR_CAL_PTR_SHIFT | 72);
	wr(tmu, TMUTTR3CR, 7<<CR_CAL_PTR_SHIFT | 97);

	/* calibration data */
	for(i = 0; i < 4; i++){
		for(j = 0; j < 12 && cfg[i][j] != 0; j++){
			wr(tmu, TMUTTCFGR, i<<16|j);
			wr(tmu, TMUTSCFGR, cfg[i][j]);
		}
	}
}

static void
lpccall(char cmd, u8int arg, void *ret)
{
	u32int con;
	int i;

	con =
		/* 8 bits burst */
		CON_BURST_LENGTH*(8-1) |
		/* clk=25Mhz, pre=1, post=2⁶ → 25Mhz/1/2⁶ ≲ 400kHz */
		CON_PRE_DIVIDER*(1-1) | CON_POST_DIVIDER*6 |
		/* master mode on channel 0 */
		CON_CHAN_MASTER<<0 |
		/* enable */
		CON_EN;
	wr(spi2, SPIx_CONREG, con);

	wr(spi2, SPIx_CONFIGREG,
		/* tx shift - rising edge; rx latch - falling edge */
		CONFIG_SCLK_PHA_1 |
		CONFIG_SS_CTL_NCSS);

	wr(spi2, SPIx_TXDATA, 0xb5);
	wr(spi2, SPIx_TXDATA, cmd);
	wr(spi2, SPIx_TXDATA, arg);
	wr(spi2, SPIx_CONREG, con | CON_XCH);

	/*
	 * LPC buffers 3 bytes without responding, but spends some time
	 * to prepare the response. 50ms should be safe, add a bit more
	 * to be sure LPC is blocked waiting for the chip select to go
	 * active again.
	 */
	sleep(80);
	while(rd(spi2, SPIx_STATREG) & STAT_RR)
		rd(spi2, SPIx_RXDATA);

	/* expecting 8 bytes, start the exchange */
	for(i = 0; i < 8; i++)
		wr(spi2, SPIx_TXDATA, 0);
	wr(spi2, SPIx_CONREG, con | CON_XCH);
	sleep(80);

	for(i = 0; i < 8; i++)
		((u8int*)ret)[i] = rd(spi2, SPIx_RXDATA);

	wr(spi2, SPIx_CONREG, con & ~CON_EN);
}

static void
readbattery(Req *r)
{
	int hh, mm, ss, full, remain, warn, safe;
	u8int st[8], ch[8];
	s16int current;
	char msg[256];
	char *state;

	lpccall('c', 0, ch);
	lpccall('q', 0, st);
	current = (s16int)(st[2] | st[3]<<8);
	remain = ch[0]|ch[1]<<8;
	warn = ch[2]|ch[3]<<8;
	full = ch[4]|ch[5]<<8;
	safe = remain - warn;

	hh = mm = ss = 0;
	state = "unknown";
	switch(st[5]){
	case Sfullycharged: state = "full"; break;
	case Sovervolted: state = "balancing"; break;
	case Scooldown: state = "cooldown"; break;
	case Spowersave: state = "powersave"; break;
	case Scharge:
		if(current < 0)
			state = "charging";
		if(current > 0)
			state = "discharging";
		if(current != 0){
			ss = (current < 0 ? full - safe : safe) * 3600 / abs(current);
			hh = ss/3600;
			ss -= 3600*(ss/3600);
			mm = ss/60;
			ss -= 60*(ss/60);
		}
		break;
	case Smissing:
		respond(r, "battery is missing");
		return;
		
	}

	snprint(msg, sizeof(msg), "%d mA %d %d %d %d ? mV %d ? %02d:%02d:%02d %s\n",
		st[4],
		remain, full, full, warn,
		st[0]|st[1]<<8,
		hh, mm, ss,
		state
	);

	readstr(r, msg);
	respond(r, nil);
}

static void
readpmctl(Req *r)
{
	char msg[256], *s, *e;
	static char lpcfw[9*3+1];
	u8int v[16], q[8];
	int i;

	if(lpcfw[0] == 0){
		lpccall('f', 0, msg+0);
		lpccall('f', 1, msg+8);
		lpccall('f', 2, msg+16);
		snprint(lpcfw, sizeof(lpcfw), "%.*s %.*s %.*s", 8, msg+0, 8, msg+8, 8, msg+16);
	}
	lpccall('v', 0, v+0);
	lpccall('v', 1, v+8);
	lpccall('q', 0, q);

	s = msg;
	e = s+sizeof(msg);
	s = seprint(s, e, "version %s\n", lpcfw);
	s = seprint(s, e, "cells(mV)");
	for(i = 0; i < 16; i += 2)
		s = seprint(s, e, " %d", v[i] | v[i+1]<<8);
	s = seprint(s, e, "\n");
	s = seprint(s, e, "current(mA) %d\n", (s16int)(q[2] | q[3]<<8));
	USED(s);

	readstr(r, msg);
	respond(r, nil);
}

static void
fsread(Req *r)
{
	char msg[256];
	void *aux;
	int c;

	msg[0] = 0;
	if(r->ifcall.offset == 0){
		aux = r->fid->file->aux;
		if(aux == (void*)Light){
			snprint(msg, sizeof(msg), "lcd %d\nkbd %d\n", getlight(Lcd), getlight(Kbd));
		}else if(aux == (void*)Temp){
			if((c = getcputemp()) < 0){
				responderror(r);
				return;
			}
			snprint(msg, sizeof(msg), "%d.0\n", c);
		}else if(aux == (void*)Battery){
			reqqueuepush(lpcreq, r, readbattery);
			return;
		}else if(aux == (void*)Pmctl){
			reqqueuepush(lpcreq, r, readpmctl);
			return;
		}
	}

	readstr(r, msg);
	respond(r, nil);
}

static void
fswrite(Req *r)
{
	char msg[256], *f[4];
	int nf, v, p, k;
	void *aux;

	aux = r->fid->file->aux;

	if(aux == (void*)Kbdoled){
		if((v = loadkbdoled(r->ifcall.data, r->ifcall.offset, r->ifcall.count)) < 0){
Err:
			responderror(r);
			return;
		}
		r->ofcall.count = v;
		respond(r, nil);
		return;
	}

	snprint(msg, sizeof(msg), "%.*s", utfnlen(r->ifcall.data, r->ifcall.count), r->ifcall.data);
	nf = tokenize(msg, f, nelem(f));
	if(aux == (void*)Light){
		if(nf < 2){
Bad:
			respond(r, "invalid ctl message");
			return;
		}
		if(strcmp(f[0], "lcd") == 0)
			k = Lcd;
		else if(strcmp(f[0], "kbd") == 0)
			k = Kbd;
		else
			goto Bad;
		v = atoi(f[1]);
		if(*f[1] == '+' || *f[1] == '-')
			v += getlight(k);
		if(setlight(k, v) != 0)
			goto Err;
	}else if(aux == (void*)Pmctl){
		p = -1;
		if(nf >= 2 && strcmp(f[0], "power") == 0){
			if(nf == 2 && strcmp(f[1], "off") == 0){
				/*
				 * LPC firmware might not be up to date so try
				 * shutting down through the keyboard first
				 */
				if(openkbdhid() == 0){
					write(kbdhidfd, "PWR0", 4);
					sleep(2000); /* give it a chance */
				}
			}
		}
		if(p < 0)
			goto Bad;
		lpccall('p', p, msg);
	}

	r->ofcall.count = r->ifcall.count;
	respond(r, nil);
}

static void
fsflush(Req *r)
{
	void *aux;
	Req *o;

	o = r->oldreq;
	aux = o->fid->file->aux;
	if(o->ifcall.type == Tread && (aux == (void*)Battery || aux == (void*)Pmctl))
		reqqueueflush(lpcreq, o);
	respond(r, nil);
}

static Srv fs = {
	.read = fsread,
	.write = fswrite,
	.flush = fsflush,
};

static void
usage(void)
{
	fprint(2, "usage: %s [-D] [-m mountpoint] [-s service]\n", argv0);
	exits("usage");
}

void
threadmain(int argc, char **argv)
{
	char *mtpt, *srv;

	mtpt = "/dev";
	srv = nil;
	ARGBEGIN{
	case 'D':
		chatty9p = 1;
		break;
	case 'm':
		mtpt = EARGF(usage());
		break;
	case 's':
		srv = EARGF(usage());
		break;
	default:
		usage();
	}ARGEND

	if((tmu = segattach(0, "tmu", 0, 0xf20)) == (void*)-1)
		sysfatal("no tmu");
	if((pwm2 = segattach(0, "pwm2", 0, 0x18)) == (void*)-1)
		sysfatal("no pwm2");
	if((spi2 = segattach(0, "ecspi2", 0, 0x20)) == (void*)-1)
		sysfatal("no spi2");
	tmuinit();
	if(memimageinit() != 0)
		sysfatal("%r");
	if((kbdoled = allocmemimage(Rect(0, 0, KbdoledW, KbdoledH), GREY1)) == nil)
		sysfatal("%r");
	lpcreq = reqqueuecreate();
	fs.tree = alloctree(uid, uid, DMDIR|0555, nil);
	createfile(fs.tree->root, "battery", uid, 0444, (void*)Battery);
	createfile(fs.tree->root, "cputemp", uid, 0444, (void*)Temp);
	createfile(fs.tree->root, "light", uid, 0666, (void*)Light);
	createfile(fs.tree->root, "kbdoled", uid, 0222, (void*)Kbdoled);
	createfile(fs.tree->root, "pmctl", uid, 0666, (void*)Pmctl);
	threadpostmountsrv(&fs, srv, mtpt, MAFTER);

	threadexits(nil);
}