Skip to content

Commit d7ce7d7

Browse files
committed
Auto merge of rust-lang#129304 - jieyouxu:rollup-gh59jxv, r=jieyouxu
Rollup of 6 pull requests Successful merges: - rust-lang#127623 (fix: fs::remove_dir_all: treat internal ENOENT as success) - rust-lang#128627 (Special case DUMMY_SP to emit line 0/column 0 locations on DWARF platforms.) - rust-lang#129187 (bootstrap: fix clean's remove_dir_all implementation) - rust-lang#129190 (Added f16 and f128 to tests/ui/consts/const-float-bits-conv.rs) - rust-lang#129231 (improve submodule updates) - rust-lang#129284 (rustdoc: animate the `:target` highlight) r? `@ghost` `@rustbot` modify labels: rollup
2 parents 4d5b3b1 + a65e367 commit d7ce7d7

File tree

18 files changed

+434
-246
lines changed

18 files changed

+434
-246
lines changed

compiler/rustc_codegen_llvm/src/debuginfo/mod.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -570,7 +570,17 @@ impl<'ll, 'tcx> DebugInfoMethods<'tcx> for CodegenCx<'ll, 'tcx> {
570570
inlined_at: Option<&'ll DILocation>,
571571
span: Span,
572572
) -> &'ll DILocation {
573-
let DebugLoc { line, col, .. } = self.lookup_debug_loc(span.lo());
573+
// When emitting debugging information, DWARF (i.e. everything but MSVC)
574+
// treats line 0 as a magic value meaning that the code could not be
575+
// attributed to any line in the source. That's also exactly what dummy
576+
// spans are. Make that equivalence here, rather than passing dummy spans
577+
// to lookup_debug_loc, which will return line 1 for them.
578+
let (line, col) = if span.is_dummy() && !self.sess().target.is_like_msvc {
579+
(0, 0)
580+
} else {
581+
let DebugLoc { line, col, .. } = self.lookup_debug_loc(span.lo());
582+
(line, col)
583+
};
574584

575585
unsafe { llvm::LLVMRustDIBuilderCreateDebugLocation(line, col, scope, inlined_at) }
576586
}

library/std/src/fs.rs

