2002-09-21 22:10:34 +04:00
|
|
|
/* $NetBSD: cd9660_vfsops.c,v 1.64 2002/09/21 18:12:43 christos Exp $ */
|
1994-06-29 10:29:24 +04:00
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 1994
|
|
|
|
* The Regents of the University of California. All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to Berkeley
|
|
|
|
* by Pace Willisson (pace@blitz.com). The Rock Ridge Extension
|
|
|
|
* Support code is derived from software contributed to Berkeley
|
|
|
|
* by Atsushi Murai (amurai@spec.co.jp).
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the University of
|
|
|
|
* California, Berkeley and its contributors.
|
|
|
|
* 4. Neither the name of the University nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
1998-03-01 05:20:01 +03:00
|
|
|
* @(#)cd9660_vfsops.c 8.18 (Berkeley) 5/22/95
|
1994-06-08 15:22:45 +04:00
|
|
|
*/
|
|
|
|
|
2001-11-12 18:25:01 +03:00
|
|
|
#include <sys/cdefs.h>
|
2002-09-21 22:10:34 +04:00
|
|
|
__KERNEL_RCSID(0, "$NetBSD: cd9660_vfsops.c,v 1.64 2002/09/21 18:12:43 christos Exp $");
|
2001-11-12 18:25:01 +03:00
|
|
|
|
2001-05-30 15:40:35 +04:00
|
|
|
#if defined(_KERNEL_OPT)
|
1998-07-05 12:49:30 +04:00
|
|
|
#include "opt_compat_netbsd.h"
|
|
|
|
#endif
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/namei.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/vnode.h>
|
|
|
|
#include <miscfs/specfs/specdev.h>
|
|
|
|
#include <sys/mount.h>
|
|
|
|
#include <sys/buf.h>
|
|
|
|
#include <sys/file.h>
|
|
|
|
#include <sys/disklabel.h>
|
1997-01-31 05:23:25 +03:00
|
|
|
#include <sys/device.h>
|
1994-06-08 15:22:45 +04:00
|
|
|
#include <sys/ioctl.h>
|
1999-01-04 18:32:08 +03:00
|
|
|
#include <sys/cdio.h>
|
1994-06-08 15:22:45 +04:00
|
|
|
#include <sys/errno.h>
|
|
|
|
#include <sys/malloc.h>
|
1998-09-01 07:40:19 +04:00
|
|
|
#include <sys/pool.h>
|
1994-07-14 02:30:07 +04:00
|
|
|
#include <sys/stat.h>
|
2002-09-06 17:18:43 +04:00
|
|
|
#include <sys/conf.h>
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
#include <isofs/cd9660/iso.h>
|
1997-01-24 03:27:29 +03:00
|
|
|
#include <isofs/cd9660/cd9660_extern.h>
|
1994-12-24 18:30:03 +03:00
|
|
|
#include <isofs/cd9660/iso_rrip.h>
|
1994-06-08 15:22:45 +04:00
|
|
|
#include <isofs/cd9660/cd9660_node.h>
|
1998-03-01 05:20:01 +03:00
|
|
|
#include <isofs/cd9660/cd9660_mount.h>
|
1994-06-08 15:22:45 +04:00
|
|
|
|
2001-01-22 16:18:29 +03:00
|
|
|
extern const struct vnodeopv_desc cd9660_vnodeop_opv_desc;
|
|
|
|
extern const struct vnodeopv_desc cd9660_specop_opv_desc;
|
|
|
|
extern const struct vnodeopv_desc cd9660_fifoop_opv_desc;
|
1998-02-18 10:05:47 +03:00
|
|
|
|
2001-01-22 16:18:29 +03:00
|
|
|
const struct vnodeopv_desc * const cd9660_vnodeopv_descs[] = {
|
1998-02-18 10:05:47 +03:00
|
|
|
&cd9660_vnodeop_opv_desc,
|
|
|
|
&cd9660_specop_opv_desc,
|
|
|
|
&cd9660_fifoop_opv_desc,
|
|
|
|
NULL,
|
|
|
|
};
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
struct vfsops cd9660_vfsops = {
|
|
|
|
MOUNT_CD9660,
|
|
|
|
cd9660_mount,
|
|
|
|
cd9660_start,
|
|
|
|
cd9660_unmount,
|
|
|
|
cd9660_root,
|
|
|
|
cd9660_quotactl,
|
|
|
|
cd9660_statfs,
|
|
|
|
cd9660_sync,
|
|
|
|
cd9660_vget,
|
|
|
|
cd9660_fhtovp,
|
|
|
|
cd9660_vptofh,
|
|
|
|
cd9660_init,
|
2001-09-15 20:12:54 +04:00
|
|
|
cd9660_reinit,
|
2000-03-16 21:08:17 +03:00
|
|
|
cd9660_done,
|
1998-03-01 05:20:01 +03:00
|
|
|
cd9660_sysctl,
|
1997-01-31 05:23:25 +03:00
|
|
|
cd9660_mountroot,
|
1999-02-27 02:44:43 +03:00
|
|
|
cd9660_check_export,
|
1998-02-18 10:05:47 +03:00
|
|
|
cd9660_vnodeopv_descs,
|
1994-06-08 15:22:45 +04:00
|
|
|
};
|
|
|
|
|
a whole bunch of changes to improve performance and robustness under load:
- remove special treatment of pager_map mappings in pmaps. this is
required now, since I've removed the globals that expose the address range.
pager_map now uses pmap_kenter_pa() instead of pmap_enter(), so there's
no longer any need to special-case it.
- eliminate struct uvm_vnode by moving its fields into struct vnode.
- rewrite the pageout path. the pager is now responsible for handling the
high-level requests instead of only getting control after a bunch of work
has already been done on its behalf. this will allow us to UBCify LFS,
which needs tighter control over its pages than other filesystems do.
writing a page to disk no longer requires making it read-only, which
allows us to write wired pages without causing all kinds of havoc.
- use a new PG_PAGEOUT flag to indicate that a page should be freed
on behalf of the pagedaemon when it's unlocked. this flag is very similar
to PG_RELEASED, but unlike PG_RELEASED, PG_PAGEOUT can be cleared if the
pageout fails due to eg. an indirect-block buffer being locked.
this allows us to remove the "version" field from struct vm_page,
and together with shrinking "loan_count" from 32 bits to 16,
struct vm_page is now 4 bytes smaller.
- no longer use PG_RELEASED for swap-backed pages. if the page is busy
because it's being paged out, we can't release the swap slot to be
reallocated until that write is complete, but unlike with vnodes we
don't keep a count of in-progress writes so there's no good way to
know when the write is done. instead, when we need to free a busy
swap-backed page, just sleep until we can get it busy ourselves.
- implement a fast-path for extending writes which allows us to avoid
zeroing new pages. this substantially reduces cpu usage.
- encapsulate the data used by the genfs code in a struct genfs_node,
which must be the first element of the filesystem-specific vnode data
for filesystems which use genfs_{get,put}pages().
- eliminate many of the UVM pagerops, since they aren't needed anymore
now that the pager "put" operation is a higher-level operation.
- enhance the genfs code to allow NFS to use the genfs_{get,put}pages
instead of a modified copy.
- clean up struct vnode by removing all the fields that used to be used by
the vfs_cluster.c code (which we don't use anymore with UBC).
- remove kmem_object and mb_object since they were useless.
instead of allocating pages to these objects, we now just allocate
pages with no object. such pages are mapped in the kernel until they
are freed, so we can use the mapping to find the page to free it.
this allows us to remove splvm() protection in several places.
The sum of all these changes improves write throughput on my
decstation 5000/200 to within 1% of the rate of NetBSD 1.5
and reduces the elapsed time for "make release" of a NetBSD 1.5
source tree on my 128MB pc to 10% less than a 1.5 kernel took.
2001-09-16 00:36:31 +04:00
|
|
|
struct genfs_ops cd9660_genfsops = {
|
|
|
|
genfs_size,
|
|
|
|
};
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
/*
|
|
|
|
* Called by vfs_mountroot when iso is going to be mounted as root.
|
|
|
|
*
|
|
|
|
* Name is updated by mount(8) after booting.
|
|
|
|
*/
|
|
|
|
#define ROOTNAME "root_device"
|
|
|
|
|
2000-05-11 00:35:35 +04:00
|
|
|
static int iso_makemp __P((struct iso_mnt *isomp, struct buf *bp, int *ea_len));
|
1994-12-24 18:30:03 +03:00
|
|
|
static int iso_mountfs __P((struct vnode *devvp, struct mount *mp,
|
|
|
|
struct proc *p, struct iso_args *argp));
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_mountroot()
|
|
|
|
{
|
1998-03-01 05:20:01 +03:00
|
|
|
struct mount *mp;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct proc *p = curproc; /* XXX */
|
|
|
|
int error;
|
|
|
|
struct iso_args args;
|
1997-01-31 05:23:25 +03:00
|
|
|
|
|
|
|
if (root_device->dv_class != DV_DISK)
|
|
|
|
return (ENODEV);
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Get vnodes for swapdev and rootdev.
|
|
|
|
*/
|
1997-06-13 19:38:58 +04:00
|
|
|
if (bdevvp(rootdev, &rootvp))
|
|
|
|
panic("cd9660_mountroot: can't setup rootvp");
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1999-07-17 05:08:28 +04:00
|
|
|
if ((error = vfs_rootmountalloc(MOUNT_CD9660, "root_device", &mp))
|
|
|
|
!= 0) {
|
|
|
|
vrele(rootvp);
|
1998-03-01 05:20:01 +03:00
|
|
|
return (error);
|
1999-07-17 05:08:28 +04:00
|
|
|
}
|
1998-03-01 05:20:01 +03:00
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
args.flags = ISOFSMNT_ROOT;
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = iso_mountfs(rootvp, mp, p, &args)) != 0) {
|
1998-03-01 05:20:01 +03:00
|
|
|
mp->mnt_op->vfs_refcount--;
|
|
|
|
vfs_unbusy(mp);
|
1994-06-08 15:22:45 +04:00
|
|
|
free(mp, M_MOUNT);
|
1999-07-17 05:08:28 +04:00
|
|
|
vrele(rootvp);
|
1994-06-08 15:22:45 +04:00
|
|
|
return (error);
|
|
|
|
}
|
1998-03-01 05:20:01 +03:00
|
|
|
simple_lock(&mountlist_slock);
|
1995-01-18 09:14:00 +03:00
|
|
|
CIRCLEQ_INSERT_TAIL(&mountlist, mp, mnt_list);
|
1998-03-01 05:20:01 +03:00
|
|
|
simple_unlock(&mountlist_slock);
|
1995-01-18 12:23:18 +03:00
|
|
|
(void)cd9660_statfs(mp, &mp->mnt_stat, p);
|
1998-03-01 05:20:01 +03:00
|
|
|
vfs_unbusy(mp);
|
2001-07-07 18:53:55 +04:00
|
|
|
inittodr(0);
|
1994-06-08 15:22:45 +04:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* VFS Operations.
|
|
|
|
*
|
|
|
|
* mount system call
|
|
|
|
*/
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_mount(mp, path, data, ndp, p)
|
1998-03-01 05:20:01 +03:00
|
|
|
struct mount *mp;
|
1996-12-22 13:10:12 +03:00
|
|
|
const char *path;
|
|
|
|
void *data;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct nameidata *ndp;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
|
|
|
struct vnode *devvp;
|
|
|
|
struct iso_args args;
|
1995-03-09 15:05:21 +03:00
|
|
|
size_t size;
|
1994-06-08 15:22:45 +04:00
|
|
|
int error;
|
1996-02-10 00:31:50 +03:00
|
|
|
struct iso_mnt *imp = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
2002-09-21 22:10:34 +04:00
|
|
|
if (mp->mnt_flag & MNT_GETARGS) {
|
|
|
|
imp = VFSTOISOFS(mp);
|
|
|
|
if (imp == NULL)
|
|
|
|
return EIO;
|
|
|
|
args.fspec = NULL;
|
|
|
|
args.flags = imp->im_flags;
|
|
|
|
vfs_showexport(mp, &args.export, &imp->im_export);
|
|
|
|
return copyout(&args, data, sizeof(args));
|
|
|
|
}
|
1996-02-10 00:31:50 +03:00
|
|
|
error = copyin(data, (caddr_t)&args, sizeof (struct iso_args));
|
|
|
|
if (error)
|
1994-06-08 15:22:45 +04:00
|
|
|
return (error);
|
|
|
|
|
|
|
|
if ((mp->mnt_flag & MNT_RDONLY) == 0)
|
|
|
|
return (EROFS);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If updating, check whether changing from read-only to
|
|
|
|
* read/write; if there is no device name, that's all we do.
|
|
|
|
*/
|
|
|
|
if (mp->mnt_flag & MNT_UPDATE) {
|
|
|
|
imp = VFSTOISOFS(mp);
|
|
|
|
if (args.fspec == 0)
|
|
|
|
return (vfs_export(mp, &imp->im_export, &args.export));
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Not an update, or updating the name: look up the name
|
|
|
|
* and verify that it refers to a sensible block device.
|
|
|
|
*/
|
|
|
|
NDINIT(ndp, LOOKUP, FOLLOW, UIO_USERSPACE, args.fspec, p);
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = namei(ndp)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
return (error);
|
|
|
|
devvp = ndp->ni_vp;
|
|
|
|
|
|
|
|
if (devvp->v_type != VBLK) {
|
|
|
|
vrele(devvp);
|
|
|
|
return ENOTBLK;
|
|
|
|
}
|
2002-09-06 17:18:43 +04:00
|
|
|
if (bdevsw_lookup(devvp->v_rdev) == NULL) {
|
1994-06-08 15:22:45 +04:00
|
|
|
vrele(devvp);
|
|
|
|
return ENXIO;
|
|
|
|
}
|
1998-09-05 08:34:47 +04:00
|
|
|
/*
|
|
|
|
* If mount by non-root, then verify that user has necessary
|
|
|
|
* permissions on the device.
|
|
|
|
*/
|
|
|
|
if (p->p_ucred->cr_uid != 0) {
|
|
|
|
vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY);
|
|
|
|
error = VOP_ACCESS(devvp, VREAD, p->p_ucred, p);
|
|
|
|
VOP_UNLOCK(devvp, 0);
|
|
|
|
if (error) {
|
|
|
|
vrele(devvp);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
if ((mp->mnt_flag & MNT_UPDATE) == 0)
|
|
|
|
error = iso_mountfs(devvp, mp, p, &args);
|
|
|
|
else {
|
|
|
|
if (devvp != imp->im_devvp)
|
|
|
|
error = EINVAL; /* needs translation */
|
|
|
|
else
|
|
|
|
vrele(devvp);
|
|
|
|
}
|
|
|
|
if (error) {
|
|
|
|
vrele(devvp);
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
imp = VFSTOISOFS(mp);
|
1995-01-18 12:23:18 +03:00
|
|
|
(void) copyinstr(path, mp->mnt_stat.f_mntonname, MNAMELEN - 1, &size);
|
1998-08-10 00:36:58 +04:00
|
|
|
memset(mp->mnt_stat.f_mntonname + size, 0, MNAMELEN - size);
|
1994-06-08 15:22:45 +04:00
|
|
|
(void) copyinstr(args.fspec, mp->mnt_stat.f_mntfromname, MNAMELEN - 1,
|
|
|
|
&size);
|
1998-08-10 00:36:58 +04:00
|
|
|
memset(mp->mnt_stat.f_mntfromname + size, 0, MNAMELEN - size);
|
1994-06-08 15:22:45 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2000-05-11 00:35:35 +04:00
|
|
|
/*
|
|
|
|
* Make a mount point from a volume descriptor
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
iso_makemp(isomp, bp, ea_len)
|
|
|
|
struct iso_mnt *isomp;
|
|
|
|
struct buf *bp;
|
|
|
|
int *ea_len;
|
|
|
|
{
|
|
|
|
struct iso_primary_descriptor *pri;
|
|
|
|
int logical_block_size;
|
|
|
|
struct iso_directory_record *rootp;
|
|
|
|
|
|
|
|
pri = (struct iso_primary_descriptor *)bp->b_data;
|
|
|
|
|
|
|
|
logical_block_size = isonum_723 (pri->logical_block_size);
|
|
|
|
|
|
|
|
if (logical_block_size < DEV_BSIZE || logical_block_size > MAXBSIZE
|
|
|
|
|| (logical_block_size & (logical_block_size - 1)) != 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
rootp = (struct iso_directory_record *)pri->root_directory_record;
|
|
|
|
|
|
|
|
isomp->logical_block_size = logical_block_size;
|
|
|
|
isomp->volume_space_size = isonum_733 (pri->volume_space_size);
|
|
|
|
memcpy(isomp->root, rootp, sizeof(isomp->root));
|
|
|
|
isomp->root_extent = isonum_733 (rootp->extent);
|
|
|
|
isomp->root_size = isonum_733 (rootp->size);
|
|
|
|
isomp->im_joliet_level = 0;
|
|
|
|
|
|
|
|
isomp->im_bmask = logical_block_size - 1;
|
|
|
|
isomp->im_bshift = 0;
|
|
|
|
while ((1 << isomp->im_bshift) < isomp->logical_block_size)
|
|
|
|
isomp->im_bshift++;
|
|
|
|
|
|
|
|
if (ea_len != NULL)
|
|
|
|
*ea_len = isonum_711(rootp->ext_attr_length);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
/*
|
|
|
|
* Common code for mount and mountroot
|
|
|
|
*/
|
1994-12-24 18:30:03 +03:00
|
|
|
static int
|
|
|
|
iso_mountfs(devvp, mp, p, argp)
|
2000-03-30 16:13:30 +04:00
|
|
|
struct vnode *devvp;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct mount *mp;
|
|
|
|
struct proc *p;
|
|
|
|
struct iso_args *argp;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_mnt *isomp = (struct iso_mnt *)0;
|
1999-07-13 15:12:05 +04:00
|
|
|
struct buf *bp = NULL, *pribp = NULL, *supbp = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
dev_t dev = devvp->v_rdev;
|
1996-02-10 00:31:50 +03:00
|
|
|
int error = EINVAL;
|
1994-06-08 15:22:45 +04:00
|
|
|
int needclose = 0;
|
|
|
|
int ronly = (mp->mnt_flag & MNT_RDONLY) != 0;
|
|
|
|
int iso_bsize;
|
|
|
|
int iso_blknum;
|
1999-07-13 15:12:05 +04:00
|
|
|
int joliet_level;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct iso_volume_descriptor *vdp;
|
1999-07-13 15:12:05 +04:00
|
|
|
struct iso_supplementary_descriptor *sup;
|
1999-01-04 18:32:08 +03:00
|
|
|
int sess = 0;
|
2000-05-11 00:35:35 +04:00
|
|
|
int ext_attr_length;
|
2002-05-27 20:44:37 +04:00
|
|
|
struct disklabel label;
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
if (!ronly)
|
|
|
|
return EROFS;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Disallow multiple mounts of the same device.
|
|
|
|
* Disallow mounting of a device that is currently in use
|
|
|
|
* (except for root, which might share swap device for miniroot).
|
|
|
|
* Flush out any old buffers remaining from a previous use.
|
|
|
|
*/
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = vfs_mountedon(devvp)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
return error;
|
|
|
|
if (vcount(devvp) > 1 && devvp != rootvp)
|
|
|
|
return EBUSY;
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = vinvalbuf(devvp, V_SAVE, p->p_ucred, p, 0, 0)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
return (error);
|
|
|
|
|
1996-02-10 00:31:50 +03:00
|
|
|
error = VOP_OPEN(devvp, ronly ? FREAD : FREAD|FWRITE, FSCRED, p);
|
|
|
|
if (error)
|
1994-06-08 15:22:45 +04:00
|
|
|
return error;
|
|
|
|
needclose = 1;
|
|
|
|
|
|
|
|
/* This is the "logical sector size". The standard says this
|
|
|
|
* should be 2048 or the physical sector size on the device,
|
|
|
|
* whichever is greater. For now, we'll just use a constant.
|
|
|
|
*/
|
|
|
|
iso_bsize = ISO_DEFAULT_BLOCK_SIZE;
|
1999-01-04 18:32:08 +03:00
|
|
|
|
2002-05-27 20:44:37 +04:00
|
|
|
error = VOP_IOCTL(devvp, DIOCGDINFO, (caddr_t)&label, FREAD, FSCRED, p);
|
|
|
|
if (!error &&
|
|
|
|
label.d_partitions[DISKPART(dev)].p_fstype == FS_ISO9660) {
|
|
|
|
/* XXX more sanity checks? */
|
|
|
|
sess = label.d_partitions[DISKPART(dev)].p_cdsession;
|
|
|
|
} else {
|
|
|
|
/* fallback to old method */
|
|
|
|
error = VOP_IOCTL(devvp, CDIOREADMSADDR, (caddr_t)&sess, 0,
|
|
|
|
FSCRED, p);
|
|
|
|
if (error)
|
|
|
|
sess = 0; /* never mind */
|
|
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
|
|
printf("isofs: session offset (part %d) %d\n", DISKPART(dev), sess);
|
1999-01-04 18:32:08 +03:00
|
|
|
#endif
|
2002-05-27 20:44:37 +04:00
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
for (iso_blknum = 16; iso_blknum < 100; iso_blknum++) {
|
1999-01-04 18:32:08 +03:00
|
|
|
if ((error = bread(devvp, (iso_blknum+sess) * btodb(iso_bsize),
|
1996-02-10 00:31:50 +03:00
|
|
|
iso_bsize, NOCRED, &bp)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
goto out;
|
|
|
|
|
1994-06-15 03:54:44 +04:00
|
|
|
vdp = (struct iso_volume_descriptor *)bp->b_data;
|
1998-08-10 00:36:58 +04:00
|
|
|
if (memcmp(vdp->id, ISO_STANDARD_ID, sizeof(vdp->id)) != 0) {
|
1994-06-08 15:22:45 +04:00
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
1999-07-13 15:12:05 +04:00
|
|
|
|
|
|
|
switch (isonum_711(vdp->type)) {
|
|
|
|
case ISO_VD_PRIMARY:
|
|
|
|
if (pribp == NULL) {
|
|
|
|
pribp = bp;
|
|
|
|
bp = NULL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISO_VD_SUPPLEMENTARY:
|
|
|
|
if (supbp == NULL) {
|
|
|
|
supbp = bp;
|
|
|
|
bp = NULL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
1999-07-13 15:12:05 +04:00
|
|
|
|
|
|
|
if (isonum_711 (vdp->type) == ISO_VD_END) {
|
|
|
|
brelse(bp);
|
|
|
|
bp = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
break;
|
1999-07-13 15:12:05 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bp != NULL) {
|
|
|
|
brelse(bp);
|
|
|
|
bp = NULL;
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
1999-07-13 15:12:05 +04:00
|
|
|
|
|
|
|
if (pribp == NULL) {
|
1994-06-08 15:22:45 +04:00
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
1999-07-13 15:12:05 +04:00
|
|
|
|
2000-05-11 00:35:35 +04:00
|
|
|
isomp = malloc(sizeof *isomp, M_ISOFSMNT, M_WAITOK);
|
|
|
|
memset((caddr_t)isomp, 0, sizeof *isomp);
|
|
|
|
if (iso_makemp(isomp, pribp, &ext_attr_length) == -1) {
|
1994-06-08 15:22:45 +04:00
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
2000-05-11 00:35:35 +04:00
|
|
|
|
2002-03-26 22:04:23 +03:00
|
|
|
isomp->volume_space_size += sess;
|
|
|
|
|
1999-07-13 15:12:05 +04:00
|
|
|
pribp->b_flags |= B_AGE;
|
|
|
|
brelse(pribp);
|
|
|
|
pribp = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
2002-07-30 11:40:07 +04:00
|
|
|
mp->mnt_data = isomp;
|
1994-06-08 15:22:45 +04:00
|
|
|
mp->mnt_stat.f_fsid.val[0] = (long)dev;
|
|
|
|
mp->mnt_stat.f_fsid.val[1] = makefstype(MOUNT_CD9660);
|
|
|
|
mp->mnt_maxsymlinklen = 0;
|
|
|
|
mp->mnt_flag |= MNT_LOCAL;
|
2000-11-27 11:39:39 +03:00
|
|
|
mp->mnt_dev_bshift = iso_bsize;
|
|
|
|
mp->mnt_fs_bshift = isomp->im_bshift;
|
1994-06-08 15:22:45 +04:00
|
|
|
isomp->im_mountp = mp;
|
|
|
|
isomp->im_dev = dev;
|
|
|
|
isomp->im_devvp = devvp;
|
|
|
|
|
1999-11-15 21:49:07 +03:00
|
|
|
devvp->v_specmountpoint = mp;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
2001-07-05 12:38:24 +04:00
|
|
|
/* Check the Rock Ridge Extension support */
|
1994-06-08 15:22:45 +04:00
|
|
|
if (!(argp->flags & ISOFSMNT_NORRIP)) {
|
2000-05-11 00:35:35 +04:00
|
|
|
struct iso_directory_record *rootp;
|
|
|
|
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = bread(isomp->im_devvp,
|
2000-05-11 00:35:35 +04:00
|
|
|
(isomp->root_extent + ext_attr_length) <<
|
1996-02-10 00:31:50 +03:00
|
|
|
(isomp->im_bshift - DEV_BSHIFT),
|
|
|
|
isomp->logical_block_size, NOCRED,
|
|
|
|
&bp)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
goto out;
|
|
|
|
|
1994-06-15 03:54:44 +04:00
|
|
|
rootp = (struct iso_directory_record *)bp->b_data;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
if ((isomp->rr_skip = cd9660_rrip_offset(rootp,isomp)) < 0) {
|
|
|
|
argp->flags |= ISOFSMNT_NORRIP;
|
|
|
|
} else {
|
|
|
|
argp->flags &= ~ISOFSMNT_GENS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The contents are valid,
|
|
|
|
* but they will get reread as part of another vnode, so...
|
|
|
|
*/
|
|
|
|
bp->b_flags |= B_AGE;
|
|
|
|
brelse(bp);
|
|
|
|
bp = NULL;
|
|
|
|
}
|
1999-07-13 15:12:05 +04:00
|
|
|
isomp->im_flags = argp->flags & (ISOFSMNT_NORRIP | ISOFSMNT_GENS |
|
2000-07-16 01:40:42 +04:00
|
|
|
ISOFSMNT_EXTATT | ISOFSMNT_NOJOLIET | ISOFSMNT_RRCASEINS);
|
2000-05-27 20:03:55 +04:00
|
|
|
|
|
|
|
if (isomp->im_flags & ISOFSMNT_GENS)
|
|
|
|
isomp->iso_ftype = ISO_FTYPE_9660;
|
|
|
|
else if (isomp->im_flags & ISOFSMNT_NORRIP) {
|
|
|
|
isomp->iso_ftype = ISO_FTYPE_DEFAULT;
|
|
|
|
if (argp->flags & ISOFSMNT_NOCASETRANS)
|
|
|
|
isomp->im_flags |= ISOFSMNT_NOCASETRANS;
|
|
|
|
} else
|
|
|
|
isomp->iso_ftype = ISO_FTYPE_RRIP;
|
2000-05-11 00:35:35 +04:00
|
|
|
|
|
|
|
/* Check the Joliet Extension support */
|
|
|
|
if ((argp->flags & ISOFSMNT_NORRIP) != 0 &&
|
|
|
|
(argp->flags & ISOFSMNT_NOJOLIET) == 0 &&
|
|
|
|
supbp != NULL) {
|
|
|
|
joliet_level = 0;
|
|
|
|
sup = (struct iso_supplementary_descriptor *)supbp->b_data;
|
|
|
|
|
|
|
|
if ((isonum_711(sup->flags) & 1) == 0) {
|
|
|
|
if (memcmp(sup->escape, "%/@", 3) == 0)
|
|
|
|
joliet_level = 1;
|
|
|
|
if (memcmp(sup->escape, "%/C", 3) == 0)
|
|
|
|
joliet_level = 2;
|
|
|
|
if (memcmp(sup->escape, "%/E", 3) == 0)
|
|
|
|
joliet_level = 3;
|
|
|
|
}
|
|
|
|
if (joliet_level != 0) {
|
|
|
|
if (iso_makemp(isomp, supbp, NULL) == -1) {
|
|
|
|
error = EINVAL;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
isomp->im_joliet_level = joliet_level;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (supbp != NULL) {
|
|
|
|
brelse(supbp);
|
|
|
|
supbp = NULL;
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
out:
|
|
|
|
if (bp)
|
|
|
|
brelse(bp);
|
1999-07-13 15:12:05 +04:00
|
|
|
if (pribp)
|
|
|
|
brelse(pribp);
|
|
|
|
if (supbp)
|
|
|
|
brelse(supbp);
|
2000-01-11 12:04:50 +03:00
|
|
|
if (needclose) {
|
|
|
|
vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY);
|
1994-06-08 15:22:45 +04:00
|
|
|
(void)VOP_CLOSE(devvp, ronly ? FREAD : FREAD|FWRITE, NOCRED, p);
|
2000-01-11 12:04:50 +03:00
|
|
|
VOP_UNLOCK(devvp, 0);
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
if (isomp) {
|
|
|
|
free((caddr_t)isomp, M_ISOFSMNT);
|
2002-07-30 11:40:07 +04:00
|
|
|
mp->mnt_data = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make a filesystem operational.
|
|
|
|
* Nothing to do at the moment.
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_start(mp, flags, p)
|
|
|
|
struct mount *mp;
|
|
|
|
int flags;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* unmount system call
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
cd9660_unmount(mp, mntflags, p)
|
|
|
|
struct mount *mp;
|
|
|
|
int mntflags;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_mnt *isomp;
|
1994-12-24 18:30:03 +03:00
|
|
|
int error, flags = 0;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1995-01-18 09:14:00 +03:00
|
|
|
if (mntflags & MNT_FORCE)
|
1994-06-08 15:22:45 +04:00
|
|
|
flags |= FORCECLOSE;
|
|
|
|
#if 0
|
|
|
|
mntflushbuf(mp, 0);
|
|
|
|
if (mntinvalbuf(mp))
|
|
|
|
return EBUSY;
|
|
|
|
#endif
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = vflush(mp, NULLVP, flags)) != 0)
|
1994-06-08 15:22:45 +04:00
|
|
|
return (error);
|
|
|
|
|
|
|
|
isomp = VFSTOISOFS(mp);
|
|
|
|
|
|
|
|
#ifdef ISODEVMAP
|
|
|
|
if (isomp->iso_ftype == ISO_FTYPE_RRIP)
|
|
|
|
iso_dunmap(isomp->im_dev);
|
|
|
|
#endif
|
1999-11-15 21:49:07 +03:00
|
|
|
|
1999-10-20 18:32:09 +04:00
|
|
|
if (isomp->im_devvp->v_type != VBAD)
|
1999-11-15 21:49:07 +03:00
|
|
|
isomp->im_devvp->v_specmountpoint = NULL;
|
|
|
|
|
1999-10-17 03:53:26 +04:00
|
|
|
vn_lock(isomp->im_devvp, LK_EXCLUSIVE | LK_RETRY);
|
1994-06-08 15:22:45 +04:00
|
|
|
error = VOP_CLOSE(isomp->im_devvp, FREAD, NOCRED, p);
|
1999-10-17 03:53:26 +04:00
|
|
|
vput(isomp->im_devvp);
|
1994-06-08 15:22:45 +04:00
|
|
|
free((caddr_t)isomp, M_ISOFSMNT);
|
2002-07-30 11:40:07 +04:00
|
|
|
mp->mnt_data = NULL;
|
1994-06-08 15:22:45 +04:00
|
|
|
mp->mnt_flag &= ~MNT_LOCAL;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return root of a filesystem
|
|
|
|
*/
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_root(mp, vpp)
|
|
|
|
struct mount *mp;
|
|
|
|
struct vnode **vpp;
|
|
|
|
{
|
1994-07-14 02:30:07 +04:00
|
|
|
struct iso_mnt *imp = VFSTOISOFS(mp);
|
1994-07-18 12:07:50 +04:00
|
|
|
struct iso_directory_record *dp =
|
|
|
|
(struct iso_directory_record *)imp->root;
|
1994-07-19 18:14:08 +04:00
|
|
|
ino_t ino = isodirino(dp, imp);
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* With RRIP we must use the `.' entry of the root directory.
|
1994-07-18 12:07:50 +04:00
|
|
|
* Simply tell vget, that it's a relocated directory.
|
1994-06-08 15:22:45 +04:00
|
|
|
*/
|
1994-07-18 12:07:50 +04:00
|
|
|
return (cd9660_vget_internal(mp, ino, vpp,
|
1996-02-10 00:31:50 +03:00
|
|
|
imp->iso_ftype == ISO_FTYPE_RRIP, dp));
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do operations associated with quotas, not supported
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
|
|
|
cd9660_quotactl(mp, cmd, uid, arg, p)
|
|
|
|
struct mount *mp;
|
|
|
|
int cmd;
|
|
|
|
uid_t uid;
|
|
|
|
caddr_t arg;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
|
|
|
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get file system statistics.
|
|
|
|
*/
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_statfs(mp, sbp, p)
|
|
|
|
struct mount *mp;
|
2000-03-30 16:13:30 +04:00
|
|
|
struct statfs *sbp;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct proc *p;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_mnt *isomp;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
isomp = VFSTOISOFS(mp);
|
|
|
|
|
|
|
|
#ifdef COMPAT_09
|
|
|
|
sbp->f_type = 5;
|
|
|
|
#else
|
|
|
|
sbp->f_type = 0;
|
|
|
|
#endif
|
|
|
|
sbp->f_bsize = isomp->logical_block_size;
|
|
|
|
sbp->f_iosize = sbp->f_bsize; /* XXX */
|
|
|
|
sbp->f_blocks = isomp->volume_space_size;
|
|
|
|
sbp->f_bfree = 0; /* total free blocks */
|
|
|
|
sbp->f_bavail = 0; /* blocks free for non superuser */
|
|
|
|
sbp->f_files = 0; /* total files */
|
|
|
|
sbp->f_ffree = 0; /* free file nodes */
|
|
|
|
if (sbp != &mp->mnt_stat) {
|
1998-08-10 00:36:58 +04:00
|
|
|
memcpy(sbp->f_mntonname, mp->mnt_stat.f_mntonname, MNAMELEN);
|
|
|
|
memcpy(sbp->f_mntfromname, mp->mnt_stat.f_mntfromname, MNAMELEN);
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
1995-01-18 12:23:18 +03:00
|
|
|
strncpy(sbp->f_fstypename, mp->mnt_op->vfs_name, MFSNAMELEN);
|
1994-06-08 15:22:45 +04:00
|
|
|
/* Use the first spare for flags: */
|
|
|
|
sbp->f_spare[0] = isomp->im_flags;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
|
|
|
cd9660_sync(mp, waitfor, cred, p)
|
|
|
|
struct mount *mp;
|
|
|
|
int waitfor;
|
|
|
|
struct ucred *cred;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* File handle to vnode
|
|
|
|
*
|
|
|
|
* Have to be really careful about stale file handles:
|
|
|
|
* - check that the inode number is in range
|
|
|
|
* - call iget() to get the locked inode
|
|
|
|
* - check for an unallocated inode (i_mode == 0)
|
|
|
|
* - check that the generation number matches
|
|
|
|
*/
|
|
|
|
|
|
|
|
struct ifid {
|
|
|
|
ushort ifid_len;
|
|
|
|
ushort ifid_pad;
|
|
|
|
int ifid_ino;
|
|
|
|
long ifid_start;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
1999-02-27 02:44:43 +03:00
|
|
|
cd9660_fhtovp(mp, fhp, vpp)
|
2000-03-30 16:13:30 +04:00
|
|
|
struct mount *mp;
|
1994-06-08 15:22:45 +04:00
|
|
|
struct fid *fhp;
|
|
|
|
struct vnode **vpp;
|
|
|
|
{
|
1994-07-14 02:30:07 +04:00
|
|
|
struct ifid *ifhp = (struct ifid *)fhp;
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_node *ip;
|
1994-07-14 02:30:07 +04:00
|
|
|
struct vnode *nvp;
|
|
|
|
int error;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
#ifdef ISOFS_DBG
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: ino %d, start %ld\n",
|
1996-10-11 02:39:39 +04:00
|
|
|
ifhp->ifid_ino, ifhp->ifid_start);
|
1994-06-08 15:22:45 +04:00
|
|
|
#endif
|
|
|
|
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = VFS_VGET(mp, ifhp->ifid_ino, &nvp)) != 0) {
|
1994-07-14 02:30:07 +04:00
|
|
|
*vpp = NULLVP;
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
ip = VTOI(nvp);
|
|
|
|
if (ip->inode.iso_mode == 0) {
|
|
|
|
vput(nvp);
|
|
|
|
*vpp = NULLVP;
|
|
|
|
return (ESTALE);
|
|
|
|
}
|
|
|
|
*vpp = nvp;
|
1999-02-27 02:44:43 +03:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* ARGSUSED */
|
|
|
|
int
|
|
|
|
cd9660_check_export(mp, nam, exflagsp, credanonp)
|
2000-03-30 16:13:30 +04:00
|
|
|
struct mount *mp;
|
1999-02-27 02:44:43 +03:00
|
|
|
struct mbuf *nam;
|
|
|
|
int *exflagsp;
|
|
|
|
struct ucred **credanonp;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct netcred *np;
|
|
|
|
struct iso_mnt *imp = VFSTOISOFS(mp);
|
1999-02-27 02:44:43 +03:00
|
|
|
|
|
|
|
#ifdef ISOFS_DBG
|
|
|
|
printf("check_export: ino %d, start %ld\n",
|
|
|
|
ifhp->ifid_ino, ifhp->ifid_start);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the export permission structure for this <mp, client> tuple.
|
|
|
|
*/
|
|
|
|
np = vfs_export_lookup(mp, &imp->im_export, nam);
|
|
|
|
if (np == NULL)
|
|
|
|
return (EACCES);
|
|
|
|
|
1994-07-14 02:30:07 +04:00
|
|
|
*exflagsp = np->netc_exflags;
|
|
|
|
*credanonp = &np->netc_anon;
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
cd9660_vget(mp, ino, vpp)
|
|
|
|
struct mount *mp;
|
|
|
|
ino_t ino;
|
|
|
|
struct vnode **vpp;
|
|
|
|
{
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXXX
|
|
|
|
* It would be nice if we didn't always set the `relocated' flag
|
|
|
|
* and force the extra read, but I don't want to think about fixing
|
|
|
|
* that right now.
|
|
|
|
*/
|
|
|
|
return (cd9660_vget_internal(mp, ino, vpp,
|
1994-07-18 12:07:50 +04:00
|
|
|
#if 0
|
1996-02-10 00:31:50 +03:00
|
|
|
VFSTOISOFS(mp)->iso_ftype == ISO_FTYPE_RRIP,
|
1994-07-18 12:07:50 +04:00
|
|
|
#else
|
1996-02-10 00:31:50 +03:00
|
|
|
0,
|
1994-07-18 12:07:50 +04:00
|
|
|
#endif
|
1996-02-10 00:31:50 +03:00
|
|
|
NULL));
|
1994-07-14 02:30:07 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
cd9660_vget_internal(mp, ino, vpp, relocated, isodir)
|
|
|
|
struct mount *mp;
|
|
|
|
ino_t ino;
|
|
|
|
struct vnode **vpp;
|
|
|
|
int relocated;
|
|
|
|
struct iso_directory_record *isodir;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_mnt *imp;
|
1994-07-14 02:30:07 +04:00
|
|
|
struct iso_node *ip;
|
2001-09-15 20:12:54 +04:00
|
|
|
#ifdef ISODEVMAP
|
|
|
|
struct iso_dnode *dp;
|
|
|
|
#endif
|
1994-07-14 02:30:07 +04:00
|
|
|
struct buf *bp;
|
|
|
|
struct vnode *vp, *nvp;
|
|
|
|
dev_t dev;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
imp = VFSTOISOFS(mp);
|
|
|
|
dev = imp->im_dev;
|
|
|
|
if ((*vpp = cd9660_ihashget(dev, ino)) != NULLVP)
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
/* Allocate a new vnode/iso_node. */
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = getnewvnode(VT_ISOFS, mp, cd9660_vnodeop_p, &vp)) != 0) {
|
1994-07-14 02:30:07 +04:00
|
|
|
*vpp = NULLVP;
|
|
|
|
return (error);
|
|
|
|
}
|
1998-09-01 07:40:19 +04:00
|
|
|
ip = pool_get(&cd9660_node_pool, PR_WAITOK);
|
1998-08-10 00:36:58 +04:00
|
|
|
memset((caddr_t)ip, 0, sizeof(struct iso_node));
|
1994-07-14 02:30:07 +04:00
|
|
|
vp->v_data = ip;
|
|
|
|
ip->i_vnode = vp;
|
|
|
|
ip->i_dev = dev;
|
|
|
|
ip->i_number = ino;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Put it onto its hash chain and lock it so that other requests for
|
|
|
|
* this inode will block if they arrive while we are sleeping waiting
|
|
|
|
* for old data structures to be purged or for the contents of the
|
|
|
|
* disk portion of this inode to be read.
|
|
|
|
*/
|
|
|
|
cd9660_ihashins(ip);
|
|
|
|
|
1994-07-18 12:07:50 +04:00
|
|
|
if (isodir == 0) {
|
|
|
|
int lbn, off;
|
|
|
|
|
1994-07-19 18:14:08 +04:00
|
|
|
lbn = lblkno(imp, ino);
|
1994-07-18 12:07:50 +04:00
|
|
|
if (lbn >= imp->volume_space_size) {
|
|
|
|
vput(vp);
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: lbn exceed volume space %d\n", lbn);
|
1994-07-18 12:07:50 +04:00
|
|
|
return (ESTALE);
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1994-07-19 18:14:08 +04:00
|
|
|
off = blkoff(imp, ino);
|
1994-07-18 12:07:50 +04:00
|
|
|
if (off + ISO_DIRECTORY_RECORD_SIZE > imp->logical_block_size) {
|
|
|
|
vput(vp);
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: crosses block boundary %d\n",
|
1996-10-11 02:39:39 +04:00
|
|
|
off + ISO_DIRECTORY_RECORD_SIZE);
|
1994-07-18 12:07:50 +04:00
|
|
|
return (ESTALE);
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1994-07-18 12:07:50 +04:00
|
|
|
error = bread(imp->im_devvp,
|
1994-07-19 18:14:08 +04:00
|
|
|
lbn << (imp->im_bshift - DEV_BSHIFT),
|
1994-07-14 02:30:07 +04:00
|
|
|
imp->logical_block_size, NOCRED, &bp);
|
|
|
|
if (error) {
|
|
|
|
vput(vp);
|
|
|
|
brelse(bp);
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: bread error %d\n",error);
|
1994-07-14 02:30:07 +04:00
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
isodir = (struct iso_directory_record *)(bp->b_data + off);
|
|
|
|
|
1994-07-18 12:07:50 +04:00
|
|
|
if (off + isonum_711(isodir->length) >
|
|
|
|
imp->logical_block_size) {
|
|
|
|
vput(vp);
|
|
|
|
if (bp != 0)
|
|
|
|
brelse(bp);
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: directory crosses block boundary %d[off=%d/len=%d]\n",
|
1996-10-11 02:39:39 +04:00
|
|
|
off +isonum_711(isodir->length), off,
|
|
|
|
isonum_711(isodir->length));
|
1994-07-18 12:07:50 +04:00
|
|
|
return (ESTALE);
|
|
|
|
}
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1994-07-14 02:30:07 +04:00
|
|
|
#if 0
|
1994-07-18 12:07:50 +04:00
|
|
|
if (isonum_733(isodir->extent) +
|
|
|
|
isonum_711(isodir->ext_attr_length) != ifhp->ifid_start) {
|
|
|
|
if (bp != 0)
|
|
|
|
brelse(bp);
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("fhtovp: file start miss %d vs %d\n",
|
1996-10-11 02:39:39 +04:00
|
|
|
isonum_733(isodir->extent) + isonum_711(isodir->ext_attr_length),
|
|
|
|
ifhp->ifid_start);
|
1994-07-18 12:07:50 +04:00
|
|
|
return (ESTALE);
|
|
|
|
}
|
1994-07-14 02:30:07 +04:00
|
|
|
#endif
|
1994-07-18 12:07:50 +04:00
|
|
|
} else
|
|
|
|
bp = 0;
|
1994-07-14 02:30:07 +04:00
|
|
|
|
|
|
|
ip->i_mnt = imp;
|
|
|
|
ip->i_devvp = imp->im_devvp;
|
|
|
|
VREF(ip->i_devvp);
|
|
|
|
|
|
|
|
if (relocated) {
|
|
|
|
/*
|
|
|
|
* On relocated directories we must
|
|
|
|
* read the `.' entry out of a dir.
|
|
|
|
*/
|
|
|
|
ip->iso_start = ino >> imp->im_bshift;
|
|
|
|
if (bp != 0)
|
|
|
|
brelse(bp);
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((error = VOP_BLKATOFF(vp, (off_t)0, NULL, &bp)) != 0) {
|
1994-07-14 02:30:07 +04:00
|
|
|
vput(vp);
|
|
|
|
return (error);
|
|
|
|
}
|
|
|
|
isodir = (struct iso_directory_record *)bp->b_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
ip->iso_extent = isonum_733(isodir->extent);
|
|
|
|
ip->i_size = isonum_733(isodir->size);
|
|
|
|
ip->iso_start = isonum_711(isodir->ext_attr_length) + ip->iso_extent;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
1994-07-14 02:30:07 +04:00
|
|
|
/*
|
|
|
|
* Setup time stamp, attribute
|
|
|
|
*/
|
|
|
|
vp->v_type = VNON;
|
|
|
|
switch (imp->iso_ftype) {
|
|
|
|
default: /* ISO_FTYPE_9660 */
|
|
|
|
{
|
|
|
|
struct buf *bp2;
|
1994-07-19 18:14:08 +04:00
|
|
|
int off;
|
1994-07-14 02:30:07 +04:00
|
|
|
if ((imp->im_flags & ISOFSMNT_EXTATT)
|
1994-07-19 18:14:08 +04:00
|
|
|
&& (off = isonum_711(isodir->ext_attr_length)))
|
1994-07-19 19:27:40 +04:00
|
|
|
VOP_BLKATOFF(vp, (off_t)-(off << imp->im_bshift), NULL,
|
1994-07-19 19:07:36 +04:00
|
|
|
&bp2);
|
1994-07-19 18:14:08 +04:00
|
|
|
else
|
|
|
|
bp2 = NULL;
|
1994-07-14 02:30:07 +04:00
|
|
|
cd9660_defattr(isodir, ip, bp2);
|
|
|
|
cd9660_deftstamp(isodir, ip, bp2);
|
1994-07-19 18:14:08 +04:00
|
|
|
if (bp2)
|
|
|
|
brelse(bp2);
|
1994-07-14 02:30:07 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISO_FTYPE_RRIP:
|
|
|
|
cd9660_rrip_analyze(isodir, ip, imp);
|
|
|
|
break;
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
1994-07-14 02:30:07 +04:00
|
|
|
|
|
|
|
if (bp != 0)
|
|
|
|
brelse(bp);
|
|
|
|
|
1994-06-08 15:22:45 +04:00
|
|
|
/*
|
1994-07-14 02:30:07 +04:00
|
|
|
* Initialize the associated vnode
|
1994-06-08 15:22:45 +04:00
|
|
|
*/
|
1994-07-14 02:30:07 +04:00
|
|
|
switch (vp->v_type = IFTOVT(ip->inode.iso_mode)) {
|
|
|
|
case VFIFO:
|
|
|
|
vp->v_op = cd9660_fifoop_p;
|
|
|
|
break;
|
|
|
|
case VCHR:
|
|
|
|
case VBLK:
|
|
|
|
/*
|
|
|
|
* if device, look at device number table for translation
|
|
|
|
*/
|
|
|
|
#ifdef ISODEVMAP
|
2001-09-15 20:12:54 +04:00
|
|
|
if ((dp = iso_dmap(dev, ino, 0)) != NULL)
|
1994-07-14 02:30:07 +04:00
|
|
|
ip->inode.iso_rdev = dp->d_dev;
|
|
|
|
#endif
|
|
|
|
vp->v_op = cd9660_specop_p;
|
1996-02-10 00:31:50 +03:00
|
|
|
if ((nvp = checkalias(vp, ip->inode.iso_rdev, mp)) != NULL) {
|
1994-07-14 02:30:07 +04:00
|
|
|
/*
|
|
|
|
* Discard unneeded vnode, but save its iso_node.
|
1998-03-01 05:20:01 +03:00
|
|
|
* Note that the lock is carried over in the iso_node
|
|
|
|
* to the replacement vnode.
|
1994-07-14 02:30:07 +04:00
|
|
|
*/
|
|
|
|
nvp->v_data = vp->v_data;
|
|
|
|
vp->v_data = NULL;
|
2001-03-24 00:10:48 +03:00
|
|
|
VOP_UNLOCK(vp, 0);
|
1994-07-14 02:30:07 +04:00
|
|
|
vp->v_op = spec_vnodeop_p;
|
2001-03-24 00:10:48 +03:00
|
|
|
vrele(vp);
|
1994-07-14 02:30:07 +04:00
|
|
|
vgone(vp);
|
1999-07-08 05:05:58 +04:00
|
|
|
lockmgr(&nvp->v_lock, LK_EXCLUSIVE, &nvp->v_interlock);
|
1994-07-14 02:30:07 +04:00
|
|
|
/*
|
|
|
|
* Reinitialize aliased inode.
|
|
|
|
*/
|
|
|
|
vp = nvp;
|
|
|
|
ip->i_vnode = vp;
|
|
|
|
}
|
|
|
|
break;
|
1996-02-10 00:31:50 +03:00
|
|
|
case VLNK:
|
|
|
|
case VNON:
|
|
|
|
case VSOCK:
|
|
|
|
case VDIR:
|
|
|
|
case VBAD:
|
2000-12-21 06:42:43 +03:00
|
|
|
break;
|
1996-02-10 00:31:50 +03:00
|
|
|
case VREG:
|
2000-12-21 06:42:43 +03:00
|
|
|
uvm_vnp_setsize(vp, ip->i_size);
|
1996-02-10 00:31:50 +03:00
|
|
|
break;
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
1994-07-14 02:30:07 +04:00
|
|
|
|
|
|
|
if (ip->iso_extent == imp->root_extent)
|
|
|
|
vp->v_flag |= VROOT;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX need generation number?
|
|
|
|
*/
|
|
|
|
|
a whole bunch of changes to improve performance and robustness under load:
- remove special treatment of pager_map mappings in pmaps. this is
required now, since I've removed the globals that expose the address range.
pager_map now uses pmap_kenter_pa() instead of pmap_enter(), so there's
no longer any need to special-case it.
- eliminate struct uvm_vnode by moving its fields into struct vnode.
- rewrite the pageout path. the pager is now responsible for handling the
high-level requests instead of only getting control after a bunch of work
has already been done on its behalf. this will allow us to UBCify LFS,
which needs tighter control over its pages than other filesystems do.
writing a page to disk no longer requires making it read-only, which
allows us to write wired pages without causing all kinds of havoc.
- use a new PG_PAGEOUT flag to indicate that a page should be freed
on behalf of the pagedaemon when it's unlocked. this flag is very similar
to PG_RELEASED, but unlike PG_RELEASED, PG_PAGEOUT can be cleared if the
pageout fails due to eg. an indirect-block buffer being locked.
this allows us to remove the "version" field from struct vm_page,
and together with shrinking "loan_count" from 32 bits to 16,
struct vm_page is now 4 bytes smaller.
- no longer use PG_RELEASED for swap-backed pages. if the page is busy
because it's being paged out, we can't release the swap slot to be
reallocated until that write is complete, but unlike with vnodes we
don't keep a count of in-progress writes so there's no good way to
know when the write is done. instead, when we need to free a busy
swap-backed page, just sleep until we can get it busy ourselves.
- implement a fast-path for extending writes which allows us to avoid
zeroing new pages. this substantially reduces cpu usage.
- encapsulate the data used by the genfs code in a struct genfs_node,
which must be the first element of the filesystem-specific vnode data
for filesystems which use genfs_{get,put}pages().
- eliminate many of the UVM pagerops, since they aren't needed anymore
now that the pager "put" operation is a higher-level operation.
- enhance the genfs code to allow NFS to use the genfs_{get,put}pages
instead of a modified copy.
- clean up struct vnode by removing all the fields that used to be used by
the vfs_cluster.c code (which we don't use anymore with UBC).
- remove kmem_object and mb_object since they were useless.
instead of allocating pages to these objects, we now just allocate
pages with no object. such pages are mapped in the kernel until they
are freed, so we can use the mapping to find the page to free it.
this allows us to remove splvm() protection in several places.
The sum of all these changes improves write throughput on my
decstation 5000/200 to within 1% of the rate of NetBSD 1.5
and reduces the elapsed time for "make release" of a NetBSD 1.5
source tree on my 128MB pc to 10% less than a 1.5 kernel took.
2001-09-16 00:36:31 +04:00
|
|
|
genfs_node_init(vp, &cd9660_genfsops);
|
1994-07-14 02:30:07 +04:00
|
|
|
*vpp = vp;
|
|
|
|
return (0);
|
1994-06-08 15:22:45 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Vnode pointer to File handle
|
|
|
|
*/
|
|
|
|
/* ARGSUSED */
|
1994-12-24 18:30:03 +03:00
|
|
|
int
|
1994-06-08 15:22:45 +04:00
|
|
|
cd9660_vptofh(vp, fhp)
|
|
|
|
struct vnode *vp;
|
|
|
|
struct fid *fhp;
|
|
|
|
{
|
2000-03-30 16:13:30 +04:00
|
|
|
struct iso_node *ip = VTOI(vp);
|
|
|
|
struct ifid *ifhp;
|
1994-06-08 15:22:45 +04:00
|
|
|
|
|
|
|
ifhp = (struct ifid *)fhp;
|
|
|
|
ifhp->ifid_len = sizeof(struct ifid);
|
|
|
|
|
|
|
|
ifhp->ifid_ino = ip->i_number;
|
|
|
|
ifhp->ifid_start = ip->iso_start;
|
|
|
|
|
|
|
|
#ifdef ISOFS_DBG
|
1996-10-13 06:28:58 +04:00
|
|
|
printf("vptofh: ino %d, start %ld\n",
|
1996-10-11 02:39:39 +04:00
|
|
|
ifhp->ifid_ino,ifhp->ifid_start);
|
1994-06-08 15:22:45 +04:00
|
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
}
|
1998-03-01 05:20:01 +03:00
|
|
|
|
|
|
|
int
|
|
|
|
cd9660_sysctl(name, namelen, oldp, oldlenp, newp, newlen, p)
|
|
|
|
int *name;
|
|
|
|
u_int namelen;
|
|
|
|
void *oldp;
|
|
|
|
size_t *oldlenp;
|
|
|
|
void *newp;
|
|
|
|
size_t newlen;
|
|
|
|
struct proc *p;
|
|
|
|
{
|
|
|
|
return (EOPNOTSUPP);
|
|
|
|
}
|