Coverage Report

Created: 2026-05-18 06:53

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/cryptofuzz-sp-math-all-8bit/modules/wolfcrypt/ecdsa_448.cpp
Line
Count
Source
1
#include "ecdsa_448.h"
2
#include "module_internal.h"
3
#include "shared.h"
4
#include "bn_ops.h"
5
#include <cryptofuzz/util.h>
6
7
extern "C" {
8
#include <wolfssl/options.h>
9
#include <wolfssl/wolfcrypt/curve448.h>
10
#include <wolfssl/wolfcrypt/ed448.h>
11
}
12
13
namespace cryptofuzz {
14
namespace module {
15
namespace wolfCrypt_detail {
16
17
#if defined(CRYPTOFUZZ_WOLFCRYPT_ALLOCATION_FAILURES)
18
    extern bool haveAllocFailure;
19
#endif
20
21
374
static bool ed448LoadPrivateKey(ed448_key& key, component::Bignum priv, Datasource& ds) {
22
374
    bool ret = false;
23
24
374
    uint8_t priv_bytes[ED448_KEY_SIZE];
25
26
374
    CF_CHECK_EQ(wolfCrypt_bignum::Bignum::ToBin(ds, priv, priv_bytes, sizeof(priv_bytes)), true);
27
357
    CF_CHECK_EQ(wc_ed448_import_private_only(priv_bytes, sizeof(priv_bytes), &key), 0);
28
29
357
    ret = true;
30
374
end:
31
374
    return ret;
32
357
}
33
34
643
static std::optional<std::vector<uint8_t>> ed448GetPublicKeyAsVector(ed448_key& key) {
35
643
    std::optional<std::vector<uint8_t>> ret = std::nullopt;
36
643
    uint8_t pub_bytes[ED448_PUB_KEY_SIZE];
37
38
643
    WC_CHECK_EQ(wc_ed448_make_public(&key, pub_bytes, sizeof(pub_bytes)), MP_OKAY);
39
40
613
    ret = std::vector<uint8_t>(pub_bytes, pub_bytes + sizeof(pub_bytes));
41
643
end:
42
643
    return ret;
43
613
}
44
45
286
static std::optional<component::ECC_PublicKey> ed448GetPublicKey(ed448_key& key, Datasource& ds) {
46
286
    std::optional<component::ECC_PublicKey> ret = std::nullopt;
47
286
    std::optional<std::vector<uint8_t>> pubv = std::nullopt;
48
286
    std::optional<component::Bignum> pub = std::nullopt;
49
50
286
    CF_CHECK_NE(pubv = ed448GetPublicKeyAsVector(key), std::nullopt);
51
274
    CF_CHECK_NE(pub = wolfCrypt_bignum::Bignum::BinToBignum(ds, pubv->data(), pubv->size()), std::nullopt);
52
53
271
    ret = {pub->ToString(), "0"};
54
55
286
end:
56
286
    return ret;
57
271
}
58
59
357
static bool ed448DerivePublicKey(ed448_key& key) {
60
357
    std::optional<std::vector<uint8_t>> pubv = std::nullopt;
61
357
    bool ret = false;
62
63
357
    CF_CHECK_NE(pubv = ed448GetPublicKeyAsVector(key), std::nullopt);
64
339
    memcpy(key.p, pubv->data(), ED448_PUB_KEY_SIZE);
65
339
    key.pubKeySet = 1;
66
67
339
    ret = true;
68
69
357
end:
70
357
    return ret;
71
339
}
72
73
521
static bool ed448LoadPublicKey(ed448_key& key, component::Bignum pub, Datasource& ds, const bool mustSucceed = false) {
74
521
    bool ret = false;
75
76
521
    uint8_t pub_bytes[ED448_PUB_KEY_SIZE];
77
521
    CF_CHECK_EQ(wolfCrypt_bignum::Bignum::ToBin(ds, pub, pub_bytes, sizeof(pub_bytes), mustSucceed), true);
78
507
    CF_CHECK_EQ(wc_ed448_import_public(pub_bytes, sizeof(pub_bytes), &key), 0);
79
80
483
    ret = true;
81
521
end:
82
521
    return ret;
83
483
}
84
85
174
std::optional<component::ECC_PublicKey> OpECC_PrivateToPublic_Curve448(operation::ECC_PrivateToPublic& op) {
86
174
    std::optional<component::ECC_PublicKey> ret = std::nullopt;
87
174
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
88
174
    wolfCrypt_detail::SetGlobalDs(&ds);
89
90
174
    static const uint8_t basepoint[CURVE448_KEY_SIZE] = {5};
91
92
174
    std::optional<std::vector<uint8_t>> priv_bytes = std::nullopt;
93
174
    uint8_t pub_bytes[CURVE448_PUB_KEY_SIZE];
94
95
    /* Load private key */
96
174
    {
97
174
        priv_bytes = util::DecToBin(op.priv.ToTrimmedString(), CURVE448_KEY_SIZE);
98
174
        CF_CHECK_NE(priv_bytes, std::nullopt);
99
100
145
        priv_bytes->data()[0] &= 0xFC;
101
145
        priv_bytes->data()[55] |= 0x80; static_assert(55 < CURVE448_KEY_SIZE);
102
145
    }
103
104
    /* Convert to public key */
105
0
    {
106
145
        CF_CHECK_EQ(curve448(pub_bytes, priv_bytes->data(), basepoint), MP_OKAY);
107
145
    }
108
109
    /* Convert public key */
110
0
    {
111
145
        ret = { util::BinToDec(pub_bytes, sizeof(pub_bytes)), "0" };
112
145
    }
113
114
174
end:
115
174
    wolfCrypt_detail::UnsetGlobalDs();
116
174
    return ret;
117
145
}
118
119
77
std::optional<component::ECC_PublicKey> OpECC_PrivateToPublic_Ed448(operation::ECC_PrivateToPublic& op) {
120
77
    std::optional<component::ECC_PublicKey> ret = std::nullopt;
121
77
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
122
77
    wolfCrypt_detail::SetGlobalDs(&ds);
123
124
77
    uint8_t pub_bytes[ED448_PUB_KEY_SIZE];
125
126
77
    ed448_key key;
127
77
    bool e448_key_inited = false;
128
129
77
    WC_CHECK_EQ(wc_ed448_init(&key), 0);
130
77
    e448_key_inited = true;
131
132
    /* Load private key */
133
77
    {
134
77
        const auto priv_bytes = util::DecToBin(op.priv.ToTrimmedString(), ED448_KEY_SIZE);
135
77
        CF_CHECK_NE(priv_bytes, std::nullopt);
136
137
61
        WC_CHECK_EQ(wc_ed448_import_private_only(priv_bytes->data(), priv_bytes->size(), &key), 0);
138
61
    }
139
140
    /* Convert to public key */
141
0
    {
142
61
        WC_CHECK_EQ(wc_ed448_make_public(&key, pub_bytes, sizeof(pub_bytes)), MP_OKAY);
143
47
    }
144
145
    /* Convert public key */
146
0
    {
147
47
        ret = { util::BinToDec(pub_bytes, sizeof(pub_bytes)), "0" };
148
47
    }
149
150
77
end:
151
77
    if ( e448_key_inited == true ) {
152
77
        wc_ed448_free(&key);
153
77
    }
154
155
77
    wolfCrypt_detail::UnsetGlobalDs();
156
77
    return ret;
157
47
}
158
159
0
std::optional<bool> OpECC_ValidatePubkey_Ed448(operation::ECC_ValidatePubkey& op) {
160
0
    std::optional<bool> ret = std::nullopt;
161
0
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
162
0
    wolfCrypt_detail::SetGlobalDs(&ds);
163
164
0
    ed448_key key;
165
0
    bool e448_key_inited = false;
166
167
0
    WC_CHECK_EQ(wc_ed448_init(&key), 0);
168
0
    e448_key_inited = true;
169
170
0
    CF_CHECK_EQ(ed448LoadPublicKey(key, op.pub.first, ds), true);
171
172
0
    haveAllocFailure = false;
173
0
    ret = wc_ed448_check_key(&key) == 0;
174
0
    if ( *ret == false && haveAllocFailure == true ) {
175
0
        ret = std::nullopt;
176
0
    }
177
178
0
end:
179
0
    if ( e448_key_inited == true ) {
180
0
        wc_ed448_free(&key);
181
0
    }
182
183
0
    wolfCrypt_detail::UnsetGlobalDs();
184
0
    return ret;
185
0
}
186
187
0
std::optional<bool> OpECC_ValidatePubkey_Curve448(operation::ECC_ValidatePubkey& op) {
188
0
    std::optional<bool> ret = std::nullopt;
189
0
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
190
0
    wolfCrypt_detail::SetGlobalDs(&ds);
191
192
0
    wolfCrypt_bignum::Bignum pub(ds);
193
0
    uint8_t pub_bytes[CURVE448_KEY_SIZE];
194
195
0
    CF_CHECK_EQ(pub.Set(op.pub.first.ToString(ds)), true);
196
0
    CF_CHECK_TRUE(pub.ToBin(pub_bytes, sizeof(pub_bytes)));
197
198
0
    haveAllocFailure = false;
199
0
    ret = wc_curve448_check_public(pub_bytes, sizeof(pub_bytes), EC448_BIG_ENDIAN) == 0;
200
0
    if ( *ret == false && haveAllocFailure == true ) {
201
0
        ret = std::nullopt;
202
0
    }
203
204
0
end:
205
206
0
    wolfCrypt_detail::UnsetGlobalDs();
207
0
    return ret;
208
0
}
209
210
374
std::optional<component::ECDSA_Signature> OpECDSA_Sign_ed448(operation::ECDSA_Sign& op) {
211
374
    std::optional<component::ECDSA_Signature> ret = std::nullopt;
212
374
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
213
374
    wolfCrypt_detail::SetGlobalDs(&ds);
214
215
374
    ed448_key key;
216
374
    bool e448_key_inited = false;
217
374
    uint8_t sig[ED448_SIG_SIZE];
218
374
    word32 sigSz = sizeof(sig);
219
220
374
    WC_CHECK_EQ(wc_ed448_init(&key), 0);
221
374
    e448_key_inited = true;
222
223
374
    CF_CHECK_EQ(wolfCrypt_detail::ed448LoadPrivateKey(key, op.priv, ds), true);
224
357
    CF_CHECK_EQ(wolfCrypt_detail::ed448DerivePublicKey(key), true);
225
226
    /* Sign message */
227
339
    WC_CHECK_EQ(wc_ed448_sign_msg(op.cleartext.GetPtr(), op.cleartext.GetSize(), sig, &sigSz, &key, nullptr, 0), MP_OKAY);
228
314
    CF_CHECK_EQ(sigSz, ED448_SIG_SIZE);
229
314
    static_assert(ED448_SIG_SIZE % 2 == 0);
230
231
314
    {
232
314
        std::optional<component::BignumPair> ret_sig;
233
314
        std::optional<component::BignumPair> ret_pub;
234
235
314
        CF_CHECK_NE(ret_sig = wolfCrypt_bignum::Bignum::BinToBignumPair(ds, sig, ED448_SIG_SIZE), std::nullopt);
236
286
        CF_CHECK_NE(ret_pub = wolfCrypt_detail::ed448GetPublicKey(key, ds), std::nullopt);
237
238
271
        ret = component::ECDSA_Signature(*ret_sig, *ret_pub);
239
271
    }
240
241
374
end:
242
374
    if ( e448_key_inited == true ) {
243
374
        wc_ed448_free(&key);
244
374
    }
245
246
374
    wolfCrypt_detail::UnsetGlobalDs();
247
374
    return ret;
248
271
}
249
250
521
std::optional<bool> OpECDSA_Verify_ed448(operation::ECDSA_Verify& op) {
251
521
    std::optional<bool> ret = std::nullopt;
252
521
    Datasource ds(op.modifier.GetPtr(), op.modifier.GetSize());
253
521
    wolfCrypt_detail::SetGlobalDs(&ds);
254
255
521
    ed448_key key;
256
521
    bool e448_key_inited = false;
257
521
    uint8_t ed448sig[ED448_SIG_SIZE];
258
521
    int verify;
259
521
    bool oneShot = true;
260
521
    bool haveEmptyPart = false;
261
262
521
    haveAllocFailure = false;
263
521
    ret = false;
264
265
521
    WC_CHECK_EQ(wc_ed448_init(&key), 0);
266
521
    e448_key_inited = true;
267
268
521
    CF_CHECK_EQ(ed448LoadPublicKey(key, op.signature.pub.first, ds, true), true);
269
483
    CF_CHECK_EQ(wolfCrypt_bignum::Bignum::ToBin(ds, op.signature.signature, ed448sig, sizeof(ed448sig), true), true);
270
271
458
#if defined(WOLFSSL_ED25519_STREAMING_VERIFY)
272
458
    try { oneShot = ds.Get<bool>(); } catch ( ... ) { }
273
458
#endif
274
275
458
    if ( oneShot == true ) {
276
314
        WC_CHECK_EQ(wc_ed448_verify_msg(ed448sig, sizeof(ed448sig), op.cleartext.GetPtr(), op.cleartext.GetSize(), &verify, &key, nullptr, 0), 0);
277
177
    } else {
278
#if !defined(WOLFSSL_ED25519_STREAMING_VERIFY)
279
        CF_UNREACHABLE();
280
#else
281
144
        const auto parts = util::ToParts(ds, op.cleartext);
282
283
144
        WC_CHECK_EQ(wc_ed448_verify_msg_init(ed448sig, sizeof(ed448sig), &key, (byte)Ed448, nullptr, 0), 0);
284
285
141k
        for (const auto& part : parts) {
286
141k
            if ( part.second == 0 ) {
287
140k
                haveEmptyPart = true;
288
140k
            }
289
141k
            WC_CHECK_EQ(wc_ed448_verify_msg_update(part.first, part.second, &key), 0);
290
141k
        }
291
292
223
        WC_CHECK_EQ(wc_ed448_verify_msg_final(ed448sig, sizeof(ed448sig), &verify, &key), 0);
293
223
#endif
294
223
    }
295
296
258
    ret = verify ? true : false;
297
298
521
end:
299
521
    if ( e448_key_inited == true ) {
300
521
        wc_ed448_free(&key);
301
521
    }
302
303
521
    wolfCrypt_detail::UnsetGlobalDs();
304
305
521
    if ( ret && *ret == false ) {
306
263
        if ( haveAllocFailure ) {
307
64
            ret = std::nullopt;
308
199
        } else if ( haveEmptyPart ) {
309
40
            ret = std::nullopt;
310
159
        } else if ( op.cleartext.IsZero() ) {
311
13
            ret = std::nullopt;
312
13
        }
313
314
263
    }
315
521
    return ret;
316
258
}
317
318
} /* namespace wolfCrypt_detail */
319
} /* namespace module */
320
} /* namespace cryptofuzz */