Ocelli is a Rust library with FFI bindings that can be used to generate high-quality entropy using a camera feed. The application supports three entropy generation methods (chop_and_tack
and pick_and_flip
) and optionally applies Van Neumann whitening for enhanced randomness.
(ported from NoiseBasedCamRng by Andika Wasisto)
The algorithm extracts entropy from two arrays with 8 bit integers that can be obtained by taking two consecutive frames from a camera feed and reading their brightness levels. It is required to cover the lens of the camera so it only sees uniform blackness. Due to thermal and quantum effects the image sensor will still sense fluctuations in brightness.
The outer 100 pixel wide edges of each frames are ignored since they can be prone to bias. Entropy is obtained by comparing the remaining pixel values in a grid pattern 30 pixels apart to avoid correlation.
(inspired by R. Li, "A True Random Number Generator algorithm from digital camera image noise for varying lighting conditions," SoutheastCon 2015, Fort Lauderdale, FL, USA, 2015, pp. 1-8, doi: 10.1109/SECON.2015.7132901.)
The algorithm extracts entropy from an array of 8-bit values (camera frame pixel brightness) by analyzing the least significant bit (LSB) of each value. This process leverages the natural variability in pixel brightness across a frame. Entropy is derived by first examining whether the brightness value of a pixel falls within the range of allowed values, to avoid bias. The LSB of qualifying pixel values is then used to form a bitstream. To avoid correlations, the bits of every second array are flipped. The resulting bits are sequentially packed into bytes, forming the output entropy.
-
chop_and_tack
takes two 8 bit integer arrayscurrent
andprevious
representing consecutive grayscale image frames, an usizewidth
and an usizeheight
of the original image frame dimensions, and aminimum_distance
usize to define the grid distance between qualifying pixels. It returns an array of random 8 bit chunks. -
pick_and_flip
takes an 8 bit integer array representing a grayscale image frame, alow
andhigh
u8 to set band pass bounds and an usizecurrent_frame_index
representing a frame count of which every even number triggers flipped bits for the frame for the output array of random 8 bit chunks.
-
is_covered
can be used to check if a camera lens is covered (a requirement for the Chop and Tack method to work properly). It takes an 8 bit integer array and an 8 bit integerthreshold
value. It checks how many unique values are present in the array. If the number of unique values lies under the threshold the method returnstrue
. -
shannon
can be used to calculate the Shannon Entropy value for an array of 8 bit integers. -
whiten
applies Van Neumann whitening to an array of 8 bit values, halving its size but increasing the entropy amount by filtering out bias.
- If Chop and Tack or Tune and Prune are to be used, utilize the
is_covered
method with a threshold of 50 to determine if the camera sensor is covered. - Read the desired amount of frames from the camera and extract the brightness levels as 8 bit integers into arrays.
- Feed the arrays into one of the main methods and make sure to provide all required arguments. If you use Chop and Tack a
minimum_distance
of 30 is recommended. If you use Pick and Flip andis_covered
returned true, then pass 3 and 252 aslow
andhigh
params. If it returned false, then 10 and 245 are recommended. - Whitening can be applied using the
whiten
method to filter out bias and increase the entropy of the result - It is recommended to check the resulting entropy quality using the
shannon
method and drop the result if it falls below a threshold (e.g. 7.9). - Loop through the previous steps and accumulate the resulting entropy until the desired amount of random bytes is reached.
Basic Rust build:
cargo build --release
Build shared libraries (.so
) for common ABIs using cargo-ndk
:
cargo ndk -t arm64-v8a -t armeabi-v7a -t x86_64 build --release
First build static libraries (.a
) for device and simulator targets:
cargo build --release --target aarch64-apple-ios # iPhone / iPad (arm64)
cargo build --release --target x86_64-apple-ios # Simulator (Intel)
cargo build --release --target aarch64-apple-ios-sim # Simulator (Apple Silicon)
Then bundle them with the public header(s) into a proper framework (.xcframework
) that Xcode can consume:
xcodebuild -create-xcframework \
-library target/aarch64-apple-ios/release/libocelli.a -headers include \
-library target/x86_64-apple-ios/release/libocelli.a -headers include \
-library target/aarch64-apple-ios-sim/release/libocelli.a -headers include \
-output Ocelli.xcframework
This produces Ocelli.xcframework/
, which you can add to your Xcode or Flutter iOS project as a vendored framework.