xref: /openbmc/libmctp/tests/test_core.c (revision 61c95992c980f343c672d23efcbcca6bdba5bdc4)
1 /* SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later */
2 #define _GNU_SOURCE
3 
4 #ifdef NDEBUG
5 #undef NDEBUG
6 #endif
7 
8 #if HAVE_CONFIG_H
9 #include "config.h"
10 #endif
11 
12 #include <assert.h>
13 #include <fcntl.h>
14 #include <stdbool.h>
15 #include <stdint.h>
16 #include <stdio.h>
17 #include <stdlib.h>
18 #include <string.h>
19 #include <unistd.h>
20 #include <errno.h>
21 
22 #include "compiler.h"
23 #include "libmctp-alloc.h"
24 #include "libmctp-log.h"
25 #include "range.h"
26 #include "test-utils.h"
27 
28 #define TEST_DEST_EID		9
29 #define TEST_DEST_NULL_EID	0
30 #define TEST_DEST_BROADCAST_EID 255
31 #define TEST_SRC_EID		10
32 
33 #ifndef ARRAY_SIZE
34 #define ARRAY_SIZE(a) (sizeof(a) / sizeof(a[0]))
35 #endif
36 
37 #define MAX_PAYLOAD_SIZE 50000
38 
39 struct pktbuf {
40 	struct mctp_hdr hdr;
41 	uint8_t *payload;
42 };
43 
44 struct test_params {
45 	bool seen;
46 	size_t message_size;
47 	uint8_t msg_tag;
48 	bool tag_owner;
49 };
50 
rx_message(uint8_t eid __unused,bool tag_owner,uint8_t msg_tag,void * data,void * msg __unused,size_t len)51 static void rx_message(uint8_t eid __unused, bool tag_owner, uint8_t msg_tag,
52 		       void *data, void *msg __unused, size_t len)
53 {
54 	struct test_params *param = (struct test_params *)data;
55 
56 	mctp_prdebug("MCTP message received: len %zd, tag %u", len, msg_tag);
57 
58 	param->seen = true;
59 	param->message_size = len;
60 	param->msg_tag = msg_tag;
61 	param->tag_owner = tag_owner;
62 }
63 
get_sequence()64 static uint8_t get_sequence()
65 {
66 	static uint8_t pkt_seq = 0;
67 
68 	return (pkt_seq++ % 4);
69 }
70 
get_tag()71 static uint8_t get_tag()
72 {
73 	static uint8_t tag = 0;
74 
75 	return (tag++ % 8);
76 }
77 
78 /*
79  * receive_pktbuf bypasses all bindings and directly invokes mctp_bus_rx.
80  * This is necessary in order invoke test cases on the core functionality.
81  * The memory allocated for the mctp packet is capped at MCTP_BTU
82  * size, however, the mimiced rx pkt still retains the len parameter.
83  * This allows to mimic packets larger than a sane memory allocator can
84  * provide.
85  */
receive_ptkbuf(struct mctp_binding_test * binding,const struct pktbuf * pktbuf,size_t len)86 static void receive_ptkbuf(struct mctp_binding_test *binding,
87 			   const struct pktbuf *pktbuf, size_t len)
88 {
89 	size_t alloc_size = MIN((size_t)MCTP_BTU, len);
90 	struct mctp_pktbuf *rx_pkt;
91 
92 	rx_pkt = __mctp_alloc(sizeof(*rx_pkt) + MCTP_PACKET_SIZE(alloc_size));
93 	assert(rx_pkt);
94 
95 	/* Preserve passed len parameter */
96 	rx_pkt->size = MCTP_PACKET_SIZE(len);
97 	rx_pkt->start = 0;
98 	rx_pkt->end = MCTP_PACKET_SIZE(len);
99 	rx_pkt->mctp_hdr_off = 0;
100 	memcpy(rx_pkt->data, &pktbuf->hdr, sizeof(pktbuf->hdr));
101 	memcpy(rx_pkt->data + sizeof(pktbuf->hdr), pktbuf->payload, alloc_size);
102 
103 	mctp_bus_rx((struct mctp_binding *)binding, rx_pkt);
104 	__mctp_free(rx_pkt);
105 }
106 
receive_one_fragment(struct mctp_binding_test * binding,uint8_t * payload,size_t fragment_size,uint8_t flags_seq_tag,struct pktbuf * pktbuf)107 static void receive_one_fragment(struct mctp_binding_test *binding,
108 				 uint8_t *payload, size_t fragment_size,
109 				 uint8_t flags_seq_tag, struct pktbuf *pktbuf)
110 {
111 	pktbuf->hdr.flags_seq_tag = flags_seq_tag;
112 	pktbuf->payload = payload;
113 	receive_ptkbuf(binding, pktbuf, fragment_size);
114 }
115 
receive_two_fragment_message(struct mctp_binding_test * binding,uint8_t * payload,size_t fragment1_size,size_t fragment2_size,struct pktbuf * pktbuf)116 static void receive_two_fragment_message(struct mctp_binding_test *binding,
117 					 uint8_t *payload,
118 					 size_t fragment1_size,
119 					 size_t fragment2_size,
120 					 struct pktbuf *pktbuf)
121 {
122 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
123 	uint8_t flags_seq_tag;
124 
125 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
126 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
127 	receive_one_fragment(binding, payload, fragment1_size, flags_seq_tag,
128 			     pktbuf);
129 
130 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
131 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
132 	receive_one_fragment(binding, payload + fragment1_size, fragment2_size,
133 			     flags_seq_tag, pktbuf);
134 }
135 
mctp_core_test_simple_rx()136 static void mctp_core_test_simple_rx()
137 {
138 	struct mctp *mctp = NULL;
139 	struct mctp_binding_test *binding = NULL;
140 	struct test_params test_param;
141 	uint8_t test_payload[2 * MCTP_BTU];
142 	struct pktbuf pktbuf;
143 
144 	memset(test_payload, 0, sizeof(test_payload));
145 	test_param.seen = false;
146 	test_param.message_size = 0;
147 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
148 	mctp_set_rx_all(mctp, rx_message, &test_param);
149 	memset(&pktbuf, 0, sizeof(pktbuf));
150 	pktbuf.hdr.dest = TEST_DEST_EID;
151 	pktbuf.hdr.src = TEST_SRC_EID;
152 
153 	/* Receive 2 fragments of equal size */
154 	receive_two_fragment_message(binding, test_payload, MCTP_BTU, MCTP_BTU,
155 				     &pktbuf);
156 
157 	assert(test_param.seen);
158 	assert(test_param.message_size == 2 * MCTP_BTU);
159 
160 	mctp_binding_test_destroy(binding);
161 	mctp_destroy(mctp);
162 }
163 
mctp_core_test_receive_equal_length_fragments()164 static void mctp_core_test_receive_equal_length_fragments()
165 {
166 	struct mctp *mctp = NULL;
167 	struct mctp_binding_test *binding = NULL;
168 	struct test_params test_param;
169 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
170 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
171 	struct pktbuf pktbuf;
172 	uint8_t flags_seq_tag;
173 
174 	memset(test_payload, 0, sizeof(test_payload));
175 	test_param.seen = false;
176 	test_param.message_size = 0;
177 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
178 	mctp_set_rx_all(mctp, rx_message, &test_param);
179 	memset(&pktbuf, 0, sizeof(pktbuf));
180 	pktbuf.hdr.dest = TEST_DEST_EID;
181 	pktbuf.hdr.src = TEST_SRC_EID;
182 
183 	/* Receive 3 fragments, each of size MCTP_BTU */
184 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
185 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
186 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
187 			     &pktbuf);
188 
189 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
190 	receive_one_fragment(binding, test_payload + MCTP_BTU, MCTP_BTU,
191 			     flags_seq_tag, &pktbuf);
192 
193 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
194 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
195 	receive_one_fragment(binding, test_payload + (2 * MCTP_BTU), MCTP_BTU,
196 			     flags_seq_tag, &pktbuf);
197 
198 	assert(test_param.seen);
199 	assert(test_param.message_size == 3 * MCTP_BTU);
200 
201 	mctp_binding_test_destroy(binding);
202 	mctp_destroy(mctp);
203 }
204 
mctp_core_test_receive_unexpected_smaller_middle_fragment()205 static void mctp_core_test_receive_unexpected_smaller_middle_fragment()
206 {
207 	struct mctp *mctp = NULL;
208 	struct mctp_binding_test *binding = NULL;
209 	struct test_params test_param;
210 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
211 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
212 	struct pktbuf pktbuf;
213 	uint8_t flags_seq_tag;
214 
215 	memset(test_payload, 0, sizeof(test_payload));
216 	test_param.seen = false;
217 	test_param.message_size = 0;
218 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
219 	mctp_set_rx_all(mctp, rx_message, &test_param);
220 	memset(&pktbuf, 0, sizeof(pktbuf));
221 	pktbuf.hdr.dest = TEST_DEST_EID;
222 	pktbuf.hdr.src = TEST_SRC_EID;
223 
224 	/* Middle fragment with size MCTP_BTU - 1 */
225 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
226 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
227 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
228 			     &pktbuf);
229 
230 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
231 	receive_one_fragment(binding, test_payload + MCTP_BTU, MCTP_BTU - 1,
232 			     flags_seq_tag, &pktbuf);
233 
234 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
235 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
236 	receive_one_fragment(binding, test_payload + (2 * MCTP_BTU), MCTP_BTU,
237 			     flags_seq_tag, &pktbuf);
238 
239 	assert(!test_param.seen);
240 
241 	mctp_binding_test_destroy(binding);
242 	mctp_destroy(mctp);
243 }
244 
mctp_core_test_receive_unexpected_bigger_middle_fragment()245 static void mctp_core_test_receive_unexpected_bigger_middle_fragment()
246 {
247 	struct mctp *mctp = NULL;
248 	struct mctp_binding_test *binding = NULL;
249 	struct test_params test_param;
250 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
251 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
252 	struct pktbuf pktbuf;
253 	uint8_t flags_seq_tag;
254 
255 	memset(test_payload, 0, sizeof(test_payload));
256 	test_param.seen = false;
257 	test_param.message_size = 0;
258 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
259 	mctp_set_rx_all(mctp, rx_message, &test_param);
260 	memset(&pktbuf, 0, sizeof(pktbuf));
261 	pktbuf.hdr.dest = TEST_DEST_EID;
262 	pktbuf.hdr.src = TEST_SRC_EID;
263 
264 	/* Middle fragment with size MCTP_BTU + 1 */
265 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
266 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
267 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
268 			     &pktbuf);
269 
270 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
271 	receive_one_fragment(binding, test_payload + MCTP_BTU, MCTP_BTU + 1,
272 			     flags_seq_tag, &pktbuf);
273 
274 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
275 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
276 	receive_one_fragment(binding, test_payload + (2 * MCTP_BTU), MCTP_BTU,
277 			     flags_seq_tag, &pktbuf);
278 
279 	assert(!test_param.seen);
280 
281 	mctp_binding_test_destroy(binding);
282 	mctp_destroy(mctp);
283 }
284 
mctp_core_test_receive_smaller_end_fragment()285 static void mctp_core_test_receive_smaller_end_fragment()
286 {
287 	struct mctp *mctp = NULL;
288 	struct mctp_binding_test *binding = NULL;
289 	struct test_params test_param;
290 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
291 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
292 	uint8_t end_frag_size = MCTP_BTU - 10;
293 	struct pktbuf pktbuf;
294 	uint8_t flags_seq_tag;
295 
296 	memset(test_payload, 0, sizeof(test_payload));
297 	test_param.seen = false;
298 	test_param.message_size = 0;
299 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
300 	mctp_set_rx_all(mctp, rx_message, &test_param);
301 	memset(&pktbuf, 0, sizeof(pktbuf));
302 	pktbuf.hdr.dest = TEST_DEST_EID;
303 	pktbuf.hdr.src = TEST_SRC_EID;
304 
305 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
306 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
307 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
308 			     &pktbuf);
309 
310 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
311 	receive_one_fragment(binding, test_payload + MCTP_BTU, MCTP_BTU,
312 			     flags_seq_tag, &pktbuf);
313 
314 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
315 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
316 	receive_one_fragment(binding, test_payload + (2 * MCTP_BTU),
317 			     end_frag_size, flags_seq_tag, &pktbuf);
318 
319 	assert(test_param.seen);
320 	assert(test_param.message_size ==
321 	       (size_t)(2 * MCTP_BTU + end_frag_size));
322 
323 	mctp_binding_test_destroy(binding);
324 	mctp_destroy(mctp);
325 }
326 
mctp_core_test_receive_bigger_end_fragment()327 static void mctp_core_test_receive_bigger_end_fragment()
328 {
329 	struct mctp *mctp = NULL;
330 	struct mctp_binding_test *binding = NULL;
331 	struct test_params test_param;
332 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
333 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
334 	uint8_t end_frag_size = MCTP_BTU + 10;
335 	struct pktbuf pktbuf;
336 	uint8_t flags_seq_tag;
337 
338 	memset(test_payload, 0, sizeof(test_payload));
339 	test_param.seen = false;
340 	test_param.message_size = 0;
341 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
342 	mctp_set_rx_all(mctp, rx_message, &test_param);
343 	memset(&pktbuf, 0, sizeof(pktbuf));
344 	pktbuf.hdr.dest = TEST_DEST_EID;
345 	pktbuf.hdr.src = TEST_SRC_EID;
346 
347 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
348 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
349 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
350 			     &pktbuf);
351 
352 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
353 	receive_one_fragment(binding, test_payload + MCTP_BTU, MCTP_BTU,
354 			     flags_seq_tag, &pktbuf);
355 
356 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
357 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
358 	receive_one_fragment(binding, test_payload + (2 * MCTP_BTU),
359 			     end_frag_size, flags_seq_tag, &pktbuf);
360 
361 	assert(!test_param.seen);
362 
363 	mctp_binding_test_destroy(binding);
364 	mctp_destroy(mctp);
365 }
366 
mctp_core_test_drop_large_fragments()367 static void mctp_core_test_drop_large_fragments()
368 {
369 	struct mctp *mctp = NULL;
370 	struct mctp_binding_test *binding = NULL;
371 	struct test_params test_param;
372 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
373 	struct pktbuf pktbuf;
374 
375 	memset(test_payload, 0, sizeof(test_payload));
376 	test_param.seen = false;
377 	test_param.message_size = 0;
378 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
379 	mctp_set_rx_all(mctp, rx_message, &test_param);
380 	memset(&pktbuf, 0, sizeof(pktbuf));
381 	pktbuf.hdr.dest = TEST_DEST_EID;
382 	pktbuf.hdr.src = TEST_SRC_EID;
383 
384 	/* Receive a large payload - first fragment with MCTP_BTU bytes,
385 	* 2nd fragment of SIZE_MAX */
386 
387 	receive_two_fragment_message(binding, test_payload, MCTP_BTU,
388 				     SIZE_MAX - sizeof(struct mctp_hdr),
389 				     &pktbuf);
390 
391 	assert(!test_param.seen);
392 
393 	mctp_binding_test_destroy(binding);
394 	mctp_destroy(mctp);
395 }
396 
mctp_core_test_exhaust_context_buffers()397 static void mctp_core_test_exhaust_context_buffers()
398 {
399 	struct mctp *mctp = NULL;
400 	struct mctp_binding_test *binding = NULL;
401 	struct test_params test_param;
402 	static uint8_t test_payload[MAX_PAYLOAD_SIZE];
403 	uint8_t tag = MCTP_HDR_FLAG_TO | get_tag();
404 	uint8_t i = 0;
405 	const uint8_t max_context_buffers = 16;
406 	struct pktbuf pktbuf;
407 	uint8_t flags_seq_tag;
408 
409 	memset(test_payload, 0, sizeof(test_payload));
410 	test_param.seen = false;
411 	test_param.message_size = 0;
412 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
413 	mctp_set_rx_all(mctp, rx_message, &test_param);
414 	memset(&pktbuf, 0, sizeof(pktbuf));
415 	pktbuf.hdr.dest = TEST_DEST_EID;
416 	pktbuf.hdr.src = TEST_SRC_EID;
417 
418 	/* Exhaust all 16 context buffers*/
419 	for (i = 0; i < max_context_buffers; i++) {
420 		flags_seq_tag = MCTP_HDR_FLAG_SOM |
421 				(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
422 		receive_one_fragment(binding, test_payload, MCTP_BTU,
423 				     flags_seq_tag, &pktbuf);
424 
425 		/* Change source EID so that different contexts are created */
426 		pktbuf.hdr.src++;
427 	}
428 
429 	/* Send a full message from a different EID */
430 	pktbuf.hdr.src++;
431 	receive_two_fragment_message(binding, test_payload, MCTP_BTU, MCTP_BTU,
432 				     &pktbuf);
433 
434 	/* Message assembly should fail */
435 	assert(!test_param.seen);
436 
437 	/* Complete message assembly for one of the messages */
438 	pktbuf.hdr.src -= max_context_buffers;
439 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
440 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) | tag;
441 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
442 			     &pktbuf);
443 
444 	assert(test_param.seen);
445 	assert(test_param.message_size == (2 * MCTP_BTU));
446 
447 	mctp_binding_test_destroy(binding);
448 	mctp_destroy(mctp);
449 }
450 
mctp_core_test_rx_with_tag()451 static void mctp_core_test_rx_with_tag()
452 {
453 	struct mctp *mctp = NULL;
454 	struct mctp_binding_test *binding = NULL;
455 	struct test_params test_param;
456 	static uint8_t test_payload[MCTP_BTU];
457 	uint8_t tag = get_tag();
458 	struct pktbuf pktbuf;
459 	uint8_t flags_seq_tag;
460 
461 	memset(test_payload, 0, sizeof(test_payload));
462 	test_param.seen = false;
463 	test_param.message_size = 0;
464 	test_param.msg_tag = 0;
465 	test_param.tag_owner = false;
466 
467 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
468 	mctp_set_rx_all(mctp, rx_message, &test_param);
469 	memset(&pktbuf, 0, sizeof(pktbuf));
470 	pktbuf.hdr.dest = TEST_DEST_EID;
471 	pktbuf.hdr.src = TEST_SRC_EID;
472 
473 	/* Set tag and tag owner fields for a recieve packet */
474 	flags_seq_tag = MCTP_HDR_FLAG_SOM | MCTP_HDR_FLAG_EOM |
475 			(1 << MCTP_HDR_TO_SHIFT) | tag;
476 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
477 			     &pktbuf);
478 
479 	assert(test_param.seen);
480 	assert(test_param.message_size == (MCTP_BTU));
481 	assert(test_param.msg_tag == tag);
482 	assert(test_param.tag_owner);
483 
484 	mctp_binding_test_destroy(binding);
485 	mctp_destroy(mctp);
486 }
487 
mctp_core_test_rx_with_tag_multifragment()488 static void mctp_core_test_rx_with_tag_multifragment()
489 {
490 	struct mctp *mctp = NULL;
491 	struct mctp_binding_test *binding = NULL;
492 	struct test_params test_param;
493 	static uint8_t test_payload[MCTP_BTU];
494 	uint8_t tag = get_tag();
495 	struct pktbuf pktbuf;
496 	uint8_t flags_seq_tag;
497 
498 	memset(test_payload, 0, sizeof(test_payload));
499 	test_param.seen = false;
500 	test_param.message_size = 0;
501 	test_param.msg_tag = 0;
502 	test_param.tag_owner = false;
503 
504 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
505 	mctp_set_rx_all(mctp, rx_message, &test_param);
506 	memset(&pktbuf, 0, sizeof(pktbuf));
507 	pktbuf.hdr.dest = TEST_DEST_EID;
508 	pktbuf.hdr.src = TEST_SRC_EID;
509 
510 	/* Set tag and tag owner fields for a 3 fragment packet */
511 	flags_seq_tag = MCTP_HDR_FLAG_SOM |
512 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) |
513 			(1 << MCTP_HDR_TO_SHIFT) | tag;
514 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
515 			     &pktbuf);
516 
517 	flags_seq_tag = (get_sequence() << MCTP_HDR_SEQ_SHIFT) |
518 			(1 << MCTP_HDR_TO_SHIFT) | tag;
519 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
520 			     &pktbuf);
521 
522 	flags_seq_tag = MCTP_HDR_FLAG_EOM |
523 			(get_sequence() << MCTP_HDR_SEQ_SHIFT) |
524 			(1 << MCTP_HDR_TO_SHIFT) | tag;
525 	receive_one_fragment(binding, test_payload, MCTP_BTU, flags_seq_tag,
526 			     &pktbuf);
527 
528 	assert(test_param.seen);
529 	assert(test_param.message_size == (3 * MCTP_BTU));
530 	assert(test_param.msg_tag == tag);
531 	assert(test_param.tag_owner);
532 
533 	mctp_binding_test_destroy(binding);
534 	mctp_destroy(mctp);
535 }
536 
537 /*
538  * This test case covers null destination eid. MCTP
539  * daemon might query endpoint (i.e., Get Endpoint
540  * ID command) by physical address requests and
541  * destination eid as 0. Endpoint shall accept and
542  * handle this request.
543  */
mctp_core_test_rx_with_null_dst_eid()544 static void mctp_core_test_rx_with_null_dst_eid()
545 {
546 	struct mctp *mctp = NULL;
547 	struct mctp_binding_test *binding = NULL;
548 	struct test_params test_param;
549 	uint8_t test_payload[2 * MCTP_BTU];
550 	struct pktbuf pktbuf;
551 
552 	memset(test_payload, 0, sizeof(test_payload));
553 	test_param.seen = false;
554 	test_param.message_size = 0;
555 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
556 	mctp_set_rx_all(mctp, rx_message, &test_param);
557 	memset(&pktbuf, 0, sizeof(pktbuf));
558 	pktbuf.hdr.dest = TEST_DEST_NULL_EID;
559 	pktbuf.hdr.src = TEST_SRC_EID;
560 
561 	/* Receive 2 fragments of equal size */
562 	receive_two_fragment_message(binding, test_payload, MCTP_BTU, MCTP_BTU,
563 				     &pktbuf);
564 
565 	assert(test_param.seen);
566 	assert(test_param.message_size == 2 * MCTP_BTU);
567 
568 	mctp_binding_test_destroy(binding);
569 	mctp_destroy(mctp);
570 }
571 
572 /*
573  * This test case covers Broadcast Request message (i.e.,
574  * `Endpoint Discovery` command). Endpoint shall accept
575  * and handle this request.
576  */
mctp_core_test_rx_with_broadcast_dst_eid()577 static void mctp_core_test_rx_with_broadcast_dst_eid()
578 {
579 	struct mctp *mctp = NULL;
580 	struct mctp_binding_test *binding = NULL;
581 	struct test_params test_param;
582 	uint8_t test_payload[2 * MCTP_BTU];
583 	struct pktbuf pktbuf;
584 
585 	memset(test_payload, 0, sizeof(test_payload));
586 	test_param.seen = false;
587 	test_param.message_size = 0;
588 	mctp_test_stack_init(&mctp, &binding, TEST_DEST_EID);
589 	mctp_set_rx_all(mctp, rx_message, &test_param);
590 	memset(&pktbuf, 0, sizeof(pktbuf));
591 	pktbuf.hdr.dest = TEST_DEST_BROADCAST_EID;
592 	pktbuf.hdr.src = TEST_SRC_EID;
593 
594 	/* Receive 2 fragments of equal size */
595 	receive_two_fragment_message(binding, test_payload, MCTP_BTU, MCTP_BTU,
596 				     &pktbuf);
597 
598 	assert(test_param.seen);
599 	assert(test_param.message_size == 2 * MCTP_BTU);
600 
601 	mctp_binding_test_destroy(binding);
602 	mctp_destroy(mctp);
603 }
604 
605 /*
606  * This test case tests tag allocation. 8 tags
607  * are allowed to be pending.
608  */
mctp_core_test_tx_alloc_tag()609 static void mctp_core_test_tx_alloc_tag()
610 {
611 	struct mctp *mctp = NULL;
612 	struct mctp_binding_test *binding = NULL;
613 	struct test_params test_param;
614 	uint8_t msg_tag;
615 	void *msg;
616 	int rc;
617 	mctp_eid_t dest_eid1 = 30;
618 	size_t msg_len = 10;
619 
620 	mctp_test_stack_init(&mctp, &binding, dest_eid1);
621 	mctp_set_rx_all(mctp, rx_message, &test_param);
622 
623 	uint8_t used = 0;
624 	for (int i = 0; i < 8; i++) {
625 		test_param.seen = false;
626 		test_param.msg_tag = 0xff;
627 		test_param.tag_owner = false;
628 
629 		msg = __mctp_alloc(msg_len);
630 		memset(msg, 0x99, msg_len);
631 		rc = mctp_message_tx_request(mctp, dest_eid1, msg, msg_len,
632 					     &msg_tag);
633 		assert(rc == 0);
634 		assert(test_param.seen == true);
635 		assert(test_param.msg_tag == msg_tag);
636 		assert(test_param.tag_owner == true);
637 		used |= (1 << msg_tag);
638 	}
639 	assert(used == 0xff);
640 
641 	/* Ran out of tags */
642 	test_param.seen = false;
643 	msg = __mctp_alloc(msg_len);
644 	memset(msg, 0x99, msg_len);
645 	rc = mctp_message_tx_request(mctp, dest_eid1, msg, msg_len, &msg_tag);
646 	assert(rc == -EBUSY);
647 	assert(test_param.seen == false);
648 
649 	/* Send/Receive a response to one of those tags */
650 	test_param.seen = false;
651 	msg = __mctp_alloc(msg_len);
652 	memset(msg, 0x99, msg_len);
653 	/* Arbitrary one */
654 	uint8_t replied_tag = 3;
655 	rc = mctp_message_tx_alloced(mctp, dest_eid1, false, replied_tag, msg,
656 				     msg_len);
657 	assert(rc == 0);
658 	assert(test_param.seen == true);
659 	assert(test_param.msg_tag == replied_tag);
660 	assert(test_param.tag_owner == false);
661 
662 	/* Now sending allocates that tag again, since it is the only spare one */
663 	test_param.seen = false;
664 	msg = __mctp_alloc(msg_len);
665 	memset(msg, 0x99, msg_len);
666 	rc = mctp_message_tx_request(mctp, dest_eid1, msg, msg_len, &msg_tag);
667 	assert(rc == 0);
668 	assert(test_param.seen == true);
669 	assert(msg_tag == replied_tag);
670 
671 	mctp_binding_test_destroy(binding);
672 	mctp_destroy(mctp);
673 }
674 
675 /* clang-format off */
676 #define TEST_CASE(test) { #test, test }
677 static const struct {
678 	const char *name;
679 	void (*test)(void);
680 } mctp_core_tests[] = {
681 	TEST_CASE(mctp_core_test_simple_rx),
682 	TEST_CASE(mctp_core_test_receive_equal_length_fragments),
683 	TEST_CASE(mctp_core_test_receive_unexpected_smaller_middle_fragment),
684 	TEST_CASE(mctp_core_test_receive_unexpected_bigger_middle_fragment),
685 	TEST_CASE(mctp_core_test_receive_smaller_end_fragment),
686 	TEST_CASE(mctp_core_test_receive_bigger_end_fragment),
687 	TEST_CASE(mctp_core_test_drop_large_fragments),
688 	TEST_CASE(mctp_core_test_exhaust_context_buffers),
689 	TEST_CASE(mctp_core_test_rx_with_tag),
690 	TEST_CASE(mctp_core_test_rx_with_tag_multifragment),
691 	TEST_CASE(mctp_core_test_rx_with_null_dst_eid),
692 	TEST_CASE(mctp_core_test_rx_with_broadcast_dst_eid),
693 	TEST_CASE(mctp_core_test_tx_alloc_tag),
694 };
695 /* clang-format on */
696 
main(void)697 int main(void)
698 {
699 	uint8_t i;
700 
701 	mctp_set_log_stdio(MCTP_LOG_DEBUG);
702 
703 	static_assert(ARRAY_SIZE(mctp_core_tests) < SIZE_MAX, "size");
704 	for (i = 0; i < ARRAY_SIZE(mctp_core_tests); i++) {
705 		mctp_prlog(MCTP_LOG_DEBUG, "begin: %s",
706 			   mctp_core_tests[i].name);
707 		mctp_core_tests[i].test();
708 		mctp_prlog(MCTP_LOG_DEBUG, "end: %s\n",
709 			   mctp_core_tests[i].name);
710 	}
711 
712 	return 0;
713 }
714