Skip to content

Commit 0182cc6

Browse files
committed
Auto merge of #116438 - ChrisDenton:truncate, r=thomcc
Windows: Allow `File::create` to work on hidden files This makes `OpenOptions::new().write(true).create(true).truncate(true).open(&path)` work if the path exists and is a hidden file. Previously it would fail with access denied. This makes it consistent with `OpenOptions::new().write(true).truncate(true).open(&path)` (note the lack of `create`) which does not have this restriction. It's also more consistent with other platforms. Fixes #115745 (see that issue for more details).
2 parents 1a3aa4a + c6f7aa0 commit 0182cc6

File tree

2 files changed

+53
-7
lines changed

2 files changed

+53
-7
lines changed

library/std/src/fs/tests.rs

+26-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ use crate::os::unix::fs::symlink as symlink_file;
2222
#[cfg(unix)]
2323
use crate::os::unix::fs::symlink as symlink_junction;
2424
#[cfg(windows)]
25-
use crate::os::windows::fs::{symlink_dir, symlink_file};
25+
use crate::os::windows::fs::{symlink_dir, symlink_file, OpenOptionsExt};
2626
#[cfg(windows)]
2727
use crate::sys::fs::symlink_junction;
2828
#[cfg(target_os = "macos")]
@@ -1793,3 +1793,28 @@ fn windows_unix_socket_exists() {
17931793
assert_eq!(socket_path.try_exists().unwrap(), true);
17941794
assert_eq!(socket_path.metadata().is_ok(), true);
17951795
}
1796+
1797+
#[cfg(windows)]
1798+
#[test]
1799+
fn test_hidden_file_truncation() {
1800+
// Make sure that File::create works on an existing hidden file. See #115745.
1801+
let tmpdir = tmpdir();
1802+
let path = tmpdir.join("hidden_file.txt");
1803+
1804+
// Create a hidden file.
1805+
const FILE_ATTRIBUTE_HIDDEN: u32 = 2;
1806+
let mut file = OpenOptions::new()
1807+
.write(true)
1808+
.create_new(true)
1809+
.attributes(FILE_ATTRIBUTE_HIDDEN)
1810+
.open(&path)
1811+
.unwrap();
1812+
file.write("hidden world!".as_bytes()).unwrap();
1813+
file.flush().unwrap();
1814+
drop(file);
1815+
1816+
// Create a new file by truncating the existing one.
1817+
let file = File::create(&path).unwrap();
1818+
let metadata = file.metadata().unwrap();
1819+
assert_eq!(metadata.len(), 0);
1820+
}

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

+27-6
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use crate::os::windows::prelude::*;
22

33
use crate::borrow::Cow;
4-
use crate::ffi::OsString;
4+
use crate::ffi::{c_void, OsString};
55
use crate::fmt;
66
use crate::io::{self, BorrowedCursor, Error, IoSlice, IoSliceMut, SeekFrom};
77
use crate::mem::{self, MaybeUninit};
@@ -16,8 +16,6 @@ use crate::sys::{c, cvt, Align8};
1616
use crate::sys_common::{AsInner, FromInner, IntoInner};
1717
use crate::thread;
1818

19-
use core::ffi::c_void;
20-
2119
use super::path::maybe_verbatim;
2220
use super::{api, to_u16s, IoResult};
2321

@@ -273,7 +271,9 @@ impl OpenOptions {
273271
(false, false, false) => c::OPEN_EXISTING,
274272
(true, false, false) => c::OPEN_ALWAYS,
275273
(false, true, false) => c::TRUNCATE_EXISTING,
276-
(true, true, false) => c::CREATE_ALWAYS,
274+
// `CREATE_ALWAYS` has weird semantics so we emulate it using
275+
// `OPEN_ALWAYS` and a manual truncation step. See #115745.
276+
(true, true, false) => c::OPEN_ALWAYS,
277277
(_, _, true) => c::CREATE_NEW,
278278
})
279279
}
@@ -289,19 +289,40 @@ impl OpenOptions {
289289
impl File {
290290
pub fn open(path: &Path, opts: &OpenOptions) -> io::Result<File> {
291291
let path = maybe_verbatim(path)?;
292+
let creation = opts.get_creation_mode()?;
292293
let handle = unsafe {
293294
c::CreateFileW(
294295
path.as_ptr(),
295296
opts.get_access_mode()?,
296297
opts.share_mode,
297298
opts.security_attributes,
298-
opts.get_creation_mode()?,
299+
creation,
299300
opts.get_flags_and_attributes(),
300301
ptr::null_mut(),
301302
)
302303
};
303304
let handle = unsafe { HandleOrInvalid::from_raw_handle(handle) };
304-
if let Ok(handle) = handle.try_into() {
305+
if let Ok(handle) = OwnedHandle::try_from(handle) {
306+
// Manual truncation. See #115745.
307+
if opts.truncate
308+
&& creation == c::OPEN_ALWAYS
309+
&& unsafe { c::GetLastError() } == c::ERROR_ALREADY_EXISTS
310+
{
311+
unsafe {
312+
// Setting the allocation size to zero also sets the
313+
// EOF position to zero.
314+
let alloc = c::FILE_ALLOCATION_INFO { AllocationSize: 0 };
315+
let result = c::SetFileInformationByHandle(
316+
handle.as_raw_handle(),
317+
c::FileAllocationInfo,
318+
ptr::addr_of!(alloc).cast::<c_void>(),
319+
mem::size_of::<c::FILE_ALLOCATION_INFO>() as u32,
320+
);
321+
if result == 0 {
322+
return Err(io::Error::last_os_error());
323+
}
324+
}
325+
}
305326
Ok(File { handle: Handle::from_inner(handle) })
306327
} else {
307328
Err(Error::last_os_error())

0 commit comments

Comments
 (0)