Page MenuHomeFreeBSD

nfsclient: upgrade vnode lock in VOP_OPEN() if we need to flush buffers
ClosedPublic

Authored by kib on Nov 15 2021, 8:53 PM.
Tags
None
Referenced Files
F108714038: D32999.id98547.diff
Mon, Jan 27, 11:58 AM
Unknown Object (File)
Sun, Jan 19, 12:24 PM
Unknown Object (File)
Sun, Jan 19, 12:23 PM
Unknown Object (File)
Sun, Jan 19, 9:23 AM
Unknown Object (File)
Mon, Jan 13, 3:38 PM
Unknown Object (File)
Mon, Jan 13, 6:56 AM
Unknown Object (File)
Mon, Jan 13, 4:30 AM
Unknown Object (File)
Dec 26 2024, 8:18 PM
Subscribers

Details

Summary

VOP_FSYNC() asserts that the vnode is exclusively locked for NFS. If we try to execute file with recently modified content, the assert is triggered.

Diff Detail

Repository
rG FreeBSD src repository
Lint
Lint Skipped
Unit
Tests Skipped

Event Timeline

kib requested review of this revision.Nov 15 2021, 8:53 PM

Oops. I have always assumed calls to VOP_OPEN/VOP_CLOSE had
exclusively locked vnodes.

I think there are reasons beyond calls the ncl_vinvalbuf()
that expects this. I'll need to take a closer look, but at least
for NFSv4 VOP_OPEN(), the upgrade will always be needed,
I think?

Maybe the upgrades will need to be done at the beginning
of both calls instead of distributed through them.

Oops. I have always assumed calls to VOP_OPEN/VOP_CLOSE had
exclusively locked vnodes.

For instance, exec() does VOP_OPEN() for shared-locked vnode.
Your note about VOP_CLOSE() makes me think that the same upgrade would be needed there, for instance execve() closes vp as well, But we do flush in open, and then text counter prevents writes.

I think there are reasons beyond calls the ncl_vinvalbuf()
that expects this. I'll need to take a closer look, but at least
for NFSv4 VOP_OPEN(), the upgrade will always be needed,
I think?

Why? I do not see it as needed unconditionally for NFSv4. There are still additional checks for mtime.

Maybe the upgrades will need to be done at the beginning
of both calls instead of distributed through them.

I tried to save the upgrades. I noted this assert happens sometimes, not always, even when I did things like "configure". Just booting multiuser from nfsroot did not. So doing upgrades always would waste a lot, IMO.

I just looked and it does look like the handling of
the NFSv4 Open state is safe for a shared vnode.
There is a separate sleep lock in the openowner
that is set appropriately (exclusive or shared)
in nfscl_open(), called from nfsrpc_open().

So, I think this patch is sufficient for nfs_open().
I'll take a closer look before I click reviewed.

Looks ok, now that I am convinced the nfsrpc_open() call
is safe for a shared vnode lock.

I also looked at nfs_close() and the other stuff is safe for
a shared vnode lock.

I'd say adding the same stuff to nfs_close() before the
ncl_vinvalbuf() and ncl_flush() calls makes sense.
As you say, with a little luck, the upgrade may never
actually happen, but having the code there just in case
seems reasonable?

This revision is now accepted and ready to land.Nov 15 2021, 9:30 PM

I'd say adding the same stuff to nfs_close() before the
ncl_vinvalbuf() and ncl_flush() calls makes sense.
As you say, with a little luck, the upgrade may never
actually happen, but having the code there just in case
seems reasonable?

It is somewhat more delicate there, we must not fail the call just due to doomed vnode, because VOP_CLOSE() is called from vgonel().
I updated the patch with some variant, which should be good enough, I believe.

Also upgrade in VOP_CLOSE()

This revision now requires review to proceed.Nov 15 2021, 9:56 PM

Looks fine to me. Btw, I did a search for any other
calls to ncl_flush()/ncl_vinvalbuf() where the vnode
might be shared locked and didn't find any.

This revision is now accepted and ready to land.Nov 16 2021, 4:51 PM