summaryrefslogtreecommitdiffstats
path: root/sys
diff options
context:
space:
mode:
authortjr <tjr@FreeBSD.org>2003-03-06 10:48:03 +0000
committertjr <tjr@FreeBSD.org>2003-03-06 10:48:03 +0000
commit82ed5e94ee42037d9b8b97e5608fe7c2875a2282 (patch)
tree5297ccaa6c8d80a6d0af2d2ab6d1a2477fafb3f5 /sys
parent96122ae0b747631268476b1c6efc61e1eea4e8b4 (diff)
downloadFreeBSD-src-82ed5e94ee42037d9b8b97e5608fe7c2875a2282.zip
FreeBSD-src-82ed5e94ee42037d9b8b97e5608fe7c2875a2282.tar.gz
Add a temporary workaround for a deadlock in Coda venus 5.3.19 that
occurs when mounting the filesystem. The problem is that venus issues the mount() syscall, which calls vfs_mount(), which calls coda_root() which attempts to communicate with venus.
Diffstat (limited to 'sys')
-rw-r--r--sys/coda/cnode.h1
-rw-r--r--sys/coda/coda_vfsops.c29
-rw-r--r--sys/fs/coda/cnode.h1
-rw-r--r--sys/fs/coda/coda_vfsops.c29
4 files changed, 56 insertions, 4 deletions
diff --git a/sys/coda/cnode.h b/sys/coda/cnode.h
index 962d329..450adc3 100644
--- a/sys/coda/cnode.h
+++ b/sys/coda/cnode.h
@@ -154,6 +154,7 @@ struct coda_mntinfo {
struct mount *mi_vfsp;
struct vcomm mi_vcomm;
dev_t dev;
+ int mi_started;
};
extern struct coda_mntinfo coda_mnttbl[]; /* indexed by minor device number */
diff --git a/sys/coda/coda_vfsops.c b/sys/coda/coda_vfsops.c
index 5040e68..54ca8bd 100644
--- a/sys/coda/coda_vfsops.c
+++ b/sys/coda/coda_vfsops.c
@@ -181,6 +181,7 @@ coda_mount(vfsp, path, data, ndp, td)
vfs_getnewfsid (vfsp);
mi->mi_vfsp = vfsp;
+ mi->mi_started = 0; /* XXX See coda_root() */
/*
* Make a root vnode to placate the Vnode interface, but don't
@@ -301,9 +302,21 @@ coda_root(vfsp, vpp)
result = NULL;
if (vfsp == mi->mi_vfsp) {
+ /*
+ * Cache the root across calls. We only need to pass the request
+ * on to Venus if the root vnode is the dummy we installed in
+ * coda_mount() with all c_fid members zeroed.
+ *
+ * XXX In addition, if we are called between coda_mount() and
+ * coda_start(), we assume that the request is from vfs_mount()
+ * (before the call to checkdirs()) and return the dummy root
+ * node to avoid a deadlock. This bug is fixed in the Coda CVS
+ * repository but not in any released versions as of 6 Mar 2003.
+ */
if ((VTOC(mi->mi_rootvp)->c_fid.Volume != 0) ||
(VTOC(mi->mi_rootvp)->c_fid.Vnode != 0) ||
- (VTOC(mi->mi_rootvp)->c_fid.Unique != 0))
+ (VTOC(mi->mi_rootvp)->c_fid.Unique != 0) ||
+ mi->mi_started == 0)
{ /* Found valid root. */
*vpp = mi->mi_rootvp;
/* On Mach, this is vref. On NetBSD, VOP_LOCK */
@@ -371,6 +384,18 @@ coda_root(vfsp, vpp)
return(error);
}
+int
+coda_start(mp, flags, td)
+ struct mount *mp;
+ int flags;
+ struct thread *td;
+{
+
+ /* XXX See coda_root(). */
+ vftomi(mp)->mi_started = 1;
+ return (0);
+}
+
/*
* Get filesystem statistics.
*/
@@ -526,7 +551,7 @@ struct mount *devtomp(dev)
struct vfsops coda_vfsops = {
coda_mount,
- vfs_stdstart,
+ coda_start,
coda_unmount,
coda_root,
vfs_stdquotactl,
diff --git a/sys/fs/coda/cnode.h b/sys/fs/coda/cnode.h
index 962d329..450adc3 100644
--- a/sys/fs/coda/cnode.h
+++ b/sys/fs/coda/cnode.h
@@ -154,6 +154,7 @@ struct coda_mntinfo {
struct mount *mi_vfsp;
struct vcomm mi_vcomm;
dev_t dev;
+ int mi_started;
};
extern struct coda_mntinfo coda_mnttbl[]; /* indexed by minor device number */
diff --git a/sys/fs/coda/coda_vfsops.c b/sys/fs/coda/coda_vfsops.c
index 5040e68..54ca8bd 100644
--- a/sys/fs/coda/coda_vfsops.c
+++ b/sys/fs/coda/coda_vfsops.c
@@ -181,6 +181,7 @@ coda_mount(vfsp, path, data, ndp, td)
vfs_getnewfsid (vfsp);
mi->mi_vfsp = vfsp;
+ mi->mi_started = 0; /* XXX See coda_root() */
/*
* Make a root vnode to placate the Vnode interface, but don't
@@ -301,9 +302,21 @@ coda_root(vfsp, vpp)
result = NULL;
if (vfsp == mi->mi_vfsp) {
+ /*
+ * Cache the root across calls. We only need to pass the request
+ * on to Venus if the root vnode is the dummy we installed in
+ * coda_mount() with all c_fid members zeroed.
+ *
+ * XXX In addition, if we are called between coda_mount() and
+ * coda_start(), we assume that the request is from vfs_mount()
+ * (before the call to checkdirs()) and return the dummy root
+ * node to avoid a deadlock. This bug is fixed in the Coda CVS
+ * repository but not in any released versions as of 6 Mar 2003.
+ */
if ((VTOC(mi->mi_rootvp)->c_fid.Volume != 0) ||
(VTOC(mi->mi_rootvp)->c_fid.Vnode != 0) ||
- (VTOC(mi->mi_rootvp)->c_fid.Unique != 0))
+ (VTOC(mi->mi_rootvp)->c_fid.Unique != 0) ||
+ mi->mi_started == 0)
{ /* Found valid root. */
*vpp = mi->mi_rootvp;
/* On Mach, this is vref. On NetBSD, VOP_LOCK */
@@ -371,6 +384,18 @@ coda_root(vfsp, vpp)
return(error);
}
+int
+coda_start(mp, flags, td)
+ struct mount *mp;
+ int flags;
+ struct thread *td;
+{
+
+ /* XXX See coda_root(). */
+ vftomi(mp)->mi_started = 1;
+ return (0);
+}
+
/*
* Get filesystem statistics.
*/
@@ -526,7 +551,7 @@ struct mount *devtomp(dev)
struct vfsops coda_vfsops = {
coda_mount,
- vfs_stdstart,
+ coda_start,
coda_unmount,
coda_root,
vfs_stdquotactl,
OpenPOWER on IntegriCloud