Skip to content

Commit e658b96

Browse files
authored
slh-dsa: adds pkcs8 support (#867)
1 parent c9eabf7 commit e658b96

9 files changed

+179
-1
lines changed

Cargo.lock

+2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

slh-dsa/Cargo.toml

+4-1
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,8 @@ signature = { version = "2.3.0-pre.4", features = ["rand_core"] }
2424
hmac = "=0.13.0-pre.4"
2525
sha2 = { version = "=0.11.0-pre.4", default-features = false }
2626
digest = "=0.11.0-pre.9"
27+
pkcs8 = { version = "=0.11.0-rc.1", default-features = false }
28+
const-oid = { version = "0.10.0-rc.1", features = ["db"] }
2729

2830
[dev-dependencies]
2931
hex-literal = "0.4.1"
@@ -41,6 +43,7 @@ paste = "1.0.15"
4143
rand = "0.8.5"
4244
serde_json = "1.0.124"
4345
serde = { version = "1.0.207", features = ["derive"] }
46+
pkcs8 = { version = "=0.11.0-rc.1", features = ["pem"] }
4447

4548
[lib]
4649
bench = false
@@ -51,4 +54,4 @@ harness = false
5154

5255
[features]
5356
alloc = []
54-
default = ["alloc"]
57+
default = ["alloc", "pkcs8/alloc"]

slh-dsa/src/hashes/sha2.rs

+7
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ use crate::{
99
xmss::XmssParams, ParameterSet,
1010
};
1111
use crate::{PkSeed, SkPrf, SkSeed};
12+
use const_oid::db::fips205;
1213
use digest::{Digest, KeyInit, Mac};
1314
use hmac::Hmac;
1415
use hybrid_array::{Array, ArraySize};
@@ -169,6 +170,7 @@ impl ForsParams for Sha2_128s {
169170
}
170171
impl ParameterSet for Sha2_128s {
171172
const NAME: &'static str = "SLH-DSA-SHA2-128s";
173+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_128_S;
172174
}
173175

174176
/// SHA2 at L1 security with fast signatures
@@ -191,6 +193,7 @@ impl ForsParams for Sha2_128f {
191193
}
192194
impl ParameterSet for Sha2_128f {
193195
const NAME: &'static str = "SLH-DSA-SHA2-128f";
196+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_128_F;
194197
}
195198

196199
/// Implementation of the component hash functions using SHA2 at Security Category 3 and 5
@@ -330,6 +333,7 @@ impl ForsParams for Sha2_192s {
330333
}
331334
impl ParameterSet for Sha2_192s {
332335
const NAME: &'static str = "SLH-DSA-SHA2-192s";
336+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_192_S;
333337
}
334338

335339
/// SHA2 at L3 security with fast signatures
@@ -352,6 +356,7 @@ impl ForsParams for Sha2_192f {
352356
}
353357
impl ParameterSet for Sha2_192f {
354358
const NAME: &'static str = "SLH-DSA-SHA2-192f";
359+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_192_F;
355360
}
356361

357362
/// SHA2 at L5 security with small signatures
@@ -374,6 +379,7 @@ impl ForsParams for Sha2_256s {
374379
}
375380
impl ParameterSet for Sha2_256s {
376381
const NAME: &'static str = "SLH-DSA-SHA2-256s";
382+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_256_S;
377383
}
378384

379385
/// SHA2 at L5 security with fast signatures
@@ -396,4 +402,5 @@ impl ForsParams for Sha2_256f {
396402
}
397403
impl ParameterSet for Sha2_256f {
398404
const NAME: &'static str = "SLH-DSA-SHA2-256f";
405+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHA_2_256_F;
399406
}

slh-dsa/src/hashes/shake.rs

+7
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ use crate::hypertree::HypertreeParams;
77
use crate::wots::WotsParams;
88
use crate::xmss::XmssParams;
99
use crate::{ParameterSet, PkSeed, SkPrf, SkSeed};
10+
use const_oid::db::fips205;
1011
use digest::{ExtendableOutput, Update};
1112
use hybrid_array::typenum::consts::{U16, U30, U32};
1213
use hybrid_array::typenum::{U24, U34, U39, U42, U47, U49};
@@ -146,6 +147,7 @@ impl ForsParams for Shake128s {
146147
}
147148
impl ParameterSet for Shake128s {
148149
const NAME: &'static str = "SLH-DSA-SHAKE-128s";
150+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_128_S;
149151
}
150152

151153
/// SHAKE256 at L1 security with fast signatures
@@ -168,6 +170,7 @@ impl ForsParams for Shake128f {
168170
}
169171
impl ParameterSet for Shake128f {
170172
const NAME: &'static str = "SLH-DSA-SHAKE-128f";
173+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_128_F;
171174
}
172175

173176
/// SHAKE256 at L3 security with small signatures
@@ -190,6 +193,7 @@ impl ForsParams for Shake192s {
190193
}
191194
impl ParameterSet for Shake192s {
192195
const NAME: &'static str = "SLH-DSA-SHAKE-192s";
196+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_192_S;
193197
}
194198

195199
/// SHAKE256 at L3 security with fast signatures
@@ -212,6 +216,7 @@ impl ForsParams for Shake192f {
212216
}
213217
impl ParameterSet for Shake192f {
214218
const NAME: &'static str = "SLH-DSA-SHAKE-192f";
219+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_192_F;
215220
}
216221

217222
/// SHAKE256 at L5 security with small signatures
@@ -234,6 +239,7 @@ impl ForsParams for Shake256s {
234239
}
235240
impl ParameterSet for Shake256s {
236241
const NAME: &'static str = "SLH-DSA-SHAKE-256s";
242+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_256_S;
237243
}
238244

239245
/// SHAKE256 at L5 security with fast signatures
@@ -256,6 +262,7 @@ impl ForsParams for Shake256f {
256262
}
257263
impl ParameterSet for Shake256f {
258264
const NAME: &'static str = "SLH-DSA-SHAKE-256f";
265+
const ALGORITHM_OID: pkcs8::ObjectIdentifier = fips205::ID_SLH_DSA_SHAKE_256_F;
259266
}
260267

261268
#[cfg(test)]

slh-dsa/src/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,9 @@ pub trait ParameterSet:
7373
{
7474
/// Human-readable name for parameter set, matching the FIPS-205 designations
7575
const NAME: &'static str;
76+
77+
/// Associated OID with the Parameter
78+
const ALGORITHM_OID: pkcs8::ObjectIdentifier;
7679
}
7780

7881
#[cfg(test)]

slh-dsa/src/signature_encoding.rs

+23
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,15 @@ use crate::{fors::ForsSignature, Shake128s};
88
use ::signature::{Error, SignatureEncoding};
99
use hybrid_array::sizes::{U16224, U17088, U29792, U35664, U49856, U7856};
1010
use hybrid_array::{Array, ArraySize};
11+
use pkcs8::{der::AnyRef, spki::AssociatedAlgorithmIdentifier, AlgorithmIdentifierRef};
1112
use typenum::Unsigned;
1213

14+
#[cfg(feature = "alloc")]
15+
use pkcs8::{
16+
der::{self, asn1::BitString},
17+
spki::SignatureBitStringEncoding,
18+
};
19+
1320
#[derive(Debug, Clone, PartialEq, Eq)]
1421
/// A parsed SLH-DSA signature for a given parameter set
1522
///
@@ -95,6 +102,22 @@ impl<P: ParameterSet> SignatureEncoding for Signature<P> {
95102
}
96103
}
97104

