This repository was archived by the owner on Mar 7, 2021. It is now read-only.
File tree 8 files changed +117
-0
lines changed
8 files changed +117
-0
lines changed Original file line number Diff line number Diff line change @@ -25,12 +25,16 @@ const INCLUDED_FUNCTIONS: &[&str] = &[
25
25
"_copy_from_user" ,
26
26
"alloc_chrdev_region" ,
27
27
"unregister_chrdev_region" ,
28
+ "wait_for_random_bytes" ,
29
+ "get_random_bytes" ,
30
+ "rng_is_initialized" ,
28
31
] ;
29
32
const INCLUDED_VARS : & [ & str ] = & [
30
33
"EINVAL" ,
31
34
"ENOMEM" ,
32
35
"ESPIPE" ,
33
36
"EFAULT" ,
37
+ "EAGAIN" ,
34
38
"__this_module" ,
35
39
"FS_REQUIRES_DEV" ,
36
40
"FS_BINARY_MOUNTDATA" ,
Original file line number Diff line number Diff line change 1
1
#include <linux/cdev.h>
2
2
#include <linux/fs.h>
3
3
#include <linux/module.h>
4
+ #include <linux/random.h>
4
5
#include <linux/slab.h>
5
6
#include <linux/uaccess.h>
6
7
#include <linux/version.h>
Original file line number Diff line number Diff line change @@ -10,6 +10,7 @@ impl Error {
10
10
pub const ENOMEM : Self = Error ( -( bindings:: ENOMEM as i32 ) ) ;
11
11
pub const EFAULT : Self = Error ( -( bindings:: EFAULT as i32 ) ) ;
12
12
pub const ESPIPE : Self = Error ( -( bindings:: ESPIPE as i32 ) ) ;
13
+ pub const EAGAIN : Self = Error ( -( bindings:: EAGAIN as i32 ) ) ;
13
14
14
15
pub fn from_kernel_errno ( errno : c_types:: c_int ) -> Error {
15
16
Error ( errno)
Original file line number Diff line number Diff line change @@ -13,6 +13,7 @@ mod error;
13
13
pub mod file_operations;
14
14
pub mod filesystem;
15
15
pub mod printk;
16
+ pub mod random;
16
17
pub mod sysctl;
17
18
mod types;
18
19
pub mod user_ptr;
Original file line number Diff line number Diff line change
1
+ use core:: convert:: TryInto ;
2
+
3
+ use crate :: { bindings, c_types, error} ;
4
+
5
+ /// Fills `dest` with random bytes generated from the kernel's CSPRNG. Ensures
6
+ /// that the CSPRNG has been seeded before generating any random bytes, and
7
+ /// will block until it's ready.
8
+ pub fn getrandom ( dest : & mut [ u8 ] ) -> error:: KernelResult < ( ) > {
9
+ let res = unsafe { bindings:: wait_for_random_bytes ( ) } ;
10
+ if res != 0 {
11
+ return Err ( error:: Error :: from_kernel_errno ( res) ) ;
12
+ }
13
+
14
+ unsafe {
15
+ bindings:: get_random_bytes (
16
+ dest. as_mut_ptr ( ) as * mut c_types:: c_void ,
17
+ dest. len ( ) . try_into ( ) ?,
18
+ ) ;
19
+ }
20
+ Ok ( ( ) )
21
+ }
22
+
23
+ /// Fills `dest` with random bytes generated from the kernel's CSPRNG. If the
24
+ /// CSPRNG is not yet seeded, returns an `Err(EAGAIN)` immediately. Only
25
+ /// available on 4.19 and later kernels.
26
+ #[ cfg( kernel_4_19_0_or_greater) ]
27
+ pub fn getrandom_nonblock ( dest : & mut [ u8 ] ) -> error:: KernelResult < ( ) > {
28
+ if !unsafe { bindings:: rng_is_initialized ( ) } {
29
+ return Err ( error:: Error :: EAGAIN ) ;
30
+ }
31
+ getrandom ( dest)
32
+ }
Original file line number Diff line number Diff line change
1
+ [package ]
2
+ name = " random-tests"
3
+ version = " 0.1.0"
4
+ authors = [
" Alex Gaynor <[email protected] >" ,
" Geoffrey Thomas <[email protected] >" ]
5
+ edition = " 2018"
6
+
7
+ [lib ]
8
+ crate-type = [" staticlib" ]
9
+ test = false
10
+
11
+ [features ]
12
+ default = [" linux-kernel-module" ]
13
+
14
+ [dependencies ]
15
+ linux-kernel-module = { path = " ../.." , optional = true }
16
+
17
+ [dev-dependencies ]
18
+ kernel-module-testlib = { path = " ../../testlib" }
Original file line number Diff line number Diff line change
1
+ #![ no_std]
2
+
3
+ use linux_kernel_module:: { self , random} ;
4
+
5
+ struct EntropySource ;
6
+
7
+ impl SysctlStorage for EntropySource {
8
+ fn store_value ( & self , data : & [ u8 ] ) -> ( usize , error:: KernelResult < ( ) > ) {
9
+ ( 0 , Err ( error:: Error :: EINVAL ) )
10
+ }
11
+
12
+ fn read_value ( & self , data : & mut UserSlicePtrWriter ) -> ( usize , error:: KernelResult < ( ) > ) {
13
+ let mut storage = vec ! [ 0 ; data. len( ) ] ;
14
+ random:: getrandom ( & mut storage) . map_err ( |e| ( 0 , Err ( e) ) ) ?;
15
+ ( storage. len ( ) , data. write ( storage) )
16
+ }
17
+ }
18
+
19
+ struct RandomTestModule {
20
+ _sysctl_entropy : Sysctl < EntropySource > ,
21
+ }
22
+
23
+ impl linux_kernel_module:: KernelModule for RandomTestModule {
24
+ fn init ( ) -> linux_kernel_module:: KernelResult < Self > {
25
+ Ok ( RandomTestModule {
26
+ _sysctl_entropy : Sysctl :: register (
27
+ cstr ! ( "rust/random-tests" ) ,
28
+ cstr ! ( "entropy" ) ,
29
+ EntropySource ,
30
+ Mode :: from_int ( 0o444 ) ,
31
+ ) ,
32
+ } )
33
+ }
34
+ }
35
+
36
+ linux_kernel_module:: kernel_module!(
37
+ RandomTestModule ,
38
+ author: "Fish in a Barrel Contributors" ,
39
+ description: "A module for testing the CSPRNG" ,
40
+ license: "GPL"
41
+ ) ;
Original file line number Diff line number Diff line change
1
+ use std:: collections:: HashSet ;
2
+ use std:: fs;
3
+ use std:: io:: Read ;
4
+
5
+ use kernel_module_testlib:: { assert_dmesg_contains, with_kernel_module} ;
6
+
7
+ #[ test]
8
+ fn test_random_entropy ( ) {
9
+ with_kernel_module ( || {
10
+ let mut keys = HashSet :: new ( ) ;
11
+ for _ in 0 ..1024 {
12
+ let mut key = [ 0 ; 16 ] ;
13
+ let mut f = fs:: File :: open ( "/proc/sys/rust/random-tests/entropy" ) . unwrap ( ) ;
14
+ f. read_exact ( & mut key) . unwrap ( ) ;
15
+ keys. insert ( key) ;
16
+ }
17
+ assert_eq ! ( keys. len( ) , 1024 ) ;
18
+ } ) ;
19
+ }
You can’t perform that action at this time.
0 commit comments