1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use anyhow::{bail, Result};
fn get_current_kernel_version() -> Result<String> {
let mut info = unsafe { std::mem::MaybeUninit::<libc::utsname>::zeroed().assume_init() };
let mut release_version = Vec::with_capacity(info.release.len());
let ret = unsafe { libc::uname(&mut info as *mut libc::utsname) };
if ret < 0 {
bail!("failed to call function: libc::uname, error code: {}", ret)
}
for i in info.release {
release_version.push(i as u8);
}
Ok(String::from_utf8(release_version)?)
}
#[derive(Debug, Clone, Copy, PartialEq, PartialOrd, Eq, Ord)]
pub struct KernelVersion {
major_revision: i32,
minor_revision: i32,
patch_number: i32,
}
impl TryFrom<&str> for KernelVersion {
type Error = anyhow::Error;
fn try_from(version_string: &str) -> Result<Self> {
let mut start_pos = 0;
let mut major_revision = 0;
let mut minor_revision = 0;
let mut patch_number = 0;
if let Some(pos) = version_string[start_pos..version_string.len()].find('.') {
let part = &version_string[start_pos..pos];
major_revision = part.parse()?;
start_pos = pos + 1;
} else {
bail!("failed to parse {}", version_string)
}
if let Some(pos) = version_string[start_pos..version_string.len()].find('.') {
let part = &version_string[start_pos..start_pos + pos];
minor_revision = part.parse()?;
start_pos = start_pos + pos + 1;
} else {
bail!("failed to parse {}", version_string)
}
for i in start_pos..version_string.len() + 1 {
let mut char = '-' as u8;
if i < version_string.len() {
char = version_string.as_bytes()[i];
}
if char < '0' as u8 || char > '9' as u8 {
let part = &version_string[start_pos..i];
patch_number = part.parse()?;
break;
}
}
Ok(KernelVersion {
major_revision,
minor_revision,
patch_number,
})
}
}
impl KernelVersion {
pub fn current() -> Result<KernelVersion> {
let str = get_current_kernel_version()?;
KernelVersion::try_from(str.as_str())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_get_current_kernel_version() {
assert_eq!(get_current_kernel_version().is_ok(), true);
}
#[test]
fn test_kernel_version_try_from1() {
let kv = KernelVersion::try_from("3.10.10").unwrap();
assert_eq!(kv.major_revision, 3);
assert_eq!(kv.minor_revision, 10);
assert_eq!(kv.patch_number, 10);
}
#[test]
fn test_kernel_version_try_from2() {
let kv = KernelVersion::try_from("3.10.10-xx").unwrap();
assert_eq!(kv.major_revision, 3);
assert_eq!(kv.minor_revision, 10);
assert_eq!(kv.patch_number, 10);
}
#[test]
fn test_kernel_version_current() {
assert_eq!(
KernelVersion::current().is_ok(),
true,
"{:?}",
KernelVersion::current()
);
}
#[test]
fn test_kernel_version_ord_eq() {
let v1 = KernelVersion::try_from("3.10.10").unwrap();
let v2 = KernelVersion::try_from("3.10.10").unwrap();
assert_eq!(v1, v2);
}
#[test]
fn test_kernel_version_ord_lt() {
let v1 = KernelVersion::try_from("2.10.10").unwrap();
let v2 = KernelVersion::try_from("3.10.10").unwrap();
assert_eq!(v1 < v2, true);
}
}