shithub: riscv

Download patch

ref: 98b1f2a75b99f1af3c7c7cecf3735bab1793e2a2
parent: 3004f058f69a16f09c07c58d0e60a1732190f0d3
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Sun Jan 21 17:55:14 EST 2018

ppp: mschapv2 support

--- a/sys/src/cmd/ip/ppp/mppc.c
+++ b/sys/src/cmd/ip/ppp/mppc.c
@@ -120,7 +120,7 @@
 static  int		ipcheck(uchar*, int);
 static  void		hischeck(Uncstate*);
 
-static	void		setkey(uchar *key, uchar *startkey);
+static	void setkey(uchar *key, uchar *startkey);
 
 Comptype cmppc = {
 	compinit,
@@ -155,8 +155,8 @@
 
 	if(ppp->sendencrypted) {
 		cs->encrypt = 1;
-		memmove(cs->startkey, ppp->key, 16);
-		memmove(cs->key, ppp->key, 16);
+		memmove(cs->startkey, ppp->sendkey, 16);
+		memmove(cs->key, cs->startkey, 16);
 		setkey(cs->key, cs->startkey);
 		setupRC4state(&cs->rc4key, cs->key, 16);
 	}
@@ -467,8 +467,8 @@
 	s = mallocz(sizeof(Uncstate), 1);
 
 	s->count = 0xfff;	/* count of non existant last packet */
-	memmove(s->startkey, ppp->key, 16);
-	memmove(s->key, ppp->key, 16);
+	memmove(s->startkey, ppp->recvkey, 16);
+	memmove(s->key, s->startkey, 16);
 	setkey(s->key, s->startkey);
 	setupRC4state(&s->rc4key, s->key, 16);
 
@@ -577,12 +577,11 @@
 	if(count & Pencrypt) {
 //netlog("mppc unencrypt count = %ux\n", count);
 		rc4(&s->rc4key, p, n);
+fprint(2, "plain=%.*H\n", n, p);
 	}
 
-	if(!(count & Pcompress)) {
-//netlog("uncompress blen = %d\n", BLEN(b));
-		return  b;
-	}
+	if(!(count & Pcompress))
+		return b;
 
 	bits = 0;
 	sreg = 0;
@@ -727,6 +726,27 @@
 	
 	s = as;	
 	free(s);
+}
+
+void
+getasymkey(uchar *key, uchar *masterkey, int send, int server)
+{
+	uchar digest[SHA1dlen];
+	uchar pad[40];
+	SHAstate *s;
+	char *m;
+
+	s = sha1(masterkey, 16, nil, nil);
+	memset(pad, 0, 40);
+	sha1(pad, 40, nil, s);
+	if(send ^ server)
+		m = "On the client side, this is the send key; on the server side, it is the receive key.";
+	else
+		m = "On the client side, this is the receive key; on the server side, it is the send key.";
+	sha1((uchar*)m, 84, nil, s);
+	memset(pad, 0xf2, 40);
+	sha1(pad, 40, digest, s);
+	memmove(key, digest, 16);
 }
 
 static void
--- a/sys/src/cmd/ip/ppp/ppp.c
+++ b/sys/src/cmd/ip/ppp/ppp.c
@@ -114,7 +114,6 @@
 static	void		terminate(PPP*, int);
 static	int		validv4(Ipaddr);
 static  void		dmppkt(char *s, uchar *a, int na);
-static	void		getauth(PPP*);
 
 void
 pppopen(PPP *ppp, int mediain, int mediaout, char *net,
@@ -191,7 +190,7 @@
 		ppp->chap = mallocz(sizeof(*ppp->chap), 1);
 		if(ppp->chap == nil)
 			abort();
-		ppp->chap->proto = APmschap;
+		ppp->chap->proto = APmschapv2;
 		ppp->chap->state = Cunauth;
 		auth_freechal(ppp->chap->cs);
 		ppp->chap->cs = nil;
@@ -252,6 +251,7 @@
 				break;
 			case APmd5:
 			case APmschap:
+			case APmschapv2:
 				break;
 			default:
 				setphase(ppp, Pnet);
@@ -713,6 +713,7 @@
 			syslog(0, "ppp", "requesting %I", ppp->local);
 			putv4o(b, Oipaddr, ppp->local);
 		}
+		primary = 1;
 		if(primary && (p->optmask & Fipdns))
 			putv4o(b, Oipdns, ppp->dns[0]);
 		if(primary && (p->optmask & Fipdns2))
