Re: [PATCH RFC 2/2] ceph: truncate the file contents when needed when file scrypted

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Mon, 2021-09-13 at 13:42 +0800, Xiubo Li wrote:
> On 9/10/21 7:46 PM, Jeff Layton wrote:
> > On Fri, 2021-09-10 at 10:30 +0800, Xiubo Li wrote:
> > > On 9/9/21 8:48 PM, Jeff Layton wrote:
> > > > On Thu, 2021-09-09 at 11:38 +0800, Xiubo Li wrote:
> > > > > On 9/8/21 9:57 PM, Jeff Layton wrote:
> > > > > > On Wed, 2021-09-08 at 17:37 +0800, Xiubo Li wrote:
> > > > > > > On 9/8/21 12:26 AM, Jeff Layton wrote:
> > > > > > > > On Fri, 2021-09-03 at 16:15 +0800, xiubli@xxxxxxxxxx wrote:
> > > > > > > > > From: Xiubo Li <xiubli@xxxxxxxxxx>
> > > > > > > > > 
> > > > > > > > > When truncating the file, it will leave the MDS to handle that,
> > > > > > > > > but the MDS won't update the file contents. So when the fscrypt
> > > > > > > > > is enabled, if the truncate size is not aligned to the block size,
> > > > > > > > > the kclient will round up the truancate size to the block size and
> > > > > > > > > leave the the last block untouched.
> > > > > > > > > 
> > > > > > > > > The opaque fscrypt_file field will be used to tricker whether the
> > > > > > > > > last block need to do the rmw to truncate the a specified block's
> > > > > > > > > contents, we can get which block needs to do the rmw by round down
> > > > > > > > > the fscrypt_file.
> > > > > > > > > 
> > > > > > > > > In kclient side, there is not need to do the rmw immediately after
> > > > > > > > > the file is truncated. We can defer doing that whenever the kclient
> > > > > > > > > will update that block in late future. And before that any kclient
> > > > > > > > > will check the fscrypt_file field when reading that block, if the
> > > > > > > > > fscrypt_file field is none zero that means the related block needs
> > > > > > > > > to be zeroed in range of [fscrypt_file, round_up(fscrypt_file + PAGE_SIZE))
> > > > > > > > > in pagecache or readed data buffer.
> > > > > > > > > 
> > > > > > > > s/PAGE_SIZE/CEPH_FSCRYPT_BLOCK_SIZE/
> > > > > > > > 
> > > > > > > > Yes, on x86_64 they are equal, but that's not the case on all arches.
> > > > > > > > Also, we are moving toward a pagecache that may hold larger pages on
> > > > > > > > x86_64 too.
> > > > > > > >      
> > > > > > > Okay.
> > > > > > > > > Once the that block contents are updated and writeback
> > > > > > > > > kclient will reset the fscrypt_file field in MDS side, then 0 means
> > > > > > > > > no need to care about the truncate stuff any more.
> > > > > > > > > 
> > > > > > > > I'm a little unclear on what the fscrypt_file actually represents here.
> > > > > > > > 
> > > > > > > > I had proposed that we just make the fscrypt_file field hold the
> > > > > > > > "actual" i_size and we'd make the old size field always be a rounded-
> > > > > > > > up version of the size. The MDS would treat that as an opaque value
> > > > > > > > under Fw caps, and the client could use that field to determine i_size.
> > > > > > > > That has a side benefit too -- if the client doesn't support fscrypt,
> > > > > > > > it'll see the rounded-up sizes which are close enough and don't violate
> > > > > > > > any POSIX rules.
> > > > > > > > 
> > > > > > > > In your version, fscrypt_file also holds the actual size of the inode,
> > > > > > > > but sometimes you're zeroing it out, and I don't understand why:
> > > > > > > I think I forgot to fix this after I adapt to multiple ftruncates case,
> > > > > > > this patch is not correctly handling the "actual" file size.
> > > > > > > 
> > > > > > > I just want the fscrypt_file field always to hold the offset from which
> > > > > > > the contents needed to be zeroed, and the range should be [fscrypt_file,
> > > > > > > round_up(fscrypt_file +CEPH_FSCRYPT_BLOCK_SIZE)).
> > > > > > > 
> > > > > > > In single ftruncate case the fscrypt_file should equal to the "actual"
> > > > > > > file size. Then the "req->r_args.setattr.size = attr->ia_size" and
> > > > > > > "req->r_args.setattr.old_size = isize", no need to round up in
> > > > > > > __ceph_setattr() in kclient side, and leave the MDS to do that, but we
> > > > > > > need to pass the CEPH_FSCRYPT_BLOCK_SIZE at the same time.
> > > > > > > 
> > > > > > I'm really not a fan of pushing this logic into the MDS. Why does it
> > > > > > need to know anything about the CEPH_FSCRYPT_BLOCK_SIZE at all?
> > > > >    From your current patch set, you are rounding up the
> > > > > "req->r_args.setattr.size" and "req->r_args.setattr.old_size" to the
> > > > > BLOCK end in __ceph_setattr().
> > > > > 
> > > > > Without considering keep the file scryption logic in kclient only, I
> > > > > need the "req->r_args.setattr.size" to keep the file's real size.
> > > > > 
> > > > > Since the MDS will do the truncate stuff. And if we won't round the
> > > > > "req->r_args.setattr.size" up to the BLOCK end any more, then the MDS
> > > > > needs to know whether and how to round up the file size to the block end
> > > > > when truncating the file. Because the fscrypt_file won't record the
> > > > > file's real size any more, it maybe zero, more detail please see the
> > > > > example below.
> > > > > 
> > > > > Yeah, but as you mentioned bellow if we will keep the file scryption
> > > > > logic in kclient only, I need one extra field to do the defer rmw:
> > > > > 
> > > > > struct fscrypt_file {
> > > > > 
> > > > >        u64 file_real_size;         // always keep the file's real size and
> > > > > the "req->r_args.setattr.size = round_up(file_real_size, BLOCK_SIZE)" as
> > > > > you do in your current patch set.
> > > > > 
> > > > >        u64 file_truncate_offset;  // this will always record in which
> > > > > BLOCK we need to do the rmw, this maybe 0 or located in the file's LAST
> > > > > block and maybe not, more detail please the example below.
> > > > > 
> > > > > }
> > > > > 
> > > > > The "file_truncate_offset" member will be what I need to do the defer rmw.
> > > > > 
> > > > > 
> > > > > > > But in multiple ftruncates case if the second ftruncate has a larger
> > > > > > > size, the fscrypt_file won't be changed and the ceph backend will help
> > > > > > > zero the extended part, but we still need to zero the contents in the
> > > > > > > first ftruncate. If the second ftruncate has a smaller size, the
> > > > > > > fscrypt_file need to be updated and always keeps the smaller size.
> > > > > > > 
> > > > > > I don't get it. Maybe you can walk me through a concrete example of how
> > > > > > racing ftruncates are a problem?
> > > > > Sorry for confusing.
> > > > > 
> > > > > For example:
> > > > > 
> > > > > 1), if there has a file named "bar", and currently the size is 100K
> > > > > bytes, the CEPH_FSCRYPT_BLOCK_SIZE size equals to 4K.
> > > > > 
> > > > > 2), first ftruncate("bar", 7K) comes, then both the file real size and
> > > > > "file_truncate_offset" will be set to 7K, then in MDS it will truncate
> > > > > the file from 8K, and the last block's [7K, 8K) contents need to be
> > > > > zeroed anyway later.
> > > > > 
> > > > > 3), immediately a second ftruncate("bar", 16K) comes, from the ftruncate
> > > > > man page it says the new extended [7K, 16K) should be zeroed when
> > > > > truncating the file. That means the OSD should help zero the [8K, 16K),
> > > > > but won't touch the block [4K, 8K), in which the [7K, 8K) contents still
> > > > > needs to be zeroed. So in this case the "file_truncate_offset" won't be
> > > > > changed and still be 7K. Then the "file_truncate_offset" won't be
> > > > > located in the last block of the file any more.
> > > > > 
> > > > Woah, why didn't the file_truncate_offset get changed when you truncated
> > > > up to 16k?
> > > > 
> > > > When you issue the truncate SETATTR to the MDS, you're changing the size
> > > > field in the inode. The fscrypt_file field _must_ be updated at the same
> > > > time. I think that means that we need to extend SETATTR to also update
> > > > fscrypt_file.
> > > > 
> > > > > 4), if the second ftruncate in step 3) the new size is 3K, then the MDS
> > > > > will truncate the file from 4K and [7K, 8K) contents will be discard
> > > > > anyway, so we need to update the "file_truncate_offset" to 3K, that
> > > > > means a new BLOCK [0K, 4K) needs to do the rmw, by zeroing the [3K, 4K).
> > > > > 
> > > > > 5), if the new truncate in step 3) the new size is 4K, since the 4K < 7K
> > > > > and 4K is aligned to the BLOCK size, so no need to rmw any block any
> > > > > more, then we can just clear the "file_truncate_offset" field.
> > > > > 
> > > > > 
> > > > > For defer RMW logic please see the following example:
> > > > > 
> > > > > 
> > > > Ok, thanks. I think I understand now how racing truncates are a problem.
> > > > Really, it comes down to the fact that we don't have a good way to
> > > > mediate the last-block RMW operation when competing clients are issuing
> > > > truncates.
> > > > 
> > > > I'm not sure adding this file_truncate_offset field really does all that
> > > > much good though. You don't really have a way to ensure that a
> > > > truncating client will see the changes to that field before it issues
> > > > its RMW op.
> > > > 
> > > > > > Suppose we have a file and client1 truncates it down from a large size
> > > > > > to 7k. client1 then sends the MDS a SETATTR to truncate it at 8k, and
> > > > > > does a RMW on the last (4k) block. client2 comes along at the same time
> > > > > > and truncates it up to 13k. client2 issues a SETATTR to extend the file
> > > > > > to 16k and does a RMW on the last block too (which would presumably
> > > > > > already be all zeroes anyway).
> > > > > I think you meant in the none defer RMW case, this is not what my defer
> > > > > RMW approach will do.
> > > > > 
> > > > > After the client1 truncated the file, it won't do the RMW if it won't
> > > > > write any data to that file, and then we assume the client1 is unmounted
> > > > > immediately.
> > > > > 
> > > > > And when the client1 is truncating the file, it will update the
> > > > > "file_truncate_offset" to 7K, which means the [7K, 8K) in the LAST block
> > > > > needs to be zeroed.
> > > > > 
> > > > > Then the client2 reads that the file size is 7K and the
> > > > > "file_truncate_offset" is 7K too, and the client2 wants to truncate the
> > > > > file up to 13K. Since the OSD should help us zero the extended part [8K,
> > > > > 13K] when truncating, but won't touch the block [4K, 8K), for which it
> > > > > still needs to do the RMW. Then the client2 is unmounted too before
> > > > > writing any data to the file. After this the "file_truncate_offset"
> > > > > won't be located in the file's LAST block any more.
> > > > > 
> > > > > After that if the client3 will update the whole file's contents, it will
> > > > > read all the file 13K bytes contents to local page buffers, since the
> > > > > "file_truncate_offset" is 7K and then in the page buffer the range [7K,
> > > > > 8K) will be zeroed just after the contents are dencrypted inplace. Then
> > > > > if the client3 successfully flushes that dirty data back and then the
> > > > > deferred RMW for block [4K, 8K) should be done at the same time, and the
> > > > > "file_truncate_offset" should be cleared too.
> > > > > 
> > > > > While if the client3 won't update the block [4K, 8K), the
> > > > > "file_truncate_offset" will be kept all the time until the above RMW is
> > > > > done in future.
> > > > > 
> > > > > 
> > > > Ok, I think I finally understand what you're saying.
> > > > 
> > > > You want to rely on the next client to do a write to handle the zeroing
> > > > at the end. You basically just want to keep track of whether and where
> > > > it should zero up to the end of the next crypto block, and defer that
> > > > until a client is writing.
> > > Yeah, the writing could also be in the same client just after the file
> > > is truncated.
> > > 
> > > 
> > > > I'll have to think about whether that's still racy. Part of the problem
> > > > is that once the client doesn't have caps, it doesn't have a way to
> > > > ensure that fscrypt_file (whatever it holds) doesn't change while it's
> > > > doing that zeroing.
> > > As my understanding, if clientA want to read a file, it will open it
> > > with RD mode, then it will get the fscrypt_file meta and "Fr" caps, then
> > > it can safely read the file contents and do the zeroing for that block.
> > Ok, so in this case, the client is just zeroing out the appropriate
> > region in the resulting read data, and not on the OSD. That should be
> > ok.
> > 
> > > Then the opened file shouldn't worry whether the fscrypt_file will be
> > > changed by others during it still holding the Fr caps, because if the
> > > clientB wants to update the fscrypt_file it must acquire the Fw caps
> > > first, before that the Fr caps must be revoked from clientA and at the
> > > same time the read pagecache in clientA will be invalidated.
> > > 
> > Are you certain that Fw caps is enough to ensure that no other client
> > holds Fr caps?
> 
> I spent hours and went through the mds Locker related code on the weekends.
> 
>  From the mds/lock.cc code, for mds filelock for example in the LOCK_MIX 
> state and some interim transition states to LOCK_MIX it will allow 
> different clients could hold any of Fw or Fr caps. But the Fb/Fc will be 
> disabled. Checked the mds/Locker.cc code, found that the mds filelock 
> could to switch LOCK_MIX state in some cases when there has one client 
> wants Fw and another client wants any of Fr and Fw.
> 
> In this case I think the Linux advisory or mandatory locks are necessary 
> to keep the file contents concurrency. In multiple processes' concurrent 
> read/write or write/write cases without the Linux advisory/mandatory 
> locks the file contents' concurrency won't be guaranteed, so the logic 
> is the same here ?
> 
> If so, couldn't we just assume the Fw vs Fw and Fr vs Fw caps should be 
> exclusive in correct use case ? For example, just after the mds filelock 
> state switches to LOCK_MIX, if clientA gets the advisory file lock and 
> the Fw caps, and even another clientB could be successfully issued the 
> Fr caps, the clientB won't do any read because it should be still stuck 
> and be waiting for the advisory file lock.
> 

