diff options
author | doug <> | 2015-06-27 23:35:52 +0000 |
---|---|---|
committer | doug <> | 2015-06-27 23:35:52 +0000 |
commit | 519b6c342127d0c210bdf99875f8afe00c6a30cf (patch) | |
tree | d71cf580866a5b49fd8f4c0d554fd7807d5aac0f | |
parent | b1e0a2287902704dfe165a7ba7d1fe23b22bf25f (diff) | |
download | openbsd-519b6c342127d0c210bdf99875f8afe00c6a30cf.tar.gz openbsd-519b6c342127d0c210bdf99875f8afe00c6a30cf.tar.bz2 openbsd-519b6c342127d0c210bdf99875f8afe00c6a30cf.zip |
Add unit tests for LibreSSL.
cipher_list.c is based on code from jsing@.
Discussed with jsing@
-rw-r--r-- | src/regress/lib/libssl/unit/Makefile | 20 | ||||
-rw-r--r-- | src/regress/lib/libssl/unit/cipher_list.c | 192 | ||||
-rw-r--r-- | src/regress/lib/libssl/unit/tests.h | 44 |
3 files changed, 256 insertions, 0 deletions
diff --git a/src/regress/lib/libssl/unit/Makefile b/src/regress/lib/libssl/unit/Makefile new file mode 100644 index 0000000000..c014615900 --- /dev/null +++ b/src/regress/lib/libssl/unit/Makefile | |||
@@ -0,0 +1,20 @@ | |||
1 | # $OpenBSD: Makefile,v 1.1 2015/06/27 23:35:52 doug Exp $ | ||
2 | |||
3 | TEST_CASES+= cipher_list | ||
4 | |||
5 | REGRESS_TARGETS= all_tests | ||
6 | |||
7 | LDADD= -lcrypto -lssl | ||
8 | DPADD= ${LIBCRYPTO} ${LIBSSL} | ||
9 | WARNINGS= Yes | ||
10 | LDFLAGS+= -lcrypto -lssl | ||
11 | CFLAGS+= -DLIBRESSL_INTERNAL -Wall -Wundef -Werror -I../../../../lib/libssl/src/ssl | ||
12 | |||
13 | CLEANFILES+= ${TEST_CASES} | ||
14 | |||
15 | all_tests: ${TEST_CASES} | ||
16 | @for test in $>; do \ | ||
17 | ./$$test; \ | ||
18 | done | ||
19 | |||
20 | .include <bsd.regress.mk> | ||
diff --git a/src/regress/lib/libssl/unit/cipher_list.c b/src/regress/lib/libssl/unit/cipher_list.c new file mode 100644 index 0000000000..b513007771 --- /dev/null +++ b/src/regress/lib/libssl/unit/cipher_list.c | |||
@@ -0,0 +1,192 @@ | |||
1 | /* $OpenBSD: cipher_list.c,v 1.1 2015/06/27 23:35:52 doug Exp $ */ | ||
2 | /* | ||
3 | * Copyright (c) 2015 Doug Hogan <doug@openbsd.org> | ||
4 | * Copyright (c) 2015 Joel Sing <jsing@openbsd.org> | ||
5 | * | ||
6 | * Permission to use, copy, modify, and distribute this software for any | ||
7 | * purpose with or without fee is hereby granted, provided that the above | ||
8 | * copyright notice and this permission notice appear in all copies. | ||
9 | * | ||
10 | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | ||
11 | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | ||
12 | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | ||
13 | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | ||
14 | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | ||
15 | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | ||
16 | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | ||
17 | */ | ||
18 | |||
19 | /* | ||
20 | * Test TLS ssl bytes (aka cipher suites) to cipher list and back. | ||
21 | * | ||
22 | * TLSv1.0 - RFC 2246 section 7.4.1.2 (ClientHello struct) | ||
23 | * TLSv1.1 - RFC 4346 section 7.4.1.2 (ClientHello struct) | ||
24 | * TLSv1.2 - RFC 5246 section 7.4.1.2 (ClientHello struct) | ||
25 | * | ||
26 | * In all of these standards, the relevant structures are: | ||
27 | * | ||
28 | * uint8 CipherSuite[2]; | ||
29 | * | ||
30 | * struct { | ||
31 | * ... | ||
32 | * CipherSuite cipher_suites<2..2^16-2> | ||
33 | * ... | ||
34 | * } ClientHello; | ||
35 | */ | ||
36 | |||
37 | #include <openssl/ssl.h> | ||
38 | |||
39 | #include <stdio.h> | ||
40 | #include <string.h> | ||
41 | |||
42 | #include "tests.h" | ||
43 | |||
44 | static uint8_t cipher_bytes[] = { | ||
45 | 0xcc, 0x14, /* ECDHE-ECDSA-CHACHA20-POLY1305 */ | ||
46 | 0xcc, 0x13, /* ECDHE-RSA-CHACHA20-POLY1305 */ | ||
47 | 0xcc, 0x15, /* DHE-RSA-CHACHA20-POLY1305 */ | ||
48 | 0x00, 0x9c, /* AES128-GCM-SHA256 */ | ||
49 | 0x00, 0x3d, /* AES256-SHA256 */ | ||
50 | 0x00, 0x09, /* DES-CBC-SHA */ | ||
51 | }; | ||
52 | |||
53 | static uint16_t cipher_values[] = { | ||
54 | 0xcc14, /* ECDHE-ECDSA-CHACHA20-POLY1305 */ | ||
55 | 0xcc13, /* ECDHE-RSA-CHACHA20-POLY1305 */ | ||
56 | 0xcc15, /* DHE-RSA-CHACHA20-POLY1305 */ | ||
57 | 0x009c, /* AES128-GCM-SHA256 */ | ||
58 | 0x003d, /* AES256-SHA256 */ | ||
59 | 0x0009, /* DES-CBC-SHA */ | ||
60 | }; | ||
61 | |||
62 | #define N_CIPHERS (sizeof(cipher_bytes) / 2) | ||
63 | |||
64 | extern STACK_OF(SSL_CIPHER) *ssl_bytes_to_cipher_list(SSL *s, | ||
65 | const unsigned char *p, int num); | ||
66 | extern int ssl_cipher_list_to_bytes(SSL *s, STACK_OF(SSL_CIPHER) *sk, | ||
67 | unsigned char *p); | ||
68 | |||
69 | static int | ||
70 | ssl_bytes_to_list_alloc(SSL *s, STACK_OF(SSL_CIPHER) **ciphers) | ||
71 | { | ||
72 | SSL_CIPHER *cipher; | ||
73 | uint16_t value; | ||
74 | int i; | ||
75 | |||
76 | *ciphers = ssl_bytes_to_cipher_list(s, cipher_bytes, | ||
77 | sizeof(cipher_bytes)); | ||
78 | CHECK(*ciphers != NULL); | ||
79 | CHECK(sk_SSL_CIPHER_num(*ciphers) == N_CIPHERS); | ||
80 | for (i = 0; i < sk_SSL_CIPHER_num(*ciphers); i++) { | ||
81 | cipher = sk_SSL_CIPHER_value(*ciphers, i); | ||
82 | CHECK(cipher != NULL); | ||
83 | value = SSL_CIPHER_get_value(cipher); | ||
84 | CHECK(value == cipher_values[i]); | ||
85 | } | ||
86 | |||
87 | return 1; | ||
88 | } | ||
89 | |||
90 | static int | ||
91 | ssl_list_to_bytes_scsv(SSL *s, STACK_OF(SSL_CIPHER) **ciphers) | ||
92 | { | ||
93 | unsigned char *buf = NULL; | ||
94 | size_t buflen; | ||
95 | int len; | ||
96 | int ret = 0; | ||
97 | |||
98 | /* Space for cipher bytes, plus reneg SCSV and two spare bytes. */ | ||
99 | CHECK(sk_SSL_CIPHER_num(*ciphers) == N_CIPHERS); | ||
100 | buflen = sizeof(cipher_bytes) + 2 + 2; | ||
101 | CHECK((buf = calloc(1, buflen)) != NULL); | ||
102 | |||
103 | len = ssl_cipher_list_to_bytes(s, *ciphers, buf); | ||
104 | CHECK_GOTO(len > 0 && (size_t)len == buflen - 2); | ||
105 | CHECK_GOTO(memcmp(buf, cipher_bytes, sizeof(cipher_bytes)) == 0); | ||
106 | CHECK_GOTO(buf[buflen - 4] == 0x00 && buf[buflen - 3] == 0xff); | ||
107 | CHECK_GOTO(buf[buflen - 2] == 0x00 && buf[buflen - 1] == 0x00); | ||
108 | |||
109 | ret = 1; | ||
110 | |||
111 | err: | ||
112 | free(buf); | ||
113 | return ret; | ||
114 | } | ||
115 | |||
116 | static int | ||
117 | ssl_list_to_bytes_no_scsv(SSL *s, STACK_OF(SSL_CIPHER) **ciphers) | ||
118 | { | ||
119 | unsigned char *buf = NULL; | ||
120 | size_t buflen; | ||
121 | int len; | ||
122 | int ret = 0; | ||
123 | |||
124 | /* Space for cipher bytes and two spare bytes */ | ||
125 | CHECK(sk_SSL_CIPHER_num(*ciphers) == N_CIPHERS); | ||
126 | buflen = sizeof(cipher_bytes) + 2; | ||
127 | CHECK((buf = calloc(1, buflen)) != NULL); | ||
128 | buf[buflen - 2] = 0xfe; | ||
129 | buf[buflen - 1] = 0xab; | ||
130 | |||
131 | /* Set renegotiate so it doesn't add SCSV */ | ||
132 | s->renegotiate = 1; | ||
133 | |||
134 | len = ssl_cipher_list_to_bytes(s, *ciphers, buf); | ||
135 | CHECK_GOTO(len > 0 && (size_t)len == buflen - 2); | ||
136 | CHECK_GOTO(memcmp(buf, cipher_bytes, sizeof(cipher_bytes)) == 0); | ||
137 | CHECK_GOTO(buf[buflen - 2] == 0xfe && buf[buflen - 1] == 0xab); | ||
138 | |||
139 | ret = 1; | ||
140 | |||
141 | err: | ||
142 | free(buf); | ||
143 | return ret; | ||
144 | } | ||
145 | |||
146 | static int | ||
147 | ssl_bytes_to_list_invalid(SSL *s, STACK_OF(SSL_CIPHER) **ciphers) | ||
148 | { | ||
149 | sk_SSL_CIPHER_free(*ciphers); | ||
150 | |||
151 | /* Invalid length: CipherSuite is 2 bytes so it must be even */ | ||
152 | *ciphers = ssl_bytes_to_cipher_list(s, cipher_bytes, | ||
153 | sizeof(cipher_bytes) - 1); | ||
154 | CHECK(*ciphers == NULL); | ||
155 | |||
156 | return 1; | ||
157 | } | ||
158 | |||
159 | int | ||
160 | main(void) | ||
161 | { | ||
162 | STACK_OF(SSL_CIPHER) *ciphers = NULL; | ||
163 | SSL_CTX *ctx = NULL; | ||
164 | SSL *s = NULL; | ||
165 | int rv = 1; | ||
166 | |||
167 | SSL_library_init(); | ||
168 | |||
169 | /* Use TLSv1.2 client to get all ciphers. */ | ||
170 | CHECK_GOTO((ctx = SSL_CTX_new(TLSv1_2_client_method())) != NULL); | ||
171 | CHECK_GOTO((s = SSL_new(ctx)) != NULL); | ||
172 | |||
173 | if (!ssl_bytes_to_list_alloc(s, &ciphers)) | ||
174 | goto err; | ||
175 | if (!ssl_list_to_bytes_scsv(s, &ciphers)) | ||
176 | goto err; | ||
177 | if (!ssl_list_to_bytes_no_scsv(s, &ciphers)) | ||
178 | goto err; | ||
179 | if (!ssl_bytes_to_list_invalid(s, &ciphers)) | ||
180 | goto err; | ||
181 | |||
182 | rv = 0; | ||
183 | |||
184 | err: | ||
185 | sk_SSL_CIPHER_free(ciphers); | ||
186 | SSL_CTX_free(ctx); | ||
187 | SSL_free(s); | ||
188 | |||
189 | if (!rv) | ||
190 | printf("PASS %s\n", __FILE__); | ||
191 | return rv; | ||
192 | } | ||
diff --git a/src/regress/lib/libssl/unit/tests.h b/src/regress/lib/libssl/unit/tests.h new file mode 100644 index 0000000000..287816946a --- /dev/null +++ b/src/regress/lib/libssl/unit/tests.h | |||
@@ -0,0 +1,44 @@ | |||
1 | /* $OpenBSD: tests.h,v 1.1 2015/06/27 23:35:52 doug Exp $ */ | ||
2 | /* | ||
3 | * Copyright (c) 2015 Doug Hogan <doug@openbsd.org> | ||
4 | * | ||
5 | * Permission to use, copy, modify, and distribute this software for any | ||
6 | * purpose with or without fee is hereby granted, provided that the above | ||
7 | * copyright notice and this permission notice appear in all copies. | ||
8 | * | ||
9 | * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | ||
10 | * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | ||
11 | * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | ||
12 | * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | ||
13 | * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | ||
14 | * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF | ||
15 | * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | ||
16 | */ | ||
17 | |||
18 | #ifndef LIBRESSL_REGRESS_TESTS_H__ | ||
19 | #define LIBRESSL_REGRESS_TESTS_H__ 1 | ||
20 | |||
21 | /* Ugly macros that are useful for regression tests. */ | ||
22 | |||
23 | #define SKIP(a) do { \ | ||
24 | printf("Skipping test in %s [%s:%d]\n", __func__, __FILE__, \ | ||
25 | __LINE__); \ | ||
26 | } while (0) | ||
27 | |||
28 | #define CHECK(a) do { \ | ||
29 | if (!(a)) { \ | ||
30 | printf("Error in %s [%s:%d]\n", __func__, __FILE__, \ | ||
31 | __LINE__); \ | ||
32 | return 0; \ | ||
33 | } \ | ||
34 | } while (0) | ||
35 | |||
36 | #define CHECK_GOTO(a) do { \ | ||
37 | if (!(a)) { \ | ||
38 | printf("Error in %s [%s:%d]\n", __func__, __FILE__, \ | ||
39 | __LINE__); \ | ||
40 | goto err; \ | ||
41 | } \ | ||
42 | } while (0) | ||
43 | |||
44 | #endif /* LIBRESSL_REGRESS_TESTS_H__ */ | ||