Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Entropy Source and DRNG Manager (ESDM) RNG support #4309

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

thillux
Copy link

@thillux thillux commented Aug 15, 2024

ESDM is a Linux-based user-space PRNG daemon, with configurable entropy sources.

See: https://github.com/smuellerDD/esdm

It currently gets used, when a high level of control over entropy sources is desirable, e.g. for VPN appliance solutions.

In order to use this source, the ESDM server daemon has to be running and botan must be configured --with-esdm.

ESDM currently works only on Linux.

@thillux
Copy link
Author

thillux commented Aug 15, 2024

cc @smuellerDD

@thillux thillux force-pushed the esdm-rng branch 3 times, most recently from b031eb7 to 4151fd1 Compare August 15, 2024 11:52
@thillux
Copy link
Author

thillux commented Aug 15, 2024

I tested the FFI-Bindings against https://github.com/thillux/botan-rs/tree/esdm-rng.

e.g. the following worked as expected:

let mut rng = RandomNumberGenerator::new_esdm_pr().unwrap();
for _ in 0..10 {
    let bytes = rng.read(32);
    println!("{bytes:x?}");
}

@thillux
Copy link
Author

thillux commented Aug 15, 2024

@reneme I reformatted the code with clang-format, after looking at the CI results.

@coveralls
Copy link

coveralls commented Aug 15, 2024

Coverage Status

coverage: 91.257% (-0.01%) from 91.271%
when pulling 8a5a8cc on thillux:esdm-rng
into ac23bcb on randombit:master.

@reneme
Copy link
Collaborator

reneme commented Aug 15, 2024

I reformatted the code with clang-format, after looking at the CI results.

Thanks! FYI: There are some Emacs, VS Code and .editorconfig config files in src/editors. In case you use any of this, you might want to symlink them into the repo's root dir and benefit from things like "format on save".

@thillux thillux force-pushed the esdm-rng branch 2 times, most recently from 756c117 to df2d5fc Compare August 15, 2024 12:50
Copy link
Collaborator

@reneme reneme left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This seems interesting and the code looks good already. Thanks for the contribution! I've left a few suggestions down below.

Is there a way to test this in CI? I'd hope that we could just install and configure the daemon in src/scripts/ci/setup_gh_actions.sh and run a few basic smoke tests against it in the unit tests.

src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/info.txt Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Outdated Show resolved Hide resolved
src/lib/utils/esdm/info.txt Outdated Show resolved Hide resolved
@randombit
Copy link
Owner

So the change already looks pretty good, I think reneme already hit most of the points I would raise on it, but I am going to have to be a downer here and ask

Can you expand more on where and how ESDM is used? It does not seem to be widely distributed, eg per https://repology.org/project/esdm/versions it is only included in AUR and nix (the Debian experimental listing is a name collision with another package). In general our integrations with other systems have been for very widely deployed (eg TPM, PKCS11, OS provided interfaces, or widely deployed libs like zlib or sqlite3) and I'm hesitant to accept something that is so niche.

@mouse07410
Copy link
Contributor

I'm pretty sure it doesn't work on Mac, for example. No idea if it runs on Windows.

@thillux
Copy link
Author

thillux commented Aug 16, 2024

So the change already looks pretty good, I think reneme already hit most of the points I would raise on it, but I am going to have to be a downer here and ask

Can you expand more on where and how ESDM is used? It does not seem to be widely distributed, eg per https://repology.org/project/esdm/versions it is only included in AUR and nix (the Debian experimental listing is a name collision with another package). In general our integrations with other systems have been for very widely deployed (eg TPM, PKCS11, OS provided interfaces, or widely deployed libs like zlib or sqlite3) and I'm hesitant to accept something that is so niche.

You're right with ESDM being niche from a major Linux distribution point of view. I cannot weaken/invalidate this argument.

ESDM is currently mostly used under the public radar I would say. My company uses it to provide randomness in a regulated/certified market, on an installed base of >= 100k Linux-based notebooks and plans to use ESDM in more products in the future.

