Skip to content

Commit 609cddb

Browse files
committed
nimble/ll: Add unit tests for CS DRBG
The BT specification provides sample data that can be used to test the correctness of a CS DRBG implementation.
1 parent 0307f73 commit 609cddb

File tree

4 files changed

+376
-0
lines changed

4 files changed

+376
-0
lines changed

nimble/controller/pkg.yml

Lines changed: 7 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,14 @@ pkg.deps:
4242
- nimble
4343
- nimble/transport
4444

45+
pkg.deps.TEST:
46+
- "@apache-mynewt-core/crypto/mbedtls"
47+
4548
pkg.init:
4649
ble_ll_init:
4750
- $before:ble_transport_hs_init
4851
- $before:ble_transport_ll_init
52+
53+
pkg.cflags.TEST:
54+
- -Irepos/mbedtls/include
55+
- -Irepos/include/mbedtls
Lines changed: 363 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,363 @@
1+
/*
2+
* Licensed to the Apache Software Foundation (ASF) under one
3+
* or more contributor license agreements. See the NOTICE file
4+
* distributed with this work for additional information
5+
* regarding copyright ownership. The ASF licenses this file
6+
* to you under the Apache License, Version 2.0 (the
7+
* "License"); you may not use this file except in compliance
8+
* with the License. You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing,
13+
* software distributed under the License is distributed on an
14+
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15+
* KIND, either express or implied. See the License for the
16+
* specific language governing permissions and limitations
17+
* under the License.
18+
*/
19+
20+
#include <stdint.h>
21+
#include <controller/ble_ll_utils.h>
22+
#include <testutil/testutil.h>
23+
#include "ble_ll_cs_drbg_priv.h"
24+
25+
static void
26+
ble_ll_cs_drbg_e_test(void)
27+
{
28+
uint8_t key[16] = { 0 };
29+
uint8_t data[16] = { 0 };
30+
uint8_t out[16] = { 0 };
31+
uint8_t expected_out[16] = { 0 };
32+
33+
/* Sample data from BLUETOOTH CORE SPECIFICATION Version 6.0 | Vol 6,
34+
* Part C, 1.1 Encrypt Command.
35+
* Swap because the copy-pasted strings are in leftmost (MSO) to rightmost
36+
* (LSO) orientation.
37+
*/
38+
swap_buf(key,
39+
(uint8_t[16]){ 0x4C, 0x68, 0x38, 0x41, 0x39, 0xF5, 0x74, 0xD8,
40+
0x36, 0xBC, 0xF3, 0x4E, 0x9D, 0xFB, 0x01, 0xBF },
41+
16);
42+
43+
swap_buf(data,
44+
(uint8_t[16]){ 0x02, 0x13, 0x24, 0x35, 0x46, 0x57, 0x68, 0x79,
45+
0xac, 0xbd, 0xce, 0xdf, 0xe0, 0xf1, 0x02, 0x13 },
46+
16);
47+
48+
swap_buf(expected_out,
49+
(uint8_t[16]){ 0x99, 0xad, 0x1b, 0x52, 0x26, 0xa3, 0x7e, 0x3e,
50+
0x05, 0x8e, 0x3b, 0x8e, 0x27, 0xc2, 0xc6, 0x66 },
51+
16);
52+
53+
ble_ll_cs_drbg_e(key, data, out);
54+
TEST_ASSERT(memcmp(out, expected_out, sizeof(out)) == 0);
55+
}
56+
57+
static void
58+
ble_ll_cs_drbg_f7_test(void)
59+
{
60+
uint8_t v_s[80];
61+
uint8_t k[16];
62+
uint8_t k2[16] = { 0 };
63+
uint8_t x[16] = { 0 };
64+
uint8_t expected_k2[16] = { 0 };
65+
uint8_t expected_x[16] = { 0 };
66+
67+
/* Sample data from BLUETOOTH CORE SPECIFICATION Version 6.0 | Vol 6,
68+
* Part C, 7. Deterministic random bit generator sample data.
69+
* Swap because the copy-pasted strings are in leftmost (MSO)
70+
* to rightmost (LSO) orientation.
71+
*/
72+
swap_buf(k,
73+
(uint8_t[16]){ 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
74+
0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F },
75+
16);
76+
77+
swap_buf(v_s,
78+
(uint8_t[80]){
79+
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
80+
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x28,
81+
0x00, 0x00, 0x00, 0x20, 0xE1, 0x0B, 0xC2, 0x8A, 0x0B, 0xFD,
82+
0xDF, 0xE9, 0x3E, 0x7F, 0x51, 0x86, 0xE0, 0xCA, 0x0B, 0x3B,
83+
0x9F, 0xF4, 0x77, 0xC1, 0x86, 0x73, 0x84, 0x0D, 0xC9, 0x80,
84+
0xDE, 0xDF, 0x98, 0x82, 0xED, 0x44, 0x64, 0xA6, 0x74, 0x96,
85+
0x78, 0x68, 0xF1, 0x43, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00,
86+
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 },
87+
80);
88+
89+
swap_buf(expected_k2,
90+
(uint8_t[16]){ 0x8B, 0x2B, 0x06, 0xDC, 0x52, 0x2D, 0x3E, 0x0A,
91+
0xF0, 0xA5, 0x0C, 0xAF, 0x48, 0x10, 0xE0, 0x35 },
92+
16);
93+
94+
TEST_ASSERT(ble_ll_cs_drbg_f7(k, v_s, sizeof(v_s), k2) == 0);
95+
TEST_ASSERT(memcmp(k2, expected_k2, sizeof(k2)) == 0);
96+
97+
v_s[76] = 0x01;
98+
swap_buf(expected_x,
99+
(uint8_t[16]){ 0xA3, 0x4F, 0xBE, 0x57, 0xF8, 0xF9, 0x7E, 0x34,
100+
0x9D, 0x15, 0xA3, 0x76, 0x79, 0x60, 0x74, 0x64 },
101+
16);
102+
103+
TEST_ASSERT(ble_ll_cs_drbg_f7(k, v_s, sizeof(v_s), x) == 0);
104+
TEST_ASSERT(memcmp(x, expected_x, sizeof(x)) == 0);
105+
}
106+
107+
static void
108+
ble_ll_cs_drbg_f8_test(void)
109+
{
110+
uint8_t input_bit_string[40] = { 0 };
111+
uint8_t expected_sm[32] = { 0 };
112+
uint8_t sm[32] = { 0 };
113+
114+
/* Sample data from BLUETOOTH CORE SPECIFICATION Version 6.0 | Vol 6,
115+
* Part C, 7. Deterministic random bit generator sample data.
116+
* Swap because the copy-pasted strings are in leftmost (MSO)
117+
* to rightmost (LSO) orientation.
118+
*/
119+
120+
/* 320-bit input bit string created from concatenated vectors
121+
* CS_IV || CS_IN || CS_PV
122+
*/
123+
swap_buf(input_bit_string,
124+
(uint8_t[40]){ 0xE1, 0x0B, 0xC2, 0x8A, 0x0B, 0xFD, 0xDF, 0xE9,
125+
0x3E, 0x7F, 0x51, 0x86, 0xE0, 0xCA, 0x0B, 0x3B,
126+
0x9F, 0xF4, 0x77, 0xC1, 0x86, 0x73, 0x84, 0x0D,
127+
0xC9, 0x80, 0xDE, 0xDF, 0x98, 0x82, 0xED, 0x44,
128+
0x64, 0xA6, 0x74, 0x96, 0x78, 0x68, 0xF1, 0x43 },
129+
40);
130+
131+
swap_buf(expected_sm,
132+
(uint8_t[32]){ 0xB6, 0x02, 0xB1, 0xB2, 0x8C, 0x6F, 0x0A, 0x3D,
133+
0xDA, 0xE6, 0x37, 0xB4, 0x84, 0x25, 0x08, 0x7D,
134+
0xDC, 0x18, 0x8C, 0x89, 0xA1, 0xB0, 0xCD, 0xFD,
135+
0xA1, 0xE8, 0xFC, 0x66, 0xC9, 0x99, 0x97, 0x50 },
136+
32);
137+
138+
TEST_ASSERT(ble_ll_cs_drbg_f8(input_bit_string, sm) == 0);
139+
TEST_ASSERT(memcmp(sm, expected_sm, sizeof(sm)) == 0);
140+
}
141+
142+
static void
143+
ble_ll_cs_drbg_f9_test(void)
144+
{
145+
uint8_t sm[32] = { 0 };
146+
uint8_t k[16] = { 0 };
147+
uint8_t v[16] = { 0 };
148+
uint8_t expected_k[16] = { 0 };
149+
uint8_t expected_v[16] = { 0 };
150+
151+
/* First call to f9 from instantiation function h9,
152+
* K and V vectors filled with zeros.
153+
*
154+
* Sample data from BLUETOOTH CORE SPECIFICATION Version 6.0 | Vol 6,
155+
* Part C, 7. Deterministic random bit generator sample data.
156+
* Swap because the copy-pasted strings are in leftmost (MSO)
157+
* to rightmost (LSO) orientation.
158+
*/
159+
160+
swap_buf(sm, (uint8_t[32]){ 0xB6, 0x02, 0xB1, 0xB2, 0x8C, 0x6F, 0x0A, 0x3D,
161+
0xDA, 0xE6, 0x37, 0xB4, 0x84, 0x25, 0x08, 0x7D,
162+
0xDC, 0x18, 0x8C, 0x89, 0xA1, 0xB0, 0xCD, 0xFD,
163+
0xA1, 0xE8, 0xFC, 0x66, 0xC9, 0x99, 0x97, 0x50 },
164+
32);
165+
166+
swap_buf(expected_k,
167+
(uint8_t[16]){ 0xEE, 0xE0, 0x4D, 0x7C, 0x76, 0x11, 0x3A, 0x5C,
168+
0xEC, 0x99, 0x2A, 0xE3, 0x20, 0xC2, 0x4D, 0x27 },
169+
16);
170+
171+
swap_buf(expected_v,
172+
(uint8_t[16]){ 0xDF, 0x90, 0x56, 0x47, 0xC1, 0x06, 0x6E, 0x6F,
173+
0x52, 0xC0, 0x3E, 0xDF, 0xB8, 0x2B, 0x69, 0x28 },
174+
16);
175+
176+
TEST_ASSERT(ble_ll_cs_drbg_f9(sm, k, v) == 0);
177+
TEST_ASSERT(memcmp(k, expected_k, sizeof(k)) == 0);
178+
TEST_ASSERT(memcmp(v, expected_v, sizeof(v)) == 0);
179+
}
180+
181+
static void
182+
cs_drbg_init(struct ble_ll_cs_drbg_ctx *ctx)
183+
{
184+
memset(ctx, 0, sizeof(*ctx));
185+
186+
/* CS_IV = CS_IV_P || CS_IV_C */
187+
swap_buf(ctx->iv,
188+
(uint8_t[16]){ 0xE1, 0x0B, 0xC2, 0x8A, 0x0B, 0xFD, 0xDF, 0xE9,
189+
0x3E, 0x7F, 0x51, 0x86, 0xE0, 0xCA, 0x0B, 0x3B },
190+
16);
191+
192+
/* CS_IN = CS_IN_P || CS_IN_C */
193+
swap_buf(ctx->in,
194+
(uint8_t[8]){ 0x9F, 0xF4, 0x77, 0xC1, 0x86, 0x73, 0x84, 0x0D }, 8);
195+
196+
/* CS_PV = CS_PV_P || CS_PV_C */
197+
swap_buf(ctx->pv,
198+
(uint8_t[16]){ 0xC9, 0x80, 0xDE, 0xDF, 0x98, 0x82, 0xED, 0x44,
199+
0x64, 0xA6, 0x74, 0x96, 0x78, 0x68, 0xF1, 0x43 },
200+
16);
201+
202+
ble_ll_cs_drbg_init(ctx);
203+
}
204+
205+
static void
206+
ble_ll_cs_drbg_rand_test(void)
207+
{
208+
struct ble_ll_cs_drbg_ctx ctx;
209+
uint8_t output[20] = { 0 };
210+
uint8_t expected_output[20] = { 0 };
211+
212+
/* Test if subsequent drgb generator calls returns expected bit sequences. */
213+
214+
cs_drbg_init(&ctx);
215+
216+
/* First round - request full 128-bit batch */
217+
swap_buf(expected_output,
218+
(uint8_t[16]){ 0x79, 0x74, 0x1F, 0xD1, 0x8F, 0x57, 0x7B, 0x45,
219+
0xD0, 0x9A, 0x66, 0x5A, 0x7F, 0x1F, 0x28, 0x58 },
220+
16);
221+
222+
TEST_ASSERT(ble_ll_cs_drbg_rand(&ctx, 0x00, BLE_LL_CS_DRBG_HOP_CHAN_NON_MODE0,
223+
output, 16) == 0);
224+
TEST_ASSERT(memcmp(output, expected_output, 16) == 0);
225+
}
226+
227+
static void
228+
ble_ll_cs_drbg_chan_selection_3b_test(void)
229+
{
230+
struct ble_ll_cs_drbg_ctx ctx;
231+
uint8_t filtered_channels[72] = { 0 };
232+
uint8_t shuffled_channels[72] = { 0 };
233+
uint8_t expected_shuffled_channels[19] = { 0 };
234+
235+
cs_drbg_init(&ctx);
236+
237+
memcpy(filtered_channels,
238+
(uint8_t[19]){ 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16,
239+
17, 18, 19, 20 },
240+
19);
241+
242+
memcpy(expected_shuffled_channels,
243+
(uint8_t[19]){ 11, 7, 14, 18, 9, 19, 10, 8, 5, 2, 4, 15, 16, 13, 12,
244+
6, 17, 20, 3 },
245+
19);
246+
assert(ble_ll_cs_drbg_shuffle_cr1(&ctx, 0x00, BLE_LL_CS_DRBG_HOP_CHAN_MODE0,
247+
filtered_channels, shuffled_channels, 19) == 0);
248+
assert(memcmp(shuffled_channels, expected_shuffled_channels, 19) == 0);
249+
250+
memcpy(expected_shuffled_channels,
251+
(uint8_t[19]){ 6, 12, 5, 10, 3, 2, 18, 17, 16, 8, 11, 7, 19, 4, 13,
252+
20, 9, 15, 14 },
253+
19);
254+
assert(ble_ll_cs_drbg_shuffle_cr1(&ctx, 0x03, BLE_LL_CS_DRBG_HOP_CHAN_NON_MODE0,
255+
filtered_channels, shuffled_channels, 19) == 0);
256+
assert(memcmp(shuffled_channels, expected_shuffled_channels, 19) == 0);
257+
}
258+
259+
static void
260+
ble_ll_cs_drbg_generate_aa_test(void)
261+
{
262+
struct ble_ll_cs_drbg_ctx ctx;
263+
uint32_t initiator_aa;
264+
uint32_t reflector_aa;
265+
uint32_t expected_initiator_aa;
266+
uint32_t expected_reflector_aa;
267+
268+
cs_drbg_init(&ctx);
269+
270+
/* Step 0 */
271+
assert(ble_ll_cs_drbg_generate_aa(&ctx, 0, &initiator_aa, &reflector_aa) == 0);
272+
expected_initiator_aa = get_be32((uint8_t[4]){ 0x6C, 0x37, 0x6A, 0xB8 });
273+
expected_reflector_aa = get_be32((uint8_t[4]){ 0xF0, 0x79, 0xBC, 0x3A });
274+
assert(initiator_aa == expected_initiator_aa);
275+
assert(reflector_aa == expected_reflector_aa);
276+
277+
/* Step 1 */
278+
assert(ble_ll_cs_drbg_generate_aa(&ctx, 1, &initiator_aa, &reflector_aa) == 0);
279+
expected_initiator_aa = get_be32((uint8_t[4]){ 0x01, 0x1C, 0xAE, 0x4E });
280+
expected_reflector_aa = get_be32((uint8_t[4]){ 0xD0, 0x6A, 0xCD, 0xDA });
281+
assert(initiator_aa == expected_initiator_aa);
282+
assert(reflector_aa == expected_reflector_aa);
283+
284+
/* Step 2 */
285+
assert(ble_ll_cs_drbg_generate_aa(&ctx, 2, &initiator_aa, &reflector_aa) == 0);
286+
expected_initiator_aa = get_be32((uint8_t[4]){ 0x64, 0x06, 0x12, 0x14 });
287+
expected_reflector_aa = get_be32((uint8_t[4]){ 0x28, 0x94, 0x2F, 0x38 });
288+
assert(initiator_aa == expected_initiator_aa);
289+
assert(reflector_aa == expected_reflector_aa);
290+
291+
/* Step 14 */
292+
assert(ble_ll_cs_drbg_generate_aa(&ctx, 14, &initiator_aa, &reflector_aa) == 0);
293+
expected_initiator_aa = get_be32((uint8_t[4]){ 0xF7, 0x21, 0x97, 0x86 });
294+
expected_reflector_aa = get_be32((uint8_t[4]){ 0x57, 0x17, 0x64, 0x70 });
295+
assert(initiator_aa == expected_initiator_aa);
296+
assert(reflector_aa == expected_reflector_aa);
297+
}
298+
299+
static void
300+
ble_ll_cs_drbg_rand_marker_position_test(void)
301+
{
302+
uint8_t position1;
303+
uint8_t position2;
304+
struct ble_ll_cs_drbg_ctx ctx;
305+
306+
cs_drbg_init(&ctx);
307+
308+
/* Step 9 */
309+
assert(ble_ll_cs_drbg_rand_marker_position(&ctx, 9, BLE_LL_CS_RTT_32_BIT_SOUNDING_SEQUENCE,
310+
&position1, &position2) == 0);
311+
assert(position1 == 12 && position2 == 0xFF);
312+
313+
assert(ble_ll_cs_drbg_rand_marker_position(&ctx, 9, BLE_LL_CS_RTT_32_BIT_SOUNDING_SEQUENCE,
314+
&position1, &position2) == 0);
315+
assert(position1 == 4 && position2 == 0xFF);
316+
317+
/* Step 14 */
318+
assert(ble_ll_cs_drbg_rand_marker_position(&ctx, 14, BLE_LL_CS_RTT_32_BIT_SOUNDING_SEQUENCE,
319+
&position1, &position2) == 0);
320+
assert(position1 == 8 && position2 == 0xFF);
321+
322+
assert(ble_ll_cs_drbg_rand_marker_position(&ctx, 14, BLE_LL_CS_RTT_32_BIT_SOUNDING_SEQUENCE,
323+
&position1, &position2) == 0);
324+
assert(position1 == 11 && position2 == 0xFF);
325+
}
326+
327+
static void
328+
ble_ll_cs_drbg_rand_marker_selection_test(void)
329+
{
330+
uint8_t marker_selection;
331+
struct ble_ll_cs_drbg_ctx ctx;
332+
333+
cs_drbg_init(&ctx);
334+
335+
/* Step 9 */
336+
assert(ble_ll_cs_drbg_rand_marker_selection(&ctx, 0x09, &marker_selection) == 0);
337+
assert(marker_selection == 0x00);
338+
339+
assert(ble_ll_cs_drbg_rand_marker_selection(&ctx, 0x09, &marker_selection) == 0);
340+
assert(marker_selection == 0x80);
341+
342+
memset(ctx.t_cache, 0, sizeof(ctx.t_cache));
343+
344+
/* Step 14 */
345+
assert(ble_ll_cs_drbg_rand_marker_selection(&ctx, 14, &marker_selection) == 0);
346+
assert(marker_selection == 0x80);
347+
348+
assert(ble_ll_cs_drbg_rand_marker_selection(&ctx, 14, &marker_selection) == 0);
349+
assert(marker_selection == 0x80);
350+
}
351+
352+
TEST_SUITE(ble_ll_cs_drbg_test_suite)
353+
{
354+
ble_ll_cs_drbg_e_test();
355+
ble_ll_cs_drbg_f7_test();
356+
ble_ll_cs_drbg_f8_test();
357+
ble_ll_cs_drbg_f9_test();
358+
ble_ll_cs_drbg_rand_test();
359+
ble_ll_cs_drbg_chan_selection_3b_test();
360+
ble_ll_cs_drbg_generate_aa_test();
361+
ble_ll_cs_drbg_rand_marker_position_test();
362+
ble_ll_cs_drbg_rand_marker_selection_test();
363+
}

nimble/controller/test/src/ble_ll_test.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ TEST_SUITE_DECL(ble_ll_crypto_test_suite);
2727
TEST_SUITE_DECL(ble_ll_csa2_test_suite);
2828
TEST_SUITE_DECL(ble_ll_isoal_test_suite);
2929
TEST_SUITE_DECL(ble_ll_iso_test_suite);
30+
TEST_SUITE_DECL(ble_ll_cs_drbg_test_suite);
3031

3132
int
3233
main(int argc, char **argv)
@@ -36,6 +37,7 @@ main(int argc, char **argv)
3637
ble_ll_csa2_test_suite();
3738
ble_ll_isoal_test_suite();
3839
ble_ll_iso_test_suite();
40+
ble_ll_cs_drbg_test_suite();
3941

4042
return tu_any_failed;
4143
}

0 commit comments

Comments
 (0)