forked from mirrors/linux
		
	Clean up the existing export namespace code along the same lines of
commit 33def8498f ("treewide: Convert macro and uses of __section(foo)
to __section("foo")") and for the same reason, it is not desired for the
namespace argument to be a macro expansion itself.
Scripted using
  git grep -l -e MODULE_IMPORT_NS -e EXPORT_SYMBOL_NS | while read file;
  do
    awk -i inplace '
      /^#define EXPORT_SYMBOL_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /^#define MODULE_IMPORT_NS/ {
        gsub(/__stringify\(ns\)/, "ns");
        print;
        next;
      }
      /MODULE_IMPORT_NS/ {
        $0 = gensub(/MODULE_IMPORT_NS\(([^)]*)\)/, "MODULE_IMPORT_NS(\"\\1\")", "g");
      }
      /EXPORT_SYMBOL_NS/ {
        if ($0 ~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+),/) {
  	if ($0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/ &&
  	    $0 !~ /(EXPORT_SYMBOL_NS[^(]*)\(\)/ &&
  	    $0 !~ /^my/) {
  	  getline line;
  	  gsub(/[[:space:]]*\\$/, "");
  	  gsub(/[[:space:]]/, "", line);
  	  $0 = $0 " " line;
  	}
  	$0 = gensub(/(EXPORT_SYMBOL_NS[^(]*)\(([^,]+), ([^)]+)\)/,
  		    "\\1(\\2, \"\\3\")", "g");
        }
      }
      { print }' $file;
  done
Requested-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org>
Link: https://mail.google.com/mail/u/2/#inbox/FMfcgzQXKWgMmjdFwwdsfgxzKpVHWPlc
Acked-by: Greg KH <gregkh@linuxfoundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
	
			
		
			
				
	
	
		
			540 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			540 lines
		
	
	
	
		
			13 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// SPDX-License-Identifier: GPL-2.0
 | 
						|
/*
 | 
						|
 * Copyright (c) 2023 Oracle and/or its affiliates.
 | 
						|
 *
 | 
						|
 * KUnit test of the handshake upcall mechanism.
 | 
						|
 */
 | 
						|
 | 
						|
#include <kunit/test.h>
 | 
						|
#include <kunit/visibility.h>
 | 
						|
 | 
						|
#include <linux/kernel.h>
 | 
						|
 | 
						|
#include <net/sock.h>
 | 
						|
#include <net/genetlink.h>
 | 
						|
#include <net/netns/generic.h>
 | 
						|
 | 
						|
#include <uapi/linux/handshake.h>
 | 
						|
#include "handshake.h"
 | 
						|
 | 
						|
MODULE_IMPORT_NS("EXPORTED_FOR_KUNIT_TESTING");
 | 
						|
 | 
						|
static int test_accept_func(struct handshake_req *req, struct genl_info *info,
 | 
						|
			    int fd)
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
 | 
						|
static void test_done_func(struct handshake_req *req, unsigned int status,
 | 
						|
			   struct genl_info *info)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
struct handshake_req_alloc_test_param {
 | 
						|
	const char			*desc;
 | 
						|
	struct handshake_proto		*proto;
 | 
						|
	gfp_t				gfp;
 | 
						|
	bool				expect_success;
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_2 = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_NONE,
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_3 = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_MAX,
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_4 = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_5 = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
 | 
						|
	.hp_accept		= test_accept_func,
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_6 = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
 | 
						|
	.hp_privsize		= UINT_MAX,
 | 
						|
	.hp_accept		= test_accept_func,
 | 
						|
	.hp_done		= test_done_func,
 | 
						|
};
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_good = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
 | 
						|
	.hp_accept		= test_accept_func,
 | 
						|
	.hp_done		= test_done_func,
 | 
						|
};
 | 
						|
 | 
						|
static const
 | 
						|
struct handshake_req_alloc_test_param handshake_req_alloc_params[] = {
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc NULL proto",
 | 
						|
		.proto			= NULL,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc CLASS_NONE",
 | 
						|
		.proto			= &handshake_req_alloc_proto_2,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc CLASS_MAX",
 | 
						|
		.proto			= &handshake_req_alloc_proto_3,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc no callbacks",
 | 
						|
		.proto			= &handshake_req_alloc_proto_4,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc no done callback",
 | 
						|
		.proto			= &handshake_req_alloc_proto_5,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc excessive privsize",
 | 
						|
		.proto			= &handshake_req_alloc_proto_6,
 | 
						|
		.gfp			= GFP_KERNEL | __GFP_NOWARN,
 | 
						|
		.expect_success		= false,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.desc			= "handshake_req_alloc all good",
 | 
						|
		.proto			= &handshake_req_alloc_proto_good,
 | 
						|
		.gfp			= GFP_KERNEL,
 | 
						|
		.expect_success		= true,
 | 
						|
	},
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
handshake_req_alloc_get_desc(const struct handshake_req_alloc_test_param *param,
 | 
						|
			     char *desc)
 | 
						|
{
 | 
						|
	strscpy(desc, param->desc, KUNIT_PARAM_DESC_SIZE);
 | 
						|
}
 | 
						|
 | 
						|
