shithub: riscv

Download patch

ref: f14ad276c25bd9ad1aec6f11d95268f5dd6d7830
parent: a208a937337d40ee1e8df69cdca0b2f073494de0
author: cinap_lenrek <cinap_lenrek@felloff.net>
date: Sat Apr 8 18:59:15 EDT 2017

upas/fs: fix dir comparsion and skip directories in mdir, avoid stat

- ignore directories in dirskip()
- use sortkey of 0 for invalid items, otherwise it could confuse qsort()
- use file size from dirreadall() instead of doing stat
- various cleanups

--- a/sys/src/cmd/upas/fs/mdir.c
+++ b/sys/src/cmd/upas/fs/mdir.c
@@ -14,11 +14,12 @@
 
 	if((fd = open(f, OREAD)) == -1)
 		return -1;
-
-	seek(fd, o, 0);
-	r = readn(fd, b, l) != l;
+	if(seek(fd, o, 0) == o)
+		r = readn(fd, b, l);
+	else
+		r = 0;
 	close(fd);
-	return r? -1: 0;
+	return r != l ? -1: 0;
 }
 
 static void
@@ -67,20 +68,6 @@
 	return 0;
 }
 
-static int
-setsize(Mailbox *mb, Message *m)
-{
-	char buf[Pathlen];
-	Dir *d;
-
-	snprint(buf, sizeof buf, "%s/%D", mb->path, m->fileid);
-	if((d = dirstat(buf)) == nil)
-		return -1;
-	m->size = d->length;
-	free(d);
-	return 0;
-}
-
 /* must be [0-9]+(\..*)? */
 int
 dirskip(Dir *a, uvlong *uv)
@@ -87,12 +74,12 @@
 {
 	char *p;
 
-	if(a->length == 0)
+	if(a->length == 0 || (a->qid.type & QTDIR) != 0)
 		return 1;
 	*uv = strtoul(a->name, &p, 0);
 	if(*uv < 1000000 || *p != '.')
 		return 1;
-	*uv = *uv<<8 | strtoul(p + 1, &p, 10);
+	*uv = *uv<<8 | strtoul(p + 1, &p, 10) & 0xFF;
 	if(*p)
 		return 1;
 	return 0;
@@ -114,8 +101,10 @@
 {
 	uvlong x, y;
 
-	dirskip(a, &x);
-	dirskip(b, &y);
+	if(dirskip(a, &x))
+		x = 0;
+	if(dirskip(b, &y))
+		y = 0;
 	return vcmp(x, y);
 }
 
@@ -128,7 +117,6 @@
 	Message *m, **ll;
 	static char err[ERRMAX];
 
-	mdprint(mdir, "mdirread()\n");
 	if((fd = open(mb->path, OREAD)) == -1){
 		errstr(err, sizeof err);
 		return err;
@@ -162,7 +150,7 @@
 	qsort(d, n, sizeof *d, (int(*)(void*, void*))dircmp);
 	ndel = 0;
 	ll = &mb->root->part;
-	for(i = 0; *ll || i < n; ){
+	for(i = 0; (m = *ll) != nil || i < n; ){
 		if(i < n && dirskip(d + i, &uv)){
 			i++;
 			continue;
@@ -170,48 +158,41 @@
 		c = -1;
 		if(i >= n)
 			c = 1;
-		else if(*ll)
-			c = vcmp(uv, (*ll)->fileid);
-		mdprint(mdir, "consider %s and %D -> %d\n", i<n? d[i].name: 0, *ll? (*ll)->fileid: 1ull, c);
+		else if(m)
+			c = vcmp(uv, m->fileid);
+		mdprint(mdir, "consider %s and %D -> %d\n", i<n? d[i].name: 0, m? m->fileid: 1ull, c);
 		if(c < 0){
 			/* new message */
-			mdprint(mdir, "new: %s (%D)\n", d[i].name, *ll? (*ll)->fileid: 0);
+			mdprint(mdir, "new: %s\n", d[i].name);
+			if(d[i].length > Maxmsg){
+				mdprint(mdir, "skipping bad size: %llud\n", d[i].length);
+				i++;
+				continue;
+			}
+			nnew++;
 			m = newmessage(mb->root);
 			m->fileid = uv;
-			if(setsize(mb, m) < 0 || m->size >= Maxmsg){
-				/* message disappeared?  unchain */
-				mdprint(mdir, "deleted → %r (%D)\n", m->fileid);
-				logmsg(m, "disappeared");
-				if(doplumb)
-					mailplumb(mb, m, 1); /* redundant */
-				unnewmessage(mb, mb->root, m);
-				/* we're out of sync; note this by dropping cached qid */
-				mb->d->qid.path = 0;
-				break;
-			}
+			m->size = d[i].length;
 			m->inmbox = 1;
-			nnew++;
 			m->next = *ll;
 			*ll = m;
 			ll = &m->next;
-			logmsg(m, "new %s", d[i].name);
-			i++;
 			newcachehash(mb, m, doplumb);
 			putcache(mb, m);
+			i++;
 		}else if(c > 0){
 			/* deleted message; */
-			mdprint(mdir, "deleted: %s (%D)\n", i<n? d[i].name: 0, *ll? (*ll)->fileid: 0);
+			mdprint(mdir, "deleted: %s (%D)\n", i<n? d[i].name: 0, m? m->fileid: 0);
 			ndel++;
-			logmsg(*ll, "deleted (refs=%d)", *ll? (*ll)->refs: -42);
 			if(doplumb)
-				mailplumb(mb, *ll, 1);
-			(*ll)->inmbox = 0;
-			(*ll)->deleted = Disappear;
-			ll = &(*ll)->next;
+				mailplumb(mb, m, 1);
+			m->inmbox = 0;
+			m->deleted = Disappear;
+			ll = &m->next;
 		}else{
 			//logmsg(*ll, "duplicate %s", d[i].name);
+			ll = &m->next;
 			i++;
-			ll = &(*ll)->next;
 		}
 	}