On Wed, Apr 26, 2017 at 12:06 PM, Miklos Szeredi <miklos@xxxxxxxxxx> wrote: > On Tue, Apr 25, 2017 at 9:11 PM, Amir Goldstein <amir73il@xxxxxxxxx> wrote: >> Or maybe like this: >> >> At mount time either set or verify the xattr in upper layer root inode: >> overlay.root.$i [i:=0..numlower-1] - ovl_root_id of lower layer i >> ovl_root_id includes for each layer: >> - sb uuid >> - fh of root inode >> >> If mount was able to set or verify that all ovl_root_id[i] match their >> respective lower layer sb and root inode, then redirect_fh can be enabled, >> otherwise it is disabled. >> >> With redirect_fh enabled, it is safe to lookup by the lower layer index, >> root fh and lower inode fh. >> With redirect_fh enabled, it is safe to store handles on copy up along >> with lower layer index and root fh. >> >> A lower layer can be used and reused by any number of overlay mounts >> at different layer index. >> >> An upper layer can be reused in an overlay mount with either copied lower >> layers or with different lower stack and will have redirect_fh disabled. >> >> An upper layer can be rotated as lower layer, because file handles are >> never followed from a lower layer. Constant inode numbers code does >> not need to follow by fh from lower layers. >> >> With this scheme, there is no need to store nor match sb_uuid a for >> every single copy up and every single lookup by fh. >> There is no need to 'lookup' the layer, just use the index and compare >> the root_fh. >> >> It is quite safe from following handles to wrong fs, except if user copies >> parts of an upper layer (without the layer root), but doing something like >> that is equivalent to a user that takes down an NFS server, brings up >> a server with the same network address and exports the same share >> name from a different filesystem. >> >> Maybe the chances are more slim, but the same interesting things could >> happen. > > Checking UUID would be O(1) and very fast, so I wouldn't worry about > that. Using is_subdir() to verify the layer is O(depth) but still > very fast. I don't think that's an issue either. > > Using is_subdir() to find the layer would be O(depth*numlower). But > we can optimize that if we really want to: have a function that > returns the first ancestor of a dentry that is a layer root (marked > with a flag). Then we just need to map that dentry to the layer, > which can be done with a hash table or whatever. > > And anyway uncached lookup will be slow, and we are only doing this > for copied up files and directories. So I don't think we need to > worry too much about optimizing this. > > So for now lets just go with the original patch but replace > ovl_is_lookable() with is_subdir(). > Just to see that I understand you correctly. I am now working on storing the following: /* * The tuple origin.{fh,layer,uuid} is a universal unique identifier * for a copy up origin, where: * origin.fh - exported file handle of the lower file * origin.root - exported file handle of the lower layer root * origin.uuid - uuid of the lower filesystem * * origin.{fh,root} are stored in format of a variable length binary blob * with struct ovl_fh header (total blob size up to 20 bytes). * uuid is stored in raw format (16 bytes) as published by sb->s_uuid. */ I intend to implement lookup as follows: - compare(origin.uuid, same_lower_sb->s_uuid) # layer root dentries cannot be DCACHE_DISCONNECTED, so # exportfs_decode_fh ignores mnt arg and returns the cached dentry - root = exportfs_decode_fh(lowerstack[0].mnt, origin.root) - find layer where lowerstack[layer].dentry == root - this = exportfs_decode_fh(lowerstack[layer].mnt, origin.fh) is_subdir() is NOT needed for decoding the layer root is_subdir() is optional for decoding the lower file, because it is not needed to identify the layer The lookup is O(numlower)+O(depth) where O(depth) is just as precousion. Amir.