/* Creates the function handshake_req_alloc_gen_params */
 | 
						|
KUNIT_ARRAY_PARAM(handshake_req_alloc, handshake_req_alloc_params,
 | 
						|
		  handshake_req_alloc_get_desc);
 | 
						|
 | 
						|
static void handshake_req_alloc_case(struct kunit *test)
 | 
						|
{
 | 
						|
	const struct handshake_req_alloc_test_param *param = test->param_value;
 | 
						|
	struct handshake_req *result;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_alloc(param->proto, param->gfp);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	if (param->expect_success)
 | 
						|
		KUNIT_EXPECT_NOT_NULL(test, result);
 | 
						|
	else
 | 
						|
		KUNIT_EXPECT_NULL(test, result);
 | 
						|
 | 
						|
	kfree(result);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test1(struct kunit *test)
 | 
						|
{
 | 
						|
	struct socket *sock;
 | 
						|
	int err, result;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_submit(sock, NULL, GFP_KERNEL);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_EQ(test, result, -EINVAL);
 | 
						|
 | 
						|
	sock_release(sock);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test2(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req;
 | 
						|
	int result;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_submit(NULL, req, GFP_KERNEL);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_EQ(test, result, -EINVAL);
 | 
						|
 | 
						|
	/* handshake_req_submit() destroys @req on error */
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test3(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req;
 | 
						|
	struct socket *sock;
 | 
						|
	int err, result;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
	sock->file = NULL;
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_EQ(test, result, -EINVAL);
 | 
						|
 | 
						|
	/* handshake_req_submit() destroys @req on error */
 | 
						|
	sock_release(sock);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test4(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req, *result;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_hash_lookup(sock->sk);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_NOT_NULL(test, result);
 | 
						|
	KUNIT_EXPECT_PTR_EQ(test, req, result);
 | 
						|
 | 
						|
	handshake_req_cancel(sock->sk);
 | 
						|
	fput(filp);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test5(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req;
 | 
						|
	struct handshake_net *hn;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	struct net *net;
 | 
						|
	int saved, err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	net = sock_net(sock->sk);
 | 
						|
	hn = handshake_pernet(net);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, hn);
 | 
						|
 | 
						|
	saved = hn->hn_pending;
 | 
						|
	hn->hn_pending = hn->hn_pending_max + 1;
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_EQ(test, err, -EAGAIN);
 | 
						|
 | 
						|
	fput(filp);
 | 
						|
	hn->hn_pending = saved;
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_submit_test6(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req1, *req2;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req1 = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req1);
 | 
						|
	req2 = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req2);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, sock->sk);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	err = handshake_req_submit(sock, req1, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
	err = handshake_req_submit(sock, req2, GFP_KERNEL);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_EQ(test, err, -EBUSY);
 | 
						|
 | 
						|
	handshake_req_cancel(sock->sk);
 | 
						|
	fput(filp);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_cancel_test1(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	bool result;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	/* NB: handshake_req hasn't been accepted */
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_cancel(sock->sk);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_TRUE(test, result);
 | 
						|
 | 
						|
	fput(filp);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_cancel_test2(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req, *next;
 | 
						|
	struct handshake_net *hn;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	struct net *net;
 | 
						|
	bool result;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	net = sock_net(sock->sk);
 | 
						|
	hn = handshake_pernet(net);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, hn);
 | 
						|
 | 
						|
	/* Pretend to accept this request */
 | 
						|
	next = handshake_req_next(hn, HANDSHAKE_HANDLER_CLASS_TLSHD);
 | 
						|
	KUNIT_ASSERT_PTR_EQ(test, req, next);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_cancel(sock->sk);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_TRUE(test, result);
 | 
						|
 | 
						|
	fput(filp);
 | 
						|
}
 | 
						|
 | 
						|
static void handshake_req_cancel_test3(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req, *next;
 | 
						|
	struct handshake_net *hn;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	struct net *net;
 | 
						|
	bool result;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_good, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	net = sock_net(sock->sk);
 | 
						|
	hn = handshake_pernet(net);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, hn);
 | 
						|
 | 
						|
	/* Pretend to accept this request */
 | 
						|
	next = handshake_req_next(hn, HANDSHAKE_HANDLER_CLASS_TLSHD);
 | 
						|
	KUNIT_ASSERT_PTR_EQ(test, req, next);
 | 
						|
 | 
						|
	/* Pretend to complete this request */
 | 
						|
	handshake_complete(next, -ETIMEDOUT, NULL);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	result = handshake_req_cancel(sock->sk);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_FALSE(test, result);
 | 
						|
 | 
						|
	fput(filp);
 | 
						|
}
 | 
						|
 | 
						|
static struct handshake_req *handshake_req_destroy_test;
 | 
						|
 | 
						|
static void test_destroy_func(struct handshake_req *req)
 | 
						|
{
 | 
						|
	handshake_req_destroy_test = req;
 | 
						|
}
 | 
						|
 | 
						|
static struct handshake_proto handshake_req_alloc_proto_destroy = {
 | 
						|
	.hp_handler_class	= HANDSHAKE_HANDLER_CLASS_TLSHD,
 | 
						|
	.hp_accept		= test_accept_func,
 | 
						|
	.hp_done		= test_done_func,
 | 
						|
	.hp_destroy		= test_destroy_func,
 | 
						|
};
 | 
						|
 | 
						|
static void handshake_req_destroy_test1(struct kunit *test)
 | 
						|
{
 | 
						|
	struct handshake_req *req;
 | 
						|
	struct socket *sock;
 | 
						|
	struct file *filp;
 | 
						|
	int err;
 | 
						|
 | 
						|
	/* Arrange */
 | 
						|
	handshake_req_destroy_test = NULL;
 | 
						|
 | 
						|
	req = handshake_req_alloc(&handshake_req_alloc_proto_destroy, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_NOT_NULL(test, req);
 | 
						|
 | 
						|
	err = __sock_create(&init_net, PF_INET, SOCK_STREAM, IPPROTO_TCP,
 | 
						|
			    &sock, 1);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	filp = sock_alloc_file(sock, O_NONBLOCK, NULL);
 | 
						|
	KUNIT_ASSERT_NOT_ERR_OR_NULL(test, filp);
 | 
						|
	sock->file = filp;
 | 
						|
 | 
						|
	err = handshake_req_submit(sock, req, GFP_KERNEL);
 | 
						|
	KUNIT_ASSERT_EQ(test, err, 0);
 | 
						|
 | 
						|
	handshake_req_cancel(sock->sk);
 | 
						|
 | 
						|
	/* Act */
 | 
						|
	/* Ensure the close/release/put process has run to
 | 
						|
	 * completion before checking the result.
 | 
						|
	 */
 | 
						|
	__fput_sync(filp);
 | 
						|
 | 
						|
	/* Assert */
 | 
						|
	KUNIT_EXPECT_PTR_EQ(test, handshake_req_destroy_test, req);
 | 
						|
}
 | 
						|
 | 
						|
static struct kunit_case handshake_api_test_cases[] = {
 | 
						|
	{
 | 
						|
		.name			= "req_alloc API fuzzing",
 | 
						|
		.run_case		= handshake_req_alloc_case,
 | 
						|
		.generate_params	= handshake_req_alloc_gen_params,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_submit NULL req arg",
 | 
						|
		.run_case		= handshake_req_submit_test1,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_submit NULL sock arg",
 | 
						|
		.run_case		= handshake_req_submit_test2,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_submit NULL sock->file",
 | 
						|
		.run_case		= handshake_req_submit_test3,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_lookup works",
 | 
						|
		.run_case		= handshake_req_submit_test4,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_submit max pending",
 | 
						|
		.run_case		= handshake_req_submit_test5,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_submit multiple",
 | 
						|
		.run_case		= handshake_req_submit_test6,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_cancel before accept",
 | 
						|
		.run_case		= handshake_req_cancel_test1,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_cancel after accept",
 | 
						|
		.run_case		= handshake_req_cancel_test2,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_cancel after done",
 | 
						|
		.run_case		= handshake_req_cancel_test3,
 | 
						|
	},
 | 
						|
	{
 | 
						|
		.name			= "req_destroy works",
 | 
						|
		.run_case		= handshake_req_destroy_test1,
 | 
						|
	},
 | 
						|
	{}
 | 
						|
};
 | 
						|
 | 
						|
static struct kunit_suite handshake_api_suite = {
 | 
						|
       .name                   = "Handshake API tests",
 | 
						|
       .test_cases             = handshake_api_test_cases,
 | 
						|
};
 | 
						|
 | 
						|
kunit_test_suites(&handshake_api_suite);
 | 
						|
 | 
						|
MODULE_DESCRIPTION("Test handshake upcall API functions");
 | 
						|
MODULE_LICENSE("GPL");
 |