From 5f585f69cf7e7b5089d4a31a98696b72440bd9d2 Mon Sep 17 00:00:00 2001 From: Matt Corallo Date: Mon, 12 Feb 2024 22:37:09 +0000 Subject: [PATCH] Add a crate which wraps `getrandom` but always compiles In the next commit we'll drop the `ahash` dependency in favor of directly calling `getrandom` to seed our hash tables. However, we'd like to depend on `getrandom` only on certain platforms *and* only when certain features (no-std) are set. This introduces an indirection crate to do so, allowing us to depend on it only when `no-std` is set but only depending on `getrandom` on platforms which it supports. --- Cargo.toml | 4 ++++ ci/check-cfg-flags.py | 2 ++ no-std-check/Cargo.toml | 3 +++ possiblyrandom/Cargo.toml | 21 ++++++++++++++++++++ possiblyrandom/src/lib.rs | 40 +++++++++++++++++++++++++++++++++++++++ 5 files changed, 70 insertions(+) create mode 100644 possiblyrandom/Cargo.toml create mode 100644 possiblyrandom/src/lib.rs diff --git a/Cargo.toml b/Cargo.toml index ddc82cd5d45..ec9edb3ac33 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -11,6 +11,7 @@ members = [ "lightning-rapid-gossip-sync", "lightning-custom-message", "lightning-transaction-sync", + "possiblyrandom", ] exclude = [ @@ -38,3 +39,6 @@ lto = "off" opt-level = 3 lto = true panic = "abort" + +[patch.crates-io.possiblyrandom] +path = "possiblyrandom" diff --git a/ci/check-cfg-flags.py b/ci/check-cfg-flags.py index d6e8e0a90fe..8952b693a60 100755 --- a/ci/check-cfg-flags.py +++ b/ci/check-cfg-flags.py @@ -15,6 +15,8 @@ def check_feature(feature): pass elif feature == "ahash": pass + elif feature == "getrandom": + pass elif feature == "hashbrown": pass elif feature == "backtrace": diff --git a/no-std-check/Cargo.toml b/no-std-check/Cargo.toml index c9d404c922f..45a70c2a6d1 100644 --- a/no-std-check/Cargo.toml +++ b/no-std-check/Cargo.toml @@ -15,3 +15,6 @@ lightning-background-processor = { path = "../lightning-background-processor", f # Obviously lightning-transaction-sync doesn't support no-std, but it should build # even if lightning is built with no-std. lightning-transaction-sync = { path = "../lightning-transaction-sync", optional = true } + +[patch.crates-io] +possiblyrandom = { path = "../possiblyrandom" } diff --git a/possiblyrandom/Cargo.toml b/possiblyrandom/Cargo.toml new file mode 100644 index 00000000000..9829007f06c --- /dev/null +++ b/possiblyrandom/Cargo.toml @@ -0,0 +1,21 @@ +[package] +name = "possiblyrandom" +version = "0.1.0" +authors = ["Matt Corallo"] +license = "MIT OR Apache-2.0" +repository = "https://github.com/lightningdevkit/rust-lightning/" +description = """ +A crate which wraps getrandom and always compiles, returning 0s when no randomness is available. +""" +edition = "2021" + +[package.metadata.docs.rs] +all-features = true +rustdoc-args = ["--cfg", "docsrs"] + +[dependencies] +getrandom = { version = "0.2", optional = true, default-features = false } + +# Enable getrandom if we are on a platform that (likely) supports it +[target.'cfg(not(any(target_os = "unknown", target_os = "none")))'.dependencies] +getrandom = { version = "0.2", default-features = false } diff --git a/possiblyrandom/src/lib.rs b/possiblyrandom/src/lib.rs new file mode 100644 index 00000000000..b76519563ef --- /dev/null +++ b/possiblyrandom/src/lib.rs @@ -0,0 +1,40 @@ +// This file is Copyright its original authors, visible in version control +// history. +// +// This file is licensed under the Apache License, Version 2.0 or the MIT license +// , at your option. +// You may not use this file except in accordance with one or both of these +// licenses. + +//! [`getrandom`] provides access to OS randomness, but will fail to compile on platforms which do +//! not support fetching OS randomness. This is exactly what you want when you're doing +//! cryptographic operations, but when you're just opportunistically randomizing, we're fine with +//! compiling and simply disabling randomization. +//! +//! This crate does that, returning only possibly-random data. +//! +//! Note that this crate only enables getrandom on a subset of platforms it supports. As getrandom +//! evolves this crate is unlikely to carefully track all getrandom-supported platforms, however +//! will use random data on popular platforms. + +#![no_std] + +#[cfg(feature = "getrandom")] +extern crate getrandom; + +/// Possibly fills `dest` with random data. May fill it with zeros. +#[cfg(feature = "getrandom")] +#[inline] +pub fn getpossiblyrandom(dest: &mut [u8]) { + if getrandom::getrandom(dest).is_err() { + dest.fill(0); + } +} + +/// Possibly fills `dest` with random data. May fill it with zeros. +#[cfg(not(feature = "getrandom"))] +#[inline] +pub fn getpossiblyrandom(dest: &mut [u8]) { + dest.fill(0); +}