xref: /openbmc/linux/net/handshake/handshake-test.c (revision 9b4469410cf9a0fcbccc92c480fd42f7c815a745)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (c) 2023 Oracle and/or its affiliates.
4  *
5  * KUnit test of the handshake upcall mechanism.
6  */
7 
8 #include <kunit/test.h>
9 #include <kunit/visibility.h>
10 
11 #include <linux/kernel.h>
12 
13 #include <net/sock.h>
14 #include <net/genetlink.h>
15 #include <net/netns/generic.h>
16 
17 #include <uapi/linux/handshake.h>
18 #include "handshake.h"
19 
20 MODULE_IMPORT_NS(EXPORTED_FOR_KUNIT_TESTING);
21 
22 static int test_accept_func(struct handshake_req *req, struct genl_info *info,
23 			    int fd)
24 {
25 	return 0;
26 }
27 
28 static void test_done_func(struct handshake_req *req, unsigned int status,
29 			   struct genl_info *info)
30 {
31 }
32 
33 struct handshake_req_alloc_test_param {
34 	const char			*desc;
35 	struct handshake_proto		*proto;
36 	gfp_t				gfp;
37 	bool				expect_success;
38 };
39 
40 static struct handshake_proto handshake_req_alloc_proto_2 = {
41 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_NONE,
42 };
43 
44 static struct handshake_proto handshake_req_alloc_proto_3 = {
45 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_MAX,
46 };
47 
48 static struct handshake_proto handshake_req_alloc_proto_4 = {
49 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
50 };
51 
52 static struct handshake_proto handshake_req_alloc_proto_5 = {
53 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
54 	.hp_accept		= test_accept_func,
55 };
56 
57 static struct handshake_proto handshake_req_alloc_proto_6 = {
58 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
59 	.hp_privsize		= UINT_MAX,
60 	.hp_accept		= test_accept_func,
61 	.hp_done		= test_done_func,
62 };
63 
64 static struct handshake_proto handshake_req_alloc_proto_good = {
65 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
66 	.hp_accept		= test_accept_func,
67 	.hp_done		= test_done_func,
68 };
69 
70 static const
71 struct handshake_req_alloc_test_param handshake_req_alloc_params[] = {
72 	{
73 		.desc			= "handshake_req_alloc NULL proto",
74 		.proto			= NULL,
75 		.gfp			= GFP_KERNEL,
76 		.expect_success		= false,
77 	},
78 	{
79 		.desc			= "handshake_req_alloc CLASS_NONE",
80 		.proto			= &handshake_req_alloc_proto_2,
81 		.gfp			= GFP_KERNEL,
82 		.expect_success		= false,
83 	},
84 	{
85 		.desc			= "handshake_req_alloc CLASS_MAX",
86 		.proto			= &handshake_req_alloc_proto_3,
87 		.gfp			= GFP_KERNEL,
88 		.expect_success		= false,
89 	},
90 	{
91 		.desc			= "handshake_req_alloc no callbacks",
92 		.proto			= &handshake_req_alloc_proto_4,
93 		.gfp			= GFP_KERNEL,
94 		.expect_success		= false,
95 	},
96 	{
97 		.desc			= "handshake_req_alloc no done callback",
98 		.proto			= &handshake_req_alloc_proto_5,
99 		.gfp			= GFP_KERNEL,
100 		.expect_success		= false,
101 	},
102 	{
103 		.desc			= "handshake_req_alloc excessive privsize",
104 		.proto			= &handshake_req_alloc_proto_6,
105 		.gfp			= GFP_KERNEL,
106 		.expect_success		= false,
107 	},
108 	{
109 		.desc			= "handshake_req_alloc all good",
110 		.proto			= &handshake_req_alloc_proto_good,
111 		.gfp			= GFP_KERNEL,
112 		.expect_success		= true,
113 	},
114 };
115 
116 static void
117 handshake_req_alloc_get_desc(const struct handshake_req_alloc_test_param *param,
118 			     char *desc)
119 {
120 	strscpy(desc, param->desc, KUNIT_PARAM_DESC_SIZE);
121 }
122 
123 /* Creates the function handshake_req_alloc_gen_params */
124 KUNIT_ARRAY_PARAM(handshake_req_alloc, handshake_req_alloc_params,
125 		  handshake_req_alloc_get_desc);
126 
127 static void handshake_req_alloc_case(struct kunit *test)
128 {
129 	const struct handshake_req_alloc_test_param *param = test->param_value;
130 	struct handshake_req *result;
131 
132 	/* Arrange */
133 
134 	/* Act */
135 	result = handshake_req_alloc(param->proto, param->gfp);
136 
137 	/* Assert */
138 	if (param->expect_success)
139 		KUNIT_EXPECT_NOT_NULL(test, result);
140 	else
141 		KUNIT_EXPECT_NULL(test, result);
142 
143 	kfree(result);
144 }
145 
146 static void handshake_req_submit_test1(struct kunit *test)
147 {
148 	struct socket *sock;
149 	int err, result;
150 
151 	/* Arrange */
152 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
153 			    &sock, 1);
154 	KUNIT_ASSERT_EQ(test, err, 0);
155 
156 	/* Act */
157 	result = handshake_req_submit(sock, NULL, GFP_KERNEL);
158 
159 	/* Assert */
160 	KUNIT_EXPECT_EQ(test, result, -EINVAL);
161 
162 	sock_release(sock);
163 }
164 
165 static void handshake_req_submit_test2(struct kunit *test)
166 {
167 	struct handshake_req *req;
168 	int result;
169 
170 	/* Arrange */
171 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
172 	KUNIT_ASSERT_NOT_NULL(test, req);
173 
174 	/* Act */
175 	result = handshake_req_submit(NULL, req, GFP_KERNEL);
176 
177 	/* Assert */
178 	KUNIT_EXPECT_EQ(test, result, -EINVAL);
179 
180 	/* handshake_req_submit() destroys @req on error */
181 }
182 
183 static void handshake_req_submit_test3(struct kunit *test)
184 {
185 	struct handshake_req *req;
186 	struct socket *sock;
187 	int err, result;
188 
189 	/* Arrange */
190 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
191 	KUNIT_ASSERT_NOT_NULL(test, req);
192 
193 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
194 			    &sock, 1);
195 	KUNIT_ASSERT_EQ(test, err, 0);
196 	sock->file = NULL;
197 
198 	/* Act */
199 	result = handshake_req_submit(sock, req, GFP_KERNEL);
200 
201 	/* Assert */
202 	KUNIT_EXPECT_EQ(test, result, -EINVAL);
203 
204 	/* handshake_req_submit() destroys @req on error */
205 	sock_release(sock);
206 }
207 
208 static void handshake_req_submit_test4(struct kunit *test)
209 {
210 	struct handshake_req *req, *result;
211 	struct socket *sock;
212 	int err;
213 
214 	/* Arrange */
215 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
216 	KUNIT_ASSERT_NOT_NULL(test, req);
217 
218 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
219 			    &sock, 1);
220 	KUNIT_ASSERT_EQ(test, err, 0);
221 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
222 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
223 	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
224 
225 	err = handshake_req_submit(sock, req, GFP_KERNEL);
226 	KUNIT_ASSERT_EQ(test, err, 0);
227 
228 	/* Act */
229 	result = handshake_req_hash_lookup(sock->sk);
230 
231 	/* Assert */
232 	KUNIT_EXPECT_NOT_NULL(test, result);
233 	KUNIT_EXPECT_PTR_EQ(test, req, result);
234 
235 	handshake_req_cancel(sock->sk);
236 	sock_release(sock);
237 }
238 
239 static void handshake_req_submit_test5(struct kunit *test)
240 {
241 	struct handshake_req *req;
242 	struct handshake_net *hn;
243 	struct socket *sock;
244 	struct net *net;
245 	int saved, err;
246 
247 	/* Arrange */
248 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
249 	KUNIT_ASSERT_NOT_NULL(test, req);
250 
251 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
252 			    &sock, 1);
253 	KUNIT_ASSERT_EQ(test, err, 0);
254 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
255 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
256 	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
257 
258 	net = sock_net(sock->sk);
259 	hn = handshake_pernet(net);
260 	KUNIT_ASSERT_NOT_NULL(test, hn);
261 
262 	saved = hn->hn_pending;
263 	hn->hn_pending = hn->hn_pending_max + 1;
264 
265 	/* Act */
266 	err = handshake_req_submit(sock, req, GFP_KERNEL);
267 
268 	/* Assert */
269 	KUNIT_EXPECT_EQ(test, err, -EAGAIN);
270 
271 	sock_release(sock);
272 	hn->hn_pending = saved;
273 }
274 
275 static void handshake_req_submit_test6(struct kunit *test)
276 {
277 	struct handshake_req *req1, *req2;
278 	struct socket *sock;
279 	int err;
280 
281 	/* Arrange */
282 	req1 = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
283 	KUNIT_ASSERT_NOT_NULL(test, req1);
284 	req2 = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
285 	KUNIT_ASSERT_NOT_NULL(test, req2);
286 
287 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
288 			    &sock, 1);
289 	KUNIT_ASSERT_EQ(test, err, 0);
290 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
291 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
292 	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
293 
294 	/* Act */
295 	err = handshake_req_submit(sock, req1, GFP_KERNEL);
296 	KUNIT_ASSERT_EQ(test, err, 0);
297 	err = handshake_req_submit(sock, req2, GFP_KERNEL);
298 
299 	/* Assert */
300 	KUNIT_EXPECT_EQ(test, err, -EBUSY);
301 
302 	handshake_req_cancel(sock->sk);
303 	sock_release(sock);
304 }
305 
306 static void handshake_req_cancel_test1(struct kunit *test)
307 {
308 	struct handshake_req *req;
309 	struct socket *sock;
310 	bool result;
311 	int err;
312 
313 	/* Arrange */
314 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
315 	KUNIT_ASSERT_NOT_NULL(test, req);
316 
317 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
318 			    &sock, 1);
319 	KUNIT_ASSERT_EQ(test, err, 0);
320 
321 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
322 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
323 
324 	err = handshake_req_submit(sock, req, GFP_KERNEL);
325 	KUNIT_ASSERT_EQ(test, err, 0);
326 
327 	/* NB: handshake_req hasn't been accepted */
328 
329 	/* Act */
330 	result = handshake_req_cancel(sock->sk);
331 
332 	/* Assert */
333 	KUNIT_EXPECT_TRUE(test, result);
334 
335 	sock_release(sock);
336 }
337 
338 static void handshake_req_cancel_test2(struct kunit *test)
339 {
340 	struct handshake_req *req, *next;
341 	struct handshake_net *hn;
342 	struct socket *sock;
343 	struct net *net;
344 	bool result;
345 	int err;
346 
347 	/* Arrange */
348 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
349 	KUNIT_ASSERT_NOT_NULL(test, req);
350 
351 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
352 			    &sock, 1);
353 	KUNIT_ASSERT_EQ(test, err, 0);
354 
355 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
356 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
357 
358 	err = handshake_req_submit(sock, req, GFP_KERNEL);
359 	KUNIT_ASSERT_EQ(test, err, 0);
360 
361 	net = sock_net(sock->sk);
362 	hn = handshake_pernet(net);
363 	KUNIT_ASSERT_NOT_NULL(test, hn);
364 
365 	/* Pretend to accept this request */
366 	next = handshake_req_next(hn, HANDSHAKE_HANDLER_CLASS_TLSHD);
367 	KUNIT_ASSERT_PTR_EQ(test, req, next);
368 
369 	/* Act */
370 	result = handshake_req_cancel(sock->sk);
371 
372 	/* Assert */
373 	KUNIT_EXPECT_TRUE(test, result);
374 
375 	sock_release(sock);
376 }
377 
378 static void handshake_req_cancel_test3(struct kunit *test)
379 {
380 	struct handshake_req *req, *next;
381 	struct handshake_net *hn;
382 	struct socket *sock;
383 	struct net *net;
384 	bool result;
385 	int err;
386 
387 	/* Arrange */
388 	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
389 	KUNIT_ASSERT_NOT_NULL(test, req);
390 
391 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
392 			    &sock, 1);
393 	KUNIT_ASSERT_EQ(test, err, 0);
394 
395 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
396 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
397 
398 	err = handshake_req_submit(sock, req, GFP_KERNEL);
399 	KUNIT_ASSERT_EQ(test, err, 0);
400 
401 	net = sock_net(sock->sk);
402 	hn = handshake_pernet(net);
403 	KUNIT_ASSERT_NOT_NULL(test, hn);
404 
405 	/* Pretend to accept this request */
406 	next = handshake_req_next(hn, HANDSHAKE_HANDLER_CLASS_TLSHD);
407 	KUNIT_ASSERT_PTR_EQ(test, req, next);
408 
409 	/* Pretend to complete this request */
410 	handshake_complete(next, -ETIMEDOUT, NULL);
411 
412 	/* Act */
413 	result = handshake_req_cancel(sock->sk);
414 
415 	/* Assert */
416 	KUNIT_EXPECT_FALSE(test, result);
417 
418 	sock_release(sock);
419 }
420 
421 static struct handshake_req *handshake_req_destroy_test;
422 
423 static void test_destroy_func(struct handshake_req *req)
424 {
425 	handshake_req_destroy_test = req;
426 }
427 
428 static struct handshake_proto handshake_req_alloc_proto_destroy = {
429 	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
430 	.hp_accept		= test_accept_func,
431 	.hp_done		= test_done_func,
432 	.hp_destroy		= test_destroy_func,
433 };
434 
435 static void handshake_req_destroy_test1(struct kunit *test)
436 {
437 	struct handshake_req *req;
438 	struct socket *sock;
439 	int err;
440 
441 	/* Arrange */
442 	handshake_req_destroy_test = NULL;
443 
444 	req = handshake_req_alloc(&handshake_req_alloc_proto_destroy, GFP_KERNEL);
445 	KUNIT_ASSERT_NOT_NULL(test, req);
446 
447 	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
448 			    &sock, 1);
449 	KUNIT_ASSERT_EQ(test, err, 0);
450 
451 	sock->file = sock_alloc_file(sock, O_NONBLOCK, NULL);
452 	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, sock->file);
453 
454 	err = handshake_req_submit(sock, req, GFP_KERNEL);
455 	KUNIT_ASSERT_EQ(test, err, 0);
456 
457 	handshake_req_cancel(sock->sk);
458 
459 	/* Act */
460 	sock_release(sock);
461 
462 	/* Assert */
463 	KUNIT_EXPECT_PTR_EQ(test, handshake_req_destroy_test, req);
464 }
465 
466 static struct kunit_case handshake_api_test_cases[] = {
467 	{
468 		.name			= "req_alloc API fuzzing",
469 		.run_case		= handshake_req_alloc_case,
470 		.generate_params	= handshake_req_alloc_gen_params,
471 	},
472 	{
473 		.name			= "req_submit NULL req arg",
474 		.run_case		= handshake_req_submit_test1,
475 	},
476 	{
477 		.name			= "req_submit NULL sock arg",
478 		.run_case		= handshake_req_submit_test2,
479 	},
480 	{
481 		.name			= "req_submit NULL sock->file",
482 		.run_case		= handshake_req_submit_test3,
483 	},
484 	{
485 		.name			= "req_lookup works",
486 		.run_case		= handshake_req_submit_test4,
487 	},
488 	{
489 		.name			= "req_submit max pending",
490 		.run_case		= handshake_req_submit_test5,
491 	},
492 	{
493 		.name			= "req_submit multiple",
494 		.run_case		= handshake_req_submit_test6,
495 	},
496 	{
497 		.name			= "req_cancel before accept",
498 		.run_case		= handshake_req_cancel_test1,
499 	},
500 	{
501 		.name			= "req_cancel after accept",
502 		.run_case		= handshake_req_cancel_test2,
503 	},
504 	{
505 		.name			= "req_cancel after done",
506 		.run_case		= handshake_req_cancel_test3,
507 	},
508 	{
509 		.name			= "req_destroy works",
510 		.run_case		= handshake_req_destroy_test1,
511 	},
512 	{}
513 };
514 
515 static struct kunit_suite handshake_api_suite = {
516        .name                   = "Handshake API tests",
517        .test_cases             = handshake_api_test_cases,
518 };
519 
520 kunit_test_suites(&handshake_api_suite);
521 
522 MODULE_DESCRIPTION("Test handshake upcall API functions");
523 MODULE_LICENSE("GPL");
524