+2
Original file line numberDiff line numberDiff line change
@@ -2491,6 +2491,8 @@ pub fn remove_dir<P: AsRef<Path>>(path: P) -> io::Result<()> {
24912491
///
24922492
/// Consider ignoring the error if validating the removal is not required for your use case.
24932493
///
2494+
/// [`io::ErrorKind::NotFound`] is only returned if no removal occurs.
2495+
///
24942496
/// [`fs::remove_file`]: remove_file
24952497
/// [`fs::remove_dir`]: remove_dir
24962498
///

library/std/src/sys/pal/solid/fs.rs

+16-7
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ use crate::sync::Arc;
1010
use crate::sys::time::SystemTime;
1111
use crate::sys::unsupported;
1212
pub use crate::sys_common::fs::exists;
13+
use crate::sys_common::ignore_notfound;
1314

1415
/// A file descriptor.
1516
#[derive(Clone, Copy)]
@@ -527,15 +528,23 @@ pub fn rmdir(p: &Path) -> io::Result<()> {
527528

528529
pub fn remove_dir_all(path: &Path) -> io::Result<()> {
529530
for child in readdir(path)? {
530-
let child = child?;
531-
let child_type = child.file_type()?;
532-
if child_type.is_dir() {
533-
remove_dir_all(&child.path())?;
534-
} else {
535-
unlink(&child.path())?;
531+
let result: io::Result<()> = try {
532+
let child = child?;
533+
let child_type = child.file_type()?;
534+
if child_type.is_dir() {
535+
remove_dir_all(&child.path())?;
536+
} else {
537+
unlink(&child.path())?;
538+
}
539+
};
540+
// ignore internal NotFound errors
541+
if let Err(err) = result
542+
&& err.kind() != io::ErrorKind::NotFound
543+
{
544+
return result;
536545
}
537546
}
538-
rmdir(path)
547+
ignore_notfound(rmdir(path))
539548
}
540549

541550
pub fn readlink(p: &Path) -> io::Result<PathBuf> {

library/std/src/sys/pal/unix/fs.rs

+34-15
Original file line numberDiff line numberDiff line change
@@ -2002,6 +2002,7 @@ mod remove_dir_impl {
20022002
use crate::path::{Path, PathBuf};
20032003
use crate::sys::common::small_c_string::run_path_with_cstr;
20042004
use crate::sys::{cvt, cvt_r};
2005+
use crate::sys_common::ignore_notfound;
20052006

20062007
pub fn openat_nofollow_dironly(parent_fd: Option<RawFd>, p: &CStr) -> io::Result<OwnedFd> {
20072008
let fd = cvt_r(|| unsafe {
@@ -2055,6 +2056,16 @@ mod remove_dir_impl {
20552056
}
20562057
}
20572058

2059+
fn is_enoent(result: &io::Result<()>) -> bool {
2060+
if let Err(err) = result
2061+
&& matches!(err.raw_os_error(), Some(libc::ENOENT))
2062+
{
2063+
true
2064+
} else {
2065+
false
2066+
}
2067+
}
2068+
20582069
fn remove_dir_all_recursive(parent_fd: Option<RawFd>, path: &CStr) -> io::Result<()> {
20592070
// try opening as directory
20602071
let fd = match openat_nofollow_dironly(parent_fd, &path) {
@@ -2078,27 +2089,35 @@ mod remove_dir_impl {
20782089
for child in dir {
20792090
let child = child?;
20802091
let child_name = child.name_cstr();
2081-
match is_dir(&child) {
2082-
Some(true) => {
2083-
remove_dir_all_recursive(Some(fd), child_name)?;
2084-
}
2085-
Some(false) => {
2086-
cvt(unsafe { unlinkat(fd, child_name.as_ptr(), 0) })?;
2087-
}
2088-
None => {
2089-
// POSIX specifies that calling unlink()/unlinkat(..., 0) on a directory can succeed
2090-
// if the process has the appropriate privileges. This however can causing orphaned
2091-
// directories requiring an fsck e.g. on Solaris and Illumos. So we try recursing
2092-
// into it first instead of trying to unlink() it.
2093-
remove_dir_all_recursive(Some(fd), child_name)?;
2092+
// we need an inner try block, because if one of these
2093+
// directories has already been deleted, then we need to
2094+
// continue the loop, not return ok.
2095+
let result: io::Result<()> = try {
2096+
match is_dir(&child) {
2097+
Some(true) => {
2098+
remove_dir_all_recursive(Some(fd), child_name)?;
2099+
}
2100+
Some(false) => {
2101+
cvt(unsafe { unlinkat(fd, child_name.as_ptr(), 0) })?;
2102+
}
2103+
None => {
2104+
// POSIX specifies that calling unlink()/unlinkat(..., 0) on a directory can succeed
2105+
// if the process has the appropriate privileges. This however can causing orphaned
2106+
// directories requiring an fsck e.g. on Solaris and Illumos. So we try recursing
2107+
// into it first instead of trying to unlink() it.
2108+
remove_dir_all_recursive(Some(fd), child_name)?;
2109+
}
20942110
}
2111+
};
2112+
if result.is_err() && !is_enoent(&result) {
2113+
return result;
20952114
}
20962115
}
20972116

20982117
// unlink the directory after removing its contents
2099-
cvt(unsafe {
2118+
ignore_notfound(cvt(unsafe {
21002119
unlinkat(parent_fd.unwrap_or(libc::AT_FDCWD), path.as_ptr(), libc::AT_REMOVEDIR)
2101-
})?;
2120+
}))?;
21022121
Ok(())
21032122
}
21042123

library/std/src/sys/pal/wasi/fs.rs

+14-6
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ use crate::sys::common::small_c_string::run_path_with_cstr;
1313
use crate::sys::time::SystemTime;
1414
use crate::sys::unsupported;
1515
pub use crate::sys_common::fs::exists;
16-
use crate::sys_common::{AsInner, FromInner, IntoInner};
16+
use crate::sys_common::{ignore_notfound, AsInner, FromInner, IntoInner};
1717
use crate::{fmt, iter, ptr};
1818

1919
pub struct File {
@@ -794,14 +794,22 @@ fn remove_dir_all_recursive(parent: &WasiFd, path: &Path) -> io::Result<()> {
794794
io::const_io_error!(io::ErrorKind::Uncategorized, "invalid utf-8 file name found")
795795
})?;
796796

797-
if entry.file_type()?.is_dir() {
798-
remove_dir_all_recursive(&entry.inner.dir.fd, path.as_ref())?;
799-
} else {
800-
entry.inner.dir.fd.unlink_file(path)?;
797+
let result: io::Result<()> = try {
798+
if entry.file_type()?.is_dir() {
799+
remove_dir_all_recursive(&entry.inner.dir.fd, path.as_ref())?;
800+
} else {
801+
entry.inner.dir.fd.unlink_file(path)?;
802+
}
803+
};
804+
// ignore internal NotFound errors
805+
if let Err(err) = &result
806+
&& err.kind() != io::ErrorKind::NotFound
807+
{
808+
return result;
801809
}
802810
}
803811

804812
// Once all this directory's contents are deleted it should be safe to
805813
// delete the directory tiself.
806-
parent.remove_directory(osstr2str(path.as_ref())?)
814+
ignore_notfound(parent.remove_directory(osstr2str(path.as_ref())?))
807815
}

library/std/src/sys/pal/windows/fs.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use crate::sys::handle::Handle;
1414
use crate::sys::path::maybe_verbatim;
1515
use crate::sys::time::SystemTime;
1616
use crate::sys::{c, cvt, Align8};
17-
use crate::sys_common::{AsInner, FromInner, IntoInner};
17+
use crate::sys_common::{ignore_notfound, AsInner, FromInner, IntoInner};
1818
use crate::{fmt, ptr, slice, thread};
1919

2020
pub struct File {
@@ -1160,7 +1160,7 @@ pub fn remove_dir_all(path: &Path) -> io::Result<()> {
11601160
return Err(io::Error::from_raw_os_error(c::ERROR_DIRECTORY as _));
11611161
}
11621162

1163-
match remove_dir_all_iterative(&file, File::posix_delete) {
1163+
match ignore_notfound(remove_dir_all_iterative(&file, File::posix_delete)) {
11641164
Err(e) => {
11651165
if let Some(code) = e.raw_os_error() {
11661166
match code as u32 {

library/std/src/sys_common/fs.rs

+15-6
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
use crate::fs;
44
use crate::io::{self, Error, ErrorKind};
55
use crate::path::Path;
6+
use crate::sys_common::ignore_notfound;
67

78
pub(crate) const NOT_FILE_ERROR: Error = io::const_io_error!(
89
ErrorKind::InvalidInput,
@@ -32,14 +33,22 @@ pub fn remove_dir_all(path: &Path) -> io::Result<()> {
3233

3334
fn remove_dir_all_recursive(path: &Path) -> io::Result<()> {
3435
for child in fs::read_dir(path)? {
35-
let child = child?;
36-
if child.file_type()?.is_dir() {
37-
remove_dir_all_recursive(&child.path())?;
38-
} else {
39-
fs::remove_file(&child.path())?;
36+
let result: io::Result<()> = try {
37+
let child = child?;
38+
if child.file_type()?.is_dir() {
39+
remove_dir_all_recursive(&child.path())?;
40+
} else {
41+
fs::remove_file(&child.path())?;
42+
}
43+
};
44+
// ignore internal NotFound errors to prevent race conditions
45+
if let Err(err) = &result
46+
&& err.kind() != io::ErrorKind::NotFound
47+
{
48+
return result;
4049
}
4150
}
42-
fs::remove_dir(path)
51+
ignore_notfound(fs::remove_dir(path))
4352
}
4453

4554
pub fn exists(path: &Path) -> io::Result<bool> {

library/std/src/sys_common/mod.rs

+8
Original file line numberDiff line numberDiff line change
@@ -80,3 +80,11 @@ pub fn mul_div_u64(value: u64, numer: u64, denom: u64) -> u64 {
8080
// r < denom, so (denom*numer) is the upper bound of (r*numer)
8181
q * numer + r * numer / denom
8282
}
83+
84+
pub fn ignore_notfound<T>(result: crate::io::Result<T>) -> crate::io::Result<()> {
85+
match result {
86+
Err(err) if err.kind() == crate::io::ErrorKind::NotFound => Ok(()),
87+
Ok(_) => Ok(()),
88+
Err(err) => Err(err),
89+
}
90+
}

src/bootstrap/src/core/build_steps/clean.rs

+15-78
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@
66
//! directory unless the `--all` flag is present.
77
88
use std::fs;
9-
use std::io::{self, ErrorKind};
109
use std::path::Path;
1110

1211
use crate::core::builder::{crate_description, Builder, RunConfig, ShouldRun, Step};
@@ -101,11 +100,11 @@ fn clean(build: &Build, all: bool, stage: Option<u32>) {
101100
return;
102101
}
103102

104-
rm_rf("tmp".as_ref());
103+
remove_dir_recursive("tmp");
105104

106105
// Clean the entire build directory
107106
if all {
108-
rm_rf(&build.out);
107+
remove_dir_recursive(&build.out);
109108
return;
110109
}
111110

@@ -136,17 +135,17 @@ fn clean_specific_stage(build: &Build, stage: u32) {
136135
}
137136

138137
let path = t!(entry.path().canonicalize());
139-
rm_rf(&path);
138+
remove_dir_recursive(&path);
140139
}
141140
}
142141
}
143142

144143
fn clean_default(build: &Build) {
145-
rm_rf(&build.out.join("tmp"));
146-
rm_rf(&build.out.join("dist"));
147-
rm_rf(&build.out.join("bootstrap").join(".last-warned-change-id"));
148-
rm_rf(&build.out.join("bootstrap-shims-dump"));
149-
rm_rf(&build.out.join("rustfmt.stamp"));
144+
remove_dir_recursive(build.out.join("tmp"));
145+
remove_dir_recursive(build.out.join("dist"));
146+
remove_dir_recursive(build.out.join("bootstrap").join(".last-warned-change-id"));
147+
remove_dir_recursive(build.out.join("bootstrap-shims-dump"));
148+
remove_dir_recursive(build.out.join("rustfmt.stamp"));
150149

151150
let mut hosts: Vec<_> = build.hosts.iter().map(|t| build.out.join(t)).collect();
152151
// After cross-compilation, artifacts of the host architecture (which may differ from build.host)
@@ -166,78 +165,16 @@ fn clean_default(build: &Build) {
166165
continue;
167166
}
168167
let path = t!(entry.path().canonicalize());
169-
rm_rf(&path);
168+
remove_dir_recursive(&path);
170169
}
171170
}
172171
}
173172

174-
fn rm_rf(path: &Path) {
175-
match path.symlink_metadata() {
176-
Err(e) => {
177-
if e.kind() == ErrorKind::NotFound {
178-
return;
179-
}
180-
panic!("failed to get metadata for file {}: {}", path.display(), e);
181-
}
182-
Ok(metadata) => {
183-
if metadata.file_type().is_file() || metadata.file_type().is_symlink() {
184-
do_op(path, "remove file", |p| match fs::remove_file(p) {
185-
#[cfg(windows)]
186-
Err(e)
187-
if e.kind() == std::io::ErrorKind::PermissionDenied
188-
&& p.file_name().and_then(std::ffi::OsStr::to_str)
189-
== Some("bootstrap.exe") =>
190-
{
191-
eprintln!("WARNING: failed to delete '{}'.", p.display());
192-
Ok(())
193-
}
194-
r => r,
195-
});
196-
197-
return;
198-
}
199-
200-
for file in t!(fs::read_dir(path)) {
201-
rm_rf(&t!(file).path());
202-
}
203-
204-
do_op(path, "remove dir", |p| match fs::remove_dir(p) {
205-
// Check for dir not empty on Windows
206-
// FIXME: Once `ErrorKind::DirectoryNotEmpty` is stabilized,
207-
// match on `e.kind()` instead.
208-
#[cfg(windows)]
209-
Err(e) if e.raw_os_error() == Some(145) => Ok(()),
210-
r => r,
211-
});
212-
}
213-
};
214-
}
215-
216-
fn do_op<F>(path: &Path, desc: &str, mut f: F)
217-
where
218-
F: FnMut(&Path) -> io::Result<()>,
219-
{
220-
match f(path) {
221-
Ok(()) => {}
222-
// On windows we can't remove a readonly file, and git will often clone files as readonly.
223-
// As a result, we have some special logic to remove readonly files on windows.
224-
// This is also the reason that we can't use things like fs::remove_dir_all().
225-
#[cfg(windows)]
226-
Err(ref e) if e.kind() == ErrorKind::PermissionDenied => {
227-
let m = t!(path.symlink_metadata());
228-
let mut p = m.permissions();
229-
p.set_readonly(false);
230-
t!(fs::set_permissions(path, p));
231-
f(path).unwrap_or_else(|e| {
232-
// Delete symlinked directories on Windows
233-
if m.file_type().is_symlink() && path.is_dir() && fs::remove_dir(path).is_ok() {
234-
return;
235-
}
236-
panic!("failed to {} {}: {}", desc, path.display(), e);
237-
});
238-
}
239-
Err(e) => {
240-
panic!("failed to {} {}: {}", desc, path.display(), e);
241-
}
173+
/// Wrapper for [`std::fs::remove_dir_all`] that panics on failure and prints the `path` we failed
174+
/// on.
175+
fn remove_dir_recursive<P: AsRef<Path>>(path: P) {
176+
let path = path.as_ref();
177+
if let Err(e) = fs::remove_dir_all(path) {
178+
panic!("failed to `remove_dir_all` at `{}`: {e}", path.display());
242179
}
243180
}

src/bootstrap/src/core/build_steps/llvm.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -110,7 +110,7 @@ pub fn prebuilt_llvm_config(builder: &Builder<'_>, target: TargetSelection) -> L
110110

111111
// Initialize the llvm submodule if not initialized already.
112112
// If submodules are disabled, this does nothing.
113-
builder.update_submodule("src/llvm-project");
113+
builder.config.update_submodule("src/llvm-project");
114114

115115
let root = "src/llvm-project/llvm";
116116
let out_dir = builder.llvm_out(target);

0 commit comments

Comments
 (0)