shithub: riscv

Download patch

ref: 8b57e59ea93b7fc4f80c33bc1560b685f392ccc4
parent: 21570a47195d90b1dc2a3634d8042929543599d3
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Wed Feb 11 18:54:28 EST 2015

libc: check name in getvent()/putenv()

passing "", "." or ".." as name caused a crash in
getenv() as it would open the directory; then seek()
to determine the file size would fail and return -1.

now checking for these special names and set
error string when its bad.

doing a single read() will not work when /env has a
9p fileserver mounted onto it and the file size is bigger
than the i/o unit. so doing incremental reads until
we get eof.

--- a/sys/src/libc/9sys/getenv.c
+++ b/sys/src/libc/9sys/getenv.c
@@ -4,33 +4,42 @@
 char*
 getenv(char *name)
 {
-	int r, f;
-	long s;
-	char *ans;
-	char *p, *ep, ename[100];
+	enum { HUNK = 100, };
+	char *s, *p;
+	int f, r, n;
 
-	if(strchr(name, '/') != nil)
+	if(name[0]=='\0' || strcmp(name, ".")==0 || strcmp(name, "..")==0 || strchr(name, '/')!=nil
+	|| strlen(name) >= HUNK-5){
+		werrstr("bad env name: %s", name);
 		return nil;
-	snprint(ename, sizeof ename, "/env/%s", name);
-	if(strcmp(ename+5, name) != 0)
+	}
+	if((s = malloc(HUNK)) == nil)
 		return nil;
-	f = open(ename, OREAD);
-	if(f < 0)
-		return 0;
-	s = seek(f, 0, 2);
-	ans = malloc(s+1);
-	if(ans) {
-		setmalloctag(ans, getcallerpc(&name));
-		seek(f, 0, 0);
-		r = read(f, ans, s);
-		if(r >= 0) {
-			ep = ans + s - 1;
-			for(p = ans; p < ep; p++)
-				if(*p == '\0')
-					*p = ' ';
-			ans[s] = '\0';
+	snprint(s, HUNK, "/env/%s", name);
+	n = 0;
+	r = -1;
+	if((f = open(s, OREAD)) >= 0){
+		while((r = read(f, s+n, HUNK)) > 0){
+			n += r;
+			r = -1;
+			if((p = realloc(s, n+HUNK)) == nil)
+				break;
+			s = p;
 		}
+		close(f);
 	}
-	close(f);
-	return ans;
+	if(r < 0 || (p = realloc(s, n+1)) == nil){
+		free(s);
+		return nil;
+	}
+	s = p;
+	setmalloctag(s, getcallerpc(&name));
+	while(n > 0 && s[n-1] == '\0')
+		n--;
+	s[n] = '\0';
+	while(--n >= 0){
+		if(s[n] == '\0')
+			s[n] = ' ';
+	}
+	return s;
 }
--- a/sys/src/libc/9sys/putenv.c
+++ b/sys/src/libc/9sys/putenv.c
@@ -4,20 +4,20 @@
 int
 putenv(char *name, char *val)
 {
-	int f;
 	char ename[100];
-	long s;
+	int f, n;
 
-	if(strchr(name, '/') != nil)
+	if(name[0]=='\0' || strcmp(name, ".")==0 || strcmp(name, "..")==0 || strchr(name, '/')!=nil
+	|| strlen(name) >= sizeof(ename)-5){
+		werrstr("bad env name: %s", name);
 		return -1;
-	snprint(ename, sizeof ename, "/env/%s", name);
-	if(strcmp(ename+5, name) != 0)
-		return -1;
+	}
+	snprint(ename, sizeof(ename), "/env/%s", name);
 	f = create(ename, OWRITE, 0664);
 	if(f < 0)
 		return -1;
-	s = strlen(val);
-	if(write(f, val, s) != s){
+	n = strlen(val);
+	if(write(f, val, n) != n){
 		close(f);
 		return -1;
 	}