This is an automated email from the ASF dual-hosted git repository.

xiaoxiang pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/nuttx-apps.git


The following commit(s) were added to refs/heads/master by this push:
     new 05def1bef testing: add case for irqprio, should only work with qemu 
MPS2_AN500
05def1bef is described below

commit 05def1befc1dddcd0ef754836759e698357d9e54
Author: buxiasen <buxia...@xiaomi.com>
AuthorDate: Sat Aug 10 01:11:32 2024 +0800

    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 <buxia...@xiaomi.com>
---
 testing/drivertest/CMakeLists.txt    |  16 ++
 testing/drivertest/Makefile          |   7 +
 testing/drivertest/drivertest_mps2.c | 494 +++++++++++++++++++++++++++++++++++
 3 files changed, 517 insertions(+)

diff --git a/testing/drivertest/CMakeLists.txt 
b/testing/drivertest/CMakeLists.txt
index 3548af95f..5b12c18d3 100644
--- a/testing/drivertest/CMakeLists.txt
+++ b/testing/drivertest/CMakeLists.txt
@@ -327,6 +327,22 @@ if(CONFIG_TESTING_DRIVER_TEST)
       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 e81783567..d7093ac4f 100644
--- a/testing/drivertest/Makefile
+++ b/testing/drivertest/Makefile
@@ -128,6 +128,13 @@ MAINSRC  += drivertest_touchpanel.c
 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 000000000..57bda24ab
--- /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);
+}

Reply via email to