1 1.105 joe /* $NetBSD: cd9660_vfsops.c,v 1.105 2025/02/16 18:38:58 joe Exp $ */ 2 1.1 jdolecek 3 1.1 jdolecek /*- 4 1.1 jdolecek * Copyright (c) 1994 5 1.1 jdolecek * The Regents of the University of California. All rights reserved. 6 1.1 jdolecek * 7 1.1 jdolecek * This code is derived from software contributed to Berkeley 8 1.1 jdolecek * by Pace Willisson (pace (at) blitz.com). The Rock Ridge Extension 9 1.1 jdolecek * Support code is derived from software contributed to Berkeley 10 1.1 jdolecek * by Atsushi Murai (amurai (at) spec.co.jp). 11 1.1 jdolecek * 12 1.1 jdolecek * Redistribution and use in source and binary forms, with or without 13 1.1 jdolecek * modification, are permitted provided that the following conditions 14 1.1 jdolecek * are met: 15 1.1 jdolecek * 1. Redistributions of source code must retain the above copyright 16 1.1 jdolecek * notice, this list of conditions and the following disclaimer. 17 1.1 jdolecek * 2. Redistributions in binary form must reproduce the above copyright 18 1.1 jdolecek * notice, this list of conditions and the following disclaimer in the 19 1.1 jdolecek * documentation and/or other materials provided with the distribution. 20 1.9 agc * 3. Neither the name of the University nor the names of its contributors 21 1.1 jdolecek * may be used to endorse or promote products derived from this software 22 1.1 jdolecek * without specific prior written permission. 23 1.1 jdolecek * 24 1.1 jdolecek * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 25 1.1 jdolecek * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 26 1.1 jdolecek * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 27 1.1 jdolecek * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 28 1.1 jdolecek * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 29 1.1 jdolecek * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 30 1.1 jdolecek * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 31 1.1 jdolecek * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 32 1.1 jdolecek * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 33 1.1 jdolecek * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 34 1.1 jdolecek * SUCH DAMAGE. 35 1.1 jdolecek * 36 1.1 jdolecek * @(#)cd9660_vfsops.c 8.18 (Berkeley) 5/22/95 37 1.1 jdolecek */ 38 1.1 jdolecek 39 1.1 jdolecek #include <sys/cdefs.h> 40 1.105 joe __KERNEL_RCSID(0, "$NetBSD: cd9660_vfsops.c,v 1.105 2025/02/16 18:38:58 joe Exp $"); 41 1.1 jdolecek 42 1.1 jdolecek #if defined(_KERNEL_OPT) 43 1.1 jdolecek #include "opt_compat_netbsd.h" 44 1.1 jdolecek #endif 45 1.1 jdolecek 46 1.1 jdolecek #include <sys/param.h> 47 1.10 atatat #include <sys/sysctl.h> 48 1.1 jdolecek #include <sys/systm.h> 49 1.1 jdolecek #include <sys/namei.h> 50 1.1 jdolecek #include <sys/proc.h> 51 1.1 jdolecek #include <sys/kernel.h> 52 1.1 jdolecek #include <sys/vnode.h> 53 1.55 dholland #include <miscfs/genfs/genfs.h> 54 1.1 jdolecek #include <miscfs/specfs/specdev.h> 55 1.1 jdolecek #include <sys/mount.h> 56 1.1 jdolecek #include <sys/buf.h> 57 1.1 jdolecek #include <sys/file.h> 58 1.1 jdolecek #include <sys/disklabel.h> 59 1.1 jdolecek #include <sys/device.h> 60 1.1 jdolecek #include <sys/ioctl.h> 61 1.1 jdolecek #include <sys/cdio.h> 62 1.1 jdolecek #include <sys/errno.h> 63 1.1 jdolecek #include <sys/malloc.h> 64 1.1 jdolecek #include <sys/pool.h> 65 1.1 jdolecek #include <sys/stat.h> 66 1.1 jdolecek #include <sys/conf.h> 67 1.18 jdolecek #include <sys/dirent.h> 68 1.32 elad #include <sys/kauth.h> 69 1.60 ad #include <sys/module.h> 70 1.1 jdolecek 71 1.1 jdolecek #include <fs/cd9660/iso.h> 72 1.1 jdolecek #include <fs/cd9660/cd9660_extern.h> 73 1.1 jdolecek #include <fs/cd9660/iso_rrip.h> 74 1.1 jdolecek #include <fs/cd9660/cd9660_node.h> 75 1.1 jdolecek #include <fs/cd9660/cd9660_mount.h> 76 1.2 thorpej 77 1.60 ad MODULE(MODULE_CLASS_VFS, cd9660, NULL); 78 1.60 ad 79 1.41 pooka MALLOC_JUSTDEFINE(M_ISOFSMNT, "ISOFS mount", "ISOFS mount structure"); 80 1.1 jdolecek 81 1.1 jdolecek extern const struct vnodeopv_desc cd9660_vnodeop_opv_desc; 82 1.1 jdolecek extern const struct vnodeopv_desc cd9660_specop_opv_desc; 83 1.1 jdolecek extern const struct vnodeopv_desc cd9660_fifoop_opv_desc; 84 1.1 jdolecek 85 1.1 jdolecek const struct vnodeopv_desc * const cd9660_vnodeopv_descs[] = { 86 1.1 jdolecek &cd9660_vnodeop_opv_desc, 87 1.1 jdolecek &cd9660_specop_opv_desc, 88 1.1 jdolecek &cd9660_fifoop_opv_desc, 89 1.1 jdolecek NULL, 90 1.1 jdolecek }; 91 1.1 jdolecek 92 1.1 jdolecek struct vfsops cd9660_vfsops = { 93 1.82 hannken .vfs_name = MOUNT_CD9660, 94 1.82 hannken .vfs_min_mount_data = sizeof (struct iso_args), 95 1.82 hannken .vfs_mount = cd9660_mount, 96 1.82 hannken .vfs_start = cd9660_start, 97 1.82 hannken .vfs_unmount = cd9660_unmount, 98 1.82 hannken .vfs_root = cd9660_root, 99 1.82 hannken .vfs_quotactl = (void *)eopnotsupp, 100 1.82 hannken .vfs_statvfs = cd9660_statvfs, 101 1.82 hannken .vfs_sync = cd9660_sync, 102 1.82 hannken .vfs_vget = cd9660_vget, 103 1.87 hannken .vfs_loadvnode = cd9660_loadvnode, 104 1.82 hannken .vfs_fhtovp = cd9660_fhtovp, 105 1.82 hannken .vfs_vptofh = cd9660_vptofh, 106 1.82 hannken .vfs_init = cd9660_init, 107 1.82 hannken .vfs_reinit = cd9660_reinit, 108 1.82 hannken .vfs_done = cd9660_done, 109 1.82 hannken .vfs_mountroot = cd9660_mountroot, 110 1.82 hannken .vfs_snapshot = (void *)eopnotsupp, 111 1.82 hannken .vfs_extattrctl = vfs_stdextattrctl, 112 1.91 hannken .vfs_suspendctl = genfs_suspendctl, 113 1.82 hannken .vfs_renamelock_enter = genfs_renamelock_enter, 114 1.82 hannken .vfs_renamelock_exit = genfs_renamelock_exit, 115 1.82 hannken .vfs_fsync = (void *)eopnotsupp, 116 1.82 hannken .vfs_opv_descs = cd9660_vnodeopv_descs 117 1.1 jdolecek }; 118 1.1 jdolecek 119 1.24 yamt static const struct genfs_ops cd9660_genfsops = { 120 1.24 yamt .gop_size = genfs_size, 121 1.1 jdolecek }; 122 1.1 jdolecek 123 1.1 jdolecek /* 124 1.1 jdolecek * Called by vfs_mountroot when iso is going to be mounted as root. 125 1.1 jdolecek * 126 1.1 jdolecek * Name is updated by mount(8) after booting. 127 1.1 jdolecek */ 128 1.26 xtraeme static int iso_makemp(struct iso_mnt *isomp, struct buf *bp, int *ea_len); 129 1.26 xtraeme static int iso_mountfs(struct vnode *devvp, struct mount *mp, 130 1.30 christos struct lwp *l, struct iso_args *argp); 131 1.1 jdolecek 132 1.95 pgoyette SYSCTL_SETUP(cd9660_sysctl_setup, "cd9660 sysctl") 133 1.60 ad { 134 1.60 ad 135 1.95 pgoyette sysctl_createv(clog, 0, NULL, NULL, 136 1.63 rumble CTLFLAG_PERMANENT, CTLTYPE_NODE, "cd9660", 137 1.63 rumble SYSCTL_DESCR("ISO-9660 file system"), 138 1.63 rumble NULL, 0, NULL, 0, 139 1.63 rumble CTL_VFS, 14, CTL_EOL); 140 1.95 pgoyette sysctl_createv(clog, 0, NULL, NULL, 141 1.63 rumble CTLFLAG_PERMANENT|CTLFLAG_READWRITE, 142 1.63 rumble CTLTYPE_INT, "utf8_joliet", 143 1.63 rumble SYSCTL_DESCR("Encode Joliet filenames to UTF-8"), 144 1.63 rumble NULL, 0, &cd9660_utf8_joliet, 0, 145 1.63 rumble CTL_VFS, 14, CD9660_UTF8_JOLIET, CTL_EOL); 146 1.63 rumble /* 147 1.63 rumble * XXX the "14" above could be dynamic, thereby eliminating 148 1.63 rumble * one more instance of the "number to vfs" mapping problem, 149 1.63 rumble * but "14" is the order as taken from sys/mount.h 150 1.63 rumble */ 151 1.95 pgoyette } 152 1.95 pgoyette 153 1.95 pgoyette static int 154 1.95 pgoyette cd9660_modcmd(modcmd_t cmd, void *arg) 155 1.95 pgoyette { 156 1.95 pgoyette int error; 157 1.95 pgoyette 158 1.95 pgoyette switch (cmd) { 159 1.95 pgoyette case MODULE_CMD_INIT: 160 1.95 pgoyette error = vfs_attach(&cd9660_vfsops); 161 1.63 rumble break; 162 1.60 ad case MODULE_CMD_FINI: 163 1.63 rumble error = vfs_detach(&cd9660_vfsops); 164 1.63 rumble break; 165 1.60 ad default: 166 1.63 rumble error = ENOTTY; 167 1.63 rumble break; 168 1.60 ad } 169 1.63 rumble 170 1.63 rumble return (error); 171 1.60 ad } 172 1.60 ad 173 1.103 christos /* Compat with pre uid/gid/fsize/dsize mount call */ 174 1.103 christos #define OSIZE sizeof(struct { \ 175 1.103 christos const char *fspec; \ 176 1.103 christos struct export_args30 _pad1; \ 177 1.103 christos int flags; \ 178 1.103 christos }) 179 1.103 christos 180 1.103 christos static int 181 1.103 christos iso_checkupdate(const struct vnode *devvp, const struct iso_mnt *imp, 182 1.103 christos const struct iso_args *args) 183 1.103 christos { 184 1.103 christos 185 1.103 christos if (devvp != imp->im_devvp && devvp->v_rdev != imp->im_devvp->v_rdev) 186 1.103 christos return EINVAL; 187 1.103 christos 188 1.103 christos if (((imp->im_flags & ISOFSMNT_UID) && args->uid != imp->im_uid) || 189 1.103 christos ((imp->im_flags & ISOFSMNT_GID) && args->gid != imp->im_gid) || 190 1.103 christos args->fmask != imp->im_fmask || args->dmask != imp->im_dmask) 191 1.103 christos return EPERM; 192 1.103 christos 193 1.103 christos return 0; 194 1.103 christos } 195 1.103 christos 196 1.103 christos static void 197 1.103 christos iso_copyidmask(struct iso_args *args, const struct iso_mnt *imp) 198 1.103 christos { 199 1.103 christos 200 1.103 christos if (imp == NULL) { 201 1.103 christos args->uid = args->gid = 0; 202 1.103 christos args->fmask = args->dmask = S_IRWXU|S_IRWXG|S_IRWXO; 203 1.103 christos return; 204 1.103 christos } 205 1.103 christos args->uid = imp->im_uid; 206 1.103 christos args->gid = imp->im_gid; 207 1.103 christos args->fmask = imp->im_fmask; 208 1.103 christos args->dmask = imp->im_dmask; 209 1.103 christos } 210 1.103 christos 211 1.1 jdolecek int 212 1.57 matt cd9660_mountroot(void) 213 1.1 jdolecek { 214 1.1 jdolecek struct mount *mp; 215 1.50 pooka struct lwp *l = curlwp; 216 1.1 jdolecek int error; 217 1.1 jdolecek struct iso_args args; 218 1.1 jdolecek 219 1.31 thorpej if (device_class(root_device) != DV_DISK) 220 1.1 jdolecek return (ENODEV); 221 1.22 perry 222 1.1 jdolecek if ((error = vfs_rootmountalloc(MOUNT_CD9660, "root_device", &mp)) 223 1.1 jdolecek != 0) { 224 1.1 jdolecek vrele(rootvp); 225 1.1 jdolecek return (error); 226 1.1 jdolecek } 227 1.1 jdolecek 228 1.1 jdolecek args.flags = ISOFSMNT_ROOT; 229 1.103 christos iso_copyidmask(&args, NULL); 230 1.30 christos if ((error = iso_mountfs(rootvp, mp, l, &args)) != 0) { 231 1.93 hannken vfs_unbusy(mp); 232 1.92 hannken vfs_rele(mp); 233 1.1 jdolecek return (error); 234 1.1 jdolecek } 235 1.80 christos mountlist_append(mp); 236 1.50 pooka (void)cd9660_statvfs(mp, &mp->mnt_stat); 237 1.93 hannken vfs_unbusy(mp); 238 1.1 jdolecek return (0); 239 1.1 jdolecek } 240 1.1 jdolecek 241 1.103 christos 242 1.1 jdolecek /* 243 1.1 jdolecek * VFS Operations. 244 1.1 jdolecek * 245 1.1 jdolecek * mount system call 246 1.1 jdolecek */ 247 1.100 christos 248 1.1 jdolecek int 249 1.57 matt cd9660_mount(struct mount *mp, const char *path, void *data, size_t *data_len) 250 1.1 jdolecek { 251 1.50 pooka struct lwp *l = curlwp; 252 1.1 jdolecek struct vnode *devvp; 253 1.101 christos struct iso_args aa, *args = data; 254 1.1 jdolecek int error; 255 1.28 dyoung struct iso_mnt *imp = VFSTOISOFS(mp); 256 1.22 perry 257 1.84 maxv if (args == NULL) 258 1.84 maxv return EINVAL; 259 1.100 christos 260 1.101 christos if (*data_len != OSIZE && *data_len < sizeof(*args)) 261 1.43 dsl return EINVAL; 262 1.43 dsl 263 1.1 jdolecek if (mp->mnt_flag & MNT_GETARGS) { 264 1.1 jdolecek if (imp == NULL) 265 1.1 jdolecek return EIO; 266 1.100 christos 267 1.43 dsl args->fspec = NULL; 268 1.43 dsl args->flags = imp->im_flags; 269 1.100 christos if (*data_len == OSIZE) 270 1.100 christos return 0; 271 1.100 christos 272 1.103 christos iso_copyidmask(args, imp); 273 1.100 christos *data_len = sizeof(*args); 274 1.43 dsl return 0; 275 1.1 jdolecek } 276 1.22 perry 277 1.101 christos if (*data_len == OSIZE) { 278 1.101 christos memcpy(&aa, args, OSIZE); 279 1.102 christos args = &aa; 280 1.103 christos iso_copyidmask(args, (mp->mnt_flag & MNT_UPDATE) ? imp : NULL); 281 1.101 christos } 282 1.101 christos 283 1.1 jdolecek if ((mp->mnt_flag & MNT_RDONLY) == 0) 284 1.101 christos return EROFS; 285 1.22 perry 286 1.43 dsl if ((mp->mnt_flag & MNT_UPDATE) && args->fspec == NULL) 287 1.27 jmmv return EINVAL; 288 1.27 jmmv 289 1.1 jdolecek /* 290 1.1 jdolecek * Not an update, or updating the name: look up the name 291 1.1 jdolecek * and verify that it refers to a sensible block device. 292 1.1 jdolecek */ 293 1.67 dholland error = namei_simple_user(args->fspec, 294 1.101 christos NSM_FOLLOW_NOEMULROOT, &devvp); 295 1.67 dholland if (error != 0) 296 1.101 christos return error; 297 1.1 jdolecek 298 1.1 jdolecek if (devvp->v_type != VBLK) { 299 1.1 jdolecek vrele(devvp); 300 1.1 jdolecek return ENOTBLK; 301 1.1 jdolecek } 302 1.1 jdolecek if (bdevsw_lookup(devvp->v_rdev) == NULL) { 303 1.1 jdolecek vrele(devvp); 304 1.1 jdolecek return ENXIO; 305 1.1 jdolecek } 306 1.1 jdolecek /* 307 1.1 jdolecek * If mount by non-root, then verify that user has necessary 308 1.1 jdolecek * permissions on the device. 309 1.1 jdolecek */ 310 1.66 elad vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY); 311 1.75 elad error = kauth_authorize_system(l->l_cred, KAUTH_SYSTEM_MOUNT, 312 1.75 elad KAUTH_REQ_SYSTEM_MOUNT_DEVICE, mp, devvp, KAUTH_ARG(VREAD)); 313 1.66 elad if (error) { 314 1.83 dholland goto fail; 315 1.1 jdolecek } 316 1.21 mycroft if ((mp->mnt_flag & MNT_UPDATE) == 0) { 317 1.50 pooka error = VOP_OPEN(devvp, FREAD, FSCRED); 318 1.21 mycroft if (error) 319 1.21 mycroft goto fail; 320 1.83 dholland VOP_UNLOCK(devvp); 321 1.43 dsl error = iso_mountfs(devvp, mp, l, args); 322 1.83 dholland vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY); 323 1.21 mycroft if (error) { 324 1.50 pooka (void)VOP_CLOSE(devvp, FREAD, NOCRED); 325 1.21 mycroft goto fail; 326 1.21 mycroft } 327 1.83 dholland VOP_UNLOCK(devvp); 328 1.83 dholland /* reference to devvp is donated through iso_mountfs */ 329 1.21 mycroft } else { 330 1.103 christos if ((error = iso_checkupdate(devvp, imp, args)) != 0) 331 1.83 dholland goto fail; 332 1.83 dholland VOP_UNLOCK(devvp); 333 1.21 mycroft vrele(devvp); 334 1.1 jdolecek } 335 1.43 dsl return set_statvfs_info(path, UIO_USERSPACE, args->fspec, UIO_USERSPACE, 336 1.44 pooka mp->mnt_op->vfs_name, mp, l); 337 1.21 mycroft 338 1.21 mycroft fail: 339 1.83 dholland VOP_UNLOCK(devvp); 340 1.21 mycroft vrele(devvp); 341 1.101 christos return error; 342 1.1 jdolecek } 343 1.1 jdolecek 344 1.1 jdolecek /* 345 1.1 jdolecek * Make a mount point from a volume descriptor 346 1.1 jdolecek */ 347 1.1 jdolecek static int 348 1.57 matt iso_makemp(struct iso_mnt *isomp, struct buf *bp, int *ea_len) 349 1.1 jdolecek { 350 1.1 jdolecek struct iso_primary_descriptor *pri; 351 1.1 jdolecek int logical_block_size; 352 1.1 jdolecek struct iso_directory_record *rootp; 353 1.1 jdolecek 354 1.1 jdolecek pri = (struct iso_primary_descriptor *)bp->b_data; 355 1.22 perry 356 1.1 jdolecek logical_block_size = isonum_723 (pri->logical_block_size); 357 1.22 perry 358 1.1 jdolecek if (logical_block_size < DEV_BSIZE || logical_block_size > MAXBSIZE 359 1.1 jdolecek || (logical_block_size & (logical_block_size - 1)) != 0) 360 1.1 jdolecek return -1; 361 1.22 perry 362 1.1 jdolecek rootp = (struct iso_directory_record *)pri->root_directory_record; 363 1.22 perry 364 1.1 jdolecek isomp->logical_block_size = logical_block_size; 365 1.1 jdolecek isomp->volume_space_size = isonum_733 (pri->volume_space_size); 366 1.1 jdolecek memcpy(isomp->root, rootp, sizeof(isomp->root)); 367 1.1 jdolecek isomp->root_extent = isonum_733 (rootp->extent); 368 1.1 jdolecek isomp->root_size = isonum_733 (rootp->size); 369 1.1 jdolecek isomp->im_joliet_level = 0; 370 1.22 perry 371 1.1 jdolecek isomp->im_bmask = logical_block_size - 1; 372 1.1 jdolecek isomp->im_bshift = 0; 373 1.1 jdolecek while ((1 << isomp->im_bshift) < isomp->logical_block_size) 374 1.1 jdolecek isomp->im_bshift++; 375 1.1 jdolecek 376 1.1 jdolecek if (ea_len != NULL) 377 1.1 jdolecek *ea_len = isonum_711(rootp->ext_attr_length); 378 1.1 jdolecek 379 1.1 jdolecek return 0; 380 1.1 jdolecek } 381 1.1 jdolecek 382 1.1 jdolecek /* 383 1.1 jdolecek * Common code for mount and mountroot 384 1.1 jdolecek */ 385 1.1 jdolecek static int 386 1.57 matt iso_mountfs(struct vnode *devvp, struct mount *mp, struct lwp *l, 387 1.57 matt struct iso_args *argp) 388 1.1 jdolecek { 389 1.1 jdolecek struct iso_mnt *isomp = (struct iso_mnt *)0; 390 1.1 jdolecek struct buf *bp = NULL, *pribp = NULL, *supbp = NULL; 391 1.1 jdolecek dev_t dev = devvp->v_rdev; 392 1.1 jdolecek int error = EINVAL; 393 1.1 jdolecek int ronly = (mp->mnt_flag & MNT_RDONLY) != 0; 394 1.1 jdolecek int iso_bsize; 395 1.1 jdolecek int iso_blknum; 396 1.1 jdolecek int joliet_level; 397 1.1 jdolecek struct iso_volume_descriptor *vdp; 398 1.1 jdolecek struct iso_supplementary_descriptor *sup; 399 1.1 jdolecek int sess = 0; 400 1.1 jdolecek int ext_attr_length; 401 1.1 jdolecek struct disklabel label; 402 1.1 jdolecek 403 1.1 jdolecek if (!ronly) 404 1.1 jdolecek return EROFS; 405 1.22 perry 406 1.21 mycroft /* Flush out any old buffers remaining from a previous use. */ 407 1.97 hannken vn_lock(devvp, LK_EXCLUSIVE | LK_RETRY); 408 1.97 hannken error = vinvalbuf(devvp, V_SAVE, l->l_cred, l, 0, 0); 409 1.97 hannken VOP_UNLOCK(devvp); 410 1.97 hannken if (error != 0) 411 1.1 jdolecek return (error); 412 1.22 perry 413 1.1 jdolecek /* This is the "logical sector size". The standard says this 414 1.1 jdolecek * should be 2048 or the physical sector size on the device, 415 1.1 jdolecek * whichever is greater. For now, we'll just use a constant. 416 1.1 jdolecek */ 417 1.1 jdolecek iso_bsize = ISO_DEFAULT_BLOCK_SIZE; 418 1.1 jdolecek 419 1.50 pooka error = VOP_IOCTL(devvp, DIOCGDINFO, &label, FREAD, FSCRED); 420 1.70 mlelstv if (!error) { 421 1.1 jdolecek /* XXX more sanity checks? */ 422 1.1 jdolecek sess = label.d_partitions[DISKPART(dev)].p_cdsession; 423 1.1 jdolecek } else { 424 1.1 jdolecek /* fallback to old method */ 425 1.50 pooka error = VOP_IOCTL(devvp, CDIOREADMSADDR, &sess, 0, FSCRED); 426 1.1 jdolecek if (error) 427 1.1 jdolecek sess = 0; /* never mind */ 428 1.1 jdolecek } 429 1.4 christos #ifdef ISO_DEBUG 430 1.65 cegger printf("isofs: session offset (part %"PRId32") %d\n", DISKPART(dev), sess); 431 1.1 jdolecek #endif 432 1.1 jdolecek 433 1.1 jdolecek for (iso_blknum = 16; iso_blknum < 100; iso_blknum++) { 434 1.1 jdolecek if ((error = bread(devvp, (iso_blknum+sess) * btodb(iso_bsize), 435 1.90 maxv iso_bsize, 0, &bp)) != 0) 436 1.1 jdolecek goto out; 437 1.22 perry 438 1.1 jdolecek vdp = (struct iso_volume_descriptor *)bp->b_data; 439 1.1 jdolecek if (memcmp(vdp->id, ISO_STANDARD_ID, sizeof(vdp->id)) != 0) { 440 1.1 jdolecek error = EINVAL; 441 1.1 jdolecek goto out; 442 1.1 jdolecek } 443 1.1 jdolecek 444 1.1 jdolecek switch (isonum_711(vdp->type)) { 445 1.1 jdolecek case ISO_VD_PRIMARY: 446 1.1 jdolecek if (pribp == NULL) { 447 1.1 jdolecek pribp = bp; 448 1.1 jdolecek bp = NULL; 449 1.1 jdolecek } 450 1.1 jdolecek break; 451 1.1 jdolecek 452 1.1 jdolecek case ISO_VD_SUPPLEMENTARY: 453 1.1 jdolecek if (supbp == NULL) { 454 1.1 jdolecek supbp = bp; 455 1.1 jdolecek bp = NULL; 456 1.1 jdolecek } 457 1.1 jdolecek break; 458 1.1 jdolecek 459 1.1 jdolecek default: 460 1.1 jdolecek break; 461 1.1 jdolecek } 462 1.1 jdolecek 463 1.1 jdolecek if (isonum_711 (vdp->type) == ISO_VD_END) { 464 1.48 ad brelse(bp, 0); 465 1.1 jdolecek bp = NULL; 466 1.1 jdolecek break; 467 1.1 jdolecek } 468 1.1 jdolecek 469 1.1 jdolecek if (bp != NULL) { 470 1.48 ad brelse(bp, 0); 471 1.1 jdolecek bp = NULL; 472 1.1 jdolecek } 473 1.1 jdolecek } 474 1.1 jdolecek 475 1.1 jdolecek if (pribp == NULL) { 476 1.1 jdolecek error = EINVAL; 477 1.1 jdolecek goto out; 478 1.1 jdolecek } 479 1.1 jdolecek 480 1.1 jdolecek isomp = malloc(sizeof *isomp, M_ISOFSMNT, M_WAITOK); 481 1.3 dsl memset(isomp, 0, sizeof *isomp); 482 1.1 jdolecek if (iso_makemp(isomp, pribp, &ext_attr_length) == -1) { 483 1.1 jdolecek error = EINVAL; 484 1.1 jdolecek goto out; 485 1.1 jdolecek } 486 1.1 jdolecek 487 1.1 jdolecek isomp->volume_space_size += sess; 488 1.1 jdolecek 489 1.48 ad brelse(pribp, BC_AGE); 490 1.1 jdolecek pribp = NULL; 491 1.22 perry 492 1.1 jdolecek mp->mnt_data = isomp; 493 1.12 christos mp->mnt_stat.f_fsidx.__fsid_val[0] = (long)dev; 494 1.12 christos mp->mnt_stat.f_fsidx.__fsid_val[1] = makefstype(MOUNT_CD9660); 495 1.12 christos mp->mnt_stat.f_fsid = mp->mnt_stat.f_fsidx.__fsid_val[0]; 496 1.73 christos mp->mnt_stat.f_namemax = ISO_MAXNAMLEN; 497 1.1 jdolecek mp->mnt_flag |= MNT_LOCAL; 498 1.96 ad mp->mnt_iflag |= IMNT_MPSAFE | IMNT_SHRLOOKUP; 499 1.1 jdolecek mp->mnt_dev_bshift = iso_bsize; 500 1.1 jdolecek mp->mnt_fs_bshift = isomp->im_bshift; 501 1.1 jdolecek isomp->im_mountp = mp; 502 1.1 jdolecek isomp->im_dev = dev; 503 1.1 jdolecek isomp->im_devvp = devvp; 504 1.22 perry 505 1.98 christos if (argp->flags & ISOFSMNT_UID) 506 1.98 christos isomp->im_uid = argp->uid; 507 1.98 christos if (argp->flags & ISOFSMNT_GID) 508 1.98 christos isomp->im_gid = argp->gid; 509 1.98 christos isomp->im_fmask = argp->fmask & ACCESSPERMS; 510 1.98 christos isomp->im_dmask = argp->dmask & ACCESSPERMS; 511 1.98 christos 512 1.1 jdolecek /* Check the Rock Ridge Extension support */ 513 1.1 jdolecek if (!(argp->flags & ISOFSMNT_NORRIP)) { 514 1.1 jdolecek struct iso_directory_record *rootp; 515 1.1 jdolecek 516 1.1 jdolecek if ((error = bread(isomp->im_devvp, 517 1.1 jdolecek (isomp->root_extent + ext_attr_length) << 518 1.1 jdolecek (isomp->im_bshift - DEV_BSHIFT), 519 1.90 maxv isomp->logical_block_size, 520 1.62 hannken 0, &bp)) != 0) 521 1.1 jdolecek goto out; 522 1.22 perry 523 1.1 jdolecek rootp = (struct iso_directory_record *)bp->b_data; 524 1.22 perry 525 1.1 jdolecek if ((isomp->rr_skip = cd9660_rrip_offset(rootp,isomp)) < 0) { 526 1.1 jdolecek argp->flags |= ISOFSMNT_NORRIP; 527 1.1 jdolecek } else { 528 1.1 jdolecek argp->flags &= ~ISOFSMNT_GENS; 529 1.1 jdolecek } 530 1.22 perry 531 1.1 jdolecek /* 532 1.1 jdolecek * The contents are valid, 533 1.1 jdolecek * but they will get reread as part of another vnode, so... 534 1.1 jdolecek */ 535 1.48 ad brelse(bp, BC_AGE); 536 1.1 jdolecek bp = NULL; 537 1.1 jdolecek } 538 1.1 jdolecek isomp->im_flags = argp->flags & (ISOFSMNT_NORRIP | ISOFSMNT_GENS | 539 1.98 christos ISOFSMNT_EXTATT | ISOFSMNT_NOJOLIET | ISOFSMNT_RRCASEINS | 540 1.98 christos ISOFSMNT_UID | ISOFSMNT_GID); 541 1.1 jdolecek 542 1.1 jdolecek if (isomp->im_flags & ISOFSMNT_GENS) 543 1.1 jdolecek isomp->iso_ftype = ISO_FTYPE_9660; 544 1.1 jdolecek else if (isomp->im_flags & ISOFSMNT_NORRIP) { 545 1.1 jdolecek isomp->iso_ftype = ISO_FTYPE_DEFAULT; 546 1.1 jdolecek if (argp->flags & ISOFSMNT_NOCASETRANS) 547 1.1 jdolecek isomp->im_flags |= ISOFSMNT_NOCASETRANS; 548 1.22 perry } else 549 1.1 jdolecek isomp->iso_ftype = ISO_FTYPE_RRIP; 550 1.1 jdolecek 551 1.1 jdolecek /* Check the Joliet Extension support */ 552 1.1 jdolecek if ((argp->flags & ISOFSMNT_NORRIP) != 0 && 553 1.1 jdolecek (argp->flags & ISOFSMNT_NOJOLIET) == 0 && 554 1.1 jdolecek supbp != NULL) { 555 1.1 jdolecek joliet_level = 0; 556 1.1 jdolecek sup = (struct iso_supplementary_descriptor *)supbp->b_data; 557 1.1 jdolecek 558 1.1 jdolecek if ((isonum_711(sup->flags) & 1) == 0) { 559 1.1 jdolecek if (memcmp(sup->escape, "%/@", 3) == 0) 560 1.1 jdolecek joliet_level = 1; 561 1.1 jdolecek if (memcmp(sup->escape, "%/C", 3) == 0) 562 1.1 jdolecek joliet_level = 2; 563 1.1 jdolecek if (memcmp(sup->escape, "%/E", 3) == 0) 564 1.1 jdolecek joliet_level = 3; 565 1.1 jdolecek } 566 1.1 jdolecek if (joliet_level != 0) { 567 1.1 jdolecek if (iso_makemp(isomp, supbp, NULL) == -1) { 568 1.1 jdolecek error = EINVAL; 569 1.1 jdolecek goto out; 570 1.1 jdolecek } 571 1.1 jdolecek isomp->im_joliet_level = joliet_level; 572 1.1 jdolecek } 573 1.1 jdolecek } 574 1.1 jdolecek 575 1.1 jdolecek if (supbp != NULL) { 576 1.48 ad brelse(supbp, 0); 577 1.1 jdolecek supbp = NULL; 578 1.1 jdolecek } 579 1.22 perry 580 1.79 hannken spec_node_setmountedfs(devvp, mp); 581 1.37 drochner 582 1.1 jdolecek return 0; 583 1.1 jdolecek out: 584 1.1 jdolecek if (bp) 585 1.48 ad brelse(bp, 0); 586 1.1 jdolecek if (pribp) 587 1.48 ad brelse(pribp, 0); 588 1.1 jdolecek if (supbp) 589 1.48 ad brelse(supbp, 0); 590 1.1 jdolecek if (isomp) { 591 1.3 dsl free(isomp, M_ISOFSMNT); 592 1.1 jdolecek mp->mnt_data = NULL; 593 1.1 jdolecek } 594 1.1 jdolecek return error; 595 1.1 jdolecek } 596 1.1 jdolecek 597 1.1 jdolecek /* 598 1.1 jdolecek * Make a filesystem operational. 599 1.1 jdolecek * Nothing to do at the moment. 600 1.1 jdolecek */ 601 1.1 jdolecek /* ARGSUSED */ 602 1.1 jdolecek int 603 1.50 pooka cd9660_start(struct mount *mp, int flags) 604 1.1 jdolecek { 605 1.1 jdolecek return 0; 606 1.1 jdolecek } 607 1.1 jdolecek 608 1.1 jdolecek /* 609 1.1 jdolecek * unmount system call 610 1.1 jdolecek */ 611 1.1 jdolecek int 612 1.57 matt cd9660_unmount(struct mount *mp, int mntflags) 613 1.1 jdolecek { 614 1.1 jdolecek struct iso_mnt *isomp; 615 1.1 jdolecek int error, flags = 0; 616 1.22 perry 617 1.1 jdolecek if (mntflags & MNT_FORCE) 618 1.1 jdolecek flags |= FORCECLOSE; 619 1.1 jdolecek if ((error = vflush(mp, NULLVP, flags)) != 0) 620 1.1 jdolecek return (error); 621 1.1 jdolecek 622 1.1 jdolecek isomp = VFSTOISOFS(mp); 623 1.1 jdolecek 624 1.1 jdolecek if (isomp->im_devvp->v_type != VBAD) 625 1.79 hannken spec_node_setmountedfs(isomp->im_devvp, NULL); 626 1.1 jdolecek 627 1.1 jdolecek vn_lock(isomp->im_devvp, LK_EXCLUSIVE | LK_RETRY); 628 1.50 pooka error = VOP_CLOSE(isomp->im_devvp, FREAD, NOCRED); 629 1.1 jdolecek vput(isomp->im_devvp); 630 1.3 dsl free(isomp, M_ISOFSMNT); 631 1.1 jdolecek mp->mnt_data = NULL; 632 1.1 jdolecek mp->mnt_flag &= ~MNT_LOCAL; 633 1.1 jdolecek return (error); 634 1.1 jdolecek } 635 1.1 jdolecek 636 1.1 jdolecek /* 637 1.1 jdolecek * Return root of a filesystem 638 1.1 jdolecek */ 639 1.1 jdolecek int 640 1.94 ad cd9660_root(struct mount *mp, int lktype, struct vnode **vpp) 641 1.1 jdolecek { 642 1.1 jdolecek struct iso_mnt *imp = VFSTOISOFS(mp); 643 1.1 jdolecek struct iso_directory_record *dp = 644 1.1 jdolecek (struct iso_directory_record *)imp->root; 645 1.1 jdolecek ino_t ino = isodirino(dp, imp); 646 1.22 perry 647 1.94 ad return cd9660_vget(mp, ino, lktype, vpp); 648 1.1 jdolecek } 649 1.1 jdolecek 650 1.1 jdolecek /* 651 1.1 jdolecek * Get file system statistics. 652 1.1 jdolecek */ 653 1.1 jdolecek int 654 1.57 matt cd9660_statvfs(struct mount *mp, struct statvfs *sbp) 655 1.1 jdolecek { 656 1.1 jdolecek struct iso_mnt *isomp; 657 1.22 perry 658 1.1 jdolecek isomp = VFSTOISOFS(mp); 659 1.1 jdolecek 660 1.1 jdolecek sbp->f_bsize = isomp->logical_block_size; 661 1.12 christos sbp->f_frsize = sbp->f_bsize; 662 1.1 jdolecek sbp->f_iosize = sbp->f_bsize; /* XXX */ 663 1.1 jdolecek sbp->f_blocks = isomp->volume_space_size; 664 1.1 jdolecek sbp->f_bfree = 0; /* total free blocks */ 665 1.1 jdolecek sbp->f_bavail = 0; /* blocks free for non superuser */ 666 1.12 christos sbp->f_bresvd = 0; /* total reserved blocks */ 667 1.1 jdolecek sbp->f_files = 0; /* total files */ 668 1.1 jdolecek sbp->f_ffree = 0; /* free file nodes */ 669 1.25 jmmv sbp->f_favail = 0; /* free file nodes for non superuser */ 670 1.12 christos sbp->f_fresvd = 0; /* reserved file nodes */ 671 1.12 christos copy_statvfs_info(sbp, mp); 672 1.1 jdolecek /* Use the first spare for flags: */ 673 1.1 jdolecek sbp->f_spare[0] = isomp->im_flags; 674 1.1 jdolecek return 0; 675 1.1 jdolecek } 676 1.1 jdolecek 677 1.1 jdolecek /* ARGSUSED */ 678 1.1 jdolecek int 679 1.57 matt cd9660_sync(struct mount *mp, int waitfor, kauth_cred_t cred) 680 1.1 jdolecek { 681 1.57 matt return 0; 682 1.1 jdolecek } 683 1.1 jdolecek 684 1.1 jdolecek /* 685 1.1 jdolecek * File handle to vnode 686 1.1 jdolecek * 687 1.1 jdolecek * Have to be really careful about stale file handles: 688 1.1 jdolecek * - check that the inode number is in range 689 1.1 jdolecek * - call iget() to get the locked inode 690 1.1 jdolecek * - check for an unallocated inode (i_mode == 0) 691 1.1 jdolecek * - check that the generation number matches 692 1.1 jdolecek */ 693 1.1 jdolecek 694 1.1 jdolecek struct ifid { 695 1.1 jdolecek ushort ifid_len; 696 1.1 jdolecek ushort ifid_pad; 697 1.85 martin ino_t ifid_ino; 698 1.85 martin #ifdef ISOFS_DBG 699 1.85 martin u_long ifid_start; 700 1.85 martin #endif 701 1.1 jdolecek }; 702 1.1 jdolecek 703 1.1 jdolecek /* ARGSUSED */ 704 1.1 jdolecek int 705 1.94 ad cd9660_fhtovp(struct mount *mp, struct fid *fhp, int lktype, struct vnode **vpp) 706 1.1 jdolecek { 707 1.33 martin struct ifid ifh; 708 1.1 jdolecek struct iso_node *ip; 709 1.1 jdolecek struct vnode *nvp; 710 1.1 jdolecek int error; 711 1.22 perry 712 1.33 martin if (fhp->fid_len != sizeof(ifh)) 713 1.33 martin return EINVAL; 714 1.33 martin 715 1.33 martin memcpy(&ifh, fhp, sizeof(ifh)); 716 1.1 jdolecek #ifdef ISOFS_DBG 717 1.85 martin printf("fhtovp: ino %"PRIu64", start %lu\n", 718 1.33 martin ifh.ifid_ino, ifh.ifid_start); 719 1.1 jdolecek #endif 720 1.22 perry 721 1.94 ad if ((error = VFS_VGET(mp, ifh.ifid_ino, lktype, &nvp)) != 0) { 722 1.1 jdolecek *vpp = NULLVP; 723 1.1 jdolecek return (error); 724 1.1 jdolecek } 725 1.1 jdolecek ip = VTOI(nvp); 726 1.1 jdolecek if (ip->inode.iso_mode == 0) { 727 1.1 jdolecek vput(nvp); 728 1.1 jdolecek *vpp = NULLVP; 729 1.1 jdolecek return (ESTALE); 730 1.1 jdolecek } 731 1.1 jdolecek *vpp = nvp; 732 1.1 jdolecek return (0); 733 1.1 jdolecek } 734 1.1 jdolecek 735 1.1 jdolecek int 736 1.94 ad cd9660_vget(struct mount *mp, ino_t ino, int lktype, struct vnode **vpp) 737 1.1 jdolecek { 738 1.87 hannken int error; 739 1.1 jdolecek 740 1.87 hannken error = vcache_get(mp, &ino, sizeof(ino), vpp); 741 1.87 hannken if (error) 742 1.87 hannken return error; 743 1.94 ad error = vn_lock(*vpp, lktype); 744 1.87 hannken if (error) { 745 1.87 hannken vrele(*vpp); 746 1.87 hannken *vpp = NULL; 747 1.87 hannken return error; 748 1.87 hannken } 749 1.87 hannken return 0; 750 1.1 jdolecek } 751 1.1 jdolecek 752 1.1 jdolecek int 753 1.87 hannken cd9660_loadvnode(struct mount *mp, struct vnode *vp, 754 1.87 hannken const void *key, size_t key_len, const void **new_key) 755 1.1 jdolecek { 756 1.1 jdolecek struct iso_mnt *imp; 757 1.1 jdolecek struct iso_node *ip; 758 1.86 hannken struct iso_directory_record *isodir; 759 1.1 jdolecek struct buf *bp; 760 1.1 jdolecek dev_t dev; 761 1.87 hannken ino_t ino; 762 1.86 hannken int lbn, off; 763 1.1 jdolecek int error; 764 1.1 jdolecek 765 1.87 hannken KASSERT(key_len == sizeof(ino)); 766 1.87 hannken memcpy(&ino, key, key_len); 767 1.1 jdolecek imp = VFSTOISOFS(mp); 768 1.1 jdolecek dev = imp->im_dev; 769 1.51 ad 770 1.1 jdolecek ip = pool_get(&cd9660_node_pool, PR_WAITOK); 771 1.51 ad 772 1.3 dsl memset(ip, 0, sizeof(struct iso_node)); 773 1.1 jdolecek ip->i_vnode = vp; 774 1.1 jdolecek ip->i_dev = dev; 775 1.1 jdolecek ip->i_number = ino; 776 1.51 ad ip->i_mnt = imp; 777 1.51 ad ip->i_devvp = imp->im_devvp; 778 1.1 jdolecek 779 1.86 hannken lbn = cd9660_lblkno(imp, ino); 780 1.86 hannken if (lbn >= imp->volume_space_size) { 781 1.87 hannken pool_put(&cd9660_node_pool, ip); 782 1.86 hannken printf("fhtovp: lbn exceed volume space %d\n", lbn); 783 1.86 hannken return (ESTALE); 784 1.86 hannken } 785 1.1 jdolecek 786 1.86 hannken off = cd9660_blkoff(imp, ino); 787 1.86 hannken if (off + ISO_DIRECTORY_RECORD_SIZE > imp->logical_block_size) { 788 1.87 hannken pool_put(&cd9660_node_pool, ip); 789 1.86 hannken printf("fhtovp: crosses block boundary %d\n", 790 1.86 hannken off + ISO_DIRECTORY_RECORD_SIZE); 791 1.86 hannken return (ESTALE); 792 1.86 hannken } 793 1.22 perry 794 1.86 hannken error = bread(imp->im_devvp, 795 1.86 hannken lbn << (imp->im_bshift - DEV_BSHIFT), 796 1.90 maxv imp->logical_block_size, 0, &bp); 797 1.86 hannken if (error) { 798 1.87 hannken pool_put(&cd9660_node_pool, ip); 799 1.86 hannken printf("fhtovp: bread error %d\n",error); 800 1.86 hannken return (error); 801 1.86 hannken } 802 1.86 hannken isodir = (struct iso_directory_record *)((char *)bp->b_data + off); 803 1.22 perry 804 1.86 hannken if (off + isonum_711(isodir->length) > imp->logical_block_size) { 805 1.87 hannken pool_put(&cd9660_node_pool, ip); 806 1.88 hannken brelse(bp, 0); 807 1.86 hannken printf("fhtovp: directory crosses block boundary %d[off=%d/len=%d]\n", 808 1.86 hannken off +isonum_711(isodir->length), off, 809 1.86 hannken isonum_711(isodir->length)); 810 1.86 hannken return (ESTALE); 811 1.86 hannken } 812 1.22 perry 813 1.1 jdolecek #if 0 814 1.86 hannken if (isonum_733(isodir->extent) + 815 1.86 hannken isonum_711(isodir->ext_attr_length) != ifhp->ifid_start) { 816 1.87 hannken pool_put(&cd9660_node_pool, ip); 817 1.1 jdolecek if (bp != 0) 818 1.48 ad brelse(bp, 0); 819 1.86 hannken printf("fhtovp: file start miss %d vs %d\n", 820 1.86 hannken isonum_733(isodir->extent) + isonum_711(isodir->ext_attr_length), 821 1.86 hannken ifhp->ifid_start); 822 1.86 hannken return (ESTALE); 823 1.1 jdolecek } 824 1.86 hannken #endif 825 1.1 jdolecek 826 1.1 jdolecek ip->iso_extent = isonum_733(isodir->extent); 827 1.1 jdolecek ip->i_size = isonum_733(isodir->size); 828 1.1 jdolecek ip->iso_start = isonum_711(isodir->ext_attr_length) + ip->iso_extent; 829 1.22 perry 830 1.87 hannken vp->v_tag = VT_ISOFS; 831 1.87 hannken vp->v_op = cd9660_vnodeop_p; 832 1.87 hannken vp->v_data = ip; 833 1.87 hannken genfs_node_init(vp, &cd9660_genfsops); 834 1.87 hannken 835 1.1 jdolecek /* 836 1.1 jdolecek * Setup time stamp, attribute 837 1.1 jdolecek */ 838 1.1 jdolecek switch (imp->iso_ftype) { 839 1.1 jdolecek default: /* ISO_FTYPE_9660 */ 840 1.1 jdolecek { 841 1.1 jdolecek struct buf *bp2; 842 1.1 jdolecek if ((imp->im_flags & ISOFSMNT_EXTATT) 843 1.1 jdolecek && (off = isonum_711(isodir->ext_attr_length))) 844 1.104 reinoud cd9660_blkatoff(vp, -((off_t) off << imp->im_bshift), 845 1.29 yamt NULL, &bp2); 846 1.1 jdolecek else 847 1.1 jdolecek bp2 = NULL; 848 1.1 jdolecek cd9660_defattr(isodir, ip, bp2); 849 1.1 jdolecek cd9660_deftstamp(isodir, ip, bp2); 850 1.1 jdolecek if (bp2) 851 1.48 ad brelse(bp2, 0); 852 1.1 jdolecek break; 853 1.1 jdolecek } 854 1.1 jdolecek case ISO_FTYPE_RRIP: 855 1.1 jdolecek cd9660_rrip_analyze(isodir, ip, imp); 856 1.1 jdolecek break; 857 1.1 jdolecek } 858 1.1 jdolecek 859 1.88 hannken brelse(bp, 0); 860 1.1 jdolecek 861 1.1 jdolecek /* 862 1.1 jdolecek * Initialize the associated vnode 863 1.1 jdolecek */ 864 1.1 jdolecek switch (vp->v_type = IFTOVT(ip->inode.iso_mode)) { 865 1.1 jdolecek case VFIFO: 866 1.1 jdolecek vp->v_op = cd9660_fifoop_p; 867 1.1 jdolecek break; 868 1.1 jdolecek case VCHR: 869 1.1 jdolecek case VBLK: 870 1.1 jdolecek /* 871 1.1 jdolecek * if device, look at device number table for translation 872 1.1 jdolecek */ 873 1.1 jdolecek vp->v_op = cd9660_specop_p; 874 1.54 ad spec_node_init(vp, ip->inode.iso_rdev); 875 1.1 jdolecek break; 876 1.1 jdolecek case VLNK: 877 1.1 jdolecek case VNON: 878 1.1 jdolecek case VSOCK: 879 1.1 jdolecek case VDIR: 880 1.1 jdolecek case VBAD: 881 1.1 jdolecek break; 882 1.1 jdolecek case VREG: 883 1.1 jdolecek uvm_vnp_setsize(vp, ip->i_size); 884 1.1 jdolecek break; 885 1.1 jdolecek } 886 1.22 perry 887 1.45 pooka if (vp->v_type != VREG) 888 1.45 pooka uvm_vnp_setsize(vp, 0); 889 1.45 pooka 890 1.1 jdolecek if (ip->iso_extent == imp->root_extent) 891 1.49 ad vp->v_vflag |= VV_ROOT; 892 1.1 jdolecek 893 1.1 jdolecek /* 894 1.1 jdolecek * XXX need generation number? 895 1.1 jdolecek */ 896 1.22 perry 897 1.87 hannken *new_key = &ip->i_number; 898 1.87 hannken return 0; 899 1.1 jdolecek } 900 1.1 jdolecek 901 1.1 jdolecek /* 902 1.1 jdolecek * Vnode pointer to File handle 903 1.1 jdolecek */ 904 1.1 jdolecek /* ARGSUSED */ 905 1.1 jdolecek int 906 1.57 matt cd9660_vptofh(struct vnode *vp, struct fid *fhp, size_t *fh_size) 907 1.1 jdolecek { 908 1.1 jdolecek struct iso_node *ip = VTOI(vp); 909 1.33 martin struct ifid ifh; 910 1.22 perry 911 1.33 martin if (*fh_size < sizeof(struct ifid)) { 912 1.33 martin *fh_size = sizeof(struct ifid); 913 1.33 martin return E2BIG; 914 1.33 martin } 915 1.33 martin *fh_size = sizeof(struct ifid); 916 1.33 martin 917 1.33 martin memset(&ifh, 0, sizeof(ifh)); 918 1.33 martin ifh.ifid_len = sizeof(struct ifid); 919 1.33 martin ifh.ifid_ino = ip->i_number; 920 1.85 martin #ifdef ISOFS_DBG 921 1.33 martin ifh.ifid_start = ip->iso_start; 922 1.85 martin #endif 923 1.33 martin memcpy(fhp, &ifh, sizeof(ifh)); 924 1.22 perry 925 1.1 jdolecek #ifdef ISOFS_DBG 926 1.85 martin printf("vptofh: ino %"PRIu64", start %lu\n", 927 1.33 martin ifh.ifid_ino,ifh.ifid_start); 928 1.1 jdolecek #endif 929 1.1 jdolecek return 0; 930 1.1 jdolecek } 931