1/* 2 * Copyright (c) 1989, 1993 3 * The Regents of the University of California. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 3. All advertising materials mentioning features or use of this software 14 * must display the following acknowledgement: 15 * This product includes software developed by the University of 16 * California, Berkeley and its contributors. 17 * 4. Neither the name of the University nor the names of its contributors 18 * may be used to endorse or promote products derived from this software 19 * without specific prior written permission. 20 * 21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31 * SUCH DAMAGE. 32 * 33 * @(#)dead_vnops.c 8.1 (Berkeley) 6/10/93
| 1/* 2 * Copyright (c) 1989, 1993 3 * The Regents of the University of California. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 3. All advertising materials mentioning features or use of this software 14 * must display the following acknowledgement: 15 * This product includes software developed by the University of 16 * California, Berkeley and its contributors. 17 * 4. Neither the name of the University nor the names of its contributors 18 * may be used to endorse or promote products derived from this software 19 * without specific prior written permission. 20 * 21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31 * SUCH DAMAGE. 32 * 33 * @(#)dead_vnops.c 8.1 (Berkeley) 6/10/93
|
92 { &vop_default_desc, (vop_t *)vn_default_error }, 93 { &vop_lookup_desc, (vop_t *)dead_lookup }, /* lookup */ 94/* XXX: vop_cachedlookup */ 95 { &vop_create_desc, (vop_t *)dead_create }, /* create */ 96/* XXX: vop_whiteout */ 97 { &vop_mknod_desc, (vop_t *)dead_mknod }, /* mknod */ 98 { &vop_open_desc, (vop_t *)dead_open }, /* open */ 99 { &vop_close_desc, (vop_t *)dead_close }, /* close */ 100 { &vop_access_desc, (vop_t *)dead_access }, /* access */ 101 { &vop_getattr_desc, (vop_t *)dead_getattr }, /* getattr */ 102 { &vop_setattr_desc, (vop_t *)dead_setattr }, /* setattr */ 103 { &vop_read_desc, (vop_t *)dead_read }, /* read */ 104 { &vop_write_desc, (vop_t *)dead_write }, /* write */ 105/* XXX: vop_lease */ 106 { &vop_ioctl_desc, (vop_t *)dead_ioctl }, /* ioctl */ 107 { &vop_poll_desc, (vop_t *)dead_poll }, /* poll */ 108/* XXX: vop_revoke */ 109 { &vop_mmap_desc, (vop_t *)dead_mmap }, /* mmap */ 110 { &vop_fsync_desc, (vop_t *)dead_fsync }, /* fsync */ 111 { &vop_seek_desc, (vop_t *)dead_seek }, /* seek */ 112 { &vop_remove_desc, (vop_t *)dead_remove }, /* remove */ 113 { &vop_link_desc, (vop_t *)dead_link }, /* link */ 114 { &vop_rename_desc, (vop_t *)dead_rename }, /* rename */ 115 { &vop_mkdir_desc, (vop_t *)dead_mkdir }, /* mkdir */ 116 { &vop_rmdir_desc, (vop_t *)dead_rmdir }, /* rmdir */ 117 { &vop_symlink_desc, (vop_t *)dead_symlink }, /* symlink */ 118 { &vop_readdir_desc, (vop_t *)dead_readdir }, /* readdir */ 119 { &vop_readlink_desc, (vop_t *)dead_readlink }, /* readlink */ 120 { &vop_abortop_desc, (vop_t *)dead_abortop }, /* abortop */ 121 { &vop_inactive_desc, (vop_t *)dead_inactive }, /* inactive */ 122 { &vop_reclaim_desc, (vop_t *)dead_reclaim }, /* reclaim */ 123 { &vop_lock_desc, (vop_t *)dead_lock }, /* lock */ 124 { &vop_unlock_desc, (vop_t *)dead_unlock }, /* unlock */ 125 { &vop_bmap_desc, (vop_t *)dead_bmap }, /* bmap */ 126 { &vop_strategy_desc, (vop_t *)dead_strategy }, /* strategy */ 127 { &vop_print_desc, (vop_t *)dead_print }, /* print */ 128 { &vop_islocked_desc, (vop_t *)dead_islocked }, /* islocked */ 129 { &vop_pathconf_desc, (vop_t *)dead_pathconf }, /* pathconf */ 130 { &vop_advlock_desc, (vop_t *)dead_advlock }, /* advlock */ 131 { &vop_blkatoff_desc, (vop_t *)dead_blkatoff }, /* blkatoff */ 132 { &vop_valloc_desc, (vop_t *)dead_valloc }, /* valloc */ 133/* XXX: vop_reallocblks */ 134 { &vop_vfree_desc, (vop_t *)dead_vfree }, /* vfree */ 135 { &vop_truncate_desc, (vop_t *)dead_truncate }, /* truncate */ 136 { &vop_update_desc, (vop_t *)dead_update }, /* update */ 137/* XXX: vop_getpages */ 138/* XXX: vop_putpages */ 139 { &vop_bwrite_desc, (vop_t *)dead_bwrite }, /* bwrite */
| 92 { &vop_default_desc, (vop_t *) vn_default_error }, 93 { &vop_abortop_desc, (vop_t *) dead_abortop }, 94 { &vop_access_desc, (vop_t *) dead_access }, 95 { &vop_advlock_desc, (vop_t *) dead_advlock }, 96 { &vop_blkatoff_desc, (vop_t *) dead_blkatoff }, 97 { &vop_bmap_desc, (vop_t *) dead_bmap }, 98 { &vop_bwrite_desc, (vop_t *) dead_bwrite }, 99 { &vop_close_desc, (vop_t *) dead_close }, 100 { &vop_create_desc, (vop_t *) dead_create }, 101 { &vop_fsync_desc, (vop_t *) dead_fsync }, 102 { &vop_getattr_desc, (vop_t *) dead_getattr }, 103 { &vop_inactive_desc, (vop_t *) dead_inactive }, 104 { &vop_ioctl_desc, (vop_t *) dead_ioctl }, 105 { &vop_islocked_desc, (vop_t *) dead_islocked }, 106 { &vop_link_desc, (vop_t *) dead_link }, 107 { &vop_lock_desc, (vop_t *) dead_lock }, 108 { &vop_lookup_desc, (vop_t *) dead_lookup }, 109 { &vop_mkdir_desc, (vop_t *) dead_mkdir }, 110 { &vop_mknod_desc, (vop_t *) dead_mknod }, 111 { &vop_mmap_desc, (vop_t *) dead_mmap }, 112 { &vop_open_desc, (vop_t *) dead_open }, 113 { &vop_pathconf_desc, (vop_t *) dead_pathconf }, 114 { &vop_poll_desc, (vop_t *) dead_poll }, 115 { &vop_print_desc, (vop_t *) dead_print }, 116 { &vop_read_desc, (vop_t *) dead_read }, 117 { &vop_readdir_desc, (vop_t *) dead_readdir }, 118 { &vop_readlink_desc, (vop_t *) dead_readlink }, 119 { &vop_reclaim_desc, (vop_t *) dead_reclaim }, 120 { &vop_remove_desc, (vop_t *) dead_remove }, 121 { &vop_rename_desc, (vop_t *) dead_rename }, 122 { &vop_rmdir_desc, (vop_t *) dead_rmdir }, 123 { &vop_seek_desc, (vop_t *) dead_seek }, 124 { &vop_setattr_desc, (vop_t *) dead_setattr }, 125 { &vop_strategy_desc, (vop_t *) dead_strategy }, 126 { &vop_symlink_desc, (vop_t *) dead_symlink }, 127 { &vop_truncate_desc, (vop_t *) dead_truncate }, 128 { &vop_unlock_desc, (vop_t *) dead_unlock }, 129 { &vop_update_desc, (vop_t *) dead_update }, 130 { &vop_valloc_desc, (vop_t *) dead_valloc }, 131 { &vop_vfree_desc, (vop_t *) dead_vfree }, 132 { &vop_write_desc, (vop_t *) dead_write },
|
140 { NULL, NULL } 141}; 142static struct vnodeopv_desc dead_vnodeop_opv_desc = 143 { &dead_vnodeop_p, dead_vnodeop_entries }; 144 145VNODEOP_SET(dead_vnodeop_opv_desc); 146 147/* 148 * Trivial lookup routine that always fails. 149 */ 150/* ARGSUSED */ 151static int 152dead_lookup(ap) 153 struct vop_lookup_args /* { 154 struct vnode * a_dvp; 155 struct vnode ** a_vpp; 156 struct componentname * a_cnp; 157 } */ *ap; 158{ 159 160 *ap->a_vpp = NULL; 161 return (ENOTDIR); 162} 163 164/* 165 * Open always fails as if device did not exist. 166 */ 167/* ARGSUSED */ 168static int 169dead_open(ap) 170 struct vop_open_args /* { 171 struct vnode *a_vp; 172 int a_mode; 173 struct ucred *a_cred; 174 struct proc *a_p; 175 } */ *ap; 176{ 177 178 return (ENXIO); 179} 180 181/* 182 * Vnode op for read 183 */ 184/* ARGSUSED */ 185static int 186dead_read(ap) 187 struct vop_read_args /* { 188 struct vnode *a_vp; 189 struct uio *a_uio; 190 int a_ioflag; 191 struct ucred *a_cred; 192 } */ *ap; 193{ 194 195 if (chkvnlock(ap->a_vp)) 196 panic("dead_read: lock"); 197#if 0 198 /* Lite2 behaviour */ 199 /* 200 * Return EOF for tty devices, EIO for others 201 */ 202 if ((ap->a_vp->v_flag & VISTTY) == 0) 203 return (EIO); 204#else 205 /* 206 * Return EOF for character devices, EIO for others 207 */ 208 if (ap->a_vp->v_type != VCHR) 209 return (EIO); 210#endif 211 return (0); 212} 213 214/* 215 * Vnode op for write 216 */ 217/* ARGSUSED */ 218static int 219dead_write(ap) 220 struct vop_write_args /* { 221 struct vnode *a_vp; 222 struct uio *a_uio; 223 int a_ioflag; 224 struct ucred *a_cred; 225 } */ *ap; 226{ 227 228 if (chkvnlock(ap->a_vp)) 229 panic("dead_write: lock"); 230 return (EIO); 231} 232 233/* 234 * Device ioctl operation. 235 */ 236/* ARGSUSED */ 237static int 238dead_ioctl(ap) 239 struct vop_ioctl_args /* { 240 struct vnode *a_vp; 241 int a_command; 242 caddr_t a_data; 243 int a_fflag; 244 struct ucred *a_cred; 245 struct proc *a_p; 246 } */ *ap; 247{ 248 249 if (!chkvnlock(ap->a_vp)) 250 return (EBADF); 251 return (VCALL(ap->a_vp, VOFFSET(vop_ioctl), ap)); 252} 253 254/* 255 * Just call the device strategy routine 256 */ 257static int 258dead_strategy(ap) 259 struct vop_strategy_args /* { 260 struct buf *a_bp; 261 } */ *ap; 262{ 263 264 if (ap->a_bp->b_vp == NULL || !chkvnlock(ap->a_bp->b_vp)) { 265 ap->a_bp->b_flags |= B_ERROR; 266 biodone(ap->a_bp); 267 return (EIO); 268 } 269 return (VOP_STRATEGY(ap->a_bp)); 270} 271 272/* 273 * Wait until the vnode has finished changing state. 274 */ 275static int 276dead_lock(ap) 277 struct vop_lock_args /* { 278 struct vnode *a_vp; 279 int a_flags; 280 struct proc *a_p; 281 } */ *ap; 282{ 283 struct vnode *vp = ap->a_vp; 284 285 /* 286 * Since we are not using the lock manager, we must clear 287 * the interlock here. 288 */ 289 if (ap->a_flags & LK_INTERLOCK) { 290 simple_unlock(&vp->v_interlock); 291 ap->a_flags &= ~LK_INTERLOCK; 292 } 293 if (!chkvnlock(vp)) 294 return (0); 295 return (VCALL(vp, VOFFSET(vop_lock), ap)); 296} 297 298/* 299 * Wait until the vnode has finished changing state. 300 */ 301static int 302dead_bmap(ap) 303 struct vop_bmap_args /* { 304 struct vnode *a_vp; 305 daddr_t a_bn; 306 struct vnode **a_vpp; 307 daddr_t *a_bnp; 308 int *a_runp; 309 int *a_runb; 310 } */ *ap; 311{ 312 313 if (!chkvnlock(ap->a_vp)) 314 return (EIO); 315 return (VOP_BMAP(ap->a_vp, ap->a_bn, ap->a_vpp, ap->a_bnp, ap->a_runp, ap->a_runb)); 316} 317 318/* 319 * Print out the contents of a dead vnode. 320 */ 321/* ARGSUSED */ 322static int 323dead_print(ap) 324 struct vop_print_args /* { 325 struct vnode *a_vp; 326 } */ *ap; 327{ 328 329 printf("tag VT_NON, dead vnode\n"); 330 return (0); 331} 332 333/* 334 * Empty vnode failed operation 335 */ 336static int 337dead_ebadf() 338{ 339 340 return (EBADF); 341} 342 343/* 344 * Empty vnode bad operation 345 */ 346static int 347dead_badop() 348{ 349 350 panic("dead_badop called"); 351 /* NOTREACHED */ 352} 353 354/* 355 * We have to wait during times when the vnode is 356 * in a state of change. 357 */ 358int 359chkvnlock(vp) 360 register struct vnode *vp; 361{ 362 int locked = 0; 363 364 while (vp->v_flag & VXLOCK) { 365 vp->v_flag |= VXWANT; 366 (void) tsleep((caddr_t)vp, PINOD, "ckvnlk", 0); 367 locked = 1; 368 } 369 return (locked); 370}
| 133 { NULL, NULL } 134}; 135static struct vnodeopv_desc dead_vnodeop_opv_desc = 136 { &dead_vnodeop_p, dead_vnodeop_entries }; 137 138VNODEOP_SET(dead_vnodeop_opv_desc); 139 140/* 141 * Trivial lookup routine that always fails. 142 */ 143/* ARGSUSED */ 144static int 145dead_lookup(ap) 146 struct vop_lookup_args /* { 147 struct vnode * a_dvp; 148 struct vnode ** a_vpp; 149 struct componentname * a_cnp; 150 } */ *ap; 151{ 152 153 *ap->a_vpp = NULL; 154 return (ENOTDIR); 155} 156 157/* 158 * Open always fails as if device did not exist. 159 */ 160/* ARGSUSED */ 161static int 162dead_open(ap) 163 struct vop_open_args /* { 164 struct vnode *a_vp; 165 int a_mode; 166 struct ucred *a_cred; 167 struct proc *a_p; 168 } */ *ap; 169{ 170 171 return (ENXIO); 172} 173 174/* 175 * Vnode op for read 176 */ 177/* ARGSUSED */ 178static int 179dead_read(ap) 180 struct vop_read_args /* { 181 struct vnode *a_vp; 182 struct uio *a_uio; 183 int a_ioflag; 184 struct ucred *a_cred; 185 } */ *ap; 186{ 187 188 if (chkvnlock(ap->a_vp)) 189 panic("dead_read: lock"); 190#if 0 191 /* Lite2 behaviour */ 192 /* 193 * Return EOF for tty devices, EIO for others 194 */ 195 if ((ap->a_vp->v_flag & VISTTY) == 0) 196 return (EIO); 197#else 198 /* 199 * Return EOF for character devices, EIO for others 200 */ 201 if (ap->a_vp->v_type != VCHR) 202 return (EIO); 203#endif 204 return (0); 205} 206 207/* 208 * Vnode op for write 209 */ 210/* ARGSUSED */ 211static int 212dead_write(ap) 213 struct vop_write_args /* { 214 struct vnode *a_vp; 215 struct uio *a_uio; 216 int a_ioflag; 217 struct ucred *a_cred; 218 } */ *ap; 219{ 220 221 if (chkvnlock(ap->a_vp)) 222 panic("dead_write: lock"); 223 return (EIO); 224} 225 226/* 227 * Device ioctl operation. 228 */ 229/* ARGSUSED */ 230static int 231dead_ioctl(ap) 232 struct vop_ioctl_args /* { 233 struct vnode *a_vp; 234 int a_command; 235 caddr_t a_data; 236 int a_fflag; 237 struct ucred *a_cred; 238 struct proc *a_p; 239 } */ *ap; 240{ 241 242 if (!chkvnlock(ap->a_vp)) 243 return (EBADF); 244 return (VCALL(ap->a_vp, VOFFSET(vop_ioctl), ap)); 245} 246 247/* 248 * Just call the device strategy routine 249 */ 250static int 251dead_strategy(ap) 252 struct vop_strategy_args /* { 253 struct buf *a_bp; 254 } */ *ap; 255{ 256 257 if (ap->a_bp->b_vp == NULL || !chkvnlock(ap->a_bp->b_vp)) { 258 ap->a_bp->b_flags |= B_ERROR; 259 biodone(ap->a_bp); 260 return (EIO); 261 } 262 return (VOP_STRATEGY(ap->a_bp)); 263} 264 265/* 266 * Wait until the vnode has finished changing state. 267 */ 268static int 269dead_lock(ap) 270 struct vop_lock_args /* { 271 struct vnode *a_vp; 272 int a_flags; 273 struct proc *a_p; 274 } */ *ap; 275{ 276 struct vnode *vp = ap->a_vp; 277 278 /* 279 * Since we are not using the lock manager, we must clear 280 * the interlock here. 281 */ 282 if (ap->a_flags & LK_INTERLOCK) { 283 simple_unlock(&vp->v_interlock); 284 ap->a_flags &= ~LK_INTERLOCK; 285 } 286 if (!chkvnlock(vp)) 287 return (0); 288 return (VCALL(vp, VOFFSET(vop_lock), ap)); 289} 290 291/* 292 * Wait until the vnode has finished changing state. 293 */ 294static int 295dead_bmap(ap) 296 struct vop_bmap_args /* { 297 struct vnode *a_vp; 298 daddr_t a_bn; 299 struct vnode **a_vpp; 300 daddr_t *a_bnp; 301 int *a_runp; 302 int *a_runb; 303 } */ *ap; 304{ 305 306 if (!chkvnlock(ap->a_vp)) 307 return (EIO); 308 return (VOP_BMAP(ap->a_vp, ap->a_bn, ap->a_vpp, ap->a_bnp, ap->a_runp, ap->a_runb)); 309} 310 311/* 312 * Print out the contents of a dead vnode. 313 */ 314/* ARGSUSED */ 315static int 316dead_print(ap) 317 struct vop_print_args /* { 318 struct vnode *a_vp; 319 } */ *ap; 320{ 321 322 printf("tag VT_NON, dead vnode\n"); 323 return (0); 324} 325 326/* 327 * Empty vnode failed operation 328 */ 329static int 330dead_ebadf() 331{ 332 333 return (EBADF); 334} 335 336/* 337 * Empty vnode bad operation 338 */ 339static int 340dead_badop() 341{ 342 343 panic("dead_badop called"); 344 /* NOTREACHED */ 345} 346 347/* 348 * We have to wait during times when the vnode is 349 * in a state of change. 350 */ 351int 352chkvnlock(vp) 353 register struct vnode *vp; 354{ 355 int locked = 0; 356 357 while (vp->v_flag & VXLOCK) { 358 vp->v_flag |= VXWANT; 359 (void) tsleep((caddr_t)vp, PINOD, "ckvnlk", 0); 360 locked = 1; 361 } 362 return (locked); 363}
|