Repository moved to codeberg.org/randogoth/ocelli.git
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_tacktakes two 8 bit integer arrayscurrentandpreviousrepresenting consecutive grayscale image frames, an usizewidthand an usizeheightof the original image frame dimensions, and aminimum_distanceusize to define the grid distance between qualifying pixels. It returns an array of random 8 bit chunks. -
pick_and_fliptakes an 8 bit integer array representing a grayscale image frame, alowandhighu8 to set band pass bounds and an usizecurrent_frame_indexrepresenting a frame count of which every even number triggers flipped bits for the frame for the output array of random 8 bit chunks.
-
is_coveredcan 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 integerthresholdvalue. It checks how many unique values are present in the array. If the number of unique values lies under the threshold the method returnstrue. -
shannoncan be used to calculate the Shannon Entropy value for an array of 8 bit integers. -
whitenapplies 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_coveredmethod 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_distanceof 30 is recommended. If you use Pick and Flip andis_coveredreturned true, then pass 3 and 252 aslowandhighparams. If it returned false, then 10 and 245 are recommended. - Whitening can be applied using the
whitenmethod to filter out bias and increase the entropy of the result - It is recommended to check the resulting entropy quality using the
shannonmethod 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 --releaseBuild shared libraries (.so) for common ABIs using cargo-ndk:
cargo ndk -t arm64-v8a -t armeabi-v7a -t x86_64 build --releaseFirst 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.xcframeworkThis produces Ocelli.xcframework/, which you can add to your Xcode or Flutter iOS project as a vendored framework.