Dear list, We've unearth an odd behaviour in cp: `cp --preserve=xattr` tries to copy both attributes of the chattr kind and extended attributes of the getfattr kind with apparently no way to disable either one of them (it's all or nothing). This is problematic in tools like mkosi where non-filesystem-specific xattributes need to be preserved whilst FS-specific attributes must be discarded for cross-filesystem support.
I have added a MWE at the end of this email after my signature: it creates two raw partitions as files (one as XFS, one as BTRFS), mounts them in local folders and creates 3 files in the BTRFS partitions later altered before copy. Referring to the script, we think there should be an option to copy files foo (no-attr), bar (setfattr) and baz (chattr), keeping the setfattr's xattr and discarding chattr's attr. Looking at the code, it seems like cp eventually defers the actual attribute copying libattr, which seems to handle both, but separately (which is what we want). Would it make sense to add a separate `attr` preserve value for the chattr case and keep `xattr` for getfattr case? Thanks Gaël #!/bin/bash workspace="/tmp/cp_attr" echo "WORKDIR: ${workspace}" echo echo 'SETTING UP WORKSPACE' mkdir -p ${workspace}/{xfs,btrfs} truncate -s 512M ${workspace}/xfs.img truncate -s 512M ${workspace}/btrfs.img mkfs.xfs -q ${workspace}/xfs.img mkfs.btrfs -q ${workspace}/btrfs.img echo 'BECOMING ROOT' sudo sh -c " cd ${workspace} echo ' - mounting filesystems' mount -o loop xfs.img xfs mount -o loop btrfs.img btrfs echo ' - creating files in BTRFS' touch btrfs/{foo,bar,baz} echo ' - original attr state:' lsattr btrfs/ echo ' - original xattr state:' getfattr btrfs/* echo echo 'CHANGING ATTRIBUTES' echo ' - unchanged: btrfs/foo' echo ' - set xattr: btrfs/bar | setfattr -n user.foobar -v BAR' setfattr -n user.foobar -v BAR btrfs/bar echo ' - set attr: btrfs/baz | chattr +c btrfs/baz' chattr +c btrfs/baz echo ' - current attr state:' lsattr btrfs/ echo ' - current xattr state:' getfattr btrfs/* echo echo 'COPIES' # Works: echo ' - copying btrfs/foo -> xfs/foo' cp --preserve=xattr btrfs/foo xfs/ && echo 'Success!' || echo 'Failed!' # Works: echo ' - copying btrfs/bar -> xfs/bar' cp --preserve=xattr btrfs/bar xfs/ && echo 'Success!' || echo 'Failed!' # PROBLEM IS RIGHT HERE: echo ' - copying btrfs/baz -> xfs/baz' cp --preserve=xattr btrfs/baz xfs/ && echo 'Success!' || echo 'Failed!' umount btrfs umount xfs " rm ${workspace}/{xfs,btrfs}.img rmdir ${workspace}/{xfs,btrfs}