495 lines
11 KiB
C
495 lines
11 KiB
C
|
/****************************************************************************
|
||
|
* 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);
|
||
|
}
|