@@ -24,20 +24,42 @@ fn slot_to_device(slot: i64) -> Option<&'static str> {
24
24
}
25
25
}
26
26
27
- fn get_whole_disk_dev_link ( node : & Node < ' _ > ) -> Result < PathBuf > {
28
- // Each of the letter-indexed references into block device
29
- // slices are represented as minor nodes.
27
+ fn get_dev_path ( node : & Node < ' _ > ) -> Result < Option < PathBuf > > {
30
28
let mut wm = node. minors ( ) ;
31
29
while let Some ( m) = wm. next ( ) . transpose ( ) ? {
32
- // Find the minor of the "whole disk"
33
- if m. name ( ) == "wd" {
34
- let links = DevLinks :: new ( false ) ?;
35
- for l in links. links_for_path ( m. devfs_path ( ) ?) ? {
36
- return Ok ( l. path ( ) . to_path_buf ( ) ) ;
30
+ if m. name ( ) != "wd" {
31
+ continue ;
32
+ }
33
+ let links = {
34
+ match DevLinks :: new ( true ) {
35
+ Ok ( links) => links,
36
+ Err ( _) => DevLinks :: new ( false ) ?,
37
37
}
38
+ } ;
39
+ let devfs_path = m. devfs_path ( ) ?;
40
+
41
+ let paths = links
42
+ . links_for_path ( & devfs_path) ?
43
+ . into_iter ( )
44
+ . filter ( |l| {
45
+ l. linktype ( ) == DevLinkType :: Primary
46
+ && l. path ( )
47
+ . file_name ( )
48
+ . map ( |f| f. to_string_lossy ( ) . ends_with ( "d0" ) )
49
+ . unwrap_or ( false )
50
+ } )
51
+ . collect :: < Vec < _ > > ( ) ;
52
+
53
+ if paths. is_empty ( ) {
54
+ return Err ( anyhow ! ( "no links for {}" , devfs_path) ) ;
38
55
}
56
+
57
+ if paths. len ( ) != 1 {
58
+ return Err ( anyhow ! ( "got weird paths {:?} for {}" , paths, devfs_path) ) ;
59
+ }
60
+ return Ok ( Some ( paths[ 0 ] . path ( ) . to_path_buf ( ) ) ) ;
39
61
}
40
- Err ( anyhow ! ( "No whole disk minor found" ) )
62
+ Ok ( None )
41
63
}
42
64
43
65
#[ derive( Copy , Clone , PartialEq , Eq , Debug , ValueEnum ) ]
@@ -60,15 +82,20 @@ struct Args {
60
82
fn main ( ) -> Result < ( ) > {
61
83
let args = Args :: parse ( ) ;
62
84
63
- let mut device_info = DevInfo :: new ( ) ?;
64
- let mut node_walker = device_info. walk_node ( ) ;
85
+ let mut device_info = {
86
+ match DevInfo :: new_force_load ( ) {
87
+ Ok ( di) => di,
88
+ Err ( _) => DevInfo :: new ( ) ?,
89
+ }
90
+ } ;
91
+ let mut node_walker = device_info. walk_driver ( "blkdev" ) ;
65
92
66
93
let mut device_descriptions = BTreeMap :: new ( ) ;
67
94
68
95
while let Some ( mut node) = node_walker. next ( ) . transpose ( ) ? {
69
96
if let Some ( driver_name) = node. driver_name ( ) {
70
97
if driver_name == "blkdev" {
71
- let dev_path = get_whole_disk_dev_link ( & node) ?;
98
+ let dev_path = get_dev_path ( & node) ?;
72
99
let devfs_path = PathBuf :: from ( format ! ( "/devices{}" , node. devfs_path( ) ?) ) ;
73
100
while let Ok ( Some ( parent) ) = node. parent ( ) {
74
101
node = parent;
@@ -83,14 +110,14 @@ fn main() -> Result<()> {
83
110
let device = slot_to_device ( slot) . unwrap_or ( "Not found" ) ;
84
111
85
112
let path = match args. flavor {
86
- PathFlavor :: Devfs => devfs_path,
87
- PathFlavor :: Dev => dev_path,
113
+ PathFlavor :: Devfs => devfs_path. to_string_lossy ( ) ,
114
+ PathFlavor :: Dev => dev_path
115
+ . map ( |p| String :: from ( p. to_string_lossy ( ) ) )
116
+ . unwrap_or ( String :: from ( "none" ) )
117
+ . into ( ) ,
88
118
} ;
89
119
90
- device_descriptions. insert (
91
- slot,
92
- format ! ( "{slot:>4}\t {device:>6}\t {path}" , path = path. display( ) ) ,
93
- ) ;
120
+ device_descriptions. insert ( slot, format ! ( "{slot:>4}\t {device:>6}\t {path}" ) ) ;
94
121
break ;
95
122
}
96
123
}
0 commit comments