1
- use super :: abi:: { self , O_APPEND , O_CREAT , O_EXCL , O_RDONLY , O_RDWR , O_TRUNC , O_WRONLY } ;
1
+ use super :: abi:: {
2
+ self , dirent64, stat as stat_struct, DT_DIR , DT_LNK , DT_REG , DT_UNKNOWN , O_APPEND , O_CREAT ,
3
+ O_EXCL , O_RDONLY , O_RDWR , O_TRUNC , O_WRONLY , S_IFDIR , S_IFLNK , S_IFMT , S_IFREG ,
4
+ } ;
2
5
use super :: fd:: FileDesc ;
3
- use crate :: ffi:: { CStr , OsString } ;
6
+ use crate :: ffi:: { CStr , OsStr , OsString } ;
4
7
use crate :: fmt;
5
8
use crate :: io:: { self , Error , ErrorKind } ;
6
9
use crate :: io:: { BorrowedCursor , IoSlice , IoSliceMut , SeekFrom } ;
7
10
use crate :: mem;
8
11
use crate :: os:: hermit:: ffi:: OsStringExt ;
9
12
use crate :: os:: hermit:: io:: { AsFd , AsRawFd , BorrowedFd , FromRawFd , IntoRawFd , RawFd } ;
10
13
use crate :: path:: { Path , PathBuf } ;
11
- use crate :: ptr;
12
14
use crate :: sync:: Arc ;
13
15
use crate :: sys:: common:: small_c_string:: run_path_with_cstr;
14
16
use crate :: sys:: cvt;
@@ -17,7 +19,6 @@ use crate::sys::unsupported;
17
19
use crate :: sys_common:: { AsInner , AsInnerMut , FromInner , IntoInner } ;
18
20
19
21
pub use crate :: sys_common:: fs:: { copy, try_exists} ;
20
- //pub use crate::sys_common::fs::remove_dir_all;
21
22
22
23
#[ derive( Debug ) ]
23
24
pub struct File ( FileDesc ) ;
@@ -34,32 +35,38 @@ impl FileAttr {
34
35
35
36
// all DirEntry's will have a reference to this struct
36
37
struct InnerReadDir {
37
- dirp : FileDesc ,
38
38
root : PathBuf ,
39
+ dir : Vec < u8 > ,
40
+ }
41
+
42
+ impl InnerReadDir {
43
+ pub fn new ( root : PathBuf , dir : Vec < u8 > ) -> Self {
44
+ Self { root, dir }
45
+ }
39
46
}
40
47
41
48
pub struct ReadDir {
42
49
inner : Arc < InnerReadDir > ,
43
- end_of_stream : bool ,
50
+ pos : i64 ,
44
51
}
45
52
46
53
impl ReadDir {
47
54
fn new ( inner : InnerReadDir ) -> Self {
48
- Self { inner : Arc :: new ( inner) , end_of_stream : false }
55
+ Self { inner : Arc :: new ( inner) , pos : 0 }
49
56
}
50
57
}
51
58
52
59
pub struct DirEntry {
53
- dir : Arc < InnerReadDir > ,
54
- entry : dirent_min ,
60
+ /// path to the entry
61
+ root : PathBuf ,
62
+ /// 64-bit inode number
63
+ ino : u64 ,
64
+ /// File type
65
+ type_ : u32 ,
66
+ /// name of the entry
55
67
name : OsString ,
56
68
}
57
69
58
- struct dirent_min {
59
- d_ino : u64 ,
60
- d_type : u32 ,
61
- }
62
-
63
70
#[ derive( Clone , Debug ) ]
64
71
pub struct OpenOptions {
65
72
// generic
@@ -105,15 +112,24 @@ pub struct DirBuilder {
105
112
106
113
impl FileAttr {
107
114
pub fn modified ( & self ) -> io:: Result < SystemTime > {
108
- Ok ( SystemTime :: new ( self . stat_val . st_mtime , self . stat_val . st_mtime_nsec ) )
115
+ Ok ( SystemTime :: new (
116
+ self . stat_val . st_mtime . try_into ( ) . unwrap ( ) ,
117
+ self . stat_val . st_mtime_nsec . try_into ( ) . unwrap ( ) ,
118
+ ) )
109
119
}
110
120
111
121
pub fn accessed ( & self ) -> io:: Result < SystemTime > {
112
- Ok ( SystemTime :: new ( self . stat_val . st_atime , self . stat_val . st_atime_nsec ) )
122
+ Ok ( SystemTime :: new (
123
+ self . stat_val . st_atime . try_into ( ) . unwrap ( ) ,
124
+ self . stat_val . st_atime_nsec . try_into ( ) . unwrap ( ) ,
125
+ ) )
113
126
}
114
127
115
128
pub fn created ( & self ) -> io:: Result < SystemTime > {
116
- Ok ( SystemTime :: new ( self . stat_val . st_ctime , self . stat_val . st_ctime_nsec ) )
129
+ Ok ( SystemTime :: new (
130
+ self . stat_val . st_ctime . try_into ( ) . unwrap ( ) ,
131
+ self . stat_val . st_ctime_nsec . try_into ( ) . unwrap ( ) ,
132
+ ) )
117
133
}
118
134
119
135
pub fn size ( & self ) -> u64 {
@@ -171,7 +187,7 @@ impl FileType {
171
187
impl fmt:: Debug for ReadDir {
172
188
fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
173
189
// This will only be called from std::fs::ReadDir, which will add a "ReadDir()" frame.
174
- // Thus the result will be e g 'ReadDir("/home")'
190
+ // Thus the result will be e.g. 'ReadDir("/home")'
175
191
fmt:: Debug :: fmt ( & * self . inner . root , f)
176
192
}
177
193
}
@@ -180,101 +196,74 @@ impl Iterator for ReadDir {
180
196
type Item = io:: Result < DirEntry > ;
181
197
182
198
fn next ( & mut self ) -> Option < io:: Result < DirEntry > > {
183
- if self . end_of_stream {
184
- return None ;
185
- }
199
+ let mut counter: usize = 0 ;
200
+ let mut offset: i64 = 0 ;
201
+
202
+ // loop over all directory entries and search the entry for the current position
203
+ loop {
204
+ // leave function, if the loop reaches the of the buffer (with all entries)
205
+ if offset >= self . inner . dir . len ( ) . try_into ( ) . unwrap ( ) {
206
+ return None ;
207
+ }
186
208
187
- unsafe {
188
- loop {
189
- // As of POSIX.1-2017, readdir() is not required to be thread safe; only
190
- // readdir_r() is. However, readdir_r() cannot correctly handle platforms
191
- // with unlimited or variable NAME_MAX. Many modern platforms guarantee
192
- // thread safety for readdir() as long an individual DIR* is not accessed
193
- // concurrently, which is sufficient for Rust.
194
- let entry_ptr = match abi:: readdir ( self . inner . dirp . as_raw_fd ( ) ) {
195
- abi:: DirectoryEntry :: Invalid ( e) => {
196
- // We either encountered an error, or reached the end. Either way,
197
- // the next call to next() should return None.
198
- self . end_of_stream = true ;
199
-
200
- return Some ( Err ( Error :: from_raw_os_error ( e) ) ) ;
201
- }
202
- abi:: DirectoryEntry :: Valid ( ptr) => {
203
- if ptr. is_null ( ) {
204
- return None ;
205
- }
206
-
207
- ptr
208
- }
209
+ let dir = unsafe {
210
+ & * ( self . inner . dir . as_ptr ( ) . offset ( offset. try_into ( ) . unwrap ( ) ) as * const dirent64 )
211
+ } ;
212
+
213
+ if counter == self . pos . try_into ( ) . unwrap ( ) {
214
+ self . pos += 1 ;
215
+
216
+ // After dirent64, the file name is stored. d_reclen represents the length of the dirent64
217
+ // plus the length of the file name. Consequently, file name has a size of d_reclen minus
218
+ // the size of dirent64. The file name is always a C string and terminated by `\0`.
219
+ // Consequently, we are able to ignore the last byte.
220
+ let name_bytes = unsafe {
221
+ core:: slice:: from_raw_parts (
222
+ & dir. d_name as * const _ as * const u8 ,
223
+ dir. d_reclen as usize - core:: mem:: size_of :: < dirent64 > ( ) - 1 ,
224
+ )
225
+ . to_vec ( )
209
226
} ;
210
-
211
- macro_rules! offset_ptr {
212
- ( $entry_ptr: expr, $field: ident) => { {
213
- const OFFSET : isize = {
214
- let delusion = MaybeUninit :: <dirent>:: uninit( ) ;
215
- let entry_ptr = delusion. as_ptr( ) ;
216
- unsafe {
217
- ptr:: addr_of!( ( * entry_ptr) . $field)
218
- . cast:: <u8 >( )
219
- . offset_from( entry_ptr. cast:: <u8 >( ) )
220
- }
221
- } ;
222
- if true {
223
- // Cast to the same type determined by the else branch.
224
- $entry_ptr. byte_offset( OFFSET ) . cast:: <_>( )
225
- } else {
226
- #[ allow( deref_nullptr) ]
227
- {
228
- ptr:: addr_of!( ( * ptr:: null:: <dirent>( ) ) . $field)
229
- }
230
- }
231
- } } ;
232
- }
233
-
234
- // d_name is NOT guaranteed to be null-terminated.
235
- let name_bytes = core:: slice:: from_raw_parts (
236
- offset_ptr ! ( entry_ptr, d_name) as * const u8 ,
237
- * offset_ptr ! ( entry_ptr, d_namelen) as usize ,
238
- )
239
- . to_vec ( ) ;
240
-
241
- if name_bytes == b"." || name_bytes == b".." {
242
- continue ;
243
- }
244
-
245
- let name = OsString :: from_vec ( name_bytes) ;
246
-
247
- let entry = dirent_min {
248
- d_ino : * offset_ptr ! ( entry_ptr, d_ino) ,
249
- d_type : * offset_ptr ! ( entry_ptr, d_type) ,
227
+ let entry = DirEntry {
228
+ root : self . inner . root . clone ( ) ,
229
+ ino : dir. d_ino ,
230
+ type_ : dir. d_type as u32 ,
231
+ name : OsString :: from_vec ( name_bytes) ,
250
232
} ;
251
233
252
- return Some ( Ok ( DirEntry { entry, name : name , dir : Arc :: clone ( & self . inner ) } ) ) ;
234
+ return Some ( Ok ( entry) ) ;
253
235
}
236
+
237
+ counter += 1 ;
238
+
239
+ // move to the next dirent64, which is directly stored after the previous one
240
+ offset = offset + dir. d_off ;
254
241
}
255
242
}
256
243
}
257
244
258
245
impl DirEntry {
259
246
pub fn path ( & self ) -> PathBuf {
260
- self . dir . root . join ( self . file_name_os_str ( ) )
247
+ self . root . join ( self . file_name_os_str ( ) )
261
248
}
262
249
263
250
pub fn file_name ( & self ) -> OsString {
264
251
self . file_name_os_str ( ) . to_os_string ( )
265
252
}
266
253
267
254
pub fn metadata ( & self ) -> io:: Result < FileAttr > {
268
- lstat ( & self . path ( ) )
255
+ let mut path = self . path ( ) ;
256
+ path. set_file_name ( self . file_name_os_str ( ) ) ;
257
+ lstat ( & path)
269
258
}
270
259
271
260
pub fn file_type ( & self ) -> io:: Result < FileType > {
272
- Ok ( FileType { mode : self . entry . d_type } )
261
+ Ok ( FileType { mode : self . type_ as u32 } )
273
262
}
274
263
275
264
#[ allow( dead_code) ]
276
265
pub fn ino ( & self ) -> u64 {
277
- self . entry . d_ino
266
+ self . ino
278
267
}
279
268
280
269
pub fn file_name_os_str ( & self ) -> & OsStr {
@@ -456,7 +445,7 @@ impl DirBuilder {
456
445
}
457
446
458
447
pub fn mkdir ( & self , path : & Path ) -> io:: Result < ( ) > {
459
- run_path_with_cstr ( path, |path| {
448
+ run_path_with_cstr ( path, & |path| {
460
449
cvt ( unsafe { abi:: mkdir ( path. as_ptr ( ) , self . mode ) } ) . map ( |_| ( ) )
461
450
} )
462
451
}
@@ -519,11 +508,42 @@ impl FromRawFd for File {
519
508
}
520
509
521
510
pub fn readdir ( path : & Path ) -> io:: Result < ReadDir > {
522
- let fd_raw = run_path_with_cstr ( path, |path| cvt ( unsafe { abi:: opendir ( path. as_ptr ( ) ) } ) ) ?;
511
+ let fd_raw = run_path_with_cstr ( path, & |path| cvt ( unsafe { abi:: opendir ( path. as_ptr ( ) ) } ) ) ?;
523
512
let fd = unsafe { FileDesc :: from_raw_fd ( fd_raw as i32 ) } ;
524
513
let root = path. to_path_buf ( ) ;
525
- let inner = InnerReadDir { dirp : fd, root } ;
526
- Ok ( ReadDir :: new ( inner) )
514
+
515
+ // read all director entries
516
+ let mut vec: Vec < u8 > = Vec :: new ( ) ;
517
+ let mut sz = 512 ;
518
+ loop {
519
+ // reserve memory to receive all directory entries
520
+ vec. resize ( sz, 0 ) ;
521
+
522
+ let readlen =
523
+ unsafe { abi:: getdents64 ( fd. as_raw_fd ( ) , vec. as_mut_ptr ( ) as * mut dirent64 , sz) } ;
524
+ if readlen > 0 {
525
+ // shrink down to the minimal size
526
+ vec. resize ( readlen. try_into ( ) . unwrap ( ) , 0 ) ;
527
+ break ;
528
+ }
529
+
530
+ // if the buffer is too small, getdents64 returns EINVAL
531
+ // otherwise, getdents64 returns an error number
532
+ if readlen != ( -abi:: errno:: EINVAL ) . into ( ) {
533
+ return Err ( Error :: from_raw_os_error ( readlen. try_into ( ) . unwrap ( ) ) ) ;
534
+ }
535
+
536
+ // we don't have enough memory => try to increase the vector size
537
+ sz = sz * 2 ;
538
+
539
+ // 1 MB for directory entries should be enough
540
+ // stop here to avoid an endless loop
541
+ if sz > 0x100000 {
542
+ return Err ( Error :: from ( ErrorKind :: Uncategorized ) ) ;
543
+ }
544
+ }
545
+
546
+ Ok ( ReadDir :: new ( InnerReadDir :: new ( root, vec) ) )
527
547
}
528
548
529
549
pub fn unlink ( path : & Path ) -> io:: Result < ( ) > {
@@ -539,12 +559,11 @@ pub fn set_perm(_p: &Path, _perm: FilePermissions) -> io::Result<()> {
539
559
}
540
560
541
561
pub fn rmdir ( path : & Path ) -> io:: Result < ( ) > {
542
- run_path_with_cstr ( path, |path| cvt ( unsafe { abi:: rmdir ( path. as_ptr ( ) ) } ) . map ( |_| ( ) ) )
562
+ run_path_with_cstr ( path, & |path| cvt ( unsafe { abi:: rmdir ( path. as_ptr ( ) ) } ) . map ( |_| ( ) ) )
543
563
}
544
564
545
565
pub fn remove_dir_all ( _path : & Path ) -> io:: Result < ( ) > {
546
- //unsupported()
547
- Ok ( ( ) )
566
+ unsupported ( )
548
567
}
549
568
550
569
pub fn readlink ( _p : & Path ) -> io:: Result < PathBuf > {
@@ -560,15 +579,15 @@ pub fn link(_original: &Path, _link: &Path) -> io::Result<()> {
560
579
}
561
580
562
581
pub fn stat ( path : & Path ) -> io:: Result < FileAttr > {
563
- run_path_with_cstr ( path, |path| {
582
+ run_path_with_cstr ( path, & |path| {
564
583
let mut stat_val: stat_struct = unsafe { mem:: zeroed ( ) } ;
565
584
cvt ( unsafe { abi:: stat ( path. as_ptr ( ) , & mut stat_val) } ) ?;
566
585
Ok ( FileAttr :: from_stat ( stat_val) )
567
586
} )
568
587
}
569
588
570
589
pub fn lstat ( path : & Path ) -> io:: Result < FileAttr > {
571
- run_path_with_cstr ( path, |path| {
590
+ run_path_with_cstr ( path, & |path| {
572
591
let mut stat_val: stat_struct = unsafe { mem:: zeroed ( ) } ;
573
592
cvt ( unsafe { abi:: lstat ( path. as_ptr ( ) , & mut stat_val) } ) ?;
574
593
Ok ( FileAttr :: from_stat ( stat_val) )
0 commit comments