/* * 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-2008 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: sys_manage.c 164 2016-03-07 11:33:50Z coas-nagasima $ */ /* * システム状態管理機能 */ #include "kernel_impl.h" #include "check.h" #include "task.h" /* * トレースログマクロのデフォルト定義 */ #ifndef LOG_ROT_RDQ_ENTER #define LOG_ROT_RDQ_ENTER(tskpri) #endif /* LOG_ROT_RDQ_ENTER */ #ifndef LOG_ROT_RDQ_LEAVE #define LOG_ROT_RDQ_LEAVE(ercd) #endif /* LOG_ROT_RDQ_LEAVE */ #ifndef LOG_IROT_RDQ_ENTER #define LOG_IROT_RDQ_ENTER(tskpri) #endif /* LOG_IROT_RDQ_ENTER */ #ifndef LOG_IROT_RDQ_LEAVE #define LOG_IROT_RDQ_LEAVE(ercd) #endif /* LOG_IROT_RDQ_LEAVE */ #ifndef LOG_GET_TID_ENTER #define LOG_GET_TID_ENTER(p_tskid) #endif /* LOG_GET_TID_ENTER */ #ifndef LOG_GET_TID_LEAVE #define LOG_GET_TID_LEAVE(ercd, tskid) #endif /* LOG_GET_TID_LEAVE */ #ifndef LOG_IGET_TID_ENTER #define LOG_IGET_TID_ENTER(p_tskid) #endif /* LOG_IGET_TID_ENTER */ #ifndef LOG_IGET_TID_LEAVE #define LOG_IGET_TID_LEAVE(ercd, tskid) #endif /* LOG_IGET_TID_LEAVE */ #ifndef LOG_LOC_CPU_ENTER #define LOG_LOC_CPU_ENTER() #endif /* LOG_LOC_CPU_ENTER */ #ifndef LOG_LOC_CPU_LEAVE #define LOG_LOC_CPU_LEAVE(ercd) #endif /* LOG_LOC_CPU_LEAVE */ #ifndef LOG_ILOC_CPU_ENTER #define LOG_ILOC_CPU_ENTER() #endif /* LOG_ILOC_CPU_ENTER */ #ifndef LOG_ILOC_CPU_LEAVE #define LOG_ILOC_CPU_LEAVE(ercd) #endif /* LOG_ILOC_CPU_LEAVE */ #ifndef LOG_UNL_CPU_ENTER #define LOG_UNL_CPU_ENTER() #endif /* LOG_UNL_CPU_ENTER */ #ifndef LOG_UNL_CPU_LEAVE #define LOG_UNL_CPU_LEAVE(ercd) #endif /* LOG_UNL_CPU_LEAVE */ #ifndef LOG_IUNL_CPU_ENTER #define LOG_IUNL_CPU_ENTER() #endif /* LOG_IUNL_CPU_ENTER */ #ifndef LOG_IUNL_CPU_LEAVE #define LOG_IUNL_CPU_LEAVE(ercd) #endif /* LOG_IUNL_CPU_LEAVE */ #ifndef LOG_DIS_DSP_ENTER #define LOG_DIS_DSP_ENTER() #endif /* LOG_DIS_DSP_ENTER */ #ifndef LOG_DIS_DSP_LEAVE #define LOG_DIS_DSP_LEAVE(ercd) #endif /* LOG_DIS_DSP_LEAVE */ #ifndef LOG_ENA_DSP_ENTER #define LOG_ENA_DSP_ENTER() #endif /* LOG_ENA_DSP_ENTER */ #ifndef LOG_ENA_DSP_LEAVE #define LOG_ENA_DSP_LEAVE(ercd) #endif /* LOG_ENA_DSP_LEAVE */ #ifndef LOG_SNS_CTX_ENTER #define LOG_SNS_CTX_ENTER() #endif /* LOG_SNS_CTX_ENTER */ #ifndef LOG_SNS_CTX_LEAVE #define LOG_SNS_CTX_LEAVE(state) #endif /* LOG_SNS_CTX_LEAVE */ #ifndef LOG_SNS_LOC_ENTER #define LOG_SNS_LOC_ENTER() #endif /* LOG_SNS_LOC_ENTER */ #ifndef LOG_SNS_LOC_LEAVE #define LOG_SNS_LOC_LEAVE(state) #endif /* LOG_SNS_LOC_LEAVE */ #ifndef LOG_SNS_DSP_ENTER #define LOG_SNS_DSP_ENTER() #endif /* LOG_SNS_DSP_ENTER */ #ifndef LOG_SNS_DSP_LEAVE #define LOG_SNS_DSP_LEAVE(state) #endif /* LOG_SNS_DSP_LEAVE */ #ifndef LOG_SNS_DPN_ENTER #define LOG_SNS_DPN_ENTER() #endif /* LOG_SNS_DPN_ENTER */ #ifndef LOG_SNS_DPN_LEAVE #define LOG_SNS_DPN_LEAVE(state) #endif /* LOG_SNS_DPN_LEAVE */ #ifndef LOG_SNS_KER_ENTER #define LOG_SNS_KER_ENTER() #endif /* LOG_SNS_KER_ENTER */ #ifndef LOG_SNS_KER_LEAVE #define LOG_SNS_KER_LEAVE(state) #endif /* LOG_SNS_KER_LEAVE */ /* * タスクの優先順位の回転 */ #ifdef TOPPERS_rot_rdq ER rot_rdq(PRI tskpri) { uint_t pri; ER ercd; LOG_ROT_RDQ_ENTER(tskpri); CHECK_TSKCTX_UNL(); CHECK_TPRI_SELF(tskpri); t_lock_cpu(); pri = (tskpri == TPRI_SELF) ? p_runtsk->priority : INT_PRIORITY(tskpri); if (rotate_ready_queue(pri)) { dispatch(); } ercd = E_OK; t_unlock_cpu(); error_exit: LOG_ROT_RDQ_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_rot_rdq */ /* * タスクの優先順位の回転(非タスクコンテキスト用) */ #ifdef TOPPERS_irot_rdq ER irot_rdq(PRI tskpri) { ER ercd; LOG_IROT_RDQ_ENTER(tskpri); CHECK_INTCTX_UNL(); CHECK_TPRI(tskpri); i_lock_cpu(); if (rotate_ready_queue(INT_PRIORITY(tskpri))) { reqflg = true; } ercd = E_OK; i_unlock_cpu(); error_exit: LOG_IROT_RDQ_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_irot_rdq */ /* * 実行状態のタスクIDの参照 */ #ifdef TOPPERS_get_tid ER get_tid(ID *p_tskid) { ER ercd; LOG_GET_TID_ENTER(p_tskid); CHECK_TSKCTX_UNL(); t_lock_cpu(); *p_tskid = TSKID(p_runtsk); ercd = E_OK; t_unlock_cpu(); error_exit: LOG_GET_TID_LEAVE(ercd, *p_tskid); return(ercd); } #endif /* TOPPERS_get_tid */ /* * 実行状態のタスクIDの参照(非タスクコンテキスト用) */ #ifdef TOPPERS_iget_tid ER iget_tid(ID *p_tskid) { ER ercd; LOG_IGET_TID_ENTER(p_tskid); CHECK_INTCTX_UNL(); i_lock_cpu(); *p_tskid = (p_runtsk == NULL) ? TSK_NONE : TSKID(p_runtsk); ercd = E_OK; i_unlock_cpu(); error_exit: LOG_IGET_TID_LEAVE(ercd, *p_tskid); return(ercd); } #endif /* TOPPERS_iget_tid */ /* * CPUロック状態への移行 */ #ifdef TOPPERS_loc_cpu ER loc_cpu(void) { ER ercd; LOG_LOC_CPU_ENTER(); CHECK_TSKCTX(); if (!t_sense_lock()) { t_lock_cpu(); } ercd = E_OK; error_exit: LOG_LOC_CPU_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_loc_cpu */ /* * CPUロック状態への移行(非タスクコンテキスト用) */ #ifdef TOPPERS_iloc_cpu ER iloc_cpu(void) { ER ercd; LOG_ILOC_CPU_ENTER(); CHECK_INTCTX(); if (!i_sense_lock()) { i_lock_cpu(); } ercd = E_OK; error_exit: LOG_ILOC_CPU_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_iloc_cpu */ /* * CPUロック状態の解除 * * CPUロック中は,ディスパッチが必要となるサービスコールを呼び出すこ * とはできないため,CPUロック状態の解除時にディスパッチャを起動する * 必要はない. */ #ifdef TOPPERS_unl_cpu ER unl_cpu(void) { ER ercd; LOG_UNL_CPU_ENTER(); CHECK_TSKCTX(); if (t_sense_lock()) { t_unlock_cpu(); } ercd = E_OK; error_exit: LOG_UNL_CPU_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_unl_cpu */ /* * CPUロック状態の解除(非タスクコンテキスト用) * * CPUロック中は,ディスパッチが必要となるサービスコールを呼び出すこ * とはできないため,CPUロック状態の解除時にディスパッチャの起動を要 * 求する必要はない. */ #ifdef TOPPERS_iunl_cpu ER iunl_cpu(void) { ER ercd; LOG_IUNL_CPU_ENTER(); CHECK_INTCTX(); if (i_sense_lock()) { i_unlock_cpu(); } ercd = E_OK; error_exit: LOG_IUNL_CPU_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_iunl_cpu */ /* * ディスパッチの禁止 */ #ifdef TOPPERS_dis_dsp ER dis_dsp(void) { ER ercd; LOG_DIS_DSP_ENTER(); CHECK_TSKCTX_UNL(); t_lock_cpu(); disdsp = true; dspflg = false; ercd = E_OK; t_unlock_cpu(); error_exit: LOG_DIS_DSP_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_dis_dsp */ /* * ディスパッチの許可 */ #ifdef TOPPERS_ena_dsp ER ena_dsp(void) { ER ercd; LOG_ENA_DSP_ENTER(); CHECK_TSKCTX_UNL(); t_lock_cpu(); disdsp = false; if (ipmflg) { dspflg = true; if (p_runtsk != p_schedtsk) { dispatch(); } } ercd = E_OK; t_unlock_cpu(); error_exit: LOG_ENA_DSP_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_ena_dsp */ /* * コンテキストの参照 */ #ifdef TOPPERS_sns_ctx bool_t sns_ctx(void) { bool_t state; LOG_SNS_CTX_ENTER(); state = sense_context() ? true : false; LOG_SNS_CTX_LEAVE(state); return(state); } #endif /* TOPPERS_sns_ctx */ /* * CPUロック状態の参照 */ #ifdef TOPPERS_sns_loc bool_t sns_loc(void) { bool_t state; LOG_SNS_LOC_ENTER(); state = x_sense_lock() ? true : false; LOG_SNS_LOC_LEAVE(state); return(state); } #endif /* TOPPERS_sns_loc */ /* * ディスパッチ禁止状態の参照 */ #ifdef TOPPERS_sns_dsp bool_t sns_dsp(void) { bool_t state; LOG_SNS_DSP_ENTER(); state = disdsp; LOG_SNS_DSP_LEAVE(state); return(state); } #endif /* TOPPERS_sns_dsp */ /* * ディスパッチ保留状態の参照 */ #ifdef TOPPERS_sns_dpn bool_t sns_dpn(void) { bool_t state; LOG_SNS_DPN_ENTER(); state = (sense_context() || t_sense_lock() || !dspflg) ? true : false; LOG_SNS_DPN_LEAVE(state); return(state); } #endif /* TOPPERS_sns_dpn */ /* * カーネル非動作状態の参照 */ #ifdef TOPPERS_sns_ker bool_t sns_ker(void) { bool_t state; LOG_SNS_KER_ENTER(); state = kerflg ? false : true; LOG_SNS_KER_LEAVE(state); return(state); } #endif /* TOPPERS_sns_ker */