"busy buffer problem" panics

Matthew Dillon dillon at apollo.backplane.com
Thu Sep 2 21:59:19 PDT 2004


:
:This box is both an nfs client (about 20 static mount points) and a server
:(about 5 clients, several mount points), and I'm running amd so it's
:automounting on /net a lot, especially during many clearcase operations.=20
:
:Not sure if 'cleartool lsact -l' does anything with /net but it is possible.
:
:I'm thinking that locking in the linuxulator is at the root of this ...
:
:Andrew.

    Try this patch with the latest kernel.  The failure case is odd, because
    it looks like the underlying lock is somehow entering the system
    shutdown code.  The code in linux_util appears to have some severe
    leaks: a vnode reference count leak and a memory leak, which thi
    s patch fixes.

    Does running clearcase crash it immediately or does it take a number of
    runs?  For the reference count to be causing the problem, the vnode
    would have to cycle through 2-4 billion accesses (to wrap the ref count).
    This is certainly possible, but it would take a while.

					-Matt
					Matthew Dillon 
					<dillon at xxxxxxxxxxxxx>

Index: linux_util.c
===================================================================
RCS file: /cvs/src/sys/emulation/linux/linux_util.c,v
retrieving revision 1.8
diff -u -r1.8 linux_util.c
--- linux_util.c	13 Nov 2003 04:04:42 -0000	1.8
+++ linux_util.c	3 Sep 2004 04:49:27 -0000
@@ -74,10 +74,8 @@
 	length = strlen(linux_emul_path);
 	bcopy(linux_emul_path, buf, length);
 	error = copyinstr(uname, buf + length, MAXPATHLEN - length, &dummy);
-	if (error) {
-		linux_free_path(kname);
-		return (error);
-	}
+	if (error)
+		goto done;
 
 	switch (flags) {
 	case LINUX_PATH_CREATE:
@@ -87,14 +85,20 @@
 		 * the last '/'.
 		 */
 		cp = buf + strlen(buf);
-		while (*--cp != '/');
-		*cp = '\0';
+		while (--cp >= buf) {
+			if (*cp == '/')
+				break;
+		}
+		if (cp < buf)
+			goto dont_translate;
+		*cp = 0;
 
 		NDINIT(&nd, NAMEI_LOOKUP, CNP_FOLLOW, UIO_SYSSPACE, buf, td);
 		error = namei(&nd);
 		if (error)
 			goto dont_translate;
-
+		NDFREE(&nd, NDF_ONLY_PNBUF);
+		vrele(nd.ni_vp);
 		*cp = '/';
 		return (0);
 	case LINUX_PATH_EXISTS:
@@ -119,30 +123,36 @@
 		NDINIT(&ndroot, NAMEI_LOOKUP, CNP_FOLLOW, UIO_SYSSPACE,
 		    linux_emul_path, td);
 		error = namei(&ndroot);
-		if (error)
+		if (error) {
+			NDFREE(&nd, NDF_ONLY_PNBUF);
+			vrele(nd.ni_vp);
 			goto dont_translate;
+		}
 		
 		error = VOP_GETATTR(nd.ni_vp, &vat, td);
+		if (error == 0) {
+			error = VOP_GETATTR(ndroot.ni_vp, &vatroot, td);
+			if (error == 0) {
+				if (vat.va_fsid == vatroot.va_fsid &&
+				    vat.va_fileid == vatroot.va_fileid)
+					error = ENOENT;
+			}
+		}
+		NDFREE(&nd, NDF_ONLY_PNBUF);
+		vrele(nd.ni_vp);
+		NDFREE(&ndroot, NDF_ONLY_PNBUF);
+		vrele(ndroot.ni_vp);
 		if (error)
 			goto dont_translate;
-
-		error = VOP_GETATTR(ndroot.ni_vp, &vatroot, td);
-		if (error)
-			goto dont_translate;
-
-		if (vat.va_fsid == vatroot.va_fsid &&
-		    vat.va_fileid == vatroot.va_fileid)
-			goto dont_translate;
-
 		return (0);
 	default:
-		linux_free_path(kname);
-		return (EINVAL);
+		error = EINVAL;
+		goto done;
 	}
 	
 dont_translate:
-
 	error = copyinstr(uname, buf, MAXPATHLEN, &dummy);
+done:
 	if (error)
 		linux_free_path(kname);
 	return (error);
@@ -176,6 +186,8 @@
 		error = 0;
 		goto cleanup;
 	}
+	NDFREE(&nd, NDF_ONLY_PNBUF);
+	vrele(nd.ni_vp);
 
 	/*
 	 * The alternate path does exist.  Return it in the buffer if





More information about the Bugs mailing list