For such a usage in a regulated market or for governmental customers, it offers the following key benefits from my point of view:

  • In combination with ESDM Botan can offer a NIST SP800-90B/C compatible RNG
  • ESDM offers more control over configured entropy sources, when needed, without the need for a developer to change Botan
  • ESDM can also use Botan for its internal cryptographic primitives, so less certification/code review work necessary for the developer
  • ESDM is currently the only way to conform to the upcoming NTG.1 definition of the AIS 20/31 2024 from BSI in Germany
  • It is easier to comply with changing governmental requirements in user-space in ESDM, than in kernel space (e.g. change /dev/random)

@thillux
Copy link
Author

thillux commented Aug 16, 2024

I'm pretty sure it doesn't work on Mac, for example. No idea if it runs on Windows.

That's correct, it runs on Unix/Linux.

@thillux
Copy link
Author

thillux commented Aug 16, 2024

I started working on your suggestions and will squash them together again, when finished.

@thillux
Copy link
Author

thillux commented Aug 16, 2024

This seems interesting and the code looks good already. Thanks for the contribution! I've left a few suggestions down below.

Is there a way to test this in CI? I'd hope that we could just install and configure the daemon in src/scripts/ci/setup_gh_actions.sh and run a few basic smoke tests against it in the unit tests.

How can I test this by myself? I think I'll need some iterations to get this running.

@reneme
Copy link
Collaborator

reneme commented Aug 16, 2024

How can I test this by myself? I think I'll need some iterations to get this running.

The repository policy doesn't let first-time contributors run the CI pipeline. You could iterate in a branch on your private fork, though. For a full end-to-end test, unfortunately the process is commit->push->fail->fix->commit 😢

You can, however, run the underlying CI script locally (that assumes that setup_gh_actions.sh -- or something else -- has already prepared the environment). Simply run: src/scripts/ci_build.py shared to run most things the CI would for the 'shared' (library) target.

@mouse07410
Copy link
Contributor

That's correct, it runs on Unix/Linux.

Does Botan aim to only run on Linux? And at that, on one (possibly major) distribution of Linux?

@thillux
Copy link
Author

thillux commented Aug 16, 2024

That's correct, it runs on Unix/Linux.

Does Botan aim to only run on Linux? And at that, on one (possibly major) distribution of Linux?

The ESDM_RNG class is added as optional dependency. On Windows and/or MacOS it just can't be used. This PR does not limit Botan to only Linux.

src/lib/rng/esdm_rng/esdm_rng.cpp Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
@thillux
Copy link
Author

thillux commented Aug 16, 2024

I'll probably try to add the test this weekend or start of next week.

Copy link
Collaborator

@reneme reneme left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One more suggestion for our little initialization-concurrency rabbit hole 🕳️, otherwise looks good to me now. Looking forward to a test.

Regarding the utility of this: I tend to agree that this is rather niche. Nevertheless, this has sparked some interest in our company as well, for similar reasons as pointed out by @thillux. FWIW.

src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.cpp Show resolved Hide resolved
@thillux thillux force-pushed the esdm-rng branch 3 times, most recently from 9884c54 to 8343523 Compare August 16, 2024 20:19
@thillux
Copy link
Author

thillux commented Aug 16, 2024

@reneme Smoke tests added and PR squashed again.

@thillux thillux requested a review from reneme August 16, 2024 20:19
@thillux thillux force-pushed the esdm-rng branch 2 times, most recently from 5a91a9c to 26d69f8 Compare August 17, 2024 07:54
@thillux
Copy link
Author

thillux commented Aug 17, 2024

I'm maybe too stupid to get esdm-server started for the coverage and sanitizer targets. I'll pushed a fix that hopefully works.

@thillux
Copy link
Author

thillux commented Aug 17, 2024

All tests green in my repo now.

@randombit randombit added this to the Botan 3.7.0 milestone Oct 10, 2024
@thillux
Copy link
Author

thillux commented Nov 7, 2024

I started rebasing, will push an updated version for 3.7 soon.

@thillux
Copy link
Author

