Skip to content

Commit b490ccc

Browse files
committed
kernel_copy: avoid panic on unexpected OS error
According to documentation, the listed errnos should only occur if the `copy_file_range` call cannot be made at all, so the assert be correct. However, since in practice file system drivers (incl. FUSE etc.) can return any errno they want, we should not panic here. Fixes #91152
1 parent 2e055d9 commit b490ccc

File tree

1 file changed

+4
-2
lines changed

1 file changed

+4
-2
lines changed

library/std/src/sys/unix/kernel_copy.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -576,20 +576,22 @@ pub(super) fn copy_regular_files(reader: RawFd, writer: RawFd, max_len: u64) ->
576576
return match err.raw_os_error() {
577577
// when file offset + max_length > u64::MAX
578578
Some(EOVERFLOW) => CopyResult::Fallback(written),
579-
Some(ENOSYS | EXDEV | EINVAL | EPERM | EOPNOTSUPP | EBADF) => {
579+
Some(ENOSYS | EXDEV | EINVAL | EPERM | EOPNOTSUPP | EBADF) if written == 0 => {
580580
// Try fallback io::copy if either:
581581
// - Kernel version is < 4.5 (ENOSYS¹)
582582
// - Files are mounted on different fs (EXDEV)
583583
// - copy_file_range is broken in various ways on RHEL/CentOS 7 (EOPNOTSUPP)
584584
// - copy_file_range file is immutable or syscall is blocked by seccomp¹ (EPERM)
585585
// - copy_file_range cannot be used with pipes or device nodes (EINVAL)
586586
// - the writer fd was opened with O_APPEND (EBADF²)
587+
// and no bytes were written successfully yet. (All these errnos should
588+
// not be returned if something was already written, but they happen in
589+
// the wild, see #91152.)
587590
//
588591
// ¹ these cases should be detected by the initial probe but we handle them here
589592
// anyway in case syscall interception changes during runtime
590593
// ² actually invalid file descriptors would cause this too, but in that case
591594
// the fallback code path is expected to encounter the same error again
592-
assert_eq!(written, 0);
593595
CopyResult::Fallback(0)
594596
}
595597
_ => CopyResult::Error(err, written),

0 commit comments

Comments
 (0)