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

fix!: zeroize temporary scalar value #204

Closed

Conversation

stringhandler
Copy link
Contributor

Zeroize a temporary array of Scalars. Scalar is Copy so these would be placed into a new Vec in memory.

I have also removed commit_scalars from the public interface, since it is a foot gun for anyone who uses it but does not zeroize the blinding factors

Comment on lines +222 to +228
impl<K: SecretKey> Drop for ExtendedWitness<K>{
fn drop(&mut self) {
self.mask.zeroize();
self.value.zeroize();
self.minimum_value_promise.zeroize();
}
}
Copy link
Contributor

Choose a reason for hiding this comment

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

You should be able to get this functionality for free by deriving ZeroizeOnDrop.

Copy link
Contributor

Choose a reason for hiding this comment

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

As a side note, ZeroizeOnDrop also has the benefit of being future-proof against structural changes.

Copy link
Contributor

@AaronFeickert AaronFeickert Sep 27, 2023

Choose a reason for hiding this comment

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

See this branch for the derived traits.

Comment on lines +110 to +114
impl<K:SecretKey> Drop for ExtendedMask<K>{
fn drop(&mut self) {
self.secrets.zeroize();
}
}
Copy link
Contributor

Choose a reason for hiding this comment

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

The SecretKey trait already enforces zeroizing on drop, so I don't think this is necessary.

Copy link
Contributor

Choose a reason for hiding this comment

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

If you still wanted to separately enforce zeroizing on drop at the ExtendedMask level as a safeguard against future structural changes, you should be able to simply derive ZeroizeOnDrop instead of this manual implementation.

Copy link
Contributor

@AaronFeickert AaronFeickert Sep 27, 2023

Choose a reason for hiding this comment

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

See this branch for the derived traits.

Copy link
Contributor

Choose a reason for hiding this comment

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

Also, if you derive ZeroizeOnDrop, I'd also do Zeroize too; this will allow a caller to manually call zeroize if they really want to. You don't get Zeroize for free for ZeroizeOnDrop.

src/ristretto/pedersen/extended_commitment_factory.rs Outdated Show resolved Hide resolved
@@ -65,9 +65,19 @@ impl borsh::BorshSerialize for RistrettoSecretKey {
impl borsh::BorshDeserialize for RistrettoSecretKey {
fn deserialize_reader<R>(reader: &mut R) -> Result<Self, borsh::maybestd::io::Error>
where R: borsh::maybestd::io::Read {
let bytes: Vec<u8> = borsh::BorshDeserialize::deserialize_reader(reader)?;
Self::from_bytes(bytes.as_slice())
let mut bytes: Vec<u8> = borsh::BorshDeserialize::deserialize_reader(reader)?;
Copy link
Contributor

Choose a reason for hiding this comment

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

You can use a Zeroizing wrapper here and remove the entire match.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I tried, but it doesn't deserialize, I could have added a wrapper after the deserialize, but it's basically the same thing

Copy link
Contributor

Choose a reason for hiding this comment

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

Which tests fail? I tried it on this branch and everything seems to work fine.

Comment on lines -237 to -242
impl From<Scalar> for RistrettoSecretKey {
fn from(s: Scalar) -> Self {
RistrettoSecretKey(s)
}
}

Copy link
Contributor

Choose a reason for hiding this comment

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

This is a breaking change. Has it been confirmed that tari doesn't use this?

Copy link
Contributor

Choose a reason for hiding this comment

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

I just checked that this doesn't introduce problems against this PR in tari (checking against its main branch isn't possible), so removing the functionality is fine.

Copy link
Contributor

Choose a reason for hiding this comment

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

Probably worth updating the PR description to clarify that this is a breaking API change.

@AaronFeickert
Copy link
Contributor

Superseded by #209.

SWvheerden pushed a commit that referenced this pull request Oct 25, 2023
This adds `Zeroize` and `ZeroizeOnDrop` to `ExtendedMask` and
`ExtendedWitness` for improved memory handling functionality. It
restricts the visibility of an internal commitment constructor and adds
zeroizing to temporary secret values. It updates secret key
deserialization to add zeroizing to a temporary byte array. Finally, it
removes a secret key constructor.

Supersedes #204.

BREAKING CHANGE: Changes the commitment and secret key APIs.
@AaronFeickert
Copy link
Contributor

Closing since #209 has been merged.

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

Successfully merging this pull request may close these issues.

3 participants