Branch data Line data Source code
1 : : /* SPDX-License-Identifier: BSD-3-Clause
2 : : * Copyright (C) 2021 Intel Corporation.
3 : : * All rights reserved.
4 : : */
5 : :
6 : : #include "spdk/stdinc.h"
7 : : #include "spdk_internal/cunit.h"
8 : : #include "spdk/nvme.h"
9 : : #include "nvme/nvme_io_msg.c"
10 : : #include "common/lib/nvme/common_stubs.h"
11 : :
12 : 4 : SPDK_LOG_REGISTER_COMPONENT(nvme)
13 : :
14 [ - + ]: 12 : DEFINE_STUB(spdk_nvme_ctrlr_free_io_qpair, int, (struct spdk_nvme_qpair *qpair), 0);
15 : :
16 [ # # ]: 0 : DEFINE_RETURN_MOCK(spdk_nvme_ctrlr_alloc_io_qpair, struct spdk_nvme_qpair *);
17 : : struct spdk_nvme_qpair *
18 : 12 : spdk_nvme_ctrlr_alloc_io_qpair(struct spdk_nvme_ctrlr *ctrlr,
19 : : const struct spdk_nvme_io_qpair_opts *user_opts,
20 : : size_t opts_size)
21 : : {
22 [ - + + + : 12 : HANDLE_RETURN_MOCK(spdk_nvme_ctrlr_alloc_io_qpair);
+ - ]
23 : 0 : return NULL;
24 : : }
25 : :
26 : : static void
27 : 32 : ut_io_msg_fn(struct spdk_nvme_ctrlr *ctrlr, uint32_t nsid, void *arg)
28 : : {
29 : : static uint32_t i = 0;
30 : :
31 : 32 : CU_ASSERT(arg == (void *)(0xDEADBEEF + sizeof(int *) * i));
32 : 32 : CU_ASSERT(nsid == i);
33 : 32 : CU_ASSERT(ctrlr->external_io_msgs_qpair == (struct spdk_nvme_qpair *)0xDBADBEEF);
34 : 32 : i++;
35 : 32 : }
36 : :
37 : : static void
38 : 4 : test_nvme_io_msg_process(void)
39 : : {
40 : 4 : struct spdk_nvme_ctrlr ctrlr = {};
41 : 4 : struct spdk_ring external_io_msgs = {};
42 : : int rc, i;
43 : :
44 : 4 : ctrlr.external_io_msgs = &external_io_msgs;
45 : 4 : ctrlr.external_io_msgs_qpair = (struct spdk_nvme_qpair *)0xDBADBEEF;
46 : 4 : TAILQ_INIT(&external_io_msgs.elements);
47 : 4 : pthread_mutex_init(&ctrlr.external_io_msgs_lock, NULL);
48 [ - + ]: 4 : pthread_mutex_init(&ctrlr.external_io_msgs->lock, NULL);
49 : :
50 : : /* Send IO processing size requests */
51 [ + + ]: 36 : for (i = 0; i < SPDK_NVME_MSG_IO_PROCESS_SIZE; i ++) {
52 : 32 : nvme_io_msg_send(&ctrlr, i, ut_io_msg_fn,
53 : 32 : (void *)(0xDEADBEEF + sizeof(int *) * i));
54 : : }
55 : :
56 : 4 : rc = nvme_io_msg_process(&ctrlr);
57 : 4 : CU_ASSERT(rc == SPDK_NVME_MSG_IO_PROCESS_SIZE);
58 : 4 : CU_ASSERT(TAILQ_EMPTY(&external_io_msgs.elements));
59 : :
60 : : /* Unavailable external_io_msgs and external_io_msgs_qpair */
61 : 4 : ctrlr.external_io_msgs = NULL;
62 : 4 : ctrlr.external_io_msgs_qpair = NULL;
63 : :
64 : 4 : rc = nvme_io_msg_process(&ctrlr);
65 [ - + ]: 4 : SPDK_CU_ASSERT_FATAL(rc == 0);
66 : 4 : }
67 : :
68 : : static void
69 : 4 : test_nvme_io_msg_send(void)
70 : : {
71 : 4 : struct spdk_nvme_ctrlr ctrlr = {};
72 : 4 : struct spdk_nvme_io_msg *request = NULL;
73 : 4 : struct spdk_ring external_io_msgs = {};
74 : : int rc;
75 : 4 : uint32_t nsid = 1;
76 : 4 : void *arg = (void *)0xDEADBEEF;
77 : :
78 : 4 : ctrlr.external_io_msgs = &external_io_msgs;
79 : 4 : TAILQ_INIT(&external_io_msgs.elements);
80 [ - + ]: 4 : pthread_mutex_init(&ctrlr.external_io_msgs_lock, NULL);
81 [ - + ]: 4 : pthread_mutex_init(&ctrlr.external_io_msgs->lock, NULL);
82 : :
83 : : /* Dequeue the request after sending io message */
84 : 4 : rc = nvme_io_msg_send(&ctrlr, nsid, ut_io_msg_fn, arg);
85 : 4 : spdk_ring_dequeue(ctrlr.external_io_msgs, (void **)&request, 1);
86 : 4 : CU_ASSERT(rc == 0);
87 : 4 : CU_ASSERT(request != NULL);
88 : 4 : CU_ASSERT(request->ctrlr == &ctrlr);
89 : 4 : CU_ASSERT(request->nsid == nsid);
90 : 4 : CU_ASSERT(request->fn == ut_io_msg_fn);
91 : 4 : CU_ASSERT(request->arg == arg);
92 : 4 : CU_ASSERT(TAILQ_EMPTY(&external_io_msgs.elements));
93 : :
94 : 4 : free(request);
95 : 4 : }
96 : :
97 : : static void
98 : 0 : ut_stop(struct spdk_nvme_ctrlr *ctrlr)
99 : : {
100 : 0 : return;
101 : : }
102 : :
103 : : static void
104 : 0 : ut_update(struct spdk_nvme_ctrlr *ctrlr)
105 : : {
106 : 0 : return;
107 : : }
108 : :
109 : : static struct nvme_io_msg_producer ut_nvme_io_msg_producer[2] = {
110 : : {
111 : : .name = "ut_test1",
112 : : .stop = ut_stop,
113 : : .update = ut_update,
114 : : }, {
115 : : .name = "ut_test2",
116 : : .stop = ut_stop,
117 : : .update = ut_update,
118 : : }
119 : : };
120 : :
121 : : static void
122 : 4 : test_nvme_io_msg_ctrlr_register_unregister(void)
123 : : {
124 : 4 : struct spdk_nvme_ctrlr ctrlr = {};
125 : : int rc;
126 : :
127 : 4 : STAILQ_INIT(&ctrlr.io_producers);
128 : 4 : MOCK_SET(spdk_nvme_ctrlr_alloc_io_qpair, (void *)0xDEADBEEF);
129 : :
130 : 4 : rc = nvme_io_msg_ctrlr_register(&ctrlr, &ut_nvme_io_msg_producer[0]);
131 : 4 : CU_ASSERT(rc == 0);
132 : 4 : CU_ASSERT(ctrlr.external_io_msgs != NULL);
133 : 4 : CU_ASSERT(!STAILQ_EMPTY(&ctrlr.io_producers));
134 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == (void *)0xDEADBEEF);
135 : :
136 : 4 : nvme_io_msg_ctrlr_unregister(&ctrlr, &ut_nvme_io_msg_producer[0]);
137 : 4 : CU_ASSERT(ctrlr.external_io_msgs == NULL);
138 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == NULL);
139 : 4 : CU_ASSERT(STAILQ_EMPTY(&ctrlr.io_producers));
140 : :
141 : : /* Multiple producer */
142 : 4 : rc = nvme_io_msg_ctrlr_register(&ctrlr, &ut_nvme_io_msg_producer[0]);
143 : 4 : CU_ASSERT(rc == 0);
144 : :
145 : 4 : rc = nvme_io_msg_ctrlr_register(&ctrlr, &ut_nvme_io_msg_producer[1]);
146 : 4 : CU_ASSERT(rc == 0);
147 : 4 : CU_ASSERT(ctrlr.external_io_msgs != NULL);
148 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == (void *)0xDEADBEEF);
149 : 4 : nvme_io_msg_ctrlr_unregister(&ctrlr, &ut_nvme_io_msg_producer[0]);
150 : 4 : CU_ASSERT(!STAILQ_EMPTY(&ctrlr.io_producers));
151 : 4 : nvme_io_msg_ctrlr_unregister(&ctrlr, &ut_nvme_io_msg_producer[1]);
152 : 4 : CU_ASSERT(STAILQ_EMPTY(&ctrlr.io_producers));
153 : 4 : CU_ASSERT(ctrlr.external_io_msgs == NULL);
154 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == NULL);
155 : :
156 : : /* The same producer exist */
157 : 4 : rc = nvme_io_msg_ctrlr_register(&ctrlr, &ut_nvme_io_msg_producer[0]);
158 : 4 : CU_ASSERT(rc == 0);
159 : 4 : CU_ASSERT(ctrlr.external_io_msgs != NULL);
160 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == (void *)0xDEADBEEF);
161 : :
162 : 4 : rc = nvme_io_msg_ctrlr_register(&ctrlr, &ut_nvme_io_msg_producer[0]);
163 : 4 : CU_ASSERT(rc == -EEXIST);
164 : 4 : nvme_io_msg_ctrlr_unregister(&ctrlr, &ut_nvme_io_msg_producer[0]);
165 : 4 : CU_ASSERT(STAILQ_EMPTY(&ctrlr.io_producers));
166 : 4 : CU_ASSERT(ctrlr.external_io_msgs == NULL);
167 : 4 : CU_ASSERT(ctrlr.external_io_msgs_qpair == NULL);
168 [ - - - + ]: 4 : MOCK_CLEAR(spdk_nvme_ctrlr_alloc_io_qpair);
169 : 4 : }
170 : :
171 : : int
172 : 4 : main(int argc, char **argv)
173 : : {
174 : 4 : CU_pSuite suite = NULL;
175 : : unsigned int num_failures;
176 : :
177 : 4 : CU_initialize_registry();
178 : :
179 : 4 : suite = CU_add_suite("nvme_io_msg", NULL, NULL);
180 : 4 : CU_ADD_TEST(suite, test_nvme_io_msg_send);
181 : 4 : CU_ADD_TEST(suite, test_nvme_io_msg_process);
182 : 4 : CU_ADD_TEST(suite, test_nvme_io_msg_ctrlr_register_unregister);
183 : :
184 : 4 : num_failures = spdk_ut_run_tests(argc, argv, NULL);
185 : 4 : CU_cleanup_registry();
186 : 4 : return num_failures;
187 : : }
|