139019Smckusick /* 239019Smckusick * Copyright (c) 1989 The Regents of the University of California. 339019Smckusick * All rights reserved. 439019Smckusick * 544536Sbostic * %sccs.include.redist.c% 639019Smckusick * 7*53594Sheideman * @(#)mfs_vnops.c 7.32 (Berkeley) 05/15/92 839019Smckusick */ 939019Smckusick 1051477Sbostic #include <sys/param.h> 1151477Sbostic #include <sys/systm.h> 1251477Sbostic #include <sys/time.h> 1351477Sbostic #include <sys/kernel.h> 1451477Sbostic #include <sys/proc.h> 1551477Sbostic #include <sys/buf.h> 1651477Sbostic #include <sys/map.h> 1751477Sbostic #include <sys/vnode.h> 1851981Smckusick #include <sys/malloc.h> 1947571Skarels 2051477Sbostic #include <machine/vmparam.h> 2152784Sralph #if defined(tahoe) 2251477Sbostic #include <machine/mtpr.h> 2352784Sralph #endif 2447571Skarels 2551477Sbostic #include <ufs/mfs/mfsnode.h> 2651477Sbostic #include <ufs/mfs/mfsiom.h> 2751477Sbostic #include <ufs/mfs/mfs_extern.h> 2839019Smckusick 2952907Storek #if !defined(hp300) && !defined(i386) && !defined(mips) && !defined(sparc) 3039019Smckusick static int mfsmap_want; /* 1 => need kernel I/O resources */ 3139019Smckusick struct map mfsmap[MFS_MAPSIZE]; 3239019Smckusick extern char mfsiobuf[]; 3341536Smckusick #endif 3439019Smckusick 3539019Smckusick /* 3639019Smckusick * mfs vnode operations. 3739019Smckusick */ 3853534Sheideman int (**mfs_vnodeop_p)(); 3953534Sheideman struct vnodeopv_entry_desc mfs_vnodeop_entries[] = { 4053534Sheideman { &vop_default_desc, vn_default_error }, 4153534Sheideman { &vop_lookup_desc, mfs_lookup }, /* lookup */ 4253534Sheideman { &vop_create_desc, mfs_create }, /* create */ 4353534Sheideman { &vop_mknod_desc, mfs_mknod }, /* mknod */ 4453534Sheideman { &vop_open_desc, mfs_open }, /* open */ 4553534Sheideman { &vop_close_desc, mfs_close }, /* close */ 4653534Sheideman { &vop_access_desc, mfs_access }, /* access */ 4753534Sheideman { &vop_getattr_desc, mfs_getattr }, /* getattr */ 4853534Sheideman { &vop_setattr_desc, mfs_setattr }, /* setattr */ 4953534Sheideman { &vop_read_desc, mfs_read }, /* read */ 5053534Sheideman { &vop_write_desc, mfs_write }, /* write */ 5153534Sheideman { &vop_ioctl_desc, mfs_ioctl }, /* ioctl */ 5253534Sheideman { &vop_select_desc, mfs_select }, /* select */ 5353534Sheideman { &vop_mmap_desc, mfs_mmap }, /* mmap */ 5453534Sheideman { &vop_fsync_desc, mfs_fsync }, /* fsync */ 5553534Sheideman { &vop_seek_desc, mfs_seek }, /* seek */ 5653534Sheideman { &vop_remove_desc, mfs_remove }, /* remove */ 5753534Sheideman { &vop_link_desc, mfs_link }, /* link */ 5853534Sheideman { &vop_rename_desc, mfs_rename }, /* rename */ 5953534Sheideman { &vop_mkdir_desc, mfs_mkdir }, /* mkdir */ 6053534Sheideman { &vop_rmdir_desc, mfs_rmdir }, /* rmdir */ 6153534Sheideman { &vop_symlink_desc, mfs_symlink }, /* symlink */ 6253534Sheideman { &vop_readdir_desc, mfs_readdir }, /* readdir */ 6353534Sheideman { &vop_readlink_desc, mfs_readlink }, /* readlink */ 6453534Sheideman { &vop_abortop_desc, mfs_abortop }, /* abortop */ 6553534Sheideman { &vop_inactive_desc, mfs_inactive }, /* inactive */ 6653534Sheideman { &vop_reclaim_desc, mfs_reclaim }, /* reclaim */ 6753534Sheideman { &vop_lock_desc, mfs_lock }, /* lock */ 6853534Sheideman { &vop_unlock_desc, mfs_unlock }, /* unlock */ 6953534Sheideman { &vop_bmap_desc, mfs_bmap }, /* bmap */ 7053534Sheideman { &vop_strategy_desc, mfs_strategy }, /* strategy */ 7153534Sheideman { &vop_print_desc, mfs_print }, /* print */ 7253534Sheideman { &vop_islocked_desc, mfs_islocked }, /* islocked */ 7353534Sheideman { &vop_advlock_desc, mfs_advlock }, /* advlock */ 7453534Sheideman { &vop_blkatoff_desc, mfs_blkatoff }, /* blkatoff */ 7553534Sheideman { &vop_vget_desc, mfs_vget }, /* vget */ 7653534Sheideman { &vop_valloc_desc, mfs_valloc }, /* valloc */ 7753534Sheideman { &vop_vfree_desc, mfs_vfree }, /* vfree */ 7853534Sheideman { &vop_truncate_desc, mfs_truncate }, /* truncate */ 7953534Sheideman { &vop_update_desc, mfs_update }, /* update */ 8053534Sheideman { &vop_bwrite_desc, mfs_bwrite }, /* bwrite */ 8153534Sheideman { (struct vnodeop_desc*)NULL, (int(*)())NULL } 8239019Smckusick }; 8353534Sheideman struct vnodeopv_desc mfs_vnodeop_opv_desc = 8453534Sheideman { &mfs_vnodeop_p, mfs_vnodeop_entries }; 8539019Smckusick 8639019Smckusick /* 8739019Smckusick * Vnode Operations. 8839019Smckusick * 8939019Smckusick * Open called to allow memory filesystem to initialize and 9039357Smckusick * validate before actual IO. Record our process identifier 9139357Smckusick * so we can tell when we are doing I/O to ourself. 9239019Smckusick */ 9339019Smckusick /* ARGSUSED */ 9451477Sbostic int 9553534Sheideman mfs_open (ap) 9653534Sheideman struct vop_open_args *ap; 9739019Smckusick { 9839019Smckusick 99*53594Sheideman if (ap->a_vp->v_type != VBLK) { 10039019Smckusick panic("mfs_ioctl not VBLK"); 10139019Smckusick /* NOTREACHED */ 10239019Smckusick } 10339019Smckusick return (0); 10439019Smckusick } 10539019Smckusick 10639019Smckusick /* 10739019Smckusick * Ioctl operation. 10839019Smckusick */ 10939019Smckusick /* ARGSUSED */ 11051477Sbostic int 11153534Sheideman mfs_ioctl (ap) 11253534Sheideman struct vop_ioctl_args *ap; 11339019Smckusick { 11439019Smckusick 11553043Sralph return (ENOTTY); 11639019Smckusick } 11739019Smckusick 11839019Smckusick /* 11939019Smckusick * Pass I/O requests to the memory filesystem process. 12039019Smckusick */ 12151477Sbostic int 12253534Sheideman mfs_strategy (ap) 12353534Sheideman struct vop_strategy_args *ap; 12439019Smckusick { 12539822Smckusick register struct mfsnode *mfsp; 12639822Smckusick struct vnode *vp; 12748038Smckusick struct proc *p = curproc; /* XXX */ 12839019Smckusick 129*53594Sheideman if (vfinddev(ap->a_bp->b_dev, VBLK, &vp) || vp->v_usecount == 0) 13039822Smckusick panic("mfs_strategy: bad dev"); 13139822Smckusick mfsp = VTOMFS(vp); 13253043Sralph /* check for mini-root access */ 13353043Sralph if (mfsp->mfs_pid == 0) { 13453043Sralph caddr_t base; 13553043Sralph 136*53594Sheideman base = mfsp->mfs_baseoff + (ap->a_bp->b_blkno << DEV_BSHIFT); 137*53594Sheideman if (ap->a_bp->b_flags & B_READ) 138*53594Sheideman bcopy(base, ap->a_bp->b_un.b_addr, ap->a_bp->b_bcount); 13953043Sralph else 140*53594Sheideman bcopy(ap->a_bp->b_un.b_addr, base, ap->a_bp->b_bcount); 141*53594Sheideman biodone(ap->a_bp); 14253043Sralph } else if (mfsp->mfs_pid == p->p_pid) { 143*53594Sheideman mfs_doio(ap->a_bp, mfsp->mfs_baseoff); 14439019Smckusick } else { 145*53594Sheideman ap->a_bp->av_forw = mfsp->mfs_buflist; 146*53594Sheideman mfsp->mfs_buflist = ap->a_bp; 14739822Smckusick wakeup((caddr_t)vp); 14839019Smckusick } 14939357Smckusick return (0); 15039019Smckusick } 15139019Smckusick 15243412Shibler #if defined(vax) || defined(tahoe) 15339019Smckusick /* 15439019Smckusick * Memory file system I/O. 15539019Smckusick * 15639019Smckusick * Essentially play ubasetup() and disk interrupt service routine by 15739019Smckusick * doing the copies to or from the memfs process. If doing physio 15839019Smckusick * (i.e. pagein), we must map the I/O through the kernel virtual 15939019Smckusick * address space. 16039019Smckusick */ 16151477Sbostic void 16239019Smckusick mfs_doio(bp, base) 16339019Smckusick register struct buf *bp; 16439019Smckusick caddr_t base; 16539019Smckusick { 16639019Smckusick register struct pte *pte, *ppte; 16739019Smckusick register caddr_t vaddr; 16839019Smckusick int off, npf, npf2, reg; 16939019Smckusick caddr_t kernaddr, offset; 17039019Smckusick 17139019Smckusick /* 17239019Smckusick * For phys I/O, map the b_addr into kernel virtual space using 17339019Smckusick * the Mfsiomap pte's. 17439019Smckusick */ 17539019Smckusick if ((bp->b_flags & B_PHYS) == 0) { 17639019Smckusick kernaddr = bp->b_un.b_addr; 17739019Smckusick } else { 17839019Smckusick if (bp->b_flags & (B_PAGET | B_UAREA | B_DIRTY)) 17939019Smckusick panic("swap on memfs?"); 18039019Smckusick off = (int)bp->b_un.b_addr & PGOFSET; 18139019Smckusick npf = btoc(bp->b_bcount + off); 18239019Smckusick /* 18339019Smckusick * Get some mapping page table entries 18439019Smckusick */ 18539019Smckusick while ((reg = rmalloc(mfsmap, (long)npf)) == 0) { 18639019Smckusick mfsmap_want++; 18739019Smckusick sleep((caddr_t)&mfsmap_want, PZERO-1); 18839019Smckusick } 18939019Smckusick reg--; 19039019Smckusick pte = vtopte(bp->b_proc, btop(bp->b_un.b_addr)); 19139019Smckusick /* 19239019Smckusick * Do vmaccess() but with the Mfsiomap page table. 19339019Smckusick */ 19439019Smckusick ppte = &Mfsiomap[reg]; 19539019Smckusick vaddr = &mfsiobuf[reg * NBPG]; 19639019Smckusick kernaddr = vaddr + off; 19739019Smckusick for (npf2 = npf; npf2; npf2--) { 19839019Smckusick mapin(ppte, (u_int)vaddr, pte->pg_pfnum, 19939019Smckusick (int)(PG_V|PG_KW)); 20039019Smckusick #if defined(tahoe) 20139019Smckusick if ((bp->b_flags & B_READ) == 0) 20239019Smckusick mtpr(P1DC, vaddr); 20339019Smckusick #endif 20439019Smckusick ppte++; 20539019Smckusick pte++; 20639019Smckusick vaddr += NBPG; 20739019Smckusick } 20839019Smckusick } 20939019Smckusick offset = base + (bp->b_blkno << DEV_BSHIFT); 21039019Smckusick if (bp->b_flags & B_READ) 21139019Smckusick bp->b_error = copyin(offset, kernaddr, bp->b_bcount); 21239019Smckusick else 21339019Smckusick bp->b_error = copyout(kernaddr, offset, bp->b_bcount); 21439019Smckusick if (bp->b_error) 21539019Smckusick bp->b_flags |= B_ERROR; 21639019Smckusick /* 21739019Smckusick * Release pte's used by physical I/O. 21839019Smckusick */ 21939019Smckusick if (bp->b_flags & B_PHYS) { 22039019Smckusick rmfree(mfsmap, (long)npf, (long)++reg); 22139019Smckusick if (mfsmap_want) { 22239019Smckusick mfsmap_want = 0; 22339019Smckusick wakeup((caddr_t)&mfsmap_want); 22439019Smckusick } 22539019Smckusick } 22639019Smckusick biodone(bp); 22739019Smckusick } 22843412Shibler #endif /* vax || tahoe */ 22939019Smckusick 23052907Storek #if defined(hp300) || defined(i386) || defined(mips) || defined(sparc) 23139019Smckusick /* 23243412Shibler * Memory file system I/O. 23343412Shibler * 23443412Shibler * Trivial on the HP since buffer has already been mapping into KVA space. 23543412Shibler */ 23651477Sbostic void 23743412Shibler mfs_doio(bp, base) 23843412Shibler register struct buf *bp; 23943412Shibler caddr_t base; 24043412Shibler { 24153043Sralph 24243412Shibler base += (bp->b_blkno << DEV_BSHIFT); 24343412Shibler if (bp->b_flags & B_READ) 24443412Shibler bp->b_error = copyin(base, bp->b_un.b_addr, bp->b_bcount); 24543412Shibler else 24643412Shibler bp->b_error = copyout(bp->b_un.b_addr, base, bp->b_bcount); 24743412Shibler if (bp->b_error) 24843412Shibler bp->b_flags |= B_ERROR; 24943412Shibler biodone(bp); 25043412Shibler } 25143412Shibler #endif 25243412Shibler 25343412Shibler /* 25439680Smckusick * This is a noop, simply returning what one has been given. 25539680Smckusick */ 25651477Sbostic int 25753534Sheideman mfs_bmap (ap) 25853534Sheideman struct vop_bmap_args *ap; 25939680Smckusick { 26039680Smckusick 261*53594Sheideman if (ap->a_vpp != NULL) 262*53594Sheideman *ap->a_vpp = ap->a_vp; 263*53594Sheideman if (ap->a_bnp != NULL) 264*53594Sheideman *ap->a_bnp = ap->a_bn; 26539680Smckusick return (0); 26639680Smckusick } 26739680Smckusick 26839680Smckusick /* 26939019Smckusick * Memory filesystem close routine 27039019Smckusick */ 27139019Smckusick /* ARGSUSED */ 27251477Sbostic int 27353534Sheideman mfs_close (ap) 27453534Sheideman struct vop_close_args *ap; 27539019Smckusick { 276*53594Sheideman register struct mfsnode *mfsp = VTOMFS(ap->a_vp); 27739438Smckusick register struct buf *bp; 27839019Smckusick 27939019Smckusick /* 28039438Smckusick * Finish any pending I/O requests. 28139438Smckusick */ 28239438Smckusick while (bp = mfsp->mfs_buflist) { 28339438Smckusick mfsp->mfs_buflist = bp->av_forw; 28439438Smckusick mfs_doio(bp, mfsp->mfs_baseoff); 28539438Smckusick wakeup((caddr_t)bp); 28639438Smckusick } 28739438Smckusick /* 28839019Smckusick * On last close of a memory filesystem 28939019Smckusick * we must invalidate any in core blocks, so that 29039019Smckusick * we can, free up its vnode. 29139019Smckusick */ 292*53594Sheideman vflushbuf(ap->a_vp, 0); 293*53594Sheideman if (vinvalbuf(ap->a_vp, 1)) 29439019Smckusick return (0); 29539019Smckusick /* 29639438Smckusick * There should be no way to have any more uses of this 29739438Smckusick * vnode, so if we find any other uses, it is a panic. 29839019Smckusick */ 299*53594Sheideman if (ap->a_vp->v_usecount > 1) 300*53594Sheideman printf("mfs_close: ref count %d > 1\n", ap->a_vp->v_usecount); 301*53594Sheideman if (ap->a_vp->v_usecount > 1 || mfsp->mfs_buflist) 30239438Smckusick panic("mfs_close"); 30339019Smckusick /* 30439019Smckusick * Send a request to the filesystem server to exit. 30539019Smckusick */ 30639389Smckusick mfsp->mfs_buflist = (struct buf *)(-1); 307*53594Sheideman wakeup((caddr_t)ap->a_vp); 30839019Smckusick return (0); 30939019Smckusick } 31039019Smckusick 31139019Smckusick /* 31239389Smckusick * Memory filesystem inactive routine 31339389Smckusick */ 31439389Smckusick /* ARGSUSED */ 31551477Sbostic int 31653534Sheideman mfs_inactive (ap) 31753534Sheideman struct vop_inactive_args *ap; 31839389Smckusick { 319*53594Sheideman register struct mfsnode *mfsp = VTOMFS(ap->a_vp); 32039389Smckusick 32153197Sralph if (mfsp->mfs_buflist && mfsp->mfs_buflist != (struct buf *)(-1)) 32253197Sralph panic("mfs_inactive: not inactive (mfs_buflist %x)", 32353197Sralph mfsp->mfs_buflist); 32439389Smckusick return (0); 32539389Smckusick } 32639389Smckusick 32739389Smckusick /* 32851981Smckusick * Reclaim a memory filesystem devvp so that it can be reused. 32951981Smckusick */ 33051981Smckusick int 33153534Sheideman mfs_reclaim (ap) 33253534Sheideman struct vop_reclaim_args *ap; 33351981Smckusick { 33451981Smckusick 335*53594Sheideman FREE(ap->a_vp->v_data, M_MFSNODE); 336*53594Sheideman ap->a_vp->v_data = NULL; 33751981Smckusick return (0); 33851981Smckusick } 33951981Smckusick 34051981Smckusick /* 34139677Smckusick * Print out the contents of an mfsnode. 34239677Smckusick */ 34351477Sbostic int 34453534Sheideman mfs_print (ap) 34553534Sheideman struct vop_print_args *ap; 34639677Smckusick { 347*53594Sheideman register struct mfsnode *mfsp = VTOMFS(ap->a_vp); 34839677Smckusick 34939677Smckusick printf("tag VT_MFS, pid %d, base %d, size %d\n", mfsp->mfs_pid, 35039677Smckusick mfsp->mfs_baseoff, mfsp->mfs_size); 35151477Sbostic return (0); 35239677Smckusick } 35339677Smckusick 35439677Smckusick /* 35539019Smckusick * Block device bad operation 35639019Smckusick */ 35751477Sbostic int 35839019Smckusick mfs_badop() 35939019Smckusick { 36039019Smckusick 36139389Smckusick panic("mfs_badop called\n"); 36239389Smckusick /* NOTREACHED */ 36339019Smckusick } 36439019Smckusick 36539019Smckusick /* 36639019Smckusick * Memory based filesystem initialization. 36739019Smckusick */ 36839019Smckusick mfs_init() 36939019Smckusick { 37039019Smckusick 37152907Storek #if !defined(hp300) && !defined(i386) && !defined(mips) && !defined(sparc) 37239019Smckusick rminit(mfsmap, (long)MFS_MAPREG, (long)1, "mfs mapreg", MFS_MAPSIZE); 37341536Smckusick #endif 37439019Smckusick } 375