139019Smckusick /* 239019Smckusick * Copyright (c) 1989 The Regents of the University of California. 339019Smckusick * All rights reserved. 439019Smckusick * 544536Sbostic * %sccs.include.redist.c% 639019Smckusick * 7*53534Sheideman * @(#)mfs_vnops.c 7.31 (Berkeley) 05/14/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 */ 38*53534Sheideman int (**mfs_vnodeop_p)(); 39*53534Sheideman struct vnodeopv_entry_desc mfs_vnodeop_entries[] = { 40*53534Sheideman { &vop_default_desc, vn_default_error }, 41*53534Sheideman { &vop_lookup_desc, mfs_lookup }, /* lookup */ 42*53534Sheideman { &vop_create_desc, mfs_create }, /* create */ 43*53534Sheideman { &vop_mknod_desc, mfs_mknod }, /* mknod */ 44*53534Sheideman { &vop_open_desc, mfs_open }, /* open */ 45*53534Sheideman { &vop_close_desc, mfs_close }, /* close */ 46*53534Sheideman { &vop_access_desc, mfs_access }, /* access */ 47*53534Sheideman { &vop_getattr_desc, mfs_getattr }, /* getattr */ 48*53534Sheideman { &vop_setattr_desc, mfs_setattr }, /* setattr */ 49*53534Sheideman { &vop_read_desc, mfs_read }, /* read */ 50*53534Sheideman { &vop_write_desc, mfs_write }, /* write */ 51*53534Sheideman { &vop_ioctl_desc, mfs_ioctl }, /* ioctl */ 52*53534Sheideman { &vop_select_desc, mfs_select }, /* select */ 53*53534Sheideman { &vop_mmap_desc, mfs_mmap }, /* mmap */ 54*53534Sheideman { &vop_fsync_desc, mfs_fsync }, /* fsync */ 55*53534Sheideman { &vop_seek_desc, mfs_seek }, /* seek */ 56*53534Sheideman { &vop_remove_desc, mfs_remove }, /* remove */ 57*53534Sheideman { &vop_link_desc, mfs_link }, /* link */ 58*53534Sheideman { &vop_rename_desc, mfs_rename }, /* rename */ 59*53534Sheideman { &vop_mkdir_desc, mfs_mkdir }, /* mkdir */ 60*53534Sheideman { &vop_rmdir_desc, mfs_rmdir }, /* rmdir */ 61*53534Sheideman { &vop_symlink_desc, mfs_symlink }, /* symlink */ 62*53534Sheideman { &vop_readdir_desc, mfs_readdir }, /* readdir */ 63*53534Sheideman { &vop_readlink_desc, mfs_readlink }, /* readlink */ 64*53534Sheideman { &vop_abortop_desc, mfs_abortop }, /* abortop */ 65*53534Sheideman { &vop_inactive_desc, mfs_inactive }, /* inactive */ 66*53534Sheideman { &vop_reclaim_desc, mfs_reclaim }, /* reclaim */ 67*53534Sheideman { &vop_lock_desc, mfs_lock }, /* lock */ 68*53534Sheideman { &vop_unlock_desc, mfs_unlock }, /* unlock */ 69*53534Sheideman { &vop_bmap_desc, mfs_bmap }, /* bmap */ 70*53534Sheideman { &vop_strategy_desc, mfs_strategy }, /* strategy */ 71*53534Sheideman { &vop_print_desc, mfs_print }, /* print */ 72*53534Sheideman { &vop_islocked_desc, mfs_islocked }, /* islocked */ 73*53534Sheideman { &vop_advlock_desc, mfs_advlock }, /* advlock */ 74*53534Sheideman { &vop_blkatoff_desc, mfs_blkatoff }, /* blkatoff */ 75*53534Sheideman { &vop_vget_desc, mfs_vget }, /* vget */ 76*53534Sheideman { &vop_valloc_desc, mfs_valloc }, /* valloc */ 77*53534Sheideman { &vop_vfree_desc, mfs_vfree }, /* vfree */ 78*53534Sheideman { &vop_truncate_desc, mfs_truncate }, /* truncate */ 79*53534Sheideman { &vop_update_desc, mfs_update }, /* update */ 80*53534Sheideman { &vop_bwrite_desc, mfs_bwrite }, /* bwrite */ 81*53534Sheideman { (struct vnodeop_desc*)NULL, (int(*)())NULL } 8239019Smckusick }; 83*53534Sheideman struct vnodeopv_desc mfs_vnodeop_opv_desc = 84*53534Sheideman { &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 95*53534Sheideman mfs_open (ap) 96*53534Sheideman struct vop_open_args *ap; 97*53534Sheideman #define vp (ap->a_vp) 98*53534Sheideman #define mode (ap->a_mode) 99*53534Sheideman #define cred (ap->a_cred) 100*53534Sheideman #define p (ap->a_p) 10139019Smckusick { 10239019Smckusick 10339019Smckusick if (vp->v_type != VBLK) { 10439019Smckusick panic("mfs_ioctl not VBLK"); 10539019Smckusick /* NOTREACHED */ 10639019Smckusick } 10739019Smckusick return (0); 10839019Smckusick } 109*53534Sheideman #undef vp 110*53534Sheideman #undef mode 111*53534Sheideman #undef cred 112*53534Sheideman #undef p 11339019Smckusick 11439019Smckusick /* 11539019Smckusick * Ioctl operation. 11639019Smckusick */ 11739019Smckusick /* ARGSUSED */ 11851477Sbostic int 119*53534Sheideman mfs_ioctl (ap) 120*53534Sheideman struct vop_ioctl_args *ap; 121*53534Sheideman #define vp (ap->a_vp) 122*53534Sheideman #define com (ap->a_command) 123*53534Sheideman #define data (ap->a_data) 124*53534Sheideman #define fflag (ap->a_fflag) 125*53534Sheideman #define cred (ap->a_cred) 126*53534Sheideman #define p (ap->a_p) 12739019Smckusick { 12839019Smckusick 12953043Sralph return (ENOTTY); 13039019Smckusick } 131*53534Sheideman #undef vp 132*53534Sheideman #undef com 133*53534Sheideman #undef data 134*53534Sheideman #undef fflag 135*53534Sheideman #undef cred 136*53534Sheideman #undef p 13739019Smckusick 13839019Smckusick /* 13939019Smckusick * Pass I/O requests to the memory filesystem process. 14039019Smckusick */ 14151477Sbostic int 142*53534Sheideman mfs_strategy (ap) 143*53534Sheideman struct vop_strategy_args *ap; 144*53534Sheideman #define bp (ap->a_bp) 14539019Smckusick { 14639822Smckusick register struct mfsnode *mfsp; 14739822Smckusick struct vnode *vp; 14848038Smckusick struct proc *p = curproc; /* XXX */ 14939019Smckusick 15039822Smckusick if (vfinddev(bp->b_dev, VBLK, &vp) || vp->v_usecount == 0) 15139822Smckusick panic("mfs_strategy: bad dev"); 15239822Smckusick mfsp = VTOMFS(vp); 15353043Sralph /* check for mini-root access */ 15453043Sralph if (mfsp->mfs_pid == 0) { 15553043Sralph caddr_t base; 15653043Sralph 15753043Sralph base = mfsp->mfs_baseoff + (bp->b_blkno << DEV_BSHIFT); 15853043Sralph if (bp->b_flags & B_READ) 15953043Sralph bcopy(base, bp->b_un.b_addr, bp->b_bcount); 16053043Sralph else 16153043Sralph bcopy(bp->b_un.b_addr, base, bp->b_bcount); 16253043Sralph biodone(bp); 16353043Sralph } else if (mfsp->mfs_pid == p->p_pid) { 16439389Smckusick mfs_doio(bp, mfsp->mfs_baseoff); 16539019Smckusick } else { 16639389Smckusick bp->av_forw = mfsp->mfs_buflist; 16739389Smckusick mfsp->mfs_buflist = bp; 16839822Smckusick wakeup((caddr_t)vp); 16939019Smckusick } 17039357Smckusick return (0); 17139019Smckusick } 172*53534Sheideman #undef bp 17339019Smckusick 17443412Shibler #if defined(vax) || defined(tahoe) 17539019Smckusick /* 17639019Smckusick * Memory file system I/O. 17739019Smckusick * 17839019Smckusick * Essentially play ubasetup() and disk interrupt service routine by 17939019Smckusick * doing the copies to or from the memfs process. If doing physio 18039019Smckusick * (i.e. pagein), we must map the I/O through the kernel virtual 18139019Smckusick * address space. 18239019Smckusick */ 18351477Sbostic void 18439019Smckusick mfs_doio(bp, base) 18539019Smckusick register struct buf *bp; 18639019Smckusick caddr_t base; 18739019Smckusick { 18839019Smckusick register struct pte *pte, *ppte; 18939019Smckusick register caddr_t vaddr; 19039019Smckusick int off, npf, npf2, reg; 19139019Smckusick caddr_t kernaddr, offset; 19239019Smckusick 19339019Smckusick /* 19439019Smckusick * For phys I/O, map the b_addr into kernel virtual space using 19539019Smckusick * the Mfsiomap pte's. 19639019Smckusick */ 19739019Smckusick if ((bp->b_flags & B_PHYS) == 0) { 19839019Smckusick kernaddr = bp->b_un.b_addr; 19939019Smckusick } else { 20039019Smckusick if (bp->b_flags & (B_PAGET | B_UAREA | B_DIRTY)) 20139019Smckusick panic("swap on memfs?"); 20239019Smckusick off = (int)bp->b_un.b_addr & PGOFSET; 20339019Smckusick npf = btoc(bp->b_bcount + off); 20439019Smckusick /* 20539019Smckusick * Get some mapping page table entries 20639019Smckusick */ 20739019Smckusick while ((reg = rmalloc(mfsmap, (long)npf)) == 0) { 20839019Smckusick mfsmap_want++; 20939019Smckusick sleep((caddr_t)&mfsmap_want, PZERO-1); 21039019Smckusick } 21139019Smckusick reg--; 21239019Smckusick pte = vtopte(bp->b_proc, btop(bp->b_un.b_addr)); 21339019Smckusick /* 21439019Smckusick * Do vmaccess() but with the Mfsiomap page table. 21539019Smckusick */ 21639019Smckusick ppte = &Mfsiomap[reg]; 21739019Smckusick vaddr = &mfsiobuf[reg * NBPG]; 21839019Smckusick kernaddr = vaddr + off; 21939019Smckusick for (npf2 = npf; npf2; npf2--) { 22039019Smckusick mapin(ppte, (u_int)vaddr, pte->pg_pfnum, 22139019Smckusick (int)(PG_V|PG_KW)); 22239019Smckusick #if defined(tahoe) 22339019Smckusick if ((bp->b_flags & B_READ) == 0) 22439019Smckusick mtpr(P1DC, vaddr); 22539019Smckusick #endif 22639019Smckusick ppte++; 22739019Smckusick pte++; 22839019Smckusick vaddr += NBPG; 22939019Smckusick } 23039019Smckusick } 23139019Smckusick offset = base + (bp->b_blkno << DEV_BSHIFT); 23239019Smckusick if (bp->b_flags & B_READ) 23339019Smckusick bp->b_error = copyin(offset, kernaddr, bp->b_bcount); 23439019Smckusick else 23539019Smckusick bp->b_error = copyout(kernaddr, offset, bp->b_bcount); 23639019Smckusick if (bp->b_error) 23739019Smckusick bp->b_flags |= B_ERROR; 23839019Smckusick /* 23939019Smckusick * Release pte's used by physical I/O. 24039019Smckusick */ 24139019Smckusick if (bp->b_flags & B_PHYS) { 24239019Smckusick rmfree(mfsmap, (long)npf, (long)++reg); 24339019Smckusick if (mfsmap_want) { 24439019Smckusick mfsmap_want = 0; 24539019Smckusick wakeup((caddr_t)&mfsmap_want); 24639019Smckusick } 24739019Smckusick } 24839019Smckusick biodone(bp); 24939019Smckusick } 25043412Shibler #endif /* vax || tahoe */ 25139019Smckusick 25252907Storek #if defined(hp300) || defined(i386) || defined(mips) || defined(sparc) 25339019Smckusick /* 25443412Shibler * Memory file system I/O. 25543412Shibler * 25643412Shibler * Trivial on the HP since buffer has already been mapping into KVA space. 25743412Shibler */ 25851477Sbostic void 25943412Shibler mfs_doio(bp, base) 26043412Shibler register struct buf *bp; 26143412Shibler caddr_t base; 26243412Shibler { 26353043Sralph 26443412Shibler base += (bp->b_blkno << DEV_BSHIFT); 26543412Shibler if (bp->b_flags & B_READ) 26643412Shibler bp->b_error = copyin(base, bp->b_un.b_addr, bp->b_bcount); 26743412Shibler else 26843412Shibler bp->b_error = copyout(bp->b_un.b_addr, base, bp->b_bcount); 26943412Shibler if (bp->b_error) 27043412Shibler bp->b_flags |= B_ERROR; 27143412Shibler biodone(bp); 27243412Shibler } 27343412Shibler #endif 27443412Shibler 27543412Shibler /* 27639680Smckusick * This is a noop, simply returning what one has been given. 27739680Smckusick */ 27851477Sbostic int 279*53534Sheideman mfs_bmap (ap) 280*53534Sheideman struct vop_bmap_args *ap; 281*53534Sheideman #define vp (ap->a_vp) 282*53534Sheideman #define bn (ap->a_bn) 283*53534Sheideman #define vpp (ap->a_vpp) 284*53534Sheideman #define bnp (ap->a_bnp) 28539680Smckusick { 28639680Smckusick 28739680Smckusick if (vpp != NULL) 28839680Smckusick *vpp = vp; 28939680Smckusick if (bnp != NULL) 29039680Smckusick *bnp = bn; 29139680Smckusick return (0); 29239680Smckusick } 293*53534Sheideman #undef vp 294*53534Sheideman #undef bn 295*53534Sheideman #undef vpp 296*53534Sheideman #undef bnp 29739680Smckusick 29839680Smckusick /* 29939019Smckusick * Memory filesystem close routine 30039019Smckusick */ 30139019Smckusick /* ARGSUSED */ 30251477Sbostic int 303*53534Sheideman mfs_close (ap) 304*53534Sheideman struct vop_close_args *ap; 305*53534Sheideman #define vp (ap->a_vp) 306*53534Sheideman #define flag (ap->a_fflag) 307*53534Sheideman #define cred (ap->a_cred) 308*53534Sheideman #define p (ap->a_p) 30939019Smckusick { 31039389Smckusick register struct mfsnode *mfsp = VTOMFS(vp); 31139438Smckusick register struct buf *bp; 31239019Smckusick 31339019Smckusick /* 31439438Smckusick * Finish any pending I/O requests. 31539438Smckusick */ 31639438Smckusick while (bp = mfsp->mfs_buflist) { 31739438Smckusick mfsp->mfs_buflist = bp->av_forw; 31839438Smckusick mfs_doio(bp, mfsp->mfs_baseoff); 31939438Smckusick wakeup((caddr_t)bp); 32039438Smckusick } 32139438Smckusick /* 32239019Smckusick * On last close of a memory filesystem 32339019Smckusick * we must invalidate any in core blocks, so that 32439019Smckusick * we can, free up its vnode. 32539019Smckusick */ 32639677Smckusick vflushbuf(vp, 0); 32739677Smckusick if (vinvalbuf(vp, 1)) 32839019Smckusick return (0); 32939019Smckusick /* 33039438Smckusick * There should be no way to have any more uses of this 33139438Smckusick * vnode, so if we find any other uses, it is a panic. 33239019Smckusick */ 33339813Smckusick if (vp->v_usecount > 1) 33439813Smckusick printf("mfs_close: ref count %d > 1\n", vp->v_usecount); 33539813Smckusick if (vp->v_usecount > 1 || mfsp->mfs_buflist) 33639438Smckusick panic("mfs_close"); 33739019Smckusick /* 33839019Smckusick * Send a request to the filesystem server to exit. 33939019Smckusick */ 34039389Smckusick mfsp->mfs_buflist = (struct buf *)(-1); 34139019Smckusick wakeup((caddr_t)vp); 34239019Smckusick return (0); 34339019Smckusick } 344*53534Sheideman #undef vp 345*53534Sheideman #undef flag 346*53534Sheideman #undef cred 347*53534Sheideman #undef p 34839019Smckusick 34939019Smckusick /* 35039389Smckusick * Memory filesystem inactive routine 35139389Smckusick */ 35239389Smckusick /* ARGSUSED */ 35351477Sbostic int 354*53534Sheideman mfs_inactive (ap) 355*53534Sheideman struct vop_inactive_args *ap; 356*53534Sheideman #define vp (ap->a_vp) 357*53534Sheideman #define p (ap->a_p) 35839389Smckusick { 35953197Sralph register struct mfsnode *mfsp = VTOMFS(vp); 36039389Smckusick 36153197Sralph if (mfsp->mfs_buflist && mfsp->mfs_buflist != (struct buf *)(-1)) 36253197Sralph panic("mfs_inactive: not inactive (mfs_buflist %x)", 36353197Sralph mfsp->mfs_buflist); 36439389Smckusick return (0); 36539389Smckusick } 366*53534Sheideman #undef vp 367*53534Sheideman #undef p 36839389Smckusick 36939389Smckusick /* 37051981Smckusick * Reclaim a memory filesystem devvp so that it can be reused. 37151981Smckusick */ 37251981Smckusick int 373*53534Sheideman mfs_reclaim (ap) 374*53534Sheideman struct vop_reclaim_args *ap; 375*53534Sheideman #define vp (ap->a_vp) 37651981Smckusick { 37751981Smckusick 37851981Smckusick FREE(vp->v_data, M_MFSNODE); 37951981Smckusick vp->v_data = NULL; 38051981Smckusick return (0); 38151981Smckusick } 382*53534Sheideman #undef vp 38351981Smckusick 38451981Smckusick /* 38539677Smckusick * Print out the contents of an mfsnode. 38639677Smckusick */ 38751477Sbostic int 388*53534Sheideman mfs_print (ap) 389*53534Sheideman struct vop_print_args *ap; 390*53534Sheideman #define vp (ap->a_vp) 39139677Smckusick { 39239677Smckusick register struct mfsnode *mfsp = VTOMFS(vp); 39339677Smckusick 39439677Smckusick printf("tag VT_MFS, pid %d, base %d, size %d\n", mfsp->mfs_pid, 39539677Smckusick mfsp->mfs_baseoff, mfsp->mfs_size); 39651477Sbostic return (0); 39739677Smckusick } 398*53534Sheideman #undef vp 39939677Smckusick 40039677Smckusick /* 40139019Smckusick * Block device bad operation 40239019Smckusick */ 40351477Sbostic int 40439019Smckusick mfs_badop() 40539019Smckusick { 40639019Smckusick 40739389Smckusick panic("mfs_badop called\n"); 40839389Smckusick /* NOTREACHED */ 40939019Smckusick } 41039019Smckusick 41139019Smckusick /* 41239019Smckusick * Memory based filesystem initialization. 41339019Smckusick */ 41439019Smckusick mfs_init() 41539019Smckusick { 41639019Smckusick 41752907Storek #if !defined(hp300) && !defined(i386) && !defined(mips) && !defined(sparc) 41839019Smckusick rminit(mfsmap, (long)MFS_MAPREG, (long)1, "mfs mapreg", MFS_MAPSIZE); 41941536Smckusick #endif 42039019Smckusick } 421