I'm not sure I like that idea. Basically, that would change the meaning
of the what Frw caps represent, in a way that is not really consistent
with how they have been used before.

We could gate that new behavior on the new feature flags, but it sounds
pretty tough.

I think we have a couple of options:

1) we could just make the clients request and wait on Fx caps when they
do a truncate. They might stall for a bit if there is contention, but it
would ensure consistency and the client could be completely in charge of
the truncate. [a]

2) we could rev the protocol, and have the client send along the last
block to be written along with the SETATTR request. Maybe we even
consider just adding a new TRUNCATE call independent of SETATTR. The MDS
would remain in complete control of it at that point.

The other ideas I've considered seem more complex and don't offer any
significant advantages that I can see.

[a]: Side question: why does buffering a truncate require Fx and not Fb?
How do Fx and Fb interact?

> 
> > IIRC, Frw don't have the same exclusionary relationship
> > that (e.g.) Asx has. To exclude Fr, you may need Fb.
> > 
> > (Side note: these rules really need to be codified into a document
> > somewhere. I started that with the capabilities doc in the ceph tree,
> > but it's light on details of the FILE caps)
> 
> Yeah, that doc is light on the details for now.
> 
> 
> > > And if after clientB have been issued the Fw caps and have modified that
> > > block and still not flushed back, a new clientC comes and wants to read
> > > the file, so the Fw caps must be revoked from clientB and the dirty data
> > > will be flushed, and then when releasing the Fw caps to the MDS, it will
> > > update the new fscrypt_file meta together.
> > > 
> > > I haven't see which case will be racy yet. Or did I miss some cases or
> > > something important ?
> > > 
> > > 
> > We also need to consider how legacy, non-fscrypt-capable clients will
> > interact with files that have this field set. If one comes in and writes
> > to or truncates one of these files, it's liable to leave a real mess.
> > The simplest solution may be to just bar them from doing anything with
> > fscrypted files aside from deleting them -- maybe we'd allow them to
> > acquire Fr caps but not Fw?
> 
> For the legacy, non-fscrypt-capable clients, the reading contents should 
> be encrypted any way, so there won't be any problem even if that 
> specified block is not RMWed yet and it should ignore this field.
> 

Right, and I think we have to allow those clients to request Fr caps so
that they have the ability to backup and archive encrypted files without
needing the key. The cephfs-mirror-daemon, in particular, may need this.

> But for write case, I think the MDS should fail it in the open() stage 
> if the mode has any of Write/Truncate, etc, and only Read/Buffer-read, 
> etc are allowed. Or if we change the mds/Locker.cc code by not allowing 
> it to issue the Fw caps to the legacy/non-fscrypt-capable clients, after 
> the file is successfully opened with Write mode, it should be stuck 
> forever when writing data to the file by waiting the Fw caps, which will 
> never come ?
> 

Yes. Those clients should be barred from making any changes to file
contents or doing anything that might result in a new dentry being
attached to an existing inode.

We need to allow them to read files, and unlink them, but that's really
about it.

> 
> > > > Really, it comes down to the fact that truncates are supposed to be an
> > > > atomic operation, but we need to perform actions in two different
> > > > places.
> > > > 
> > > > Hmmm...it's worse than that even -- if the truncate changes it so that
> > > > the last block is in an entirely different object, then there are 3
> > > > places that will need to coordinate access.
> > > > 
> > > > Tricky.
> 

-- 
Jeff Layton <jlayton@xxxxxxxxxx>




[Index of Archives]     [CEPH Users]     [Ceph Large]     [Ceph Dev]     [Information on CEPH]     [Linux BTRFS]     [Linux USB Devel]     [Video for Linux]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux