// SPDX-License-Identifier: GPL-2.0 #include #include "protocol.h" static struct mptcp_subflow_request_sock *build_req_sock(struct kunit *test) { struct mptcp_subflow_request_sock *req; req = kunit_kzalloc(test, sizeof(struct mptcp_subflow_request_sock), GFP_USER); KUNIT_EXPECT_NOT_ERR_OR_NULL(test, req); mptcp_token_init_request((struct request_sock *)req); return req; } static void mptcp_token_test_req_basic(struct kunit *test) { struct mptcp_subflow_request_sock *req = build_req_sock(test); struct mptcp_sock *null_msk = NULL; KUNIT_ASSERT_EQ(test, 0, mptcp_token_new_request((struct request_sock *)req)); KUNIT_EXPECT_NE(test, 0, (int)req->token); KUNIT_EXPECT_PTR_EQ(test, null_msk, mptcp_token_get_sock(req->token)); /* cleanup */ mptcp_token_destroy_request((struct request_sock *)req); } static struct inet_connection_sock *build_icsk(struct kunit *test) { struct inet_connection_sock *icsk; icsk = kunit_kzalloc(test, sizeof(struct inet_connection_sock), GFP_USER); KUNIT_EXPECT_NOT_ERR_OR_NULL(test, icsk); return icsk; } static struct mptcp_subflow_context *build_ctx(struct kunit *test) { struct mptcp_subflow_context *ctx; ctx = kunit_kzalloc(test, sizeof(struct mptcp_subflow_context), GFP_USER); KUNIT_EXPECT_NOT_ERR_OR_NULL(test, ctx); return ctx; } static struct mptcp_sock *build_msk(struct kunit *test) { struct mptcp_sock *msk; msk = kunit_kzalloc(test, sizeof(struct mptcp_sock), GFP_USER); KUNIT_EXPECT_NOT_ERR_OR_NULL(test, msk); refcount_set(&((struct sock *)msk)->sk_refcnt, 1); return msk; } static void mptcp_token_test_msk_basic(struct kunit *test) { struct inet_connection_sock *icsk = build_icsk(test); struct mptcp_subflow_context *ctx = build_ctx(test); struct mptcp_sock *msk = build_msk(test); struct mptcp_sock *null_msk = NULL; struct sock *sk; rcu_assign_pointer(icsk->icsk_ulp_data, ctx); ctx->conn = (struct sock *)msk; sk = (struct sock *)msk; KUNIT_ASSERT_EQ(test, 0, mptcp_token_new_connect((struct sock *)icsk)); KUNIT_EXPECT_NE(test, 0, (int)ctx->token); KUNIT_EXPECT_EQ(test, ctx->token, msk->token); KUNIT_EXPECT_PTR_EQ(test, msk, mptcp_token_get_sock(ctx->token)); KUNIT_EXPECT_EQ(test, 2, (int)refcount_read(&sk->sk_refcnt)); mptcp_token_destroy(msk); KUNIT_EXPECT_PTR_EQ(test, null_msk, mptcp_token_get_sock(ctx->token)); } static void mptcp_token_test_accept(struct kunit *test) { struct mptcp_subflow_request_sock *req = build_req_sock(test); struct mptcp_sock *msk = build_msk(test); KUNIT_ASSERT_EQ(test, 0, mptcp_token_new_request((struct request_sock *)req)); msk->token = req->token; mptcp_token_accept(req, msk); KUNIT_EXPECT_PTR_EQ(test, msk, mptcp_token_get_sock(msk->token)); /* this is now a no-op */ mptcp_token_destroy_request((struct request_sock *)req); KUNIT_EXPECT_PTR_EQ(test, msk, mptcp_token_get_sock(msk->token)); /* cleanup */ mptcp_token_destroy(msk); } static void mptcp_token_test_destroyed(struct kunit *test) { struct mptcp_subflow_request_sock *req = build_req_sock(test); struct mptcp_sock *msk = build_msk(test); struct mptcp_sock *null_msk = NULL; struct sock *sk; sk = (struct sock *)msk; KUNIT_ASSERT_EQ(test, 0, mptcp_token_new_request((struct request_sock *)req)); msk->token = req->token; mptcp_token_accept(req, msk); /* simulate race on removal */ refcount_set(&sk->sk_refcnt, 0); KUNIT_EXPECT_PTR_EQ(test, null_msk, mptcp_token_get_sock(msk->token)); /* cleanup */ mptcp_token_destroy(msk); } static struct kunit_case mptcp_token_test_cases[] = { KUNIT_CASE(mptcp_token_test_req_basic), KUNIT_CASE(mptcp_token_test_msk_basic), KUNIT_CASE(mptcp_token_test_accept), KUNIT_CASE(mptcp_token_test_destroyed), {} }; static struct kunit_suite mptcp_token_suite = { .name = "mptcp-token", .test_cases = mptcp_token_test_cases, }; kunit_test_suite(mptcp_token_suite); MODULE_LICENSE("GPL");