139019Smckusick /* 263375Sbostic * Copyright (c) 1989, 1993 363375Sbostic * The Regents of the University of California. All rights reserved. 439019Smckusick * 544536Sbostic * %sccs.include.redist.c% 639019Smckusick * 7*67408Smckusick * @(#)mfs_vnops.c 8.4 (Berkeley) 06/16/94 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 2055093Spendry #include <miscfs/specfs/specdev.h> 2155093Spendry 2251477Sbostic #include <machine/vmparam.h> 2347571Skarels 2451477Sbostic #include <ufs/mfs/mfsnode.h> 2551477Sbostic #include <ufs/mfs/mfsiom.h> 2651477Sbostic #include <ufs/mfs/mfs_extern.h> 2739019Smckusick 2853875Smckusick #if !defined(hp300) && !defined(i386) && !defined(mips) && !defined(sparc) && !defined(luna68k) 2939019Smckusick static int mfsmap_want; /* 1 => need kernel I/O resources */ 3039019Smckusick struct map mfsmap[MFS_MAPSIZE]; 3139019Smckusick extern char mfsiobuf[]; 3241536Smckusick #endif 3339019Smckusick 3439019Smckusick /* 3539019Smckusick * mfs vnode operations. 3639019Smckusick */ 3753534Sheideman int (**mfs_vnodeop_p)(); 3853534Sheideman struct vnodeopv_entry_desc mfs_vnodeop_entries[] = { 3953534Sheideman { &vop_default_desc, vn_default_error }, 4053534Sheideman { &vop_lookup_desc, mfs_lookup }, /* lookup */ 4153534Sheideman { &vop_create_desc, mfs_create }, /* create */ 4254457Smckusick { &vop_mknod_desc, mfs_mknod }, /* mknod */ 4354457Smckusick { &vop_open_desc, mfs_open }, /* open */ 4454457Smckusick { &vop_close_desc, mfs_close }, /* close */ 4553534Sheideman { &vop_access_desc, mfs_access }, /* access */ 4653534Sheideman { &vop_getattr_desc, mfs_getattr }, /* getattr */ 4753534Sheideman { &vop_setattr_desc, mfs_setattr }, /* setattr */ 4854457Smckusick { &vop_read_desc, mfs_read }, /* read */ 4954457Smckusick { &vop_write_desc, mfs_write }, /* write */ 5054457Smckusick { &vop_ioctl_desc, mfs_ioctl }, /* ioctl */ 5153534Sheideman { &vop_select_desc, mfs_select }, /* select */ 5254457Smckusick { &vop_mmap_desc, mfs_mmap }, /* mmap */ 5354458Smckusick { &vop_fsync_desc, spec_fsync }, /* fsync */ 5454457Smckusick { &vop_seek_desc, mfs_seek }, /* seek */ 5553534Sheideman { &vop_remove_desc, mfs_remove }, /* remove */ 5654457Smckusick { &vop_link_desc, mfs_link }, /* link */ 5753534Sheideman { &vop_rename_desc, mfs_rename }, /* rename */ 5854457Smckusick { &vop_mkdir_desc, mfs_mkdir }, /* mkdir */ 5954457Smckusick { &vop_rmdir_desc, mfs_rmdir }, /* rmdir */ 6053534Sheideman { &vop_symlink_desc, mfs_symlink }, /* symlink */ 6153534Sheideman { &vop_readdir_desc, mfs_readdir }, /* readdir */ 6253534Sheideman { &vop_readlink_desc, mfs_readlink }, /* readlink */ 6353534Sheideman { &vop_abortop_desc, mfs_abortop }, /* abortop */ 6453534Sheideman { &vop_inactive_desc, mfs_inactive }, /* inactive */ 6553534Sheideman { &vop_reclaim_desc, mfs_reclaim }, /* reclaim */ 6654457Smckusick { &vop_lock_desc, mfs_lock }, /* lock */ 6753534Sheideman { &vop_unlock_desc, mfs_unlock }, /* unlock */ 6854457Smckusick { &vop_bmap_desc, mfs_bmap }, /* bmap */ 6953534Sheideman { &vop_strategy_desc, mfs_strategy }, /* strategy */ 7054457Smckusick { &vop_print_desc, mfs_print }, /* print */ 7153534Sheideman { &vop_islocked_desc, mfs_islocked }, /* islocked */ 7264179Smckusick { &vop_pathconf_desc, mfs_pathconf }, /* pathconf */ 7353534Sheideman { &vop_advlock_desc, mfs_advlock }, /* advlock */ 7453534Sheideman { &vop_blkatoff_desc, mfs_blkatoff }, /* blkatoff */ 7553534Sheideman { &vop_valloc_desc, mfs_valloc }, /* valloc */ 7654457Smckusick { &vop_vfree_desc, mfs_vfree }, /* vfree */ 7753534Sheideman { &vop_truncate_desc, mfs_truncate }, /* truncate */ 7853534Sheideman { &vop_update_desc, mfs_update }, /* update */ 7953534Sheideman { &vop_bwrite_desc, mfs_bwrite }, /* bwrite */ 8053534Sheideman { (struct vnodeop_desc*)NULL, (int(*)())NULL } 8139019Smckusick }; 8253534Sheideman struct vnodeopv_desc mfs_vnodeop_opv_desc = 8353534Sheideman { &mfs_vnodeop_p, mfs_vnodeop_entries }; 8439019Smckusick 8539019Smckusick /* 8639019Smckusick * Vnode Operations. 8739019Smckusick * 8839019Smckusick * Open called to allow memory filesystem to initialize and 8939357Smckusick * validate before actual IO. Record our process identifier 9039357Smckusick * so we can tell when we are doing I/O to ourself. 9139019Smckusick */ 9239019Smckusick /* ARGSUSED */ 9351477Sbostic int 9454660Smckusick mfs_open(ap) 9554660Smckusick struct vop_open_args /* { 9654660Smckusick struct vnode *a_vp; 9754660Smckusick int a_mode; 9854660Smckusick struct ucred *a_cred; 9954660Smckusick struct proc *a_p; 10054660Smckusick } */ *ap; 10139019Smckusick { 10239019Smckusick 10353594Sheideman if (ap->a_vp->v_type != VBLK) { 10439019Smckusick panic("mfs_ioctl not VBLK"); 10539019Smckusick /* NOTREACHED */ 10639019Smckusick } 10739019Smckusick return (0); 10839019Smckusick } 10939019Smckusick 11039019Smckusick /* 11139019Smckusick * Ioctl operation. 11239019Smckusick */ 11339019Smckusick /* ARGSUSED */ 11451477Sbostic int 11554660Smckusick mfs_ioctl(ap) 11654660Smckusick struct vop_ioctl_args /* { 11754660Smckusick struct vnode *a_vp; 11854660Smckusick int a_command; 11954660Smckusick caddr_t a_data; 12054660Smckusick int a_fflag; 12154660Smckusick struct ucred *a_cred; 12254660Smckusick struct proc *a_p; 12354660Smckusick } */ *ap; 12439019Smckusick { 12539019Smckusick 12653043Sralph return (ENOTTY); 12739019Smckusick } 12839019Smckusick 12939019Smckusick /* 13039019Smckusick * Pass I/O requests to the memory filesystem process. 13139019Smckusick */ 13251477Sbostic int 13354660Smckusick mfs_strategy(ap) 13454660Smckusick struct vop_strategy_args /* { 13554660Smckusick struct buf *a_bp; 13654660Smckusick } */ *ap; 13739019Smckusick { 13853866Sheideman register struct buf *bp = ap->a_bp; 13939822Smckusick register struct mfsnode *mfsp; 14039822Smckusick struct vnode *vp; 14148038Smckusick struct proc *p = curproc; /* XXX */ 14239019Smckusick 14359485Smckusick if (!vfinddev(bp->b_dev, VBLK, &vp) || vp->v_usecount == 0) 14439822Smckusick panic("mfs_strategy: bad dev"); 14539822Smckusick mfsp = VTOMFS(vp); 14653043Sralph /* check for mini-root access */ 14753043Sralph if (mfsp->mfs_pid == 0) { 14853043Sralph caddr_t base; 14953043Sralph 15053866Sheideman base = mfsp->mfs_baseoff + (bp->b_blkno << DEV_BSHIFT); 15153866Sheideman if (bp->b_flags & B_READ) 15264545Sbostic bcopy(base, bp->b_data, bp->b_bcount); 15353043Sralph else 15464545Sbostic bcopy(bp->b_data, base, bp->b_bcount); 15553866Sheideman biodone(bp); 15653043Sralph } else if (mfsp->mfs_pid == p->p_pid) { 15753866Sheideman mfs_doio(bp, mfsp->mfs_baseoff); 15839019Smckusick } else { 15956392Smckusick bp->b_actf = mfsp->mfs_buflist; 16053866Sheideman mfsp->mfs_buflist = bp; 16139822Smckusick wakeup((caddr_t)vp); 16239019Smckusick } 16339357Smckusick return (0); 16439019Smckusick } 16539019Smckusick 16643412Shibler #if defined(vax) || defined(tahoe) 16739019Smckusick /* 16839019Smckusick * Memory file system I/O. 16939019Smckusick * 17039019Smckusick * Essentially play ubasetup() and disk interrupt service routine by 17139019Smckusick * doing the copies to or from the memfs process. If doing physio 17239019Smckusick * (i.e. pagein), we must map the I/O through the kernel virtual 17339019Smckusick * address space. 17439019Smckusick */ 17551477Sbostic void 17639019Smckusick mfs_doio(bp, base) 17739019Smckusick register struct buf *bp; 17839019Smckusick caddr_t base; 17939019Smckusick { 18039019Smckusick register struct pte *pte, *ppte; 18139019Smckusick register caddr_t vaddr; 18239019Smckusick int off, npf, npf2, reg; 18339019Smckusick caddr_t kernaddr, offset; 18439019Smckusick 18539019Smckusick /* 18664545Sbostic * For phys I/O, map the b_data into kernel virtual space using 18739019Smckusick * the Mfsiomap pte's. 18839019Smckusick */ 18939019Smckusick if ((bp->b_flags & B_PHYS) == 0) { 19064545Sbostic kernaddr = bp->b_data; 19139019Smckusick } else { 19239019Smckusick if (bp->b_flags & (B_PAGET | B_UAREA | B_DIRTY)) 19339019Smckusick panic("swap on memfs?"); 19464545Sbostic off = (int)bp->b_data & PGOFSET; 19539019Smckusick npf = btoc(bp->b_bcount + off); 19639019Smckusick /* 19739019Smckusick * Get some mapping page table entries 19839019Smckusick */ 19939019Smckusick while ((reg = rmalloc(mfsmap, (long)npf)) == 0) { 20039019Smckusick mfsmap_want++; 20139019Smckusick sleep((caddr_t)&mfsmap_want, PZERO-1); 20239019Smckusick } 20339019Smckusick reg--; 20464545Sbostic pte = vtopte(bp->b_proc, btop(bp->b_data)); 20539019Smckusick /* 20639019Smckusick * Do vmaccess() but with the Mfsiomap page table. 20739019Smckusick */ 20839019Smckusick ppte = &Mfsiomap[reg]; 20939019Smckusick vaddr = &mfsiobuf[reg * NBPG]; 21039019Smckusick kernaddr = vaddr + off; 21139019Smckusick for (npf2 = npf; npf2; npf2--) { 21239019Smckusick mapin(ppte, (u_int)vaddr, pte->pg_pfnum, 21339019Smckusick (int)(PG_V|PG_KW)); 21439019Smckusick #if defined(tahoe) 21539019Smckusick if ((bp->b_flags & B_READ) == 0) 21639019Smckusick mtpr(P1DC, vaddr); 21739019Smckusick #endif 21839019Smckusick ppte++; 21939019Smckusick pte++; 22039019Smckusick vaddr += NBPG; 22139019Smckusick } 22239019Smckusick } 22339019Smckusick offset = base + (bp->b_blkno << DEV_BSHIFT); 22439019Smckusick if (bp->b_flags & B_READ) 22539019Smckusick bp->b_error = copyin(offset, kernaddr, bp->b_bcount); 22639019Smckusick else 22739019Smckusick bp->b_error = copyout(kernaddr, offset, bp->b_bcount); 22839019Smckusick if (bp->b_error) 22939019Smckusick bp->b_flags |= B_ERROR; 23039019Smckusick /* 23139019Smckusick * Release pte's used by physical I/O. 23239019Smckusick */ 23339019Smckusick if (bp->b_flags & B_PHYS) { 23439019Smckusick rmfree(mfsmap, (long)npf, (long)++reg); 23539019Smckusick if (mfsmap_want) { 23639019Smckusick mfsmap_want = 0; 23739019Smckusick wakeup((caddr_t)&mfsmap_want); 23839019Smckusick } 23939019Smckusick } 24039019Smckusick biodone(bp); 24139019Smckusick } 24243412Shibler #endif /* vax || tahoe */ 24339019Smckusick 24453875Smckusick #if defined(hp300) || defined(i386) || defined(mips) || defined(sparc) || defined(luna68k) 24539019Smckusick /* 24643412Shibler * Memory file system I/O. 24743412Shibler * 24843412Shibler * Trivial on the HP since buffer has already been mapping into KVA space. 24943412Shibler */ 25051477Sbostic void 25143412Shibler mfs_doio(bp, base) 25243412Shibler register struct buf *bp; 25343412Shibler caddr_t base; 25443412Shibler { 25553043Sralph 25643412Shibler base += (bp->b_blkno << DEV_BSHIFT); 25743412Shibler if (bp->b_flags & B_READ) 25864545Sbostic bp->b_error = copyin(base, bp->b_data, bp->b_bcount); 25943412Shibler else 26064545Sbostic bp->b_error = copyout(bp->b_data, base, bp->b_bcount); 26143412Shibler if (bp->b_error) 26243412Shibler bp->b_flags |= B_ERROR; 26343412Shibler biodone(bp); 26443412Shibler } 26543412Shibler #endif 26643412Shibler 26743412Shibler /* 26839680Smckusick * This is a noop, simply returning what one has been given. 26939680Smckusick */ 27051477Sbostic int 27154660Smckusick mfs_bmap(ap) 27254660Smckusick struct vop_bmap_args /* { 27354660Smckusick struct vnode *a_vp; 27454660Smckusick daddr_t a_bn; 27554660Smckusick struct vnode **a_vpp; 27654660Smckusick daddr_t *a_bnp; 27756454Smargo int *a_runp; 27854660Smckusick } */ *ap; 27939680Smckusick { 28039680Smckusick 28153594Sheideman if (ap->a_vpp != NULL) 28253594Sheideman *ap->a_vpp = ap->a_vp; 28353594Sheideman if (ap->a_bnp != NULL) 28453594Sheideman *ap->a_bnp = ap->a_bn; 28539680Smckusick return (0); 28639680Smckusick } 28739680Smckusick 28839680Smckusick /* 28939019Smckusick * Memory filesystem close routine 29039019Smckusick */ 29139019Smckusick /* ARGSUSED */ 29251477Sbostic int 29354660Smckusick mfs_close(ap) 29454660Smckusick struct vop_close_args /* { 29554660Smckusick struct vnode *a_vp; 29654660Smckusick int a_fflag; 29754660Smckusick struct ucred *a_cred; 29854660Smckusick struct proc *a_p; 29954660Smckusick } */ *ap; 30039019Smckusick { 30153866Sheideman register struct vnode *vp = ap->a_vp; 30253866Sheideman register struct mfsnode *mfsp = VTOMFS(vp); 30339438Smckusick register struct buf *bp; 30454458Smckusick int error; 30539019Smckusick 30639019Smckusick /* 30739438Smckusick * Finish any pending I/O requests. 30839438Smckusick */ 30939438Smckusick while (bp = mfsp->mfs_buflist) { 31056392Smckusick mfsp->mfs_buflist = bp->b_actf; 31139438Smckusick mfs_doio(bp, mfsp->mfs_baseoff); 31239438Smckusick wakeup((caddr_t)bp); 31339438Smckusick } 31439438Smckusick /* 31539019Smckusick * On last close of a memory filesystem 31639019Smckusick * we must invalidate any in core blocks, so that 31739019Smckusick * we can, free up its vnode. 31839019Smckusick */ 31957807Smckusick if (error = vinvalbuf(vp, 1, ap->a_cred, ap->a_p, 0, 0)) 32054458Smckusick return (error); 32139019Smckusick /* 32239438Smckusick * There should be no way to have any more uses of this 32339438Smckusick * vnode, so if we find any other uses, it is a panic. 32439019Smckusick */ 32553866Sheideman if (vp->v_usecount > 1) 32653866Sheideman printf("mfs_close: ref count %d > 1\n", vp->v_usecount); 32753866Sheideman if (vp->v_usecount > 1 || mfsp->mfs_buflist) 32839438Smckusick panic("mfs_close"); 32939019Smckusick /* 33039019Smckusick * Send a request to the filesystem server to exit. 33139019Smckusick */ 33239389Smckusick mfsp->mfs_buflist = (struct buf *)(-1); 33353866Sheideman wakeup((caddr_t)vp); 33439019Smckusick return (0); 33539019Smckusick } 33639019Smckusick 33739019Smckusick /* 33839389Smckusick * Memory filesystem inactive routine 33939389Smckusick */ 34039389Smckusick /* ARGSUSED */ 34151477Sbostic int 34254660Smckusick mfs_inactive(ap) 34354660Smckusick struct vop_inactive_args /* { 34454660Smckusick struct vnode *a_vp; 34554660Smckusick } */ *ap; 34639389Smckusick { 34753594Sheideman register struct mfsnode *mfsp = VTOMFS(ap->a_vp); 34839389Smckusick 34953197Sralph if (mfsp->mfs_buflist && mfsp->mfs_buflist != (struct buf *)(-1)) 35053197Sralph panic("mfs_inactive: not inactive (mfs_buflist %x)", 35153197Sralph mfsp->mfs_buflist); 35239389Smckusick return (0); 35339389Smckusick } 35439389Smckusick 35539389Smckusick /* 35651981Smckusick * Reclaim a memory filesystem devvp so that it can be reused. 35751981Smckusick */ 35851981Smckusick int 35954660Smckusick mfs_reclaim(ap) 36054660Smckusick struct vop_reclaim_args /* { 36154660Smckusick struct vnode *a_vp; 36254660Smckusick } */ *ap; 36351981Smckusick { 364*67408Smckusick register struct vnode *vp = ap->a_vp; 36551981Smckusick 366*67408Smckusick FREE(vp->v_data, M_MFSNODE); 367*67408Smckusick vp->v_data = NULL; 36851981Smckusick return (0); 36951981Smckusick } 37051981Smckusick 37151981Smckusick /* 37239677Smckusick * Print out the contents of an mfsnode. 37339677Smckusick */ 37451477Sbostic int 37554660Smckusick mfs_print(ap) 37654660Smckusick struct vop_print_args /* { 37754660Smckusick struct vnode *a_vp; 37854660Smckusick } */ *ap; 37939677Smckusick { 38053594Sheideman register struct mfsnode *mfsp = VTOMFS(ap->a_vp); 38139677Smckusick 38239677Smckusick printf("tag VT_MFS, pid %d, base %d, size %d\n", mfsp->mfs_pid, 38339677Smckusick mfsp->mfs_baseoff, mfsp->mfs_size); 38451477Sbostic return (0); 38539677Smckusick } 38639677Smckusick 38739677Smckusick /* 38839019Smckusick * Block device bad operation 38939019Smckusick */ 39051477Sbostic int 39139019Smckusick mfs_badop() 39239019Smckusick { 39339019Smckusick 39439389Smckusick panic("mfs_badop called\n"); 39539389Smckusick /* NOTREACHED */ 39639019Smckusick } 39739019Smckusick 39839019Smckusick /* 39939019Smckusick * Memory based filesystem initialization. 40039019Smckusick */ 40139019Smckusick mfs_init() 40239019Smckusick { 40339019Smckusick 40453875Smckusick #if !defined(hp300) && !defined(i386) && !defined(mips) && !defined(sparc) && !defined(luna68k) 40539019Smckusick rminit(mfsmap, (long)MFS_MAPREG, (long)1, "mfs mapreg", MFS_MAPSIZE); 40641536Smckusick #endif 40739019Smckusick } 408