/* * TOPPERS/ASP Kernel * Toyohashi Open Platform for Embedded Real-Time Systems/ * Advanced Standard Profile Kernel * * Copyright (C) 2000-2003 by Embedded and Real-Time Systems Laboratory * Toyohashi Univ. of Technology, JAPAN * Copyright (C) 2005-2014 by Embedded and Real-Time Systems Laboratory * Graduate School of Information Science, Nagoya Univ., JAPAN * * 上記著作権者は,以下の(1)〜(4)の条件を満たす場合に限り,本ソフトウェ * ア(本ソフトウェアを改変したものを含む.以下同じ)を使用・複製・改 * 変・再配布(以下,利用と呼ぶ)することを無償で許諾する. * (1) 本ソフトウェアをソースコードの形で利用する場合には,上記の著作 * 権表示,この利用条件および下記の無保証規定が,そのままの形でソー * スコード中に含まれていること. * (2) 本ソフトウェアを,ライブラリ形式など,他のソフトウェア開発に使 * 用できる形で再配布する場合には,再配布に伴うドキュメント(利用 * 者マニュアルなど)に,上記の著作権表示,この利用条件および下記 * の無保証規定を掲載すること. * (3) 本ソフトウェアを,機器に組み込むなど,他のソフトウェア開発に使 * 用できない形で再配布する場合には,次のいずれかの条件を満たすこ * と. * (a) 再配布に伴うドキュメント(利用者マニュアルなど)に,上記の著 * 作権表示,この利用条件および下記の無保証規定を掲載すること. * (b) 再配布の形態を,別に定める方法によって,TOPPERSプロジェクトに * 報告すること. * (4) 本ソフトウェアの利用により直接的または間接的に生じるいかなる損 * 害からも,上記著作権者およびTOPPERSプロジェクトを免責すること. * また,本ソフトウェアのユーザまたはエンドユーザからのいかなる理 * 由に基づく請求からも,上記著作権者およびTOPPERSプロジェクトを * 免責すること. * * 本ソフトウェアは,無保証で提供されているものである.上記著作権者お * よびTOPPERSプロジェクトは,本ソフトウェアに関して,特定の使用目的 * に対する適合性も含めて,いかなる保証も行わない.また,本ソフトウェ * アの利用により直接的または間接的に生じたいかなる損害に関しても,そ * の責任を負わない. * * @(#) $Id: cyclic.c 167 2016-03-08 11:37:45Z coas-nagasima $ */ /* * 周期ハンドラ機能 */ #include "kernel_impl.h" #include "check.h" #include "cyclic.h" /* * トレースログマクロのデフォルト定義 */ #ifndef LOG_CYC_ENTER #define LOG_CYC_ENTER(p_cyccb) #endif /* LOG_CYC_ENTER */ #ifndef LOG_CYC_LEAVE #define LOG_CYC_LEAVE(p_cyccb) #endif /* LOG_CYC_LEAVE */ #ifndef LOG_ACRE_CYC_ENTER #define LOG_ACRE_CYC_ENTER(pk_ccyc) #endif /* LOG_ACRE_CYC_ENTER */ #ifndef LOG_ACRE_CYC_LEAVE #define LOG_ACRE_CYC_LEAVE(ercd) #endif /* LOG_ACRE_CYC_LEAVE */ #ifndef LOG_DEL_CYC_ENTER #define LOG_DEL_CYC_ENTER(cycid) #endif /* LOG_DEL_CYC_ENTER */ #ifndef LOG_DEL_CYC_LEAVE #define LOG_DEL_CYC_LEAVE(ercd) #endif /* LOG_DEL_CYC_LEAVE */ #ifndef LOG_STA_CYC_ENTER #define LOG_STA_CYC_ENTER(cycid) #endif /* LOG_STA_CYC_ENTER */ #ifndef LOG_STA_CYC_LEAVE #define LOG_STA_CYC_LEAVE(ercd) #endif /* LOG_STA_CYC_LEAVE */ #ifndef LOG_STP_CYC_ENTER #define LOG_STP_CYC_ENTER(cycid) #endif /* LOG_STP_CYC_ENTER */ #ifndef LOG_STP_CYC_LEAVE #define LOG_STP_CYC_LEAVE(ercd) #endif /* LOG_STP_CYC_LEAVE */ #ifndef LOG_REF_CYC_ENTER #define LOG_REF_CYC_ENTER(cycid, pk_rcyc) #endif /* LOG_REF_CYC_ENTER */ #ifndef LOG_REF_CYC_LEAVE #define LOG_REF_CYC_LEAVE(ercd, pk_rcyc) #endif /* LOG_REF_CYC_LEAVE */ /* * 周期ハンドラの数 */ #define tnum_cyc ((uint_t)(tmax_cycid - TMIN_CYCID + 1)) #define tnum_scyc ((uint_t)(tmax_scycid - TMIN_CYCID + 1)) /* * 周期ハンドラIDから周期ハンドラ管理ブロックを取り出すためのマクロ */ #define INDEX_CYC(cycid) ((uint_t)((cycid) - TMIN_CYCID)) #define get_cyccb(cycid) (&(cyccb_table[INDEX_CYC(cycid)])) /* * 周期ハンドラ起動のためのタイムイベントブロックの登録 */ Inline void tmevtb_enqueue_cyc(CYCCB *p_cyccb, EVTTIM evttim) { tmevtb_enqueue_evttim(&(p_cyccb->tmevtb), evttim, (CBACK) call_cychdr, (void *) p_cyccb); p_cyccb->evttim = evttim; } #ifdef TOPPERS_cycini /* * 使用していない周期ハンドラ管理ブロックのリスト * * 周期ハンドラ管理ブロックの先頭にはキューにつなぐための領域がないた * め,タイムイベントブロック(tmevtb)の領域を用いる. */ QUEUE free_cyccb; /* * 周期ハンドラ機能の初期化 */ void initialize_cyclic(void) { uint_t i, j; CYCCB *p_cyccb; CYCINIB *p_cycinib; for (i = 0; i < tnum_scyc; i++) { p_cyccb = &(cyccb_table[i]); p_cyccb->p_cycinib = &(cycinib_table[i]); if ((p_cyccb->p_cycinib->cycatr & TA_STA) != 0U) { p_cyccb->cycsta = true; tmevtb_enqueue_cyc(p_cyccb, (EVTTIM)(p_cyccb->p_cycinib->cycphs)); } else { p_cyccb->cycsta = false; } } queue_initialize(&free_cyccb); for (j = 0; i < tnum_cyc; i++, j++) { p_cyccb = &(cyccb_table[i]); p_cycinib = &(acycinib_table[j]); p_cycinib->cycatr = TA_NOEXS; p_cyccb->p_cycinib = ((const CYCINIB *) p_cycinib); queue_insert_prev(&free_cyccb, ((QUEUE *) &(p_cyccb->tmevtb))); } } #endif /* TOPPERS_cycini */ /* * 周期ハンドラの生成 */ #ifdef TOPPERS_acre_cyc ER_UINT acre_cyc(const T_CCYC *pk_ccyc) { CYCCB *p_cyccb; CYCINIB *p_cycinib; ER ercd; LOG_ACRE_CYC_ENTER(pk_ccyc); CHECK_TSKCTX_UNL(); CHECK_RSATR(pk_ccyc->cycatr, TA_STA); CHECK_ALIGN_FUNC(pk_ccyc->cychdr); CHECK_NONNULL_FUNC(pk_ccyc->cychdr); CHECK_PAR(0 < pk_ccyc->cyctim && pk_ccyc->cyctim <= TMAX_RELTIM); CHECK_PAR(0 <= pk_ccyc->cycphs && pk_ccyc->cycphs <= TMAX_RELTIM); t_lock_cpu(); if (tnum_cyc == 0 || queue_empty(&free_cyccb)) { ercd = E_NOID; } else { p_cyccb = ((CYCCB *)(((char *) queue_delete_next(&free_cyccb)) - offsetof(CYCCB, tmevtb))); p_cycinib = (CYCINIB *)(p_cyccb->p_cycinib); p_cycinib->cycatr = pk_ccyc->cycatr; p_cycinib->exinf = pk_ccyc->exinf; p_cycinib->cychdr = pk_ccyc->cychdr; p_cycinib->cyctim = pk_ccyc->cyctim; p_cycinib->cycphs = pk_ccyc->cycphs; if ((p_cyccb->p_cycinib->cycatr & TA_STA) != 0U) { p_cyccb->cycsta = true; tmevtb_enqueue_cyc(p_cyccb, base_time + p_cyccb->p_cycinib->cycphs); } else { p_cyccb->cycsta = false; } ercd = CYCID(p_cyccb); } t_unlock_cpu(); error_exit: LOG_ACRE_CYC_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_acre_cyc */ /* * 周期ハンドラの削除 */ #ifdef TOPPERS_del_cyc ER del_cyc(ID cycid) { CYCCB *p_cyccb; CYCINIB *p_cycinib; ER ercd; LOG_DEL_CYC_ENTER(cycid); CHECK_TSKCTX_UNL(); CHECK_CYCID(cycid); p_cyccb = get_cyccb(cycid); t_lock_cpu(); if (p_cyccb->p_cycinib->cycatr == TA_NOEXS) { ercd = E_NOEXS; } else if (CYCID(p_cyccb) > tmax_scycid) { if (p_cyccb->cycsta) { p_cyccb->cycsta = false; tmevtb_dequeue(&(p_cyccb->tmevtb)); } p_cycinib = (CYCINIB *)(p_cyccb->p_cycinib); p_cycinib->cycatr = TA_NOEXS; queue_insert_prev(&free_cyccb, ((QUEUE *) &(p_cyccb->tmevtb))); ercd = E_OK; } else { ercd = E_OBJ; } t_unlock_cpu(); error_exit: LOG_DEL_CYC_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_del_cyc */ /* * 周期ハンドラの動作開始 */ #ifdef TOPPERS_sta_cyc ER sta_cyc(ID cycid) { CYCCB *p_cyccb; ER ercd; LOG_STA_CYC_ENTER(cycid); CHECK_TSKCTX_UNL(); CHECK_CYCID(cycid); p_cyccb = get_cyccb(cycid); t_lock_cpu(); if (p_cyccb->p_cycinib->cycatr == TA_NOEXS) { ercd = E_NOEXS; } else { if (p_cyccb->cycsta) { tmevtb_dequeue(&(p_cyccb->tmevtb)); } else { p_cyccb->cycsta = true; } tmevtb_enqueue_cyc(p_cyccb, base_time + p_cyccb->p_cycinib->cycphs); ercd = E_OK; } t_unlock_cpu(); error_exit: LOG_STA_CYC_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_sta_cyc */ /* * 周期ハンドラの動作停止 */ #ifdef TOPPERS_stp_cyc ER stp_cyc(ID cycid) { CYCCB *p_cyccb; ER ercd; LOG_STP_CYC_ENTER(cycid); CHECK_TSKCTX_UNL(); CHECK_CYCID(cycid); p_cyccb = get_cyccb(cycid); t_lock_cpu(); if (p_cyccb->p_cycinib->cycatr == TA_NOEXS) { ercd = E_NOEXS; } else { if (p_cyccb->cycsta) { p_cyccb->cycsta = false; tmevtb_dequeue(&(p_cyccb->tmevtb)); } ercd = E_OK; } t_unlock_cpu(); error_exit: LOG_STP_CYC_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_stp_cyc */ /* * 周期ハンドラの状態参照 */ #ifdef TOPPERS_ref_cyc ER ref_cyc(ID cycid, T_RCYC *pk_rcyc) { CYCCB *p_cyccb; ER ercd; LOG_REF_CYC_ENTER(cycid, pk_rcyc); CHECK_TSKCTX_UNL(); CHECK_CYCID(cycid); p_cyccb = get_cyccb(cycid); t_lock_cpu(); if (p_cyccb->p_cycinib->cycatr == TA_NOEXS) { ercd = E_NOEXS; } else { if (p_cyccb->cycsta) { pk_rcyc->cycstat = TCYC_STA; pk_rcyc->lefttim = tmevt_lefttim(&(p_cyccb->tmevtb)); } else { pk_rcyc->cycstat = TCYC_STP; } ercd = E_OK; } t_unlock_cpu(); error_exit: LOG_REF_CYC_LEAVE(ercd, pk_rcyc); return(ercd); } #endif /* TOPPERS_ref_cyc */ /* * 周期ハンドラ起動ルーチン */ #ifdef TOPPERS_cyccal void call_cychdr(CYCCB *p_cyccb) { PRI saved_ipm; /* * 次回の起動のためのタイムイベントブロックを登録する. * * 同じタイムティックで周期ハンドラを再度起動すべき場合には,この * 関数からsignal_timeに戻った後に,再度この関数が呼ばれることにな * る. */ tmevtb_enqueue_cyc(p_cyccb, p_cyccb->evttim + p_cyccb->p_cycinib->cyctim); /* * 周期ハンドラを,CPUロック解除状態で呼び出す. */ saved_ipm = i_get_ipm(); i_unlock_cpu(); LOG_CYC_ENTER(p_cyccb); (*((CYCHDR)(p_cyccb->p_cycinib->cychdr)))(p_cyccb->p_cycinib->exinf); LOG_CYC_LEAVE(p_cyccb); if (!i_sense_lock()) { i_lock_cpu(); } i_set_ipm(saved_ipm); } #endif /* TOPPERS_cyccal */