@@ -852,7 +853,9 @@
 				}
 				if(proto != Pchap)
 					break;
-				if(o->data[2] != APmd5 && o->data[2] != APmschap)
+				if(o->data[2] != APmd5
+				&& o->data[2] != APmschap
+				&& o->data[2] != APmschapv2)
 					break;
 				chapproto = o->data[2];
 				continue;
@@ -859,8 +862,6 @@
 			}
 			break;
 		case Pccp:
-			if(nocompress)
-				break;
 			switch(o->type){
 			case Octhwack:
 				break;
@@ -880,10 +881,6 @@
 			case Ocmppc:
 				x = nhgetl(o->data);
 
-				// hack for Mac
-				// if(x == 0)
-				//	continue;
-
 				/* stop ppp loops */
 				if((x&0x41) == 0 || ppp->ctries++ > 5) {
 					/*
@@ -895,7 +892,7 @@
 				}
 				if(rejecting)
 					continue;
-				if(x & 1) {
+				if((x & 0x01000001) == 1){
 					ctype = &cmppc;
 					ppp->sendencrypted = (o->data[3]&0x40) == 0x40;
 					continue;
@@ -1965,6 +1962,20 @@
 	ppp->out.reports++;
 }
 
+static char*
+getaproto(int proto)
+{
+	switch(proto){
+	case APmd5:
+		return "chap";
+	case APmschap:
+		return "mschap";
+	case APmschapv2:
+		return "mschapv2";
+	}
+	return nil;
+}
+
 /*
  * init challenge response dialog
  */
@@ -1975,24 +1986,14 @@
 	Lcpmsg *m;
 	Chap *c;
 	int len;
-	char *aproto;
 
-	getauth(ppp);
-
 	c = ppp->chap;
 	c->id++;
-
-	switch(c->proto){
-	default:
-		abort();
-	case APmd5:
-		aproto = "chap";
-		break;
-	case APmschap:
-		aproto = "mschap";
-		break;
+	if(c->ai != nil){
+		auth_freeAI(c->ai);
+		c->ai = nil;
 	}
-	if((c->cs = auth_challenge("proto=%q role=server", aproto)) == nil)
+	if((c->cs = auth_challenge("proto=%q role=server", getaproto(c->proto))) == nil)
 		sysfatal("auth_challenge: %r");
 	syslog(0, LOG, ": remote=%I: sending %d byte challenge", ppp->remote, c->cs->nchal);
 	len = 4 + 1 + c->cs->nchal + strlen(ppp->chapname);
@@ -2011,72 +2012,42 @@
 }
 
 /*
- * BUG factotum should do this
+ *  challenge response dialog
  */
-enum {
-	MShashlen = 16,
-	MSresplen = 24,
-	MSchallen = 8,
-};
-
-void
-desencrypt(uchar data[8], uchar key[7])
+static void
+setppekey(PPP *ppp, int isserver)
 {
-	ulong ekey[32];
+	Chap *c = ppp->chap;
 
-	key_setup(key, ekey);
-	block_cipher(ekey, data, 0);
-}
-
-void
-nthash(uchar hash[MShashlen], char *passwd)
-{
-	uchar buf[512];
-	int i;
-	
-	for(i=0; *passwd && i<sizeof(buf); passwd++) {
-		buf[i++] = *passwd;
-		buf[i++] = 0;
+	switch(c->proto){
+	case APmschap:
+		if(c->ai == nil || c->ai->nsecret != 16)
+			sysfatal("could not get the encryption key");
+		memmove(ppp->sendkey, c->ai->secret, 16);
+		memmove(ppp->recvkey, c->ai->secret, 16);
+		break;
+	case APmschapv2:
+		if(c->ai == nil || c->ai->nsecret != 16+20)
+			sysfatal("could not get the encryption key + authenticator");
+		getasymkey(ppp->sendkey, c->ai->secret, 1, isserver);
+		getasymkey(ppp->recvkey, c->ai->secret, 0, isserver);
+		break;
 	}
-	memset(hash, 0, 16);
-	md4(buf, i, hash, 0);
+	auth_freeAI(c->ai);
+	c->ai = nil;
 }
 
-void
-mschalresp(uchar resp[MSresplen], uchar hash[MShashlen], uchar chal[MSchallen])
-{
-	int i;
-	uchar buf[21];
-	
-	memset(buf, 0, sizeof(buf));
-	memcpy(buf, hash, MShashlen);
-
-	for(i=0; i<3; i++) {
-		memmove(resp+i*MSchallen, chal, MSchallen);
-		desencrypt(resp+i*MSchallen, buf+i*7);
-	}
-}
-
-/*
- *  challenge response dialog
- */
-extern	int	_asrdresp(int, uchar*, int);
-
 static void
 getchap(PPP *ppp, Block *b)
 {
-	AuthInfo *ai;
 	Lcpmsg *m;
 	int len, vlen, i, id, n, nresp;
-	char md5buf[512], code;
+	char code;
 	Chap *c;
 	Chapreply cr;
 	MSchapreply mscr;
 	char uid[PATH];
-	uchar digest[16], *p, *resp, sdigest[SHA1dlen];
-	uchar mshash[MShashlen], mshash2[MShashlen];
-	DigestState *s;
-	uchar msresp[2*MSresplen+1];
+	uchar resp[256], *p;
 
 	m = (Lcpmsg*)b->rptr;
 	len = nhgets(m->len);
@@ -2087,50 +2058,23 @@
 	}
 
 	qlock(ppp);
-
+	c = ppp->chap;
+	vlen = m->data[0];
 	switch(m->code){
 	case Cchallenge:
-		getauth(ppp);
-
-		vlen = m->data[0];
-		if(vlen > len - 5) {
-			netlog("PPP: chap: bad challenge len\n");
-			break;
-		}
-
 		id = m->id;
-		switch(ppp->chap->proto){
-		default:
-			abort();
-		case APmd5:
-			n = strlen(ppp->secret);
-			if(n + vlen + 1 > sizeof(md5buf)) {
-				netlog("PPP: chap: bad challenge len\n");
-				goto end;
-			}
-			md5buf[0] = m->id;
-			memcpy(md5buf+1, ppp->secret, n);
-			memcpy(md5buf+1+n, m->data+1, vlen);
-			md5((uchar*)md5buf, n + vlen + 1, digest, nil);
-			resp = digest;
-			nresp = 16;
-			break;
-		case APmschap:
-			nthash(mshash, ppp->secret);
-			memset(msresp, 0, sizeof msresp);
-			mschalresp(msresp+MSresplen, mshash, m->data+1);
-			resp = msresp;
-			nresp = sizeof msresp;
-			nthash(mshash, ppp->secret);
-			md4(mshash, 16, mshash2, 0);
-			s = sha1(mshash2, 16, 0, 0);
-			sha1(mshash2, 16, 0, s);
-			sha1(m->data+1, 8, sdigest, s);
-			memmove(ppp->key, sdigest, 16);
-			break;
-		}
-		len = 4 + 1 + nresp + strlen(ppp->chapname);
+		memset(ppp->chapname, 0, sizeof(ppp->chapname));
+		nresp = auth_respondAI(m->data+1, vlen,
+			ppp->chapname, sizeof(ppp->chapname), 
+			resp, sizeof(resp), &c->ai,
+			auth_getkey,
+			"proto=%s role=client service=ppp %s", getaproto(c->proto), keyspec);
+		if(nresp < 0)
+			sysfatal("auth_respond: %r");
+		if(c->proto == APmschap || c->proto == APmschapv2)
+			while(nresp < 49) resp[nresp++] = 0;
 		freeb(b);
+		len = 4 + 1 + nresp + strlen(ppp->chapname);
 		b = alloclcp(Cresponse, id, len, &m);
 		*b->wptr++ = nresp;
 		memmove(b->wptr, resp, nresp);
@@ -2138,14 +2082,12 @@
 		memmove(b->wptr, ppp->chapname, strlen(ppp->chapname));
 		b->wptr += strlen(ppp->chapname);
 		hnputs(m->len, len);
-		netlog("PPP: sending response len %d\n", len);
+		netlog("ppp: sending response len %d\n", len);
 		putframe(ppp, Pchap, b);
 		break;
 	case Cresponse:
-		c = ppp->chap;
-		vlen = m->data[0];
-		if(m->id != c->id) {
-			netlog("PPP: chap: bad response id\n");
+		if(m->id != c->id || c->cs == nil) {
+			netlog("ppp: chap: bad response id\n");
 			break;
 		}
 		switch(c->proto) {
@@ -2153,10 +2095,9 @@
 			sysfatal("unknown chap protocol: %d", c->proto);
 		case APmd5:
 			if(vlen > len - 5 || vlen != 16) {
-				netlog("PPP: chap: bad response len\n");
+				netlog("ppp: chap: bad response len\n");
 				break;
 			}
-
 			cr.id = m->id;
 			memmove(cr.resp, m->data+1, 16);
 			memset(uid, 0, sizeof(uid));
@@ -2169,8 +2110,9 @@
 			c->cs->nresp = sizeof cr;
 			break;
 		case APmschap:
-			if(vlen > len - 5 || vlen != 49) {
-				netlog("PPP: chap: bad response len\n");
+		case APmschapv2:
+			if(vlen > len - 5 || vlen < 48) {
+				netlog("ppp: chap: bad response len\n");
 				break;
 			}
 			memset(&mscr, 0, sizeof(mscr));
@@ -2196,42 +2138,53 @@
 			break;
 		} 
 
-		syslog(0, LOG, ": remote=%I vlen %d proto %d response user %s nresp %d", ppp->remote, vlen, c->proto, c->cs->user, c->cs->nresp);
-		if((ai = auth_response(c->cs)) == nil || auth_chuid(ai, nil) < 0){
+		syslog(0, LOG, ": remote=%I vlen %d proto %d response user %s nresp %d",
+			ppp->remote, vlen, c->proto, c->cs->user, c->cs->nresp);
+
+		if((c->ai = auth_response(c->cs)) == nil || auth_chuid(c->ai, nil) < 0){
 			c->state = Cunauth;
 			code = Cfailure;
-			syslog(0, LOG, ": remote=%I: auth failed: %r, uid=%s", ppp->remote, uid);
+			syslog(0, LOG, ": remote=%I: auth failed: %r, uid=%s",
+				ppp->remote, uid);
 		}else{
 			c->state = Cauthok;
 			code = Csuccess;
-			syslog(0, LOG, ": remote=%I: auth ok: uid=%s nsecret=%d", ppp->remote, uid, ai->nsecret);
-			if(c->proto == APmschap){
-				if(ai->nsecret != sizeof(ppp->key))
-					sysfatal("could not get the encryption key");
-				memmove(ppp->key, ai->secret, sizeof(ppp->key));
-			}
+			syslog(0, LOG, ": remote=%I: auth ok: uid=%s nsecret=%d",
+				ppp->remote, uid, c->ai->nsecret);
 		}
-		auth_freeAI(ai);
 		auth_freechal(c->cs);
 		c->cs = nil;
 		freeb(b);
 
 		/* send reply */
-		len = 4;
-		b = alloclcp(code, c->id, len, &m);
-		hnputs(m->len, len);
+		if(code == Csuccess && c->proto == APmschapv2 && c->ai->nsecret == 16+20){
+			b = alloclcp(code, c->id, 4+2+2*20+1, &m);
+			b->wptr += sprint((char*)m->data, "S=%.20H", c->ai->secret+16);
+		} else {
+			b = alloclcp(code, c->id, 4, &m);
+		}
+		hnputs(m->len, BLEN(b));
 		putframe(ppp, Pchap, b);
 
-		if(c->state == Cauthok) {
+		if(c->state == Cauthok){
+			setppekey(ppp, 1);
 			setphase(ppp, Pnet);
 		} else {
 			/* restart chapp negotiation */
 			chapinit(ppp);
 		}
-		
 		break;
 	case Csuccess:
+		if(c->proto == APmschapv2 && c->ai != nil && c->ai->nsecret == 16+20){
+			n = snprint((char*)resp, sizeof(resp), "S=%.20H", c->ai->secret+16);
+			if(len - 4 < n || tsmemcmp(m->data, resp, n) != 0){
+				netlog("ppp: chap: bad authenticator\n");
+				terminate(ppp, 0);
+				break;
+			}
+		}
 		netlog("ppp: chap succeeded\n");
+		setppekey(ppp, 0);
 		setphase(ppp, Pnet);
 		break;
 	case Cfailure:
@@ -2242,7 +2195,6 @@
 		syslog(0, LOG, "chap code %d?", m->code);
 		break;
 	}
-end:
 	qunlock(ppp);
 	freeb(b);
 }
@@ -2253,27 +2205,32 @@
 	Block *b;
 	Lcpmsg *m;
 	Chap *c;
+	UserPasswd *up;
 	int len, nlen, slen;
 
-	getauth(ppp);
+	up = auth_getuserpasswd(auth_getkey, "proto=pass service=ppp %s", keyspec);
+	if(up == nil)
+		sysfatal("auth_getuserpasswd: %r");
 
 	c = ppp->chap;
 	c->id++;
-	netlog("PPP: pap: send authreq %d %s %s\n", c->id, ppp->chapname, "****");
+	netlog("ppp: pap: send authreq %d %s %s\n", c->id, up->user, "****");
 
-	nlen = strlen(ppp->chapname);
-	slen = strlen(ppp->secret);
+	nlen = strlen(up->user);
+	slen = strlen(up->passwd);
 	len = 4 + 1 + nlen + 1 + slen;
 	b = alloclcp(Pauthreq, c->id, len, &m);
 
 	*b->wptr++ = nlen;
-	memmove(b->wptr, ppp->chapname, nlen);
+	memmove(b->wptr, up->user, nlen);
 	b->wptr += nlen;
 	*b->wptr++ = slen;
-	memmove(b->wptr, ppp->secret, slen);
+	memmove(b->wptr, up->passwd, slen);
 	b->wptr += slen;
 	hnputs(m->len, len);
 
+	free(up);
+
 	putframe(ppp, Ppasswd, b);
 	freeb(b);
 }
@@ -2304,13 +2261,13 @@
 	qlock(ppp);
 	switch(m->code){
 	case Pauthreq:
-		netlog("PPP: pap auth request, not supported\n");
+		netlog("ppp: pap auth request, not supported\n");
 		break;
 	case Pauthack:
 		if(ppp->phase == Pauth
 		&& ppp->chap->proto == APpasswd
 		&& m->id <= ppp-> chap->id){
-			netlog("PPP: pap succeeded\n");
+			netlog("ppp: pap succeeded\n");
 			setphase(ppp, Pnet);
 		}
 		break;
@@ -2318,13 +2275,13 @@
 		if(ppp->phase == Pauth
 		&& ppp->chap->proto == APpasswd
 		&& m->id <= ppp-> chap->id){
-			netlog("PPP: pap failed (%d:%.*s)\n",
+			netlog("ppp: pap failed (%d:%.*s)\n",
 				m->data[0], m->data[0], (char*)m->data+1);
 			terminate(ppp, 0);
 		}
 		break;
 	default:
-		netlog("PPP: unknown pap messsage %d\n", m->code);
+		netlog("ppp: unknown pap messsage %d\n", m->code);
 	}
 	qunlock(ppp);
 	freeb(b);
@@ -2702,6 +2659,7 @@
 	fmtinstall('I', eipfmt);
 	fmtinstall('V', eipfmt);
 	fmtinstall('E', eipfmt);
+	fmtinstall('H', encodefmt);
 
 	dev = nil;
 
@@ -2969,19 +2927,4 @@
 		write(fd, "refresh", 7);
 		close(fd);
 	}
-}
-
-static void
-getauth(PPP *ppp)
-{
-	UserPasswd *up;
-
-	if(*ppp->chapname)
-		return;
-
-	up = auth_getuserpasswd(auth_getkey,"proto=pass service=ppp %s", keyspec);
-	if(up != nil){
-		strcpy(ppp->chapname, up->user);
-		strcpy(ppp->secret, up->passwd);
-	}		
 }
--- a/sys/src/cmd/ip/ppp/ppp.h
+++ b/sys/src/cmd/ip/ppp/ppp.h
@@ -113,6 +113,7 @@
 	/* authentication protocols */
 	APmd5=		5,
 	APmschap=	128,
+	APmschapv2=	129,
 	APpasswd=	Ppasswd,		/* use Pap, not Chap */
 
 	/* lcp flags */
@@ -221,7 +222,8 @@
 	int	state;		/* chap state */
 	uchar	id;		/* id of current message */
 	int	timeout;	/* for current state */
-	Chalstate *cs;
+	AuthInfo	*ai;
+	Chalstate	*cs;
 };
 
 struct Qualstats
@@ -292,11 +294,11 @@
 	void		*uncstate;	/* uncompression state */
 	
 	/* encryption key */
-	uchar		key[16];
+	uchar		sendkey[16];
+	uchar		recvkey[16];
 	int		sendencrypted;
 
 	/* authentication */
-	char		secret[256];	/* md5 key */
 	char		chapname[256];	/* chap system name */
 
 	/* link quality monitoring */
@@ -373,6 +375,8 @@
 extern ushort	ptclcsum(Block*, int, int);
 extern ushort	ptclbsum(uchar*, int);
 extern ushort	ipcsum(uchar*);
+
+void getasymkey(uchar *key, uchar *masterkey, int send, int server);
 
 extern	Comptype	cmppc;
 extern	Uncomptype	uncmppc;