⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 cpu.c

📁 这是一个同样来自贝尔实验室的和UNIX有着渊源的操作系统, 其简洁的设计和实现易于我们学习和理解
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * cpu.c - Make a connection to a cpu server * *	   Invoked by listen as 'cpu -R | -N service net netdir' *	    	   by users  as 'cpu [-h system] [-c cmd args ...]' */#include <u.h>#include <libc.h>#include <bio.h>#include <auth.h>#include <fcall.h>#include <libsec.h>#define	Maxfdata 8192#define MaxStr 128void	remoteside(int);void	fatal(int, char*, ...);void	lclnoteproc(int);void	rmtnoteproc(void);void	catcher(void*, char*);void	usage(void);void	writestr(int, char*, char*, int);int	readstr(int, char*, int);char	*rexcall(int*, char*, char*);int	setamalg(char*);char *keyspec = "";int 	notechan;int	exportpid;char	*system;int	cflag;int	dbg;char	*user;char	*patternfile;char	*srvname = "ncpu";char	*exportfs = "/bin/exportfs";char	*ealgs = "rc4_256 sha1";/* message size for exportfs; may be larger so we can do big graphics in CPU window */int	msgsize = Maxfdata+IOHDRSZ;/* authentication mechanisms */static int	netkeyauth(int);static int	netkeysrvauth(int, char*);static int	p9auth(int);static int	srvp9auth(int, char*);static int	noauth(int);static int	srvnoauth(int, char*);typedef struct AuthMethod AuthMethod;struct AuthMethod {	char	*name;			/* name of method */	int	(*cf)(int);		/* client side authentication */	int	(*sf)(int, char*);	/* server side authentication */} authmethod[] ={	{ "p9",		p9auth,		srvp9auth,},	{ "netkey",	netkeyauth,	netkeysrvauth,},//	{ "none",	noauth,		srvnoauth,},	{ nil,	nil}};AuthMethod *am = authmethod;	/* default is p9 */char *p9authproto = "p9any";int setam(char*);voidusage(void){	fprint(2, "usage: cpu [-h system] [-u user] [-a authmethod] [-e 'crypt hash'] [-k keypattern] [-P patternfile] [-c cmd args ...]\n");	exits("usage");}voidmain(int argc, char **argv){	char dat[MaxStr], buf[MaxStr], cmd[MaxStr], *p, *err;	int ac, fd, ms, data;	char *av[10];	/* see if we should use a larger message size */	fd = open("/dev/draw", OREAD);	if(fd > 0){		ms = iounit(fd);		if(msgsize < ms+IOHDRSZ)			msgsize = ms+IOHDRSZ;		close(fd);	}	user = getuser();	if(user == nil)		fatal(1, "can't read user name");	ARGBEGIN{	case 'a':		p = EARGF(usage());		if(setam(p) < 0)			fatal(0, "unknown auth method %s", p);		break;	case 'e':		ealgs = EARGF(usage());		if(*ealgs == 0 || strcmp(ealgs, "clear") == 0)			ealgs = nil;		break;	case 'd':		dbg++;		break;	case 'f':		/* ignored but accepted for compatibility */		break;	case 'O':		p9authproto = "p9sk2";		remoteside(1);				/* From listen */		break;	case 'R':				/* From listen */		remoteside(0);		break;	case 'h':		system = EARGF(usage());		break;	case 'c':		cflag++;		cmd[0] = '!';		cmd[1] = '\0';		while(p = ARGF()) {			strcat(cmd, " ");			strcat(cmd, p);		}		break;	case 'k':		keyspec = EARGF(usage());		break;	case 'P':		patternfile = EARGF(usage());		break;	case 'u':		user = EARGF(usage());		keyspec = smprint("user=%s", user);		break;	default:		usage();	}ARGEND;	if(argc != 0)		usage();	if(system == nil) {		p = getenv("cpu");		if(p == 0)			fatal(0, "set $cpu");		system = p;	}	if(err = rexcall(&data, system, srvname))		fatal(1, "%s: %s", err, system);	/* Tell the remote side the command to execute and where our working directory is */	if(cflag)		writestr(data, cmd, "command", 0);	if(getwd(dat, sizeof(dat)) == 0)		writestr(data, "NO", "dir", 0);	else		writestr(data, dat, "dir", 0);	/* start up a process to pass along notes */	lclnoteproc(data);	/* 	 *  Wait for the other end to execute and start our file service	 *  of /mnt/term	 */	if(readstr(data, buf, sizeof(buf)) < 0)		fatal(1, "waiting for FS: %r");	if(strncmp("FS", buf, 2) != 0) {		print("remote cpu: %s", buf);		exits(buf);	}	/* Begin serving the gnot namespace */	close(0);	dup(data, 0);	close(data);	sprint(buf, "%d", msgsize);	ac = 0;	av[ac++] = exportfs;	av[ac++] = "-m";	av[ac++] = buf;	if(dbg)		av[ac++] = "-d";	if(patternfile != nil){		av[ac++] = "-P";		av[ac++] = patternfile;	}	av[ac] = nil;	exec(exportfs, av);	fatal(1, "starting exportfs");}voidfatal(int syserr, char *fmt, ...){	Fmt f;	char *str;	va_list arg;	fmtstrinit(&f);	fmtprint(&f, "cpu: ");	va_start(arg, fmt);	fmtvprint(&f, fmt, arg);	va_end(arg);	if(syserr)		fmtprint(&f, ": %r");	fmtprint(&f, "\n");	str = fmtstrflush(&f);	write(2, str, strlen(str));	exits(str);}char *negstr = "negotiating authentication method";char bug[256];intold9p(int fd){	int p[2];	if(pipe(p) < 0)		fatal(1, "pipe");	switch(rfork(RFPROC|RFFDG|RFNAMEG)) {	case -1:		fatal(1, "rfork srvold9p");	case 0:		if(fd != 1){			dup(fd, 1);			close(fd);		}		if(p[0] != 0){			dup(p[0], 0);			close(p[0]);		}		close(p[1]);		if(0){			fd = open("/sys/log/cpu", OWRITE);			if(fd != 2){				dup(fd, 2);				close(fd);			}			execl("/bin/srvold9p", "srvold9p", "-ds", nil);		} else			execl("/bin/srvold9p", "srvold9p", "-s", nil);		fatal(1, "exec srvold9p");	default:		close(fd);		close(p[0]);	}	return p[1];	}/* Invoked with stdin, stdout and stderr connected to the network connection */voidremoteside(int old){	char user[MaxStr], home[MaxStr], buf[MaxStr], xdir[MaxStr], cmd[MaxStr];	int i, n, fd, badchdir, gotcmd;	rfork(RFENVG);	putenv("service", "cpu");	fd = 0;	/* negotiate authentication mechanism */	n = readstr(fd, cmd, sizeof(cmd));	if(n < 0)		fatal(1, "authenticating");	if(setamalg(cmd) < 0){		writestr(fd, "unsupported auth method", nil, 0);		fatal(1, "bad auth method %s", cmd);	} else		writestr(fd, "", "", 1);	fd = (*am->sf)(fd, user);	if(fd < 0)		fatal(1, "srvauth");	/* Set environment values for the user */	putenv("user", user);	sprint(home, "/usr/%s", user);	putenv("home", home);	/* Now collect invoking cpu's current directory or possibly a command */	gotcmd = 0;	if(readstr(fd, xdir, sizeof(xdir)) < 0)		fatal(1, "dir/cmd");	if(xdir[0] == '!') {		strcpy(cmd, &xdir[1]);		gotcmd = 1;		if(readstr(fd, xdir, sizeof(xdir)) < 0)			fatal(1, "dir");	}	/* Establish the new process at the current working directory of the	 * gnot */	badchdir = 0;	if(strcmp(xdir, "NO") == 0)		chdir(home);	else if(chdir(xdir) < 0) {		badchdir = 1;		chdir(home);	}	/* Start the gnot serving its namespace */	writestr(fd, "FS", "FS", 0);	writestr(fd, "/", "exportfs dir", 0);	n = read(fd, buf, sizeof(buf));	if(n != 2 || buf[0] != 'O' || buf[1] != 'K')		exits("remote tree");	if(old)		fd = old9p(fd);	/* make sure buffers are big by doing fversion explicitly; pick a huge number; other side will trim */	strcpy(buf, VERSION9P);	if(fversion(fd, 64*1024, buf, sizeof buf) < 0)		exits("fversion failed");	if(mount(fd, -1, "/mnt/term", MCREATE|MREPL, "") < 0)		exits("mount failed");	close(fd);	/* the remote noteproc uses the mount so it must follow it */	rmtnoteproc();	for(i = 0; i < 3; i++)		close(i);	if(open("/mnt/term/dev/cons", OREAD) != 0)		exits("open stdin");	if(open("/mnt/term/dev/cons", OWRITE) != 1)		exits("open stdout");	dup(1, 2);	if(badchdir)		print("cpu: failed to chdir to '%s'\n", xdir);	if(gotcmd)		execl("/bin/rc", "rc", "-lc", cmd, nil);	else		execl("/bin/rc", "rc", "-li", nil);	fatal(1, "exec shell");}char*rexcall(int *fd, char *host, char *service){	char *na;	char dir[MaxStr];	char err[ERRMAX];	char msg[MaxStr];	int n;	na = netmkaddr(host, 0, service);	if((*fd = dial(na, 0, dir, 0)) < 0)		return "can't dial";	/* negotiate authentication mechanism */	if(ealgs != nil)		snprint(msg, sizeof(msg), "%s %s", am->name, ealgs);	else		snprint(msg, sizeof(msg), "%s", am->name);	writestr(*fd, msg, negstr, 0);	n = readstr(*fd, err, sizeof err);	if(n < 0)		return negstr;	if(*err){		werrstr(err);		return negstr;	}	/* authenticate */	*fd = (*am->cf)(*fd);	if(*fd < 0)		return "can't authenticate";	return 0;}voidwritestr(int fd, char *str, char *thing, int ignore){	int l, n;	l = strlen(str);	n = write(fd, str, l+1);	if(!ignore && n < 0)		fatal(1, "writing network: %s", thing);}intreadstr(int fd, char *str, int len){	int n;	while(len) {		n = read(fd, str, 1);		if(n < 0) 			return -1;		if(*str == '\0')			return 0;		str++;		len--;	}	return -1;}static intreadln(char *buf, int n){	int i;	char *p;	n--;	/* room for \0 */	p = buf;	for(i=0; i<n; i++){		if(read(0, p, 1) != 1)			break;		if(*p == '\n' || *p == '\r')			break;		p++;	}	*p = '\0';	return p-buf;}/* *  user level challenge/response */static intnetkeyauth(int fd){	char chall[32];	char resp[32];	strecpy(chall, chall+sizeof chall, getuser());	print("user[%s]: ", chall);	if(readln(resp, sizeof(resp)) < 0)		return -1;	if(*resp != 0)		strcpy(chall, resp);	writestr(fd, chall, "challenge/response", 1);	for(;;){		if(readstr(fd, chall, sizeof chall) < 0)			break;		if(*chall == 0)			return fd;		print("challenge: %s\nresponse: ", chall);		if(readln(resp, sizeof(resp)) < 0)			break;		writestr(fd, resp, "challenge/response", 1);	}	return -1;}static intnetkeysrvauth(int fd, char *user){	char response[32];	Chalstate *ch;	int tries;	AuthInfo *ai;	if(readstr(fd, user, 32) < 0)		return -1;	ai = nil;	ch = nil;	for(tries = 0; tries < 10; tries++){		if((ch = auth_challenge("proto=p9cr role=server user=%q", user)) == nil)			return -1;		writestr(fd, ch->chal, "challenge", 1);		if(readstr(fd, response, sizeof response) < 0)			return -1;		ch->resp = response;		ch->nresp = strlen(response);		if((ai = auth_response(ch)) != nil)			break;	}	auth_freechal(ch);	if(ai == nil)		return -1;	writestr(fd, "", "challenge", 1);	if(auth_chuid(ai, 0) < 0)		fatal(1, "newns");	auth_freeAI(ai);	return fd;}static voidmksecret(char *t, uchar *f){	sprint(t, "%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux%2.2ux",		f[0], f[1], f[2], f[3], f[4], f[5], f[6], f[7], f[8], f[9]);}/* *  plan9 authentication followed by rc4 encryption */static intp9auth(int fd){	uchar key[16];	uchar digest[SHA1dlen];	char fromclientsecret[21];	char fromserversecret[21];	int i;	AuthInfo *ai;	ai = auth_proxy(fd, auth_getkey, "proto=%q role=client %s", p9authproto, keyspec);	if(ai == nil)		return -1;	memmove(key+4, ai->secret, ai->nsecret);	if(ealgs == nil)		return fd;	/* exchange random numbers */	srand(truerand());	for(i = 0; i < 4; i++)		key[i] = rand();	if(write(fd, key, 4) != 4)		return -1;	if(readn(fd, key+12, 4) != 4)		return -1;	/* scramble into two secrets */	sha1(key, sizeof(key), digest, nil);	mksecret(fromclientsecret, digest);	mksecret(fromserversecret, digest+10);	/* set up encryption */	i = pushssl(fd, ealgs, fromclientsecret, fromserversecret, nil);	if(i < 0)		werrstr("can't establish ssl connection: %r");	return i;}static intnoauth(int fd){	ealgs = nil;	return fd;}static intsrvnoauth(int fd, char *user){	strecpy(user, user+MaxStr, getuser());	ealgs = nil;	return fd;}voidloghex(uchar *p, int n){	char buf[100];	int i;	for(i = 0; i < n; i++)		sprint(buf+2*i, "%2.2ux", p[i]);	syslog(0, "cpu", buf);}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -