code: drawterm

Download patch

ref: 9731b2c5e105e92d14a7899fddd7142f6f99fc3f
parent: 39347ab47dab6c556bb37ced7526bfcc8de4cf9b
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Thu Feb 25 03:31:01 EST 2016

fix devfs-*.c to not use cname hack to get file path, get rid of MAXPATH limit, make root accessible under /mnt/term/root

under linux, it was not possible to access the clients /mnt directory as it
gets hidden by #/mnt, to solve this, we first bind #U to /root, and then
let cpumain() bind any subdirectory (-r option) in there and finally bind
/root after /. for this to work, i had to get rid of the dreadfull cname hack,
where the c->name is used to derive the unix file path. this did not work when
the device was accessed form a different bind... and it also leaked memory in
cclone() as the c->name's where doubly maintained in fswalk().

also get rid of the fixed MAXPATH sized stack buffers. each devfs now maintains
a dynamic buffer in ther Ufsinfo structure. under windows, this is a wide string.

remove the #ifdef UNICODE maze from devfs-posix.c, we always assume unicode
now with wide strings.

--- a/cpu.c
+++ b/cpu.c
@@ -29,7 +29,6 @@
 static char	*system;
 static int	cflag;
 extern int	dbg;
-extern char*   base;   // fs base for devroot 
 
 static char	*srvname = "ncpu";
 static char	*ealgs = "rc4_256 sha1";
@@ -69,7 +68,7 @@
 void
 usage(void)
 {
-	fprint(2, "usage: drawterm [-a authserver] [-c cpuserver] [-s secstore] [-u user]\n");
+	fprint(2, "usage: drawterm [-a authserver] [-c cpuserver] [-s secstore] [-u user] [-r root]\n");
 	exits("usage");
 }
 int fdd;
@@ -168,6 +167,12 @@
 		if(*ealgs == 0 || strcmp(ealgs, "clear") == 0)
 			ealgs = nil;
 		break;
+	case 'r':
+		snprint(buf, sizeof(buf), "/root/%s", EARGF(usage()));
+		cleanname(buf);
+		if(bind(buf, "/root", MREPL) < 0)
+			panic("bind /root: %r");
+		break;
 	case 'C':
 		cflag++;
 		cmd[0] = '!';
@@ -180,9 +185,6 @@
 	case 'k':
 		keyspec = EARGF(usage());
 		break;
-	case 'r':
-		base = EARGF(usage());
-		break;
 	case 's':
 		secstoreserver = EARGF(usage());
 		break;
@@ -195,6 +197,9 @@
 
 	if(argc != 0)
 		usage();
+
+	if(bind("/root", "/", MAFTER) < 0)
+		panic("bind /root: %r");
 
 	if(system == nil)
 		system = readcons("cpu", "cpu", 0);
--- a/kern/devfs-posix.c
+++ b/kern/devfs-posix.c
@@ -15,17 +15,7 @@
 #include	"fns.h"
 #include	"error.h"
 
-
 typedef	struct Ufsinfo	Ufsinfo;
-
-enum
-{
-	NUID	= 256,
-	NGID	= 256,
-	MAXPATH	= 1024,
-	MAXCOMP	= 128
-};
-
 struct Ufsinfo
 {
 	int	mode;
@@ -35,36 +25,20 @@
 	DIR*	dir;
 	vlong	offset;
 	QLock	oq;
+	char*	path;
 	char nextname[NAME_MAX];
 };
 
-char	*base = "/";
-
 static	Qid	fsqid(struct stat *);
-static	void	fspath(Chan*, char*, char*);
+static	char*	catpath(char*, char*);
 static	ulong	fsdirread(Chan*, uchar*, int, ulong);
 static	int	fsomode(int);
 
-/* clumsy hack, but not worse than the Path stuff in the last one */
 static char*
-uc2name(Chan *c)
+lastelem(char *s)
 {
-	char *s;
+	char *t;
 
-	if(c->name == nil)
-		return "/";
-	s = c2name(c);
-	if(s[0]=='#' && s[1]=='U')
-		return s+2;
-	return s;
-}
-
-static char*
-lastelem(Chan *c)
-{
-	char *s, *t;
-
-	s = uc2name(c);
 	if((t = strrchr(s, '/')) == nil)
 		return s;
 	if(t[1] == 0)
@@ -80,9 +54,6 @@
 	static int devno;
 	Ufsinfo *uif;
 
-	if(stat(base, &stbuf) < 0)
-		error(strerror(errno));
-
 	c = devattach('U', spec);
 
 	uif = mallocz(sizeof(Ufsinfo), 1);
@@ -89,6 +60,7 @@
 	uif->mode = stbuf.st_mode;
 	uif->uid = stbuf.st_uid;
 	uif->gid = stbuf.st_gid;
+	uif->path = strdup("/");
 
 	c->aux = uif;
 	c->dev = devno++;
@@ -105,6 +77,7 @@
 
 	uif = mallocz(sizeof(Ufsinfo), 1);
 	*uif = *(Ufsinfo*)c->aux;
+	uif->path = strdup(uif->path);
 	nc->aux = uif;
 
 	return nc;
@@ -114,18 +87,20 @@
 fswalk1(Chan *c, char *name)
 {
 	struct stat stbuf;
-	char path[MAXPATH];
 	Ufsinfo *uif;
+	char *path;
 
-	fspath(c, name, path);
-
 	/*print("** fs walk '%s' -> %s\n", path, name);  */
 
-	if(stat(path, &stbuf) < 0)
+	uif = c->aux;
+	path = catpath(uif->path, name);
+	if(stat(path, &stbuf) < 0){
+		free(path);
 		return 0;
+	}
+	free(uif->path);
+	uif->path = path;
 
-	uif = c->aux;
-
 	uif->mode = stbuf.st_mode;
 	uif->uid = stbuf.st_uid;
 	uif->gid = stbuf.st_gid;
@@ -135,13 +110,10 @@
 	return 1;
 }
 
-extern Cname* addelem(Cname*, char*);
-
 static Walkqid*
 fswalk(Chan *c, Chan *nc, char **name, int nname)
 {
 	int i;
-	Cname *cname;
 	Walkqid *wq;
 
 	if(nc != nil)
@@ -148,19 +120,13 @@
 		panic("fswalk: nc != nil");
 	wq = smalloc(sizeof(Walkqid)+(nname-1)*sizeof(Qid));
 	nc = devclone(c);
-	cname = c->name;
-	incref(&cname->ref);
-
 	fsclone(c, nc);
 	wq->clone = nc;
 	for(i=0; i<nname; i++){
-		nc->name = cname;
 		if(fswalk1(nc, name[i]) == 0)
 			break;
-		cname = addelem(cname, name[i]);
 		wq->qid[i] = nc->qid;
 	}
-	nc->name = cname;
 	if(i != nname){
 		cclose(nc);
 		wq->clone = nil;
@@ -174,19 +140,19 @@
 {
 	Dir d;
 	struct stat stbuf;
-	char path[MAXPATH];
+	Ufsinfo *uif;
 
 	if(n < BIT16SZ)
 		error(Eshortstat);
 
-	fspath(c, 0, path);
-	if(stat(path, &stbuf) < 0)
+	uif = c->aux;
+	if(stat(uif->path, &stbuf) < 0)
 		error(strerror(errno));
 
-	d.name = lastelem(c);
-	d.uid = "unknown";
-	d.gid = "unknown";
-	d.muid = "unknown";
+	d.name = lastelem(uif->path);
+	d.uid = eve;
+	d.gid = eve;
+	d.muid = eve;
 	d.qid = c->qid;
 	d.mode = (c->qid.type<<24)|(stbuf.st_mode&0777);
 	d.atime = stbuf.st_atime;
@@ -200,7 +166,6 @@
 static Chan*
 fsopen(Chan *c, int mode)
 {
-	char path[MAXPATH];
 	int m, isdir;
 	Ufsinfo *uif;
 
@@ -231,10 +196,8 @@
 	c->mode = openmode(mode);
 
 	uif = c->aux;
-
-	fspath(c, 0, path);
 	if(isdir) {
-		uif->dir = opendir(path);
+		uif->dir = opendir(uif->path);
 		if(uif->dir == 0)
 			error(strerror(errno));
 	}	
@@ -241,8 +204,7 @@
 	else {
 		if(mode & OTRUNC)
 			m |= O_TRUNC;
-		uif->fd = open(path, m, 0666);
-
+		uif->fd = open(uif->path, m, 0666);
 		if(uif->fd < 0)
 			error(strerror(errno));
 	}
@@ -257,16 +219,18 @@
 fscreate(Chan *c, char *name, int mode, ulong perm)
 {
 	int fd, m;
-	char path[MAXPATH];
+	char *path;
 	struct stat stbuf;
 	Ufsinfo *uif;
 
 	m = fsomode(mode&3);
 
-	fspath(c, name, path);
-
 	uif = c->aux;
-
+	path = catpath(uif->path, name);
+	if(waserror()){
+		free(path);
+		nexterror();
+	}
 	if(perm & DMDIR) {
 		if(m)
 			error(Eperm);
@@ -299,9 +263,13 @@
 			error(strerror(errno));
 		uif->fd = fd;
 	}
-
 	if(stat(path, &stbuf) < 0)
 		error(strerror(errno));
+
+	free(uif->path);
+	uif->path = path;
+	poperror();
+
 	c->qid = fsqid(&stbuf);
 	c->offset = 0;
 	c->flag |= COPEN;
@@ -314,7 +282,6 @@
 	Ufsinfo *uif;
 
 	uif = c->aux;
-
 	if(c->flag & COPEN) {
 		if(c->qid.type & QTDIR)
 			closedir(uif->dir);
@@ -321,7 +288,7 @@
 		else
 			close(uif->fd);
 	}
-
+	free(uif->path);
 	free(uif);
 }
 
@@ -367,7 +334,6 @@
 	Ufsinfo *uif;
 
 	uif = c->aux;
-
 	qlock(&uif->oq);
 	if(waserror()) {
 		qunlock(&uif->oq);
@@ -396,13 +362,13 @@
 fsremove(Chan *c)
 {
 	int n;
-	char path[MAXPATH];
+	Ufsinfo *uif;
 
-	fspath(c, 0, path);
+	uif = c->aux;
 	if(c->qid.type & QTDIR)
-		n = rmdir(path);
+		n = rmdir(uif->path);
 	else
-		n = remove(path);
+		n = remove(uif->path);
 	if(n < 0)
 		error(strerror(errno));
 }
@@ -412,34 +378,39 @@
 {
 	Dir d;
 	struct stat stbuf;
-	char old[MAXPATH], new[MAXPATH];
-	char strs[MAXPATH*3], *p;
+	char strs[NAME_MAX*3];
 	Ufsinfo *uif;
 
 	if(convM2D(buf, n, &d, strs) != n)
 		error(Ebadstat);
 	
-	fspath(c, 0, old);
-	if(stat(old, &stbuf) < 0)
+	uif = c->aux;
+	if(stat(uif->path, &stbuf) < 0)
 		error(strerror(errno));
 
-	uif = c->aux;
-
-	fspath(c, 0, old);
 	if(~d.mode != 0 && (int)(d.mode&0777) != (int)(stbuf.st_mode&0777)) {
-		if(chmod(old, d.mode&0777) < 0)
+		if(chmod(uif->path, d.mode&0777) < 0)
 			error(strerror(errno));
 		uif->mode &= ~0777;
 		uif->mode |= d.mode&0777;
 	}
 
-	if(d.name[0] && strcmp(d.name, lastelem(c)) != 0) {
-		fspath(c, 0, old);
-		strcpy(new, old);
-		p = strrchr(new, '/');
-		strcpy(p+1, d.name);
-		if(rename(old, new) < 0)
+	if(d.name[0] && strcmp(d.name, lastelem(uif->path)) != 0) {
+		char *base, *newpath;
+
+		base = strdup(uif->path);
+		if(waserror()){
+			free(base);
+			nexterror();
+		}
+		*lastelem(base) = 0;
+		newpath = catpath(base, d.name);
+		free(base), base = newpath;
+		if(rename(uif->path, newpath) < 0)
 			error(strerror(errno));
+		free(uif->path);
+		uif->path = newpath;
+		poperror();
 	}
 
 /*
@@ -472,17 +443,20 @@
 	return q;
 }
 
-static void
-fspath(Chan *c, char *ext, char *path)
+static char*
+catpath(char *base, char *ext)
 {
-	strcpy(path, base);
-	strcat(path, "/");
-	strcat(path, uc2name(c));
-	if(ext){
-		strcat(path, "/");
-		strcat(path, ext);
-	}
-	cleanname(path);
+	char *path;
+	int n, m;
+
+	n = strlen(base);
+	m = strlen(ext);
+	path = malloc(n+m+2);
+	memmove(path, base, n);
+	if(n > 0 && path[n-1] != '/')
+		path[n++] = '/';
+	memmove(path+n, ext, m+1);
+	return path;
 }
 
 static int
@@ -526,7 +500,6 @@
 	long n;
 	char de[NAME_MAX];
 	struct stat stbuf;
-	char path[MAXPATH], dirpath[MAXPATH];
 	Ufsinfo *uif;
 
 /*print("fsdirread %s\n", c2name(c));*/
@@ -542,9 +515,9 @@
 		rewinddir(uif->dir);
 	}
 
-	fspath(c, 0, dirpath);
-
 	while(i+BIT16SZ < count) {
+		char *p;
+
 		if(!p9readdir(de, uif))
 			break;
 
@@ -552,17 +525,17 @@
 			continue;
 
 		d.name = de;
-		sprint(path, "%s/%s", dirpath, de);
-		memset(&stbuf, 0, sizeof stbuf);
-
-		if(stat(path, &stbuf) < 0) {
+		p = catpath(uif->path, de);
+		if(stat(p, &stbuf) < 0) {
 			/* fprint(2, "dir: bad path %s\n", path); */
 			/* but continue... probably a bad symlink */
+			memset(&stbuf, 0, sizeof stbuf);
 		}
+		free(p);
 
-		d.uid = "unknown";
-		d.gid = "unknown";
-		d.muid = "unknown";
+		d.uid = eve;
+		d.gid = eve;
+		d.muid = eve;
 		d.qid = fsqid(&stbuf);
 		d.mode = (d.qid.type<<24)|(stbuf.st_mode&0777);
 		d.atime = stbuf.st_atime;
--- a/kern/devfs-win32.c
+++ b/kern/devfs-win32.c
@@ -14,10 +14,8 @@
 typedef struct DIR	DIR;
 typedef	struct Ufsinfo	Ufsinfo;
 
-
 enum
 {
-	MAXPATH	= 1024,
 	TPATH_ROOT	= 0,	// ""
 	TPATH_VOLUME	= 1,	// "C:"
 	TPATH_FILE	= 2,	// "C:\bla"
@@ -30,8 +28,8 @@
 	WIN32_FIND_DATA	wfd;
 
 	// for GetLogicalDriveStrings()
-	TCHAR		*drivebuf;
-	TCHAR		*drivep;
+	wchar_t		*drivebuf;
+	wchar_t		*drivep;
 
 	// dont move to the next item
 	int		keep;
@@ -44,17 +42,16 @@
 	DIR*	dir;
 	vlong	offset;
 	QLock	oq;
+	wchar_t	*path;
 };
 
-static	void	fspath(Chan *, char *, TCHAR *, int);
+static	wchar_t *catpath(wchar_t *, char *, wchar_t *);
 static	ulong	fsdirread(Chan*, uchar*, int, vlong);
 static	int	fsomode(int);
 static	ulong	fsaccess(int);
-static	ulong	pathtype(TCHAR *);
-static	int	checkvolume(TCHAR *);
+static	ulong	pathtype(wchar_t *);
+static	int	checkvolume(wchar_t *);
 
-char *base = "";
-
 static ulong
 unixtime(FILETIME *ft)
 {
@@ -65,12 +62,10 @@
 }
 
 static uvlong
-pathhash(TCHAR *p)
+pathhash(wchar_t *p)
 {
 	uchar digest[SHA1dlen];
-	int n;
-	for(n=0; p[n] != 0; n++);
-	sha1((uchar*)p, n*sizeof(TCHAR), digest, nil);
+	sha1((uchar*)p, wcslen(p)*sizeof(wchar_t), digest, nil);
 	return *(uvlong*)digest;
 }
 
@@ -89,7 +84,7 @@
 }
 
 static Qid
-wfdtoqid(TCHAR *path, WIN32_FIND_DATA *wfd)
+wfdtoqid(wchar_t *path, WIN32_FIND_DATA *wfd)
 {
 	ulong t;
 	WIN32_FIND_DATA f;
@@ -96,7 +91,6 @@
 	Qid q;
 
 	t = pathtype(path);
-
 	switch(t){
 	case TPATH_VOLUME:
 	case TPATH_ROOT:
@@ -122,14 +116,10 @@
 }
 
 static void
-wfdtodir(TCHAR *path, Dir *d, WIN32_FIND_DATA *wfd)
+wfdtodir(wchar_t *path, Dir *d, WIN32_FIND_DATA *wfd)
 {
 	WIN32_FIND_DATA f;
 
-	d->uid	= "nul";
-	d->gid	= "nul";
-	d->muid	= "nul";
-
 	switch(pathtype(path)){
 	case TPATH_VOLUME:
 	case TPATH_ROOT:
@@ -141,7 +131,7 @@
 		break;
 
 	case TPATH_FILE:
-		if(!wfd){
+		if(wfd == nil){
 			HANDLE h;
 			if((h = FindFirstFile(path, &f))==INVALID_HANDLE_VALUE)
 				oserror();
@@ -155,30 +145,17 @@
 		break;
 	}
 	d->qid = wfdtoqid(path, wfd);
+	d->uid = eve;
+	d->gid = eve;
+	d->muid = eve;
 }
 
-/* clumsy hack, but not worse than the Path stuff in the last one */
 static char*
-uc2name(Chan *c)
-{
-	char *s;
-
-	if(c->name == nil)
-		return "";
-	s = c2name(c);
-	if(s[0]=='#' && s[1]=='U')
-		s += 2;
-	if(*s=='/')
-		s++;
-	return s;
-}
-
-static char*
 lastelem(Chan *c)
 {
 	char *s, *t;
 
-	s = uc2name(c);
+	s = c2name(c);
 	if((t = strrchr(s, '/')) == nil)
 		return s;
 	if(t[1] == 0)
@@ -187,35 +164,26 @@
 }
 
 static ulong
-pathtype(TCHAR *path)
+pathtype(wchar_t *path)
 {
-	int n;
-	n = wcslen(path);
-	if(n < 2){
+	if(path[0] == 0 || path[1] == 0)
 		return TPATH_ROOT;
-	}
-	if(n==2){
+	if(path[1] == ':' && path[2] == 0)
 		return TPATH_VOLUME;
-	}
 	return TPATH_FILE;
 }
 
 static int
-checkvolume(TCHAR *path)
+checkvolume(wchar_t *path)
 {
-	TCHAR vol[MAX_PATH];
-	TCHAR volname[MAX_PATH];
-	TCHAR fsysname[MAX_PATH];
+	wchar_t vol[MAX_PATH];
+	wchar_t volname[MAX_PATH];
+	wchar_t fsysname[MAX_PATH];
 	DWORD complen;
 	DWORD flags;
 
-	wcsncpy(vol, path, MAX_PATH);
-#ifdef UNICODE
-	wcsncat(vol, L"\\", MAXPATH);
-#else
-	wcsncat(vol, "\\", MAXPATH);
-#endif
-
+	wcscpy(vol, path);
+	wcscat(vol, L"\\");
 	if(!GetVolumeInformation(
 		vol,
 		volname,
@@ -224,27 +192,35 @@
 		&complen,
 		&flags,
 		fsysname,
-		MAX_PATH)){
+		MAX_PATH))
 		return 0;
-	}
+
 	return 1;	
 }
 
-static int
-getfileowner(TCHAR *path, char *owner, int nowner)
+static wchar_t*
+wstrdup(wchar_t *s)
 {
-	strncpy(owner, "Bill", nowner);
-	return 1;
+	wchar_t *d;
+	long n;
+
+	n = (wcslen(s)+1)*sizeof(wchar_t);
+	d = mallocz(n, 0);
+	memmove(d, s, n);
+	return d;
 }
 	
 static Chan*
 fsattach(char *spec)
 {
-	Chan *c;
 	static int devno;
 	Ufsinfo *uif;
-	c = devattach('U', spec);
+	Chan *c;
+
 	uif = mallocz(sizeof(Ufsinfo), 1);
+	uif->path = wstrdup(L"");
+
+	c = devattach('U', spec);
 	c->aux = uif;
 	c->dev = devno++;
 	c->qid.type = QTDIR;
@@ -259,6 +235,7 @@
 
 	uif = mallocz(sizeof(Ufsinfo), 1);
 	*uif = *(Ufsinfo*)c->aux;
+	uif->path = wstrdup(uif->path);
 	nc->aux = uif;
 
 	return nc;
@@ -267,37 +244,41 @@
 static int
 fswalk1(Chan *c, char *name)
 {
-	HANDLE h;
 	WIN32_FIND_DATA wfd;
-	TCHAR path[MAXPATH];
-
-	fspath(c, name, path, MAXPATH);
-
-	switch(pathtype(path)){
+	HANDLE h;
+	wchar_t *p;
+	Ufsinfo *uif;
+	
+	uif = c->aux;
+	p = catpath(uif->path, name, nil);
+	switch(pathtype(p)){
 	case TPATH_VOLUME:
-		if(!checkvolume(path))
+		if(!checkvolume(p)){
+			free(p);
 			return 0;
+		}
 	case TPATH_ROOT:
-		c->qid = wfdtoqid(path, nil);
+		c->qid = wfdtoqid(p, nil);
 		break;
 
 	case TPATH_FILE:
-		if((h = FindFirstFile(path, &wfd)) == INVALID_HANDLE_VALUE)
+		if((h = FindFirstFile(p, &wfd)) == INVALID_HANDLE_VALUE){
+			free(p);
 			return 0;
+		}
 		FindClose(h);
-		c->qid = wfdtoqid(path, &wfd);
+		c->qid = wfdtoqid(p, &wfd);
 		break;
 	}
+	free(uif->path);
+	uif->path = p;
 	return 1;
 }
 
-extern Cname* addelem(Cname*, char*);
-
 static Walkqid*
 fswalk(Chan *c, Chan *nc, char **name, int nname)
 {
 	int i;
-	Cname *cname;
 	Walkqid *wq;
 
 	if(nc != nil)
@@ -304,19 +285,13 @@
 		panic("fswalk: nc != nil");
 	wq = smalloc(sizeof(Walkqid)+(nname-1)*sizeof(Qid));
 	nc = devclone(c);
-	cname = c->name;
-	incref(&cname->ref);
-
 	fsclone(c, nc);
 	wq->clone = nc;
 	for(i=0; i<nname; i++){
-		nc->name = cname;
 		if(fswalk1(nc, name[i]) == 0)
 			break;
-		cname = addelem(cname, name[i]);
 		wq->qid[i] = nc->qid;
 	}
-	nc->name = cname;
 	if(i != nname){
 		cclose(nc);
 		wq->clone = nil;
@@ -329,21 +304,13 @@
 fsstat(Chan *c, uchar *buf, int n)
 {
 	Dir d;
-	TCHAR path[MAXPATH];
-	char owner[MAXPATH];
+	Ufsinfo *uif;
 
 	if(n < BIT16SZ)
 		error(Eshortstat);
-	fspath(c, 0, path, MAXPATH);
+	uif = c->aux;
 	d.name = lastelem(c);
-	wfdtodir(path, &d, nil);
-
-	if(getfileowner(path, owner, MAXPATH)){
-		d.uid = owner;
-		d.gid = owner;
-		d.muid = owner;
-	}
-
+	wfdtodir(uif->path, &d, nil);
 	d.type = 'U';
 	d.dev = c->dev;
 	return convD2M(&d, buf, n);
@@ -352,9 +319,9 @@
 static Chan*
 fsopen(Chan *c, int mode)
 {
-	TCHAR path[MAXPATH];
 	ulong t;
 	int m, isdir;
+	wchar_t *p;
 	Ufsinfo *uif;
 
 	m = mode & (OTRUNC|3);
@@ -381,14 +348,13 @@
 	c->mode = openmode(mode);
 	uif = c->aux;
 	uif->offset = 0;
-	fspath(c, 0, path, MAXPATH);
-	t = pathtype(path);
+	t = pathtype(uif->path);
 	if(isdir){
 		DIR *d;
 		d = malloc(sizeof(*d));
 		switch(t){
 		case TPATH_ROOT:
-			d->drivebuf = malloc(sizeof(TCHAR)*MAX_PATH);
+			d->drivebuf = malloc(sizeof(wchar_t)*MAX_PATH);
 			if(GetLogicalDriveStrings(MAX_PATH-1, d->drivebuf) == 0){
 				free(d->drivebuf);
 				d->drivebuf = nil;
@@ -398,12 +364,10 @@
 			break;
 		case TPATH_VOLUME:
 		case TPATH_FILE:
-#ifdef UNICODE
-			wcsncat(path, L"\\*.*", MAXPATH);
-#else
-			wcsncat(path, "\\*.*", MAXPATH);
-#endif
-			if((d->handle = FindFirstFile(path, &d->wfd)) == INVALID_HANDLE_VALUE){
+			p = catpath(uif->path, "*.*", nil);
+			d->handle = FindFirstFile(p, &d->wfd);
+			free(p);
+			if(d->handle == INVALID_HANDLE_VALUE){
 				free(d);
 				oserror();
 			}
@@ -414,15 +378,14 @@
 	} else {
 		uif->dir = nil;
 		if((uif->fh = CreateFile(
-			path,
+			uif->path,
 			fsaccess(mode),
 			FILE_SHARE_READ | FILE_SHARE_WRITE,
 			NULL,
 			(mode & OTRUNC) ? TRUNCATE_EXISTING : OPEN_EXISTING,
 			FILE_ATTRIBUTE_NORMAL,
-			0)) == INVALID_HANDLE_VALUE){
+			0)) == INVALID_HANDLE_VALUE)
 			oserror();
-		}	
 	}
 	c->offset = 0;
 	c->flag |= COPEN;
@@ -433,52 +396,51 @@
 fscreate(Chan *c, char *name, int mode, ulong perm)
 {
 	int m;
-	TCHAR path[MAXPATH];
-	Ufsinfo *uif;
 	ulong t;
+	wchar_t *newpath;
+	Ufsinfo *uif;
 
 	m = fsomode(mode&3);
-
-	fspath(c, name, path, MAXPATH);
-	t = pathtype(path);
-
 	uif = c->aux;
-
+	t = pathtype(uif->path);
+	newpath = catpath(uif->path, name, nil);
+	if(waserror()){
+		free(newpath);
+		nexterror();
+	}
 	if(perm & DMDIR) {
-		TCHAR *p;
+		wchar_t *p;
 		DIR *d;
 		if(m || t!=TPATH_FILE)
 			error(Eperm);
-		if(!CreateDirectory(path, NULL))
+		if(!CreateDirectory(newpath, NULL))
 			oserror();
 		d = malloc(sizeof(*d));
-		p = &path[wcslen(path)];
-#ifdef UNICODE
-		wcsncat(path, L"\\*.*", MAXPATH);
-#else
-		wcsncat(path, "\\*.*", MAXPATH);
-#endif
-		if((d->handle = FindFirstFile(path, &d->wfd)) == INVALID_HANDLE_VALUE){
+		p = catpath(newpath, "*.*", nil);
+		d->handle = FindFirstFile(p, &d->wfd);
+		free(p);
+		if(d->handle == INVALID_HANDLE_VALUE){
 			free(d);
 			oserror();
 		}
-		*p = 0;
 		d->keep = 1;
 		uif->dir = d;
 	} else {
 		uif->dir = nil;
 		if((uif->fh = CreateFile(
-			path,
+			newpath,
 			fsaccess(mode),
 			FILE_SHARE_READ | FILE_SHARE_WRITE,
 			NULL,
 			CREATE_NEW,
 			FILE_ATTRIBUTE_NORMAL,
-			0)) == INVALID_HANDLE_VALUE){
+			0)) == INVALID_HANDLE_VALUE)
 			oserror();
-		}	
 	}
-	c->qid = wfdtoqid(path, nil);
+	free(uif->path);
+	uif->path = newpath;
+	poperror();
+	c->qid = wfdtoqid(newpath, nil);
 	c->offset = 0;
 	c->flag |= COPEN;
 	c->mode = openmode(mode);
@@ -492,8 +454,8 @@
 
 	uif = c->aux;
 	if(c->flag & COPEN) {
-		if(uif->dir){
-			if(uif->dir->drivebuf){
+		if(uif->dir != nil){
+			if(uif->dir->drivebuf != nil){
 				free(uif->dir->drivebuf);
 				uif->dir->drivebuf = nil;
 			} else {
@@ -504,6 +466,7 @@
 			CloseHandle(uif->fh);
 		}
 	}
+	free(uif->path);
 	free(uif);
 }
 
@@ -532,9 +495,8 @@
 		uif->offset = offset;
 	}
 	r = 0;
-	if(!ReadFile(fh, va, (DWORD)n, &r, NULL)){
+	if(!ReadFile(fh, va, (DWORD)n, &r, NULL))
 		oserror();
-	}
 	n = r;
 	uif->offset += n;
 	qunlock(&uif->oq);
@@ -567,9 +529,8 @@
 		uif->offset = offset;
 	}
 	w = 0;
-	if(!WriteFile(fh, va, (DWORD)n, &w, NULL)){
+	if(!WriteFile(fh, va, (DWORD)n, &w, NULL))
 		oserror();
-	}
 	n = w;
 	uif->offset += n;
 	qunlock(&uif->oq);
@@ -580,15 +541,15 @@
 static void
 fsremove(Chan *c)
 {
-	TCHAR path[MAXPATH];
+	Ufsinfo *uif;
 
-	fspath(c, 0, path, MAXPATH);
+	uif = c->aux;
 	if(c->qid.type & QTDIR){
-		if(!RemoveDirectory(path))
-			error("RemoveDirectory...");
+		if(!RemoveDirectory(uif->path))
+			oserror();
 	} else {
-		if(!DeleteFile(path))
-			error("DeleteFile...");
+		if(!DeleteFile(uif->path))
+			oserror();
 	}
 }
 
@@ -595,86 +556,69 @@
 static int
 fswstat(Chan *c, uchar *buf, int n)
 {
-	TCHAR path[MAXPATH];
-	char strs[MAXPATH*3];
+	char strs[MAX_PATH*3];
+	Ufsinfo *uif;
 	Dir d;
-	ulong t;
 
 	if (convM2D(buf, n, &d, strs) != n)
 		error(Ebadstat);
-	fspath(c, 0, path, MAXPATH);
-	t = pathtype(path);
-	if(t != TPATH_FILE)
+	uif = c->aux;
+	if(pathtype(uif->path) != TPATH_FILE)
 		error(Ebadstat);
 	/* change name */
 	if(d.name[0]){
+		wchar_t *base, *newpath;
 		int l;
-		TCHAR newpath[MAXPATH];
-		wcsncpy(newpath, path, MAXPATH);
 
+		base = wstrdup(uif->path);
+		if(waserror()){
+			free(base);
+			nexterror();
+		}
 		/* replace last path-element with d.name */
-		l = wcslen(newpath)-1;
+		l = wcslen(base)-1;
 		if(l <= 0)
 			error(Ebadstat);
 		for(;l>0; l--){
-			if(newpath[l-1]=='\\')
+			if(base[l-1]=='\\')
 				break;
 		}
 		if(l <= 0)
 			error(Ebadstat);
-		newpath[l] = 0;
-#ifdef UNICODE
-		if(!MultiByteToWideChar(CP_UTF8,0,d.name,-1,&newpath[l],MAXPATH-l-1))
-			oserror();
-#else
-		wcsncpy(&newpath[l], d.name, MAXPATH-l-1);
-#endif
-		if(wcscmp(path, newpath)!=0){
-			if(!MoveFile(path, newpath))
+		base[l] = 0;
+		newpath = catpath(base, d.name, nil);
+		free(base), base = newpath;
+		if(wcscmp(uif->path, newpath)!=0){
+			if(!MoveFile(uif->path, newpath))
 				oserror();
-			wcsncpy(path, newpath, MAXPATH);
 		}
+		free(uif->path);
+		uif->path = newpath;
+		poperror();
 	}
 
 	/* fixme: change attributes */
-	c->qid = wfdtoqid(path, nil);
+	c->qid = wfdtoqid(uif->path, nil);
 	return n;	
 }
 
-
-static void
-_fspath(Chan *c, char *ext, char *path, int npath)
+static wchar_t*
+catpath(wchar_t *base, char *cext, wchar_t *wext)
 {
-	*path = 0;
-	strncat(path, uc2name(c), npath);
-	if(ext) {
-		if(*path)
-			strncat(path, "/", npath);
-		strncat(path, ext, npath);
-	}
-	cleanname(path);
-	if(*path == '.')
-		*path = 0;
-}
+	wchar_t *path;
+	long n, m;
 
-static void
-fspath(Chan *c, char *ext, TCHAR *path, int npath)
-{
-	TCHAR *p;
-#ifdef UNICODE
-	char buf[MAXPATH];
-	_fspath(c, ext, buf, sizeof(buf));
-	if(!MultiByteToWideChar(CP_UTF8,0,buf,-1,path,npath)){
-		oserror();
-	}
-#else
-	_fspath(c, ext, path, npath);
-#endif
-	/* make a DOS path */
-	for(p=path; *p; p++){
-		if(*p == '/')
-			*p = '\\';
-	}
+	n = wcslen(base);
+	m = wext!=nil ? wcslen(wext) : strlen(cext)*4;
+	path = malloc((n+m+2)*sizeof(wchar_t));
+	memmove(path, base, n*sizeof(wchar_t));
+	if(n > 0 && path[n-1] != '\\') path[n++] = '\\';
+	if(wext != nil)
+		memmove(path+n, wext, m*sizeof(wchar_t));
+	else
+		m = MultiByteToWideChar(CP_UTF8,0,cext,-1,path+n,m);
+	path[n+m] = 0;
+	return path;
 }
 
 static int
@@ -700,17 +644,13 @@
 	long n;
 	Ufsinfo *uif;
 	char de[MAX_PATH*3];
-	int ndirpath;
-	TCHAR dirpath[MAXPATH];
+	wchar_t *p;
 
 	i = 0;
 	uif = c->aux;
 	errno = 0;
 
-	fspath(c, 0, dirpath, MAXPATH);
-	ndirpath = wcslen(dirpath);
-	t = pathtype(dirpath);
-
+	t = pathtype(uif->path);
 	if(uif->offset != offset) {
 		if(offset != 0)
 			error("bad offset in fsdirread");
@@ -722,14 +662,11 @@
 		case TPATH_VOLUME:
 		case TPATH_FILE:
 			FindClose(uif->dir->handle);
-#ifdef UNICODE
-			wcsncat(dirpath, L"\\*.*", MAXPATH);
-#else
-			wcsncat(dirpath, "\\*.*", MAXPATH);
-#endif
-			if((uif->dir->handle = FindFirstFile(dirpath, &uif->dir->wfd))==INVALID_HANDLE_VALUE){
+			p = catpath(uif->path, "*.*", nil);
+			uif->dir->handle = FindFirstFile(p, &uif->dir->wfd);
+			free(p);
+			if(uif->dir->handle == INVALID_HANDLE_VALUE)
 				oserror();
-			}
 			break;
 		}
 		uif->dir->keep = 1;
@@ -736,8 +673,6 @@
 	}
 
 	while(i+BIT16SZ < count) {
-		char owner[MAXPATH];
-
 		if(!uif->dir->keep) {
 			switch(t){
 			case TPATH_ROOT:
@@ -756,41 +691,21 @@
 		}
 		if(t == TPATH_ROOT){
 			uif->dir->drivep[2] = 0;
-#ifdef UNICODE
 			WideCharToMultiByte(CP_UTF8,0,uif->dir->drivep,-1,de,sizeof(de),0,0);
-#else
-			strncpy(de, uif->dir->drivep, sizeof(de));
-#endif
 		} else {
-#ifdef UNICODE
 			WideCharToMultiByte(CP_UTF8,0,uif->dir->wfd.cFileName,-1,de,sizeof(de),0,0);
-#else
-			strncpy(de, uif->dir->wfd.cFileName, sizeof(de));
-#endif
 		}
 		if(de[0]==0 || isdots(de))
 			continue;
 		d.name = de;
-		dirpath[ndirpath] = 0;		
 		if(t == TPATH_ROOT){
-			wcsncat(dirpath, uif->dir->drivep, MAXPATH);
-			wfdtodir(dirpath, &d, nil);
+			p = catpath(uif->path, nil, uif->dir->drivep);
+			wfdtodir(p, &d, nil);
 		} else {
-#ifdef UNICODE
-			wcsncat(dirpath, L"\\", MAXPATH);
-#else
-			wcsncat(dirpath, "\\", MAXPATH);
-#endif
-			wcsncat(dirpath, uif->dir->wfd.cFileName, MAXPATH);
-			wfdtodir(dirpath, &d, &uif->dir->wfd);
+			p = catpath(uif->path, nil, uif->dir->wfd.cFileName);
+			wfdtodir(p, &d, &uif->dir->wfd);
 		}
-
-		if(getfileowner(dirpath, owner, MAXPATH)){
-			d.uid = owner;
-			d.gid = owner;
-			d.muid = owner;
-		}
-
+		free(p);
 		d.type = 'U';
 		d.dev = c->dev;
 		n = convD2M(&d, (uchar*)va+i, count-i);
--- a/kern/devroot.c
+++ b/kern/devroot.c
@@ -27,7 +27,7 @@
 };
 
 static Dirtab rootdir[Nrootfiles] = {
-	"#/",		{Qdir, 0, QTDIR},	0,		DMDIR|0555,
+	"#/",	{Qdir, 0, QTDIR},	0,		DMDIR|0555,
 	"boot",	{Qboot, 0, QTDIR},	0,		DMDIR|0555,
 	"mnt",	{Qmnt, 0, QTDIR},	0,		DMDIR|0555,
 };
--- a/main.c
+++ b/main.c
@@ -55,10 +55,8 @@
 		panic("bind #i: %r");
 	if(bind("#I", "/net", MBEFORE) < 0)
 		panic("bind #I: %r");
-	if(bind("#U", "/", MAFTER) < 0)
+	if(bind("#U", "/root", MREPL) < 0)
 		panic("bind #U: %r");
-	bind("#A", "/dev", MAFTER);
-
 	if(open("/dev/cons", OREAD) != 0)
 		panic("open0: %r");
 	if(open("/dev/cons", OWRITE) != 1)