/* ChibiOS - Copyright (C) 2006..2018 Giovanni Di Sirio Licensed 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. */ /** * @file irq_storm.c * @brief IRQ Storm stress test code. * * @addtogroup IRQ_STORM * @{ */ #include #include "ch.h" #include "hal.h" #include "chprintf.h" #include "vt_storm.h" /*===========================================================================*/ /* Module local definitions. */ /*===========================================================================*/ /*===========================================================================*/ /* Module exported variables. */ /*===========================================================================*/ /*===========================================================================*/ /* Module local types. */ /*===========================================================================*/ /*===========================================================================*/ /* Module local variables. */ /*===========================================================================*/ static const vt_storm_config_t *config; static virtual_timer_t watchdog; static virtual_timer_t wrapper; static virtual_timer_t sweeper0, sweeperm1, sweeperp1, sweeperm3, sweeperp3; static volatile sysinterval_t delay; static volatile bool saturated; /*===========================================================================*/ /* Module local functions. */ /*===========================================================================*/ static void watchdog_cb(void *p) { (void)p; chSysLockFromISR(); chVTResetI(&sweeper0); chVTResetI(&sweeperm1); chVTResetI(&sweeperp1); chVTResetI(&sweeperm3); chVTResetI(&sweeperp3); chVTResetI(&wrapper); chSysUnlockFromISR(); saturated = true; } static void wrapper_cb(void *p) { (void)p; } static void sweeper0_cb(void *p) { (void)p; chSysLockFromISR(); chVTSetI(&wrapper, (sysinterval_t)-1, wrapper_cb, NULL); chVTSetI(&sweeper0, delay, sweeper0_cb, NULL); chSysUnlockFromISR(); } static void sweeperm1_cb(void *p) { (void)p; chSysLockFromISR(); chVTSetI(&sweeperm1, delay - 1, sweeperm1_cb, NULL); chSysUnlockFromISR(); } static void sweeperp1_cb(void *p) { (void)p; chSysLockFromISR(); chVTSetI(&sweeperp1, delay + 1, sweeperp1_cb, NULL); chSysUnlockFromISR(); } static void sweeperm3_cb(void *p) { (void)p; chSysLockFromISR(); chVTSetI(&sweeperm3, delay - 3, sweeperm3_cb, NULL); chSysUnlockFromISR(); } static void sweeperp3_cb(void *p) { (void)p; chSysLockFromISR(); chVTSetI(&sweeperp3, delay + 3, sweeperp3_cb, NULL); chSysUnlockFromISR(); } /*===========================================================================*/ /* Module exported functions. */ /*===========================================================================*/ /** * @brief VT storm execution. * * @param[in] cfg pointer to the test configuration structure * * @api */ void vt_storm_execute(const vt_storm_config_t *cfg) { unsigned i; config = cfg; /* Printing environment information.*/ chprintf(cfg->out, ""); chprintf(cfg->out, "\r\n*** ChibiOS/RT VT-STORM long duration test\r\n***\r\n"); chprintf(cfg->out, "*** Kernel: %s\r\n", CH_KERNEL_VERSION); chprintf(cfg->out, "*** Compiled: %s\r\n", __DATE__ " - " __TIME__); #ifdef PORT_COMPILER_NAME chprintf(cfg->out, "*** Compiler: %s\r\n", PORT_COMPILER_NAME); #endif chprintf(cfg->out, "*** Architecture: %s\r\n", PORT_ARCHITECTURE_NAME); #ifdef PORT_CORE_VARIANT_NAME chprintf(cfg->out, "*** Core Variant: %s\r\n", PORT_CORE_VARIANT_NAME); #endif chprintf(cfg->out, "*** System Clock: %d\r\n", cfg->sysclk); #ifdef PORT_INFO chprintf(cfg->out, "*** Port Info: %s\r\n", PORT_INFO); #endif #ifdef PLATFORM_NAME chprintf(cfg->out, "*** Platform: %s\r\n", PLATFORM_NAME); #endif #ifdef BOARD_NAME chprintf(cfg->out, "*** Test Board: %s\r\n", BOARD_NAME); #endif chprintf(cfg->out, "***\r\n"); chprintf(cfg->out, "*** Randomize: %d\r\n", VT_STORM_CFG_RANDOMIZE); chprintf(cfg->out, "*** System Time: %d\r\n", CH_CFG_ST_RESOLUTION); chprintf(cfg->out, "*** Intervals: %d\r\n", CH_CFG_INTERVALS_SIZE); chprintf(cfg->out, "*** SysTick: %d\r\n", CH_CFG_ST_FREQUENCY); chprintf(cfg->out, "*** Delta: %d\r\n", CH_CFG_ST_TIMEDELTA); chprintf(cfg->out, "\r\n"); for (i = 1; i <= VT_STORM_CFG_ITERATIONS; i++){ chprintf(cfg->out, "Iteration %d\r\n", i); delay = 120; saturated = false; do { /* Starting sweepers.*/ chSysLock(); chVTSetI(&watchdog, TIME_MS2I(501), watchdog_cb, NULL); chVTSetI(&sweeper0, delay, sweeper0_cb, NULL); chVTSetI(&sweeperm1, delay - 1, sweeperm1_cb, NULL); chVTSetI(&sweeperp1, delay + 1, sweeperp1_cb, NULL); chVTSetI(&sweeperm3, delay - 3, sweeperm3_cb, NULL); chVTSetI(&sweeperp3, delay + 3, sweeperp3_cb, NULL); chVTSetI(&wrapper, (sysinterval_t) - 1, wrapper_cb, NULL); /* Letting them run for half second.*/ chThdSleepS(TIME_MS2I(500)); /* Stopping everything.*/ chVTResetI(&watchdog); chVTResetI(&sweeper0); chVTResetI(&sweeperm1); chVTResetI(&sweeperp1); chVTResetI(&sweeperm3); chVTResetI(&sweeperp3); chVTResetI(&wrapper); chSysUnlock(); if (saturated) { chprintf(cfg->out, "#"); break; } palToggleLine(config->line); chprintf(cfg->out, "."); delay--; } while (delay >= 10); if (saturated) { chprintf(cfg->out, "\r\nSaturated at %d uS\r\n\r\n", TIME_I2US(delay)); } else { chprintf(cfg->out, "\r\n\r\n"); } } } /** @} */