105+
#[cfg(feature = "alloc")]
106+
impl<P: ParameterSet> SignatureBitStringEncoding for Signature<P> {
107+
fn to_bitstring(&self) -> der::Result<BitString> {
108+
BitString::new(0, self.to_vec())
109+
}
110+
}
111+
112+
impl<P: ParameterSet> AssociatedAlgorithmIdentifier for Signature<P> {
113+
type Params = AnyRef<'static>;
114+
115+
const ALGORITHM_IDENTIFIER: AlgorithmIdentifierRef<'static> = AlgorithmIdentifierRef {
116+
oid: P::ALGORITHM_OID,
117+
parameters: None,
118+
};
119+
}
120+
98121
impl<P: ParameterSet> From<Signature<P>> for Array<u8, P::SigLen> {
99122
fn from(sig: Signature<P>) -> Array<u8, P::SigLen> {
100123
sig.to_bytes()

slh-dsa/src/signing_key.rs

+51
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,18 @@ use crate::verifying_key::VerifyingKey;
55
use crate::{ParameterSet, PkSeed, Sha2L1, Sha2L35, Shake, VerifyingKeyLen};
66
use ::signature::{Error, KeypairRef, RandomizedSigner, Signer};
77
use hybrid_array::{Array, ArraySize};
8+
use pkcs8::{
9+
der::AnyRef,
10+
spki::{AlgorithmIdentifier, AssociatedAlgorithmIdentifier, SignatureAlgorithmIdentifier},
11+
};
812
use typenum::{Unsigned, U, U16, U24, U32};
913

14+
#[cfg(feature = "alloc")]
15+
use pkcs8::{
16+
der::{self, asn1::OctetStringRef},
17+
EncodePrivateKey,
18+
};
19+
1020
// NewTypes for ensuring hash argument order correctness
1121
#[derive(Clone, PartialEq, Eq, Debug)]
1222
pub(crate) struct SkSeed<N: ArraySize>(pub(crate) Array<u8, N>);
@@ -215,6 +225,47 @@ impl<P: ParameterSet> KeypairRef for SigningKey<P> {
215225
type VerifyingKey = VerifyingKey<P>;
216226
}
217227

228+
impl<P> TryFrom<pkcs8::PrivateKeyInfoRef<'_>> for SigningKey<P>
229+
where
230+
P: ParameterSet,
231+
{
232+
type Error = pkcs8::Error;
233+
234+
fn try_from(private_key_info: pkcs8::PrivateKeyInfoRef<'_>) -> pkcs8::Result<Self> {
235+
private_key_info
236+
.algorithm
237+
.assert_algorithm_oid(P::ALGORITHM_OID)?;
238+
239+
Self::try_from(private_key_info.private_key.as_bytes())
240+
.map_err(|_| pkcs8::Error::KeyMalformed)
241+
}
242+
}
243+
244+
#[cfg(feature = "alloc")]
245+
impl<P> EncodePrivateKey for SigningKey<P>
246+
where
247+
P: ParameterSet,
248+
{
249+
fn to_pkcs8_der(&self) -> pkcs8::Result<der::SecretDocument> {
250+
let algorithm_identifier = pkcs8::AlgorithmIdentifierRef {
251+
oid: P::ALGORITHM_OID,
252+
parameters: None,
253+
};
254+
255+
let private_key = self.to_bytes();
256+
let pkcs8_key =
257+
pkcs8::PrivateKeyInfoRef::new(algorithm_identifier, OctetStringRef::new(&private_key)?);
258+
Ok(der::SecretDocument::encode_msg(&pkcs8_key)?)
259+
}
260+
}
261+
262+
impl<P: ParameterSet> SignatureAlgorithmIdentifier for SigningKey<P> {
263+
type Params = AnyRef<'static>;
264+
265+
const SIGNATURE_ALGORITHM_IDENTIFIER: AlgorithmIdentifier<Self::Params> =
266+
Signature::<P>::ALGORITHM_IDENTIFIER;
267+
}
268+
218269
impl<M> SigningKeyLen for Sha2L1<U16, M> {
219270
type SkLen = U<{ 4 * 16 }>;
220271
}

slh-dsa/src/verifying_key.rs

+38
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,12 @@ use crate::Sha2L35;
77
use crate::Shake;
88
use ::signature::{Error, Verifier};
99
use hybrid_array::{Array, ArraySize};
10+
use pkcs8::{der, spki};
1011
use typenum::{Unsigned, U, U16, U24, U32};
1112

13+
#[cfg(feature = "alloc")]
14+
use pkcs8::EncodePublicKey;
15+
1216
/// A trait specifying the length of a serialized verifying key for a given parameter set
1317
pub trait VerifyingKeyLen {
1418
/// The length of the serialized verifying key in bytes
@@ -150,6 +154,40 @@ impl<P: ParameterSet> Verifier<Signature<P>> for VerifyingKey<P> {
150154
}
151155
}
152156

157+
#[cfg(feature = "alloc")]
158+
impl<P: ParameterSet> EncodePublicKey for VerifyingKey<P> {
159+
fn to_public_key_der(&self) -> pkcs8::spki::Result<der::Document> {
160+
let algorithm_identifier = pkcs8::AlgorithmIdentifierRef {
161+
oid: P::ALGORITHM_OID,
162+
parameters: None,
163+
};
164+
165+
let public_key = self.to_bytes();
166+
let subject_public_key = der::asn1::BitStringRef::new(0, &public_key)?;
167+
168+
pkcs8::SubjectPublicKeyInfo {
169+
algorithm: algorithm_identifier,
170+
subject_public_key,
171+
}
172+
.try_into()
173+
}
174+
}
175+
176+
impl<P: ParameterSet> TryFrom<pkcs8::SubjectPublicKeyInfoRef<'_>> for VerifyingKey<P> {
177+
type Error = spki::Error;
178+
179+
fn try_from(spki: pkcs8::SubjectPublicKeyInfoRef<'_>) -> spki::Result<Self> {
180+
spki.algorithm.assert_algorithm_oid(P::ALGORITHM_OID)?;
181+
182+
Ok(Self::try_from(
183+
spki.subject_public_key
184+
.as_bytes()
185+
.ok_or_else(|| der::Tag::BitString.value_error())?,
186+
)
187+
.map_err(|_| pkcs8::Error::KeyMalformed)?)
188+
}
189+
}
190+
153191
impl<M> VerifyingKeyLen for Sha2L1<U16, M> {
154192
type VkLen = U<32>;
155193
}

slh-dsa/tests/pkcs8.rs

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
#![cfg(feature = "alloc")]
2+
3+
use hex_literal::hex;
4+
use pkcs8::{DecodePrivateKey, EncodePrivateKey, EncodePublicKey, LineEnding};
5+
use slh_dsa::{Sha2_128s, SigningKey, VerifyingKey};
6+
use std::ops::Deref;
7+
8+
// Serialization of the SLH-DSA keys is still a draft
9+
// The vectors used here are taken from the draft-ietf-lamps-x509-slhdsa
10+
// https://github.com/lamps-wg/x509-slhdsa/commit/128c68b6b141e109e3e0ec8f3f47c832a4baaa30
11+
#[test]
12+
fn pkcs8_output() {
13+
let signing = SigningKey::<Sha2_128s>::try_from(&hex!("A2263BCA45860836523160049523D621677FAD90D51EB6067A327E0D1E64A5012B8109EC777CAA4E1F024CCFCF9497D99180509280F4256AF2B07AF80289B494")[..]).unwrap();
14+
15+
let out = signing.to_pkcs8_pem(LineEnding::LF).unwrap();
16+
17+
// https://github.com/lamps-wg/x509-slhdsa/blob/main/id-slh-dsa-sha2-128s.priv
18+
assert_eq!(
19+
out.deref(),
20+
r#"-----BEGIN PRIVATE KEY-----
21+
MFICAQAwCwYJYIZIAWUDBAMUBECiJjvKRYYINlIxYASVI9YhZ3+tkNUetgZ6Mn4N
22+
HmSlASuBCex3fKpOHwJMz8+Ul9mRgFCSgPQlavKwevgCibSU
23+
-----END PRIVATE KEY-----
24+
"#
25+
);
26+
27+
let parsed = SigningKey::<Sha2_128s>::from_pkcs8_pem(out.deref()).unwrap();
28+
29+
assert_eq!(parsed, signing);
30+
31+
let public: VerifyingKey<Sha2_128s> = parsed.as_ref().clone();
32+
33+
let out = public.to_public_key_pem(LineEnding::LF).unwrap();
34+
35+
// https://github.com/lamps-wg/x509-slhdsa/blob/main/id-slh-dsa-sha2-128s.pub
36+
assert_eq!(
37+
out.deref(),
38+
r#"-----BEGIN PUBLIC KEY-----
39+
MDAwCwYJYIZIAWUDBAMUAyEAK4EJ7Hd8qk4fAkzPz5SX2ZGAUJKA9CVq8rB6+AKJ
40+
tJQ=
41+
-----END PUBLIC KEY-----
42+
"#
43+
);
44+
}

0 commit comments

Comments
 (0)