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
#![no_std]
#[cfg(not(target_os = "windows"))]
compile_error!("This crate is for querying Windows, but the target isn't Windows.");
#[cfg(not(feature = "fallback"))]
mod internal {
#[link(name = "ntdll")]
extern "system" {
pub fn RtlGetNtVersionNumbers(major: *mut u32, minor: *mut u32, build: *mut u32);
}
}
#[cfg(feature = "fallback")]
mod internal {
#![allow(non_snake_case)]
use core::{mem::transmute, ptr};
use winapi::um::libloaderapi::{GetModuleHandleA, GetProcAddress};
static mut NTDLL_FUNCTION: *const () = ptr::null();
pub unsafe fn RtlGetNtVersionNumbers(major: *mut u32, minor: *mut u32, build: *mut u32) {
if NTDLL_FUNCTION.is_null() {
NTDLL_FUNCTION = GetProcAddress(
GetModuleHandleA(b"ntdll.dll\0".as_ptr() as *const _),
b"RtlGetNtVersionNumbers\0".as_ptr() as *const _,
) as *const _;
}
transmute::<_, extern "system" fn(*mut u32, *mut u32, *mut u32)>(NTDLL_FUNCTION)(
major, minor, build,
);
}
}
pub fn get() -> (u32, u32, u32) {
let (mut major, mut minor, mut build) = (0u32, 0u32, 0u32);
unsafe {
internal::RtlGetNtVersionNumbers(&mut major as _, &mut minor as _, &mut build as _);
}
(major, minor, build)
}