Changing v_op for vnode on the fly
    Andrey Simonenko 
    simon at comsys.ntu-kpi.kiev.ua
       
    Fri Feb 13 01:08:46 PST 2004
    
    
  
Hello all,
I want to control in a KLD module when any process make any
VOPs, which can change the content of some file.  For this I change
v_op field in the needed vnode to my vnodeop_p, currently my VOPs
print some debug information and call original VOPs for the vnode.
I can't simply wrap syscall entries for a process to control write()
and read() syscalls, because it is unknown in my task which process
can access a file and processes can mmap a file and modify or read it
directly in their memory.
Is it enough to get exclusive lock on vnode, before changing
v_op pointer?  Here is my code:
	vn_lock(cvp->vp, LK_EXCLUSIVE | LK_RETRY, p);
	if (flag > 0)
		cvp->vp->v_op = catch_vnode_vnodeop_p;	/* My vnodeop_p. */
	else
		cvp->vp->v_op = cvp->vnodeop_p;		/* Original v_op. */
	VOP_UNLOCK(cvp->vp, 0, p);
I made some tests and see that most of VOP_xxx require lock (shared
or exclusive) on vnode, as well this is documented in the manual pages.
Another my question.
Below I include one of my tests on FreeBSD 4.8: ">>>..." means start of the
operation, "<<<" means end of the operation, "mod:..." are messages from
the KLD module.
According to documentation "VOP_GETATTR expects the vnode to be locked
on entry and will leave the vnode locked on return".  In my test a file
is mmap'ed as MAP_SHARED with PROT_WRITE and one char is put in mmap'ed
memory.
Why mmap() system call (vm/vm_mmap.c:mmap() and vm/vm_mmap.c:vm_mmap())
doesn't lock the vnode before VOP_GETATTR?
[simon at comp1 test1]% ./test1
mod: fd = 3, flag = catch -> vnode 0xc89ce0c0
mod: vnode_unlock 0xc89ce0c0 ()
>>> mmap file
mod: vnode_getvobject 0xc89ce0c0
mod: vnode_getattr 0xc89ce0c0
mod: vnode_getattr 0xc89ce0c0
<<<
>>> put char to mmap'ed file
mod: vnode_lock 0xc89ce0c0 (LK_SHARED | LK_CANRECURSE | LK_NOPAUSE | LK_INTERLOC
K | LK_RETRY)
mod: vnode_bmap 0xc89ce0c0
mod: vnode_getpages 0xc89ce0c0
mod: vnode_unlock 0xc89ce0c0 ( | LK_INTERLOCK)
<<<
mod: vnode_islocked 0xc89ce0c0
mod: vnode_lock 0xc89ce0c0 (LK_EXCLUSIVE | LK_NOPAUSE | LK_INTERLOCK | LK_RETRY
| LK_NOOBJ)
mod: vnode_getvobject 0xc89ce0c0
mod: vnode_putpages 0xc89ce0c0
mod: vnode_write 0xc89ce0c0
mod: vnode_balloc 0xc89ce0c0
mod: vnode_unlock 0xc89ce0c0 ( | LK_INTERLOCK)
mod: vnode_lock 0xc89ce0c0 (LK_EXCLUSIVE | LK_NOWAIT | LK_NOPAUSE | LK_INTERLOCK
)
mod: vnode_fsync 0xc89ce0c0
mod: vnode_bwrite 0xc89ce0c0
mod: vnode_getvobject 0xc89ce0c0
mod: vnode_strategy 0xc89ce0c0
mod: vnode_unlock 0xc89ce0c0 ()
mod: vnode_getvobject 0xc89ce0c0
^Cmod: fd = 3, flag = restore -> vnode 0xc89ce0c0
mod: vnode_lock 0xc89ce0c0 (LK_EXCLUSIVE | LK_NOPAUSE | LK_INTERLOCK | LK_RETRY)
 
[simon at comp1 test1]%
    
    
More information about the freebsd-hackers
mailing list