testing: add case for irqprio, should only work with qemu MPS2_AN500
MPS2_AN500 have UART1,2,3,4, TIMER0,1, is a great board to do irqprio
test.
enable CONFIG_ARCH_IRQPRIO, and use
'qemu-system-arm -M mps2-an500 -nographic -kernel nuttx.bin'
to lauch qemu
Signed-off-by: buxiasen <buxiasen@xiaomi.com>
diff --git a/testing/drivertest/CMakeLists.txt b/testing/drivertest/CMakeLists.txt
index 3548af9..5b12c18 100644
--- a/testing/drivertest/CMakeLists.txt
+++ b/testing/drivertest/CMakeLists.txt
@@ -327,6 +327,22 @@
drivertest_touchpanel.c)
endif()
+ if(CONFIG_ARCH_CHIP_MPS2_AN500 AND CONFIG_ARCH_IRQPRIO)
+ nuttx_add_application(
+ NAME
+ cmocka_driver_mps2
+ PRIORITY
+ ${CONFIG_TESTING_DRIVER_TEST_PRIORITY}
+ STACKSIZE
+ ${CONFIG_TESTING_DRIVER_TEST_STACKSIZE}
+ MODULE
+ ${CONFIG_TESTING_DRIVER_TEST}
+ DEPENDS
+ cmocka
+ SRCS
+ drivertest_mps2.c)
+ endif()
+
if(CONFIG_PM)
nuttx_add_application(
NAME
diff --git a/testing/drivertest/Makefile b/testing/drivertest/Makefile
index e817835..d7093ac 100644
--- a/testing/drivertest/Makefile
+++ b/testing/drivertest/Makefile
@@ -128,6 +128,13 @@
PROGNAME += cmocka_driver_touchpanel
endif
+ifeq ($(CONFIG_ARCH_CHIP_MPS2_AN500),y)
+ifeq ($(CONFIG_ARCH_IRQPRIO),y)
+MAINSRC += drivertest_mps2.c
+PROGNAME += cmocka_driver_mps2
+endif
+endif
+
ifneq ($(CONFIG_PM),)
MAINSRC += drivertest_pm.c
PROGNAME += cmocka_driver_pm
diff --git a/testing/drivertest/drivertest_mps2.c b/testing/drivertest/drivertest_mps2.c
new file mode 100644
index 0000000..57bda24
--- /dev/null
+++ b/testing/drivertest/drivertest_mps2.c
@@ -0,0 +1,494 @@
+/****************************************************************************
+ * apps/testing/drivertest/drivertest_mps2.c
+ *
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed with
+ * this work for additional information regarding copyright ownership. The
+ * ASF licenses this file to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance with the
+ * License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
+ * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
+ * License for the specific language governing permissions and limitations
+ * under the License.
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/nuttx.h>
+#include <stdbool.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <stdarg.h>
+#include <stddef.h>
+#include <setjmp.h>
+#include <string.h>
+#include <cmocka.h>
+
+#include <nuttx/arch.h>
+#include <nuttx/irq.h>
+#include <nuttx/semaphore.h>
+#include <nuttx/serial/uart_cmsdk.h>
+
+#include <pthread.h>
+
+/****************************************************************************
+ * Pre-processor Definitions
+ ****************************************************************************/
+
+#define CONFIG_TEST_IRQPRIO_TTHREAD 8
+#define CONFIG_TEST_IRQPRIO_LOOP_CNT 5000
+#define CONFIG_TEST_IRQPRIO_LOG 0
+
+#define MPS2_ADDR2REG_PTR(base, off) (uint32_t*)((uint32_t*)(base) + (off))
+#define MPS2_IRQ_FROMBASE(base, off) ((base) + (off))
+
+/* https://developer.arm.com/documentation/101104/0200/programmers-model/
+ * base-element/cmsdk-timer
+ */
+
+#define MPS_TIMER_CTRL_OFFSET 0
+#define MPS_TIMER_VALUE_OFFSET 1
+#define MPS_TIMER_RELOAD_OFFSET 2
+#define MPS_TIMER_CLEAR_OFFSET 3
+
+#define MPS_TIMER_CTRL_ENABLE (1<<0)
+#define MPS_TIMER_CTRL_IE (1<<3)
+
+#if CONFIG_TEST_IRQPRIO_LOG
+# define TAG_BEGIN(v) \
+do { \
+ if ((v)->begin) \
+ { \
+ up_puts((v)->begin); \
+ } \
+} while(0)
+
+# define TAG_END(v) \
+do { \
+ if ((v)->end) \
+ { \
+ up_puts((v)->end); \
+ } \
+} while(0)
+#else
+#define TAG_BEGIN(v)
+#define TAG_END(v)
+#endif
+
+static_assert(NVIC_SYSH_PRIORITY_DEFAULT == 0x80, "prio");
+
+/****************************************************************************
+ * Private Types
+ ****************************************************************************/
+
+typedef struct mps2_an500_uart_s
+{
+ volatile uint32_t *ctrl;
+ volatile uint32_t *tx;
+ volatile uint32_t *clear;
+ int irq;
+ int before;
+ sem_t *sem;
+ struct mps2_an500_uart_s *trigger;
+ int after;
+#if CONFIG_TEST_IRQPRIO_LOG
+ const char *begin;
+ const char *end;
+#endif
+} mps2_an500_uart_t;
+
+typedef struct mps2_an500_timer_s
+{
+ volatile uint32_t *reload;
+ volatile uint32_t *ctrl;
+ volatile uint32_t *clear;
+ int irq;
+ int before;
+ sem_t *sem;
+ int after;
+#if CONFIG_TEST_IRQPRIO_LOG
+ const char *begin;
+ const char *end;
+#endif
+} mps2_an500_timer_t;
+
+/****************************************************************************
+ * Private Functions Prototypes
+ ****************************************************************************/
+
+static int uart_irq_tx_handle(int irq, void *context, void *arg);
+static int timer_irq_handle(int irq, void *context, void *arg);
+static uint32_t uart_random_test(mps2_an500_uart_t *uart);
+
+/****************************************************************************
+ * Private Data
+ ****************************************************************************/
+
+static mps2_an500_uart_t uarts[5];
+static mps2_an500_timer_t timer[2];
+static sem_t test_sem = SEM_INITIALIZER(0);
+
+static const int armv7m_gpio_base = 16;
+
+/****************************************************************************
+ * Private Functions
+ ****************************************************************************/
+
+static void mps_uart_init(void)
+{
+ static const uint32_t uartbase[] =
+ {
+ 0x40004000, 0x40005000, 0x40006000, 0x40007000, 0x40009000
+ };
+
+ /* static const int an500_uart_tx_irq_offset = 1; */
+
+ static const int uarttxirq[] =
+ {
+ 1, 3, 5, 19, 21
+ };
+
+ static const int uarttxirq_prio[] =
+ {
+ 0x80, 0x90, 0xb0, 0xc0, 0xd0,
+ };
+
+#if CONFIG_TEST_IRQPRIO_LOG
+ static const char *begin_tag[] =
+ {
+ "0", "1", "2", "3", "4"
+ };
+
+ static const char *end_tag[] =
+ {
+ " ", "A", "B", "C", "D"
+ };
+#endif
+
+ mps2_an500_uart_t *prev = NULL;
+
+ for (int i = 0; i < 5; i++)
+ {
+ mps2_an500_uart_t *u = &uarts[i];
+
+ u->ctrl = MPS2_ADDR2REG_PTR(uartbase[i], UART_CTRL_OFFSET);
+ u->tx = MPS2_ADDR2REG_PTR(uartbase[i], UART_THR_OFFSET);
+ u->clear = MPS2_ADDR2REG_PTR(uartbase[i], UART_INTSTS_OFFSET);
+ u->irq = MPS2_IRQ_FROMBASE(armv7m_gpio_base, uarttxirq[i]);
+
+ if (i > 0)
+ {
+ irq_attach(u->irq, uart_irq_tx_handle, u);
+ up_enable_irq(u->irq);
+ *u->ctrl = UART_CTRL_TX_ENABLE | UART_CTRL_TX_INT_ENABLE;
+ u->trigger = prev;
+ prev = u;
+#if CONFIG_TEST_IRQPRIO_LOG
+ u->begin = begin_tag[i];
+ u->end = end_tag[i];
+#endif
+ }
+
+ up_prioritize_irq(u->irq, uarttxirq_prio[i]);
+ }
+}
+
+static void mps_timer_init(void)
+{
+ static const uint32_t timerbase[] =
+ {
+ 0x40000000, 0x40001000
+ };
+
+ static const int timerirq[] =
+ {
+ 8, 9
+ };
+
+ static const int timer_irq_prio[] =
+ {
+ 0x80, 0xa0
+ };
+
+#if CONFIG_TEST_IRQPRIO_LOG
+ static const char *begin_tag[] =
+ {
+ "t", "u"
+ };
+
+ static const char *end_tag[] =
+ {
+ "T", "U"
+ };
+#endif
+
+ for (int i = 0; i < 2; i++)
+ {
+ mps2_an500_timer_t *t = &timer[i];
+
+ t->reload = MPS2_ADDR2REG_PTR(timerbase[i], MPS_TIMER_RELOAD_OFFSET);
+ t->ctrl = MPS2_ADDR2REG_PTR(timerbase[i], MPS_TIMER_CTRL_OFFSET);
+ t->clear = MPS2_ADDR2REG_PTR(timerbase[i], MPS_TIMER_CLEAR_OFFSET);
+ t->irq = MPS2_IRQ_FROMBASE(armv7m_gpio_base, timerirq[i]);
+
+ irq_attach(t->irq, timer_irq_handle, t);
+ up_enable_irq(t->irq);
+
+ up_prioritize_irq(t->irq, timer_irq_prio[i]);
+
+#if CONFIG_TEST_IRQPRIO_LOG
+ t->begin = begin_tag[i];
+ t->end = end_tag[i];
+#endif
+ }
+}
+
+static int uart_irq_tx_handle(int irq, void *context, void *arg)
+{
+ mps2_an500_uart_t *u = arg;
+ *u->clear = UART_INTSTATUS_TX;
+
+ TAG_BEGIN(u);
+
+ up_udelay(u->before);
+
+ if (u->sem != NULL)
+ {
+ sem_post(u->sem);
+ }
+
+ if (u->trigger != NULL)
+ {
+ uart_random_test(u->trigger);
+ }
+
+ up_udelay(u->after);
+
+ TAG_END(u);
+
+ return 0;
+}
+
+static int timer_irq_handle(int irq, void *context, void *arg)
+{
+ mps2_an500_timer_t *t = arg;
+ *t->clear = 1;
+
+ TAG_BEGIN(t);
+
+ up_udelay(t->before);
+
+ if (t->sem != NULL)
+ {
+ sem_post(t->sem);
+ }
+
+ up_udelay(t->after);
+
+ TAG_END(t);
+
+ return 0;
+}
+
+static void timer_begin_test(mps2_an500_timer_t *t, uint32_t reload_us)
+{
+ uint32_t reload = reload_us * 25;
+ *t->reload = reload;
+ t->sem = &test_sem;
+
+ *t->ctrl = MPS_TIMER_CTRL_IE | MPS_TIMER_CTRL_ENABLE;
+}
+
+static void timer_end_test(mps2_an500_timer_t *t)
+{
+ *t->ctrl = 0;
+ *t->clear = 1;
+}
+
+static void uart_simple_test(mps2_an500_uart_t *uart)
+{
+ uart->before = 0;
+ uart->sem = NULL;
+ uart->after = 0;
+ *uart->tx = 0;
+}
+
+static uint32_t uart_random_test(mps2_an500_uart_t *uart)
+{
+ /* 31bits random
+ * | thread usleep | after udelay | before udelay | uart_sel | use sem |
+ * | 30...19 (12b) | 18..11 (8b) | 10..3 (8b) | 2..1 (2b) | 0 (1b) |
+ */
+
+ uint32_t r = random();
+ int use_sem = (r >> 0) & 0x1;
+ int u_before = (r >> 3) & (0xff);
+ int u_after = (r >> 12) & (0xff);
+
+ mps2_an500_uart_t *u;
+
+ if (uart != NULL)
+ {
+ u = uart;
+ if (!use_sem)
+ {
+ return r;
+ }
+ }
+ else
+ {
+ int uart_select = (r >> 1) & 0x3;
+ u = &uarts[uart_select + 1];
+ }
+
+ if (use_sem)
+ {
+ u->sem = &test_sem;
+ }
+ else
+ {
+ u->sem = NULL;
+ }
+
+ u->before = u_before;
+ u->after = u_after;
+ *u->tx = 0;
+
+ if (!up_interrupt_context())
+ {
+ int u_thread = (r >> 19) & (0xfff);
+ usleep(u_thread);
+ }
+
+ return r;
+}
+
+static void *test_irq_awaker_thread_entry(void *arg)
+{
+ struct timespec ts;
+ int cnt = 0;
+ int ret;
+ while (true)
+ {
+ clock_gettime(CLOCK_REALTIME, &ts);
+ ts.tv_sec++;
+ ret = sem_timedwait(&test_sem, &ts);
+ if (ret != OK)
+ {
+ break;
+ }
+ else
+ {
+ cnt++;
+ if (cnt % 10000 == 0)
+ {
+ printf("%d, recv:%d\n", gettid(), cnt);
+ }
+#if CONFIG_TEST_IRQPRIO_LOG
+ else
+ {
+ printf(".");
+ }
+#endif
+ }
+ }
+
+ printf("timeoutquit %d\n", cnt);
+ return 0;
+}
+
+static void *test_irq_prio_thread_entry(void *arg)
+{
+ int i;
+ for (i = 0; i < CONFIG_TEST_IRQPRIO_LOOP_CNT; i++)
+ {
+ uart_random_test(NULL);
+ }
+
+ return NULL;
+}
+
+static void test_irqprio(void **argv)
+{
+ pid_t tid[CONFIG_TEST_IRQPRIO_TTHREAD + 1];
+ pthread_attr_t attr;
+ int i;
+
+ printf("init done\n");
+
+ for (i = 1; i < 5; i++)
+ {
+ uart_simple_test(&uarts[i]);
+ }
+
+ printf("simple_test done\n");
+ timer[0].before = 1;
+ timer[0].after = 1;
+ timer_begin_test(&timer[0], 1000);
+ timer[1].before = 10;
+ timer[1].after = 10;
+ timer_begin_test(&timer[1], 100 * 1000);
+
+ pthread_attr_init(&attr);
+
+ attr.priority = 1;
+ for (i = 0; i < CONFIG_TEST_IRQPRIO_TTHREAD; i++)
+ {
+ attr.priority++;
+ pthread_create(&tid[i], &attr, test_irq_prio_thread_entry, NULL);
+ }
+
+ attr.priority = 255;
+ pthread_create(&tid[CONFIG_TEST_IRQPRIO_TTHREAD],
+ &attr,
+ test_irq_awaker_thread_entry,
+ NULL);
+ printf("thread init done\n");
+
+ for (i = 0; i < CONFIG_TEST_IRQPRIO_TTHREAD; i++)
+ {
+ pthread_join(tid[i], NULL);
+ }
+
+ printf("uart join done\n");
+ timer_end_test(&timer[0]);
+ timer_end_test(&timer[1]);
+ printf("timer end done\n");
+ pthread_join(tid[CONFIG_TEST_IRQPRIO_TTHREAD], NULL);
+ printf("sem thread join done\n");
+}
+
+static int setup(void **argv)
+{
+ mps_uart_init();
+ mps_timer_init();
+ return 0;
+}
+
+static int teardown(void **argv)
+{
+ return 0;
+}
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+int main(int argc, char *argv[])
+{
+ const struct CMUnitTest tests[] = {
+ cmocka_unit_test_prestate_setup_teardown(
+ test_irqprio, setup, teardown, NULL),
+ };
+
+ return cmocka_run_group_tests(tests, NULL, NULL);
+}