/* * TOPPERS/ASP Kernel * Toyohashi Open Platform for Embedded Real-Time Systems/ * Advanced Standard Profile Kernel * * Copyright (C) 2005-2009 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: mutex.c 1694 2010-01-01 15:59:09Z ertl-hiro $ */ /* * ミューテックス機能 */ #include "kernel_impl.h" #include "check.h" #include "task.h" #include "wait.h" #include "mutex.h" /* * トレースログマクロのデフォルト定義 */ #ifndef LOG_LOC_MTX_ENTER #define LOG_LOC_MTX_ENTER(mtxid) #endif /* LOG_LOC_MTX_ENTER */ #ifndef LOG_LOC_MTX_LEAVE #define LOG_LOC_MTX_LEAVE(ercd) #endif /* LOG_LOC_MTX_LEAVE */ #ifndef LOG_PLOC_MTX_ENTER #define LOG_PLOC_MTX_ENTER(mtxid) #endif /* LOG_PLOC_MTX_ENTER */ #ifndef LOG_PLOC_MTX_LEAVE #define LOG_PLOC_MTX_LEAVE(ercd) #endif /* LOG_PLOC_MTX_LEAVE */ #ifndef LOG_TLOC_MTX_ENTER #define LOG_TLOC_MTX_ENTER(mtxid, tmout) #endif /* LOG_TLOC_MTX_ENTER */ #ifndef LOG_TLOC_MTX_LEAVE #define LOG_TLOC_MTX_LEAVE(ercd) #endif /* LOG_TLOC_MTX_LEAVE */ #ifndef LOG_UNL_MTX_ENTER #define LOG_UNL_MTX_ENTER(mtxid) #endif /* LOG_UNL_MTX_ENTER */ #ifndef LOG_UNL_MTX_LEAVE #define LOG_UNL_MTX_LEAVE(ercd) #endif /* LOG_UNL_MTX_LEAVE */ #ifndef LOG_INI_MTX_ENTER #define LOG_INI_MTX_ENTER(mtxid) #endif /* LOG_INI_MTX_ENTER */ #ifndef LOG_INI_MTX_LEAVE #define LOG_INI_MTX_LEAVE(ercd) #endif /* LOG_INI_MTX_LEAVE */ #ifndef LOG_REF_MTX_ENTER #define LOG_REF_MTX_ENTER(mtxid, pk_rmtx) #endif /* LOG_REF_MTX_ENTER */ #ifndef LOG_REF_MTX_LEAVE #define LOG_REF_MTX_LEAVE(ercd, pk_rmtx) #endif /* LOG_REF_MTX_LEAVE */ /* * ミューテックスの数 */ #define tnum_mtx ((uint_t)(tmax_mtxid - TMIN_MTXID + 1)) /* * ミューテックスIDからミューテックス管理ブロックを取り出すためのマクロ */ #define INDEX_MTX(mtxid) ((uint_t)((mtxid) - TMIN_MTXID)) #define get_mtxcb(mtxid) (&(mtxcb_table[INDEX_MTX(mtxid)])) /* * ミューテックス管理ブロック中のmutex_queueへのポインタから,ミューテッ * クス管理ブロックへのポインタを取り出すためのマクロ */ #define MTXCB_QUEUE(p_queue) \ ((MTXCB *)(((char *) p_queue) - offsetof(MTXCB, mutex_queue))) /* * ミューテックスのプロトコルを判断するマクロ */ #define MTXPROTO_MASK 0x03U #define MTXPROTO(p_mtxcb) ((p_mtxcb)->p_mtxinib->mtxatr & MTXPROTO_MASK) #define MTX_CEILING(p_mtxcb) (MTXPROTO(p_mtxcb) == TA_CEILING) /* * フックルーチン呼出し用の変数 */ #ifdef TOPPERS_mtxhook bool_t (*mtxhook_check_ceilpri)(TCB *p_tcb, uint_t bpriority) = NULL; bool_t (*mtxhook_scan_ceilmtx)(TCB *p_tcb) = NULL; bool_t (*mtxhook_release_all)(TCB *p_tcb) = NULL; #endif /* TOPPERS_mtxhook */ /* * ミューテックス機能の初期化 */ #ifdef TOPPERS_mtxini void initialize_mutex(void) { uint_t i; MTXCB *p_mtxcb; mtxhook_check_ceilpri = mutex_check_ceilpri; mtxhook_scan_ceilmtx = mutex_scan_ceilmtx; mtxhook_release_all = mutex_release_all; for (p_mtxcb = mtxcb_table, i = 0; i < tnum_mtx; p_mtxcb++, i++) { queue_initialize(&(p_mtxcb->wait_queue)); p_mtxcb->p_mtxinib = &(mtxinib_table[i]); p_mtxcb->p_loctsk = NULL; } } #endif /* TOPPERS_mtxini */ /* * 上限優先度違反のチェック */ #ifdef TOPPERS_mtxchk bool_t mutex_check_ceilpri(TCB *p_tcb, uint_t bpriority) { QUEUE *p_queue; MTXCB *p_mtxcb; /* * タスクがロックしている優先度上限ミューテックスの中で,上限優先 * 度がbpriorityよりも低いものがあれば,falseを返す. */ p_queue = p_tcb->mutex_queue.p_next; while (p_queue != &(p_tcb->mutex_queue)) { p_mtxcb = MTXCB_QUEUE(p_queue); if (MTX_CEILING(p_mtxcb) && bpriority < p_mtxcb->p_mtxinib->ceilpri) { return(false); } p_queue = p_queue->p_next; } /* * タスクが優先度上限ミューテックスのロックを待っている場合に,そ * の上限優先度がbpriorityよりも低くければ,falseを返す. */ if (TSTAT_WAIT_MTX(p_tcb->tstat)) { p_mtxcb = ((WINFO_MTX *)(p_tcb->p_winfo))->p_mtxcb; if (MTX_CEILING(p_mtxcb) && bpriority < p_mtxcb->p_mtxinib->ceilpri) { return(false); } } /* * いずれの条件にも当てはまらなければtrueを返す. */ return(true); } #endif /* TOPPERS_mtxchk */ /* * 優先度上限ミューテックスをロックしているかのチェック */ #ifdef TOPPERS_mtxscan bool_t mutex_scan_ceilmtx(TCB *p_tcb) { QUEUE *p_queue; MTXCB *p_mtxcb; p_queue = p_tcb->mutex_queue.p_next; while (p_queue != &(p_tcb->mutex_queue)) { p_mtxcb = MTXCB_QUEUE(p_queue); if (MTX_CEILING(p_mtxcb)) { return(true); } p_queue = p_queue->p_next; } return(false); } #endif /* TOPPERS_mtxscan */ /* * タスクの現在優先度の計算 */ #ifdef TOPPERS_mtxcalc uint_t mutex_calc_priority(TCB *p_tcb) { uint_t priority; QUEUE *p_queue; MTXCB *p_mtxcb; priority = p_tcb->bpriority; p_queue = p_tcb->mutex_queue.p_next; while (p_queue != &(p_tcb->mutex_queue)) { p_mtxcb = MTXCB_QUEUE(p_queue); if (MTX_CEILING(p_mtxcb) && p_mtxcb->p_mtxinib->ceilpri < priority) { priority = p_mtxcb->p_mtxinib->ceilpri; } p_queue = p_queue->p_next; } return(priority); } #endif /* TOPPERS_mtxcalc */ /* * 要素優先度が上がる(または増える)場合の現在優先度変更処理 */ Inline bool_t mutex_raise_priority(TCB *p_tcb, uint_t newpri) { if (newpri < p_tcb->priority) { return(change_priority(p_tcb, newpri, true)); } return(false); } /* * 要素優先度が下がる(または減る)場合の現在優先度変更処理 */ Inline bool_t mutex_drop_priority(TCB *p_tcb, uint_t oldpri) { uint_t newpri; if (oldpri == p_tcb->priority) { newpri = mutex_calc_priority(p_tcb); if (newpri != p_tcb->priority) { return(change_priority(p_tcb, newpri, true)); } } return(false); } /* * ミューテックスをロックした場合の処理 */ Inline bool_t mutex_acquire(TCB *p_loctsk, MTXCB *p_mtxcb) { p_mtxcb->p_loctsk = p_loctsk; queue_insert_prev(&(p_loctsk->mutex_queue), &(p_mtxcb->mutex_queue)); if (MTX_CEILING(p_mtxcb)) { return(mutex_raise_priority(p_loctsk, p_mtxcb->p_mtxinib->ceilpri)); } return(false); } /* * ミューテックスのロック解除 */ #ifdef TOPPERS_mtxrel bool_t mutex_release(MTXCB *p_mtxcb) { TCB *p_tcb; if (queue_empty(&(p_mtxcb->wait_queue))) { p_mtxcb->p_loctsk = NULL; return(false); } else { /* * ミューテックス待ちキューの先頭タスク(p_tcb)に,ミューテッ * クスをロックさせる. */ p_tcb = (TCB *) queue_delete_next(&(p_mtxcb->wait_queue)); wait_dequeue_tmevtb(p_tcb); p_tcb->p_winfo->wercd = E_OK; p_mtxcb->p_loctsk = p_tcb; queue_insert_prev(&(p_tcb->mutex_queue), &(p_mtxcb->mutex_queue)); if (MTX_CEILING(p_mtxcb)) { if (p_mtxcb->p_mtxinib->ceilpri < p_tcb->priority) { p_tcb->priority = p_mtxcb->p_mtxinib->ceilpri; } } return(make_non_wait(p_tcb)); } } #endif /* TOPPERS_mtxrel */ /* * タスクがロックしているすべてのミューテックスのロック解除 */ #ifdef TOPPERS_mtxrela bool_t mutex_release_all(TCB *p_tcb) { MTXCB *p_mtxcb; bool_t dspreq = false; while (!queue_empty(&(p_tcb->mutex_queue))) { p_mtxcb = MTXCB_QUEUE(p_tcb->mutex_queue.p_next); queue_delete(&(p_mtxcb->mutex_queue)); if (mutex_release(p_mtxcb)) { dspreq = true; } } return(dspreq); } #endif /* TOPPERS_mtxrela */ /* * ミューテックスのロック */ #ifdef TOPPERS_loc_mtx ER loc_mtx(ID mtxid) { MTXCB *p_mtxcb; WINFO_MTX winfo_mtx; ER ercd; LOG_LOC_MTX_ENTER(mtxid); CHECK_DISPATCH(); CHECK_MTXID(mtxid); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); if (MTX_CEILING(p_mtxcb) && p_runtsk->bpriority < p_mtxcb->p_mtxinib->ceilpri) { ercd = E_ILUSE; } else if (p_mtxcb->p_loctsk == NULL) { (void) mutex_acquire(p_runtsk, p_mtxcb); /* * 優先度上限ミューテックスをロックした場合,p_runtskの優先度 * が上がる可能性があるが,ディスパッチが必要になることはない. */ assert(!(p_runtsk != p_schedtsk && dspflg)); ercd = E_OK; } else if (p_mtxcb->p_loctsk == p_runtsk) { ercd = E_ILUSE; } else { p_runtsk->tstat = (TS_WAITING | TS_WAIT_MTX); wobj_make_wait((WOBJCB *) p_mtxcb, (WINFO_WOBJ *) &winfo_mtx); dispatch(); ercd = winfo_mtx.winfo.wercd; } t_unlock_cpu(); error_exit: LOG_LOC_MTX_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_loc_mtx */ /* * ミューテックスのロック(ポーリング) */ #ifdef TOPPERS_ploc_mtx ER ploc_mtx(ID mtxid) { MTXCB *p_mtxcb; ER ercd; LOG_PLOC_MTX_ENTER(mtxid); CHECK_TSKCTX_UNL(); CHECK_MTXID(mtxid); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); if (MTX_CEILING(p_mtxcb) && p_runtsk->bpriority < p_mtxcb->p_mtxinib->ceilpri) { ercd = E_ILUSE; } else if (p_mtxcb->p_loctsk == NULL) { (void) mutex_acquire(p_runtsk, p_mtxcb); /* * 優先度上限ミューテックスをロックした場合,p_runtskの優先度 * が上がる可能性があるが,ディスパッチが必要になることはない. */ assert(!(p_runtsk != p_schedtsk && dspflg)); ercd = E_OK; } else if (p_mtxcb->p_loctsk == p_runtsk) { ercd = E_ILUSE; } else { ercd = E_TMOUT; } t_unlock_cpu(); error_exit: LOG_PLOC_MTX_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_ploc_mtx */ /* * ミューテックスのロック(タイムアウトあり) */ #ifdef TOPPERS_tloc_mtx ER tloc_mtx(ID mtxid, TMO tmout) { MTXCB *p_mtxcb; WINFO_MTX winfo_mtx; TMEVTB tmevtb; ER ercd; LOG_TLOC_MTX_ENTER(mtxid, tmout); CHECK_DISPATCH(); CHECK_MTXID(mtxid); CHECK_TMOUT(tmout); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); if (MTX_CEILING(p_mtxcb) && p_runtsk->bpriority < p_mtxcb->p_mtxinib->ceilpri) { ercd = E_ILUSE; } else if (p_mtxcb->p_loctsk == NULL) { (void) mutex_acquire(p_runtsk, p_mtxcb); /* * 優先度上限ミューテックスをロックした場合,p_runtskの優先度 * が上がる可能性があるが,ディスパッチが必要になることはない. */ assert(!(p_runtsk != p_schedtsk && dspflg)); ercd = E_OK; } else if (p_mtxcb->p_loctsk == p_runtsk) { ercd = E_ILUSE; } else if (tmout == TMO_POL) { ercd = E_TMOUT; } else { p_runtsk->tstat = (TS_WAITING | TS_WAIT_MTX); wobj_make_wait_tmout((WOBJCB *) p_mtxcb, (WINFO_WOBJ *) &winfo_mtx, &tmevtb, tmout); dispatch(); ercd = winfo_mtx.winfo.wercd; } t_unlock_cpu(); error_exit: LOG_TLOC_MTX_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_tloc_mtx */ /* * ミューテックスのロック解除 */ #ifdef TOPPERS_unl_mtx ER unl_mtx(ID mtxid) { MTXCB *p_mtxcb; bool_t dspreq = false; ER ercd; LOG_UNL_MTX_ENTER(mtxid); CHECK_TSKCTX_UNL(); CHECK_MTXID(mtxid); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); if (p_mtxcb->p_loctsk != p_runtsk) { ercd = E_ILUSE; } else { queue_delete(&(p_mtxcb->mutex_queue)); if (MTX_CEILING(p_mtxcb)) { if (mutex_drop_priority(p_runtsk, p_mtxcb->p_mtxinib->ceilpri)) { dspreq = true; } } if (mutex_release(p_mtxcb)) { dspreq = true; } if (dspreq) { dispatch(); } ercd = E_OK; } t_unlock_cpu(); error_exit: LOG_UNL_MTX_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_unl_mtx */ /* * ミューテックスの初期化 */ #ifdef TOPPERS_ini_mtx ER ini_mtx(ID mtxid) { MTXCB *p_mtxcb; TCB *p_loctsk; bool_t dspreq; ER ercd; LOG_INI_MTX_ENTER(mtxid); CHECK_TSKCTX_UNL(); CHECK_MTXID(mtxid); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); dspreq = init_wait_queue(&(p_mtxcb->wait_queue)); p_loctsk = p_mtxcb->p_loctsk; if (p_loctsk != NULL) { queue_delete(&(p_mtxcb->mutex_queue)); p_mtxcb->p_loctsk = NULL; if (MTX_CEILING(p_mtxcb)) { if (mutex_drop_priority(p_loctsk, p_mtxcb->p_mtxinib->ceilpri)) { dspreq = true; } } } if (dspreq) { dispatch(); } ercd = E_OK; t_unlock_cpu(); error_exit: LOG_INI_MTX_LEAVE(ercd); return(ercd); } #endif /* TOPPERS_ini_mtx */ /* * ミューテックスの状態参照 */ #ifdef TOPPERS_ref_mtx ER ref_mtx(ID mtxid, T_RMTX *pk_rmtx) { MTXCB *p_mtxcb; ER ercd; LOG_REF_MTX_ENTER(mtxid, pk_rmtx); CHECK_TSKCTX_UNL(); CHECK_MTXID(mtxid); p_mtxcb = get_mtxcb(mtxid); t_lock_cpu(); pk_rmtx->htskid = (p_mtxcb->p_loctsk != NULL) ? TSKID(p_mtxcb->p_loctsk) : TSK_NONE; pk_rmtx->wtskid = wait_tskid(&(p_mtxcb->wait_queue)); ercd = E_OK; t_unlock_cpu(); error_exit: LOG_REF_MTX_LEAVE(ercd, pk_rmtx); return(ercd); } #endif /* TOPPERS_ref_mtx */