Skip to content

Commit 77acda1

Browse files
committed
Auto merge of #24598 - lfairy:impl-debug-for-file, r=alexcrichton
This patch adds a `Debug` impl for `std::fs::File`. On all platforms (Unix and Windows) it shows the file descriptor. On Linux, it displays the path and access mode as well. Ideally we should show the path/mode for all platforms, not just Linux, but this will do for now. cc #24570
2 parents 3860240 + 1131bc0 commit 77acda1

File tree

4 files changed

+68
-0
lines changed

4 files changed

+68
-0
lines changed

src/liblibc/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -3296,6 +3296,8 @@ pub mod consts {
32963296
pub const F_GETFL : c_int = 3;
32973297
pub const F_SETFL : c_int = 4;
32983298

3299+
pub const O_ACCMODE : c_int = 3;
3300+
32993301
pub const SIGTRAP : c_int = 5;
33003302
pub const SIG_IGN: size_t = 1;
33013303

src/libstd/fs.rs

+7
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919

2020
use core::prelude::*;
2121

22+
use fmt;
2223
use io::{self, Error, ErrorKind, SeekFrom, Seek, Read, Write};
2324
use path::{Path, PathBuf};
2425
use sys::fs2 as fs_imp;
@@ -305,6 +306,12 @@ impl FromInner<fs_imp::File> for File {
305306
}
306307
}
307308

309+
impl fmt::Debug for File {
310+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
311+
self.inner.fmt(f)
312+
}
313+
}
314+
308315
#[stable(feature = "rust1", since = "1.0.0")]
309316
impl Read for File {
310317
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {

src/libstd/sys/unix/fs2.rs

+49
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ use io::prelude::*;
1313
use os::unix::prelude::*;
1414

1515
use ffi::{CString, CStr, OsString, OsStr};
16+
use fmt;
1617
use io::{self, Error, SeekFrom};
1718
use libc::{self, c_int, size_t, off_t, c_char, mode_t};
1819
use mem;
@@ -294,6 +295,54 @@ impl FromInner<c_int> for File {
294295
}
295296
}
296297

298+
impl fmt::Debug for File {
299+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
300+
#[cfg(target_os = "linux")]
301+
fn get_path(fd: c_int) -> Option<PathBuf> {
302+
use string::ToString;
303+
let mut p = PathBuf::from("/proc/self/fd");
304+
p.push(&fd.to_string());
305+
readlink(&p).ok()
306+
}
307+
308+
#[cfg(not(target_os = "linux"))]
309+
fn get_path(_fd: c_int) -> Option<PathBuf> {
310+
// FIXME(#24570): implement this for other Unix platforms
311+
None
312+
}
313+
314+
#[cfg(target_os = "linux")]
315+
fn get_mode(fd: c_int) -> Option<(bool, bool)> {
316+
let mode = unsafe { libc::fcntl(fd, libc::F_GETFL) };
317+
if mode == -1 {
318+
return None;
319+
}
320+
match mode & libc::O_ACCMODE {
321+
libc::O_RDONLY => Some((true, false)),
322+
libc::O_RDWR => Some((true, true)),
323+
libc::O_WRONLY => Some((false, true)),
324+
_ => None
325+
}
326+
}
327+
328+
#[cfg(not(target_os = "linux"))]
329+
fn get_mode(_fd: c_int) -> Option<(bool, bool)> {
330+
// FIXME(#24570): implement this for other Unix platforms
331+
None
332+
}
333+
334+
let fd = self.0.raw();
335+
let mut b = f.debug_struct("File").field("fd", &fd);
336+
if let Some(path) = get_path(fd) {
337+
b = b.field("path", &path);
338+
}
339+
if let Some((read, write)) = get_mode(fd) {
340+
b = b.field("read", &read).field("write", &write);
341+
}
342+
b.finish()
343+
}
344+
}
345+
297346
pub fn mkdir(p: &Path) -> io::Result<()> {
298347
let p = try!(cstr(p));
299348
try!(cvt(unsafe { libc::mkdir(p.as_ptr(), 0o777) }));

src/libstd/sys/windows/fs2.rs

+10
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ use os::windows::prelude::*;
1414

1515
use default::Default;
1616
use ffi::{OsString, AsOsStr};
17+
use fmt;
1718
use io::{self, Error, SeekFrom};
1819
use libc::{self, HANDLE};
1920
use mem;
@@ -271,6 +272,15 @@ impl FromInner<libc::HANDLE> for File {
271272
}
272273
}
273274

275+
impl fmt::Debug for File {
276+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
277+
// FIXME(#24570): add more info here (e.g. path, mode)
278+
f.debug_struct("File")
279+
.field("handle", &self.handle.raw())
280+
.finish()
281+
}
282+
}
283+
274284
pub fn to_utf16(s: &Path) -> Vec<u16> {
275285
s.as_os_str().encode_wide().chain(Some(0).into_iter()).collect()
276286
}

0 commit comments

Comments
 (0)