|
1 | 1 | //! Implementation for Windows
|
2 | 2 | use crate::Error;
|
3 |
| -use core::mem::MaybeUninit; |
4 |
| - |
5 |
| -extern crate std; |
6 |
| -use std::println; |
| 3 | +use core::{ |
| 4 | + ffi::c_void, |
| 5 | + mem, |
| 6 | + mem::MaybeUninit, |
| 7 | + ptr, |
| 8 | + sync::atomic::{fence, AtomicPtr, Ordering}, |
| 9 | +}; |
7 | 10 |
|
8 | 11 | type HMODULE = isize;
|
9 | 12 | type BOOL = i32;
|
10 | 13 | const TRUE: BOOL = 1;
|
11 | 14 |
|
12 |
| -type ProcessPrng = unsafe extern "system" fn(*mut u8, usize) -> BOOL; |
13 |
| - |
14 | 15 | #[link(name = "kernel32")]
|
15 | 16 | extern "system" {
|
16 | 17 | fn LoadLibraryA(libfilename: *const u8) -> HMODULE;
|
17 |
| - fn GetProcAddress(hmodule: HMODULE, procname: *const u8) -> Option<ProcessPrng>; |
| 18 | + fn GetProcAddress(hmodule: HMODULE, procname: *const u8) -> *mut c_void; |
18 | 19 | }
|
19 | 20 |
|
20 |
| -pub fn getrandom_inner(dest: &mut [MaybeUninit<u8>]) -> Result<(), Error> { |
21 |
| - let dll = unsafe { LoadLibraryA(b"bcryptprimitives.dll\0".as_ptr()) }; |
22 |
| - println!("Got DLL: {}", dll); |
23 |
| - if dll == 0 { |
24 |
| - return Err(Error::WINDOWS_LOAD_DLL); |
| 21 | +type ProcessPrng = unsafe extern "system" fn(*mut u8, usize) -> BOOL; |
| 22 | + |
| 23 | +static PROCESS_PRNG_PTR: AtomicPtr<c_void> = AtomicPtr::new(ptr::null_mut()); |
| 24 | + |
| 25 | +fn get_process_prng() -> Result<ProcessPrng, Error> { |
| 26 | + let mut p = PROCESS_PRNG_PTR.load(Ordering::Relaxed); |
| 27 | + if p.is_null() { |
| 28 | + let dll = unsafe { LoadLibraryA(b"bcryptprimitives.dll\0".as_ptr()) }; |
| 29 | + if dll == 0 { |
| 30 | + return Err(Error::WINDOWS_LOAD_DLL); |
| 31 | + } |
| 32 | + p = unsafe { GetProcAddress(dll, b"ProcessPrng\0".as_ptr()) }; |
| 33 | + if p.is_null() { |
| 34 | + return Err(Error::WINDOWS_LOAD_DLL); |
| 35 | + } |
| 36 | + PROCESS_PRNG_PTR.store(p, Ordering::Release); |
| 37 | + } else { |
| 38 | + fence(Ordering::Acquire); |
25 | 39 | }
|
26 |
| - let process_prng = match unsafe { GetProcAddress(dll, b"ProcessPrng\0".as_ptr()) } { |
27 |
| - Some(p) => p, |
28 |
| - None => return Err(Error::WINDOWS_LOAD_DLL), |
29 |
| - }; |
30 |
| - println!("Got ProcessPrng: {:p}", process_prng); |
| 40 | + Ok(unsafe { mem::transmute(p) }) |
| 41 | +} |
| 42 | + |
| 43 | +pub fn getrandom_inner(dest: &mut [MaybeUninit<u8>]) -> Result<(), Error> { |
| 44 | + let process_prng = get_process_prng()?; |
31 | 45 | match unsafe { process_prng(dest.as_mut_ptr() as *mut u8, dest.len()) } {
|
32 | 46 | TRUE => Ok(()),
|
33 | 47 | _ => Err(Error::WINDOWS_PROCESS_PRNG),
|
|
0 commit comments