| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276 |
- /*
- * Copyright (c) 2006-2021, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2021-08-28 Sherman the first version
- * 2023-09-15 xqyjlj change stack size in cpu64
- * fix in smp
- * 2025-11-17 Ze-Hou add standardized utest documentation block
- */
- /**
- * Test Case Name: Kernel Core Message Queue Test
- *
- * Test Objectives:
- * - Validate the RT-Thread kernel message queue (rt_messagequeue) functionality
- * - Test static and dynamic message queue creation, initialization, sending,
- * receiving, urgent send, priority send, reset, detach, and delete APIs
- *
- * Test Scenarios:
- * - Initialize a static message queue and verify correct setup
- * - Create a dynamic message queue and verify allocation
- * - Send messages to the queue until full, test blocking and non-blocking send,
- * urgent send, and priority send (if enabled)
- * - Receive messages from the queue, verify order and data integrity, including
- * priority receive (if enabled)
- * - Reset the message queue and verify it is empty
- * - Detach and delete message queues, ensuring resources are released
- * - Use multiple threads to simulate concurrent send/receive operations
- *
- * Verification Metrics:
- * - All uassert assertions pass without failure
- * - Message queues are created, initialized, detached, and deleted successfully
- * - Messages are sent and received in correct order and with correct data
- * - Blocking and non-blocking operations behave as expected
- * - Urgent and priority send/receive functions work correctly (if enabled)
- *
- * Dependencies:
- * - Enable message queue priority (RT-Thread Kernel -> Inter-Thread communication
- * -> Enable message queue priority)
- * - Enable Message Queue Test (RT-Thread Utestcases -> Kernel Core -> Message Queue Test)
- * - Test on any RT-Thread supported platform (e.g., qemu-virt64-riscv)
- *
- * Expected Results:
- * - After executing this test in msh, the expected output should be:
- * "[ PASSED ] [ result ] testcase (core.messagequeue)"
- */
- #include <rtthread.h>
- #include "utest.h"
- #define THREAD_STACKSIZE UTEST_THR_STACK_SIZE
- #define MSG_SIZE 4
- #define MAX_MSGS 5
- static struct rt_messagequeue static_mq;
- static rt_uint8_t mq_buf[RT_MQ_BUF_SIZE(MSG_SIZE, MAX_MSGS)];
- static struct rt_thread mq_send_thread;
- static struct rt_thread mq_recv_thread;
- static rt_uint8_t mq_send_stack[UTEST_THR_STACK_SIZE];
- static rt_uint8_t mq_recv_stack[UTEST_THR_STACK_SIZE];
- static struct rt_event finish_e;
- #define MQSEND_FINISH 0x01
- #define MQRECV_FINIHS 0x02
- #ifdef RT_USING_HEAP
- static rt_mq_t dynamic_mq;
- #endif /* RT_USING_HEAP */
- static void test_mq_init(void)
- {
- rt_err_t ret;
- ret = rt_mq_init(&static_mq,"testmq1", mq_buf, MSG_SIZE, sizeof(mq_buf), RT_IPC_FLAG_FIFO);
- uassert_true(ret == RT_EOK);
- }
- static void test_mq_create(void)
- {
- #ifdef RT_USING_HEAP
- dynamic_mq = rt_mq_create("testmq2", MSG_SIZE, MAX_MSGS, RT_IPC_FLAG_FIFO);
- uassert_true(dynamic_mq != RT_NULL);
- #endif /* RT_USING_HEAP */
- }
- static void mq_send_case(rt_mq_t testmq)
- {
- rt_uint32_t send_buf[MAX_MSGS+1] = {0};
- rt_err_t ret = RT_EOK;
- for (int var = 0; var < MAX_MSGS; ++var)
- {
- send_buf[var] = var + 1;
- ret = rt_mq_send_wait(testmq, &send_buf[var], sizeof(send_buf[0]), RT_WAITING_FOREVER);
- uassert_true(ret == RT_EOK);
- }
- send_buf[MAX_MSGS] = MAX_MSGS + 1;
- ret = rt_mq_send(testmq, &send_buf[MAX_MSGS], sizeof(send_buf[0]));
- uassert_true(ret == -RT_EFULL);
- ret = rt_mq_send_wait(testmq, &send_buf[MAX_MSGS], sizeof(send_buf[0]), RT_WAITING_FOREVER);
- uassert_true(ret == RT_EOK);
- while (testmq->entry != 0)
- {
- rt_thread_delay(100);
- }
- ret = rt_mq_send(testmq, &send_buf[1], sizeof(send_buf[0]));
- uassert_true(ret == RT_EOK);
- ret = rt_mq_send(testmq, &send_buf[2], sizeof(send_buf[0]));
- uassert_true(ret == RT_EOK);
- ret = rt_mq_urgent(testmq, &send_buf[0], sizeof(send_buf[0]));
- uassert_true(ret == RT_EOK);
- while (testmq->entry != 0)
- {
- rt_thread_delay(100);
- }
- #ifdef RT_USING_MESSAGEQUEUE_PRIORITY
- ret = rt_mq_send_wait_prio(testmq, &send_buf[3], sizeof(send_buf[0]), 3, 0, RT_UNINTERRUPTIBLE);
- uassert_true(ret == RT_EOK);
- ret = rt_mq_send_wait_prio(testmq, &send_buf[0], sizeof(send_buf[0]), 0, 0, RT_UNINTERRUPTIBLE);
- uassert_true(ret == RT_EOK);
- ret = rt_mq_send_wait_prio(testmq, &send_buf[2], sizeof(send_buf[0]), 1, 0, RT_UNINTERRUPTIBLE);
- uassert_true(ret == RT_EOK);
- ret = rt_mq_send_wait_prio(testmq, &send_buf[4], sizeof(send_buf[0]), 4, 0, RT_UNINTERRUPTIBLE);
- uassert_true(ret == RT_EOK);
- ret = rt_mq_send_wait_prio(testmq, &send_buf[1], sizeof(send_buf[0]), 1, 0, RT_UNINTERRUPTIBLE);
- uassert_true(ret == RT_EOK);
- while (testmq->entry != 0)
- {
- rt_thread_delay(100);
- }
- #endif
- ret = rt_mq_send(testmq, &send_buf[1], sizeof(send_buf[0]));
- uassert_true(ret == RT_EOK);
- ret = rt_mq_control(testmq, RT_IPC_CMD_RESET, RT_NULL);
- uassert_true(ret == RT_EOK);
- uassert_true(testmq->entry == 0);
- }
- static void mq_send_entry(void *param)
- {
- mq_send_case(&static_mq);
- #ifdef RT_USING_HEAP
- if(dynamic_mq != RT_NULL)
- {
- mq_send_case(dynamic_mq);
- }
- #endif /* RT_USING_HEAP */
- rt_event_send(&finish_e, MQSEND_FINISH);
- }
- static void mq_recv_case(rt_mq_t testmq)
- {
- rt_uint32_t recv_buf[MAX_MSGS+1] = {0};
- rt_ssize_t ret = RT_EOK;
- for (int var = 0; var < MAX_MSGS + 1; ++var)
- {
- ret = rt_mq_recv(testmq, &recv_buf[var], sizeof(recv_buf[0]), RT_WAITING_FOREVER);
- uassert_true(ret >= 0);
- uassert_true(recv_buf[var] == (var + 1));
- }
- for (int var = 0; var < 3; ++var)
- {
- ret = rt_mq_recv(testmq, &recv_buf[var], sizeof(recv_buf[0]), RT_WAITING_FOREVER);
- uassert_true(ret >= 0);
- uassert_true(recv_buf[var] == (var + 1));
- }
- #ifdef RT_USING_MESSAGEQUEUE_PRIORITY
- rt_int32_t msg_prio;
- while (testmq->entry == MAX_MSGS)
- {
- rt_thread_delay(100);
- }
- for (int var = 0; var < MAX_MSGS; ++var)
- {
- ret = rt_mq_recv_prio(testmq, &recv_buf[var], sizeof(recv_buf[0]), &msg_prio, RT_WAITING_FOREVER, RT_UNINTERRUPTIBLE);
- rt_kprintf("msg_prio = %d\r\n", msg_prio);
- uassert_true(ret >= 0);
- uassert_true(recv_buf[var] == (MAX_MSGS - var));
- }
- #endif
- }
- static void mq_recv_entry(void *param)
- {
- mq_recv_case(&static_mq);
- #ifdef RT_USING_HEAP
- if(dynamic_mq != RT_NULL)
- {
- mq_recv_case(dynamic_mq);
- }
- #endif /* RT_USING_HEAP */
- rt_event_send(&finish_e, MQRECV_FINIHS);
- }
- static void test_mq_testcase(void)
- {
- rt_thread_startup(&mq_send_thread);
- rt_thread_startup(&mq_recv_thread);
- rt_event_recv(&finish_e, MQSEND_FINISH | MQRECV_FINIHS, RT_EVENT_FLAG_AND, RT_WAITING_FOREVER, RT_NULL);
- }
- static void test_mq_detach(void)
- {
- rt_err_t ret = rt_mq_detach(&static_mq);
- uassert_true(ret == RT_EOK);
- }
- static void test_mq_delete(void)
- {
- #ifdef RT_USING_HEAP
- rt_err_t ret = rt_mq_delete(dynamic_mq);
- uassert_true(ret == RT_EOK);
- #endif /* RT_USING_HEAP */
- }
- static rt_err_t utest_tc_init(void)
- {
- rt_err_t ret ;
- ret = rt_thread_init(&mq_send_thread, "mq_send", mq_send_entry, RT_NULL, mq_send_stack, sizeof(mq_send_stack), 22, 20);
- if(ret != RT_EOK)
- return -RT_ERROR;
- ret = rt_thread_init(&mq_recv_thread, "mq_recv", mq_recv_entry, RT_NULL, mq_recv_stack, sizeof(mq_recv_stack), 23, 20);
- if(ret != RT_EOK)
- return -RT_ERROR;
- #ifdef RT_USING_SMP
- rt_thread_control(&mq_send_thread, RT_THREAD_CTRL_BIND_CPU, (void *)0);
- rt_thread_control(&mq_recv_thread, RT_THREAD_CTRL_BIND_CPU, (void *)0);
- #endif
- ret = rt_event_init(&finish_e, "finish", RT_IPC_FLAG_FIFO);
- if(ret != RT_EOK)
- return -RT_ERROR;
- return RT_EOK;
- }
- static rt_err_t utest_tc_cleanup(void)
- {
- /* wait for threads to finish */
- rt_thread_mdelay(100);
- /* detach event object */
- rt_event_detach(&finish_e);
- return RT_EOK;
- }
- static void testcase(void)
- {
- UTEST_UNIT_RUN(test_mq_init);
- UTEST_UNIT_RUN(test_mq_create);
- UTEST_UNIT_RUN(test_mq_testcase);
- UTEST_UNIT_RUN(test_mq_detach);
- UTEST_UNIT_RUN(test_mq_delete);
- }
- UTEST_TC_EXPORT(testcase, "core.messagequeue", utest_tc_init, utest_tc_cleanup, 1000);
|