mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
Generalise the Montgomery-curve DH support.
This gets rid of the magic constants we apply to the top and bottom bytes of the random data to make the Curve25519 private DH value. Or rather, one of the magic constants is completely gone (we can infer it from curve->fieldBits), and the other is moved into the curve structure instead of being hardwired into the private-key-inventing function. With this change, it will be easy to add the similar Curve448 kex method, because it's now just a matter of adding the protocol names and curve constants.
This commit is contained in:
parent
319d4b5075
commit
7fa0749fcb
1
ssh.h
1
ssh.h
@ -451,6 +451,7 @@ struct ec_mcurve
|
|||||||
{
|
{
|
||||||
MontgomeryCurve *mc;
|
MontgomeryCurve *mc;
|
||||||
MontgomeryPoint *G;
|
MontgomeryPoint *G;
|
||||||
|
unsigned log2_cofactor;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Edwards form curve */
|
/* Edwards form curve */
|
||||||
|
18
sshecc.c
18
sshecc.c
@ -66,11 +66,12 @@ static void initialise_wcurve(
|
|||||||
|
|
||||||
static void initialise_mcurve(
|
static void initialise_mcurve(
|
||||||
struct ec_curve *curve, mp_int *p, mp_int *a, mp_int *b,
|
struct ec_curve *curve, mp_int *p, mp_int *a, mp_int *b,
|
||||||
mp_int *G_x)
|
mp_int *G_x, unsigned log2_cofactor)
|
||||||
{
|
{
|
||||||
initialise_common(curve, EC_MONTGOMERY, p);
|
initialise_common(curve, EC_MONTGOMERY, p);
|
||||||
|
|
||||||
curve->m.mc = ecc_montgomery_curve(p, a, b);
|
curve->m.mc = ecc_montgomery_curve(p, a, b);
|
||||||
|
curve->m.log2_cofactor = log2_cofactor;
|
||||||
|
|
||||||
curve->m.G = ecc_montgomery_point_new(curve->m.mc, G_x);
|
curve->m.G = ecc_montgomery_point_new(curve->m.mc, G_x);
|
||||||
}
|
}
|
||||||
@ -194,7 +195,7 @@ static struct ec_curve *ec_curve25519(void)
|
|||||||
mp_int *a = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000076d06);
|
mp_int *a = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000076d06);
|
||||||
mp_int *b = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000000001);
|
mp_int *b = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000000001);
|
||||||
mp_int *G_x = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000000009);
|
mp_int *G_x = MP_LITERAL(0x0000000000000000000000000000000000000000000000000000000000000009);
|
||||||
initialise_mcurve(&curve, p, a, b, G_x);
|
initialise_mcurve(&curve, p, a, b, G_x, 3);
|
||||||
mp_free(p);
|
mp_free(p);
|
||||||
mp_free(a);
|
mp_free(a);
|
||||||
mp_free(b);
|
mp_free(b);
|
||||||
@ -1283,10 +1284,17 @@ static void ssh_ecdhkex_m_setup(ecdh_key *dh)
|
|||||||
random_read(strbuf_append(bytes, dh->curve->fieldBytes),
|
random_read(strbuf_append(bytes, dh->curve->fieldBytes),
|
||||||
dh->curve->fieldBytes);
|
dh->curve->fieldBytes);
|
||||||
|
|
||||||
bytes->u[0] &= 0xF8;
|
|
||||||
bytes->u[bytes->len-1] &= 0x7F;
|
|
||||||
bytes->u[bytes->len-1] |= 0x40;
|
|
||||||
dh->private = mp_from_bytes_le(ptrlen_from_strbuf(bytes));
|
dh->private = mp_from_bytes_le(ptrlen_from_strbuf(bytes));
|
||||||
|
|
||||||
|
/* Ensure the private key has the highest valid bit set, and no
|
||||||
|
* bits _above_ the highest valid one */
|
||||||
|
mp_reduce_mod_2to(dh->private, dh->curve->fieldBits);
|
||||||
|
mp_set_bit(dh->private, dh->curve->fieldBits - 1, 1);
|
||||||
|
|
||||||
|
/* Clear a curve-specific number of low bits */
|
||||||
|
for (unsigned bit = 0; bit < dh->curve->m.log2_cofactor; bit++)
|
||||||
|
mp_set_bit(dh->private, bit, 0);
|
||||||
|
|
||||||
strbuf_free(bytes);
|
strbuf_free(bytes);
|
||||||
|
|
||||||
dh->m_public = ecc_montgomery_multiply(dh->curve->m.G, dh->private);
|
dh->m_public = ecc_montgomery_multiply(dh->curve->m.G, dh->private);
|
||||||
|
Loading…
Reference in New Issue
Block a user