thillux commented Nov 8, 2024

@reneme @randombit CI was green on my testing branch yesterday.

@thillux
Copy link
Author

thillux commented Nov 26, 2024

@reneme @randombit Is there something left in this PR which I have to adapt to and was not aware of?

Copy link
Collaborator

@reneme reneme left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for your patience and effort that went into that! I left a few minor nits but generally this looks really good to me and also works fine on my local machine. I'll prepare a patch integrating the CI dependency as mentioned in one of the comments.

src/lib/rng/esdm_rng/esdm_rng.cpp Outdated Show resolved Hide resolved
src/lib/rng/esdm_rng/esdm_rng.h Outdated Show resolved Hide resolved
src/scripts/ci/setup_gh_actions.sh Outdated Show resolved Hide resolved
src/scripts/test_python.py Outdated Show resolved Hide resolved
@thillux thillux force-pushed the esdm-rng branch 2 times, most recently from 20c08cc to f3d19f7 Compare December 12, 2024 08:02
@thillux
Copy link
Author

thillux commented Dec 12, 2024

I applied your suggestions and squashed them together with my last commit as suggested. Thanks for your review @reneme!

Copy link
Owner

@randombit randombit left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@thillux Generally looks good and we can probably merge for 3.7.0, some comments to address first though.

*/
esdm_invoke(esdm_rpcc_write_data(in.data(), in.size()));
if(ret != 0) {
throw Botan::System_Error("Writing additional input to ESDM failed");
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is pretty confusing on first read - is esdm_invoke a macro that sets ret? 😬 If so please add a block comment here calling out this.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The comment mentioning this, was already at the initial declaration of ret in line 64. I added a short note above every check against ret nevertheless, as this behavior is probably not expected.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry I missed this

@@ -29,7 +29,7 @@
"--build-fuzzers=libfuzzer",
"--build-targets=static",
"--without-os-features=getrandom,getentropy",
"--disable-modules=system_rng,processor_rng",
"--disable-modules=system_rng,processor_rng,esdm_rng",
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This shouldn’t be necessary - this is only built if explicitly requested. (Right??)

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. I removed this, as you are right here.

@@ -179,6 +179,27 @@ def test_rng(self):

user_rng.add_entropy('seed material...')

if platform.system() != "Linux":
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Don’t do this. This breaks anyone (such as distributions or end users) which want to run the Python tests but are not using ESDM. Instead try to call ESDM. If it succeeds, great. If not, verify that we throw something indicating not implemented / not available.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done, I moved this to a own unit test function, which checks for "not implemented" and gets skipped accordingly.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perfect thanks

}

private:
std::mutex m_mutex;
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Couple of things here

  1. If using std::mutex then include <mutex> in the file
  2. This then requires threads, which is ok BUT
    2a) If we require threads here then add os_features block listing threads
    2b) If it’s ok to use this on a machine without threads then use the wrappers in utils/mutex.h

I think the context ESDM is used in it probably doesn’t make sense to support systems without threads. This is typically just for people doing very niche baremetal/RTOS/etc kind of things so probably the correct fix is 2a.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. I second to use 1 & 2a.

@thillux
Copy link
Author

thillux commented Jan 2, 2025

@randombit Thanks for your feedback! If you can live with the current state, I'll squash the WIP commit. I just left it for now, in order to make it easier to look at the changes related to your suggestions.

Copy link
Owner

@randombit randombit left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks @thillux LGTM pending a final squash

ESDM is a Linux-based user-space PRNG daemon, with configurable entropy
sources.

See: https://github.com/smuellerDD/esdm

It currently gets used, when a high level of control over entropy
sources is desirable, e.g. for VPN appliance solutions.

In order to use this source, the ESDM server daemon has to be running
and botan must be configured --with-esdm.

ESDM currently works only on Linux.

Signed-off-by: Markus Theil <[email protected]>
@thillux
Copy link
Author

thillux commented Jan 6, 2025

Thanks @thillux LGTM pending a final squash

Thanks for your review, squash is now pushed.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants