📄 eventflag.c
字号:
/*
* TOPPERS/JSP Kernel
* Toyohashi Open Platform for Embedded Real-Time Systems/
* Just Standard Profile Kernel
*
* Copyright (C) 2000-2003 by Embedded and Real-Time Systems Laboratory
* Toyohashi Univ. of Technology, JAPAN
*
* 忋婰挊嶌尃幰偼丆埲壓偺 (1)乣(4) 偺忦審偐丆Free Software Foundation
* 偵傛偭偰岞昞偝傟偰偄傞 GNU General Public License 偺 Version 2 偵婰
* 弎偝傟偰偄傞忦審傪枮偨偡応崌偵尷傝丆杮僜僼僩僂僃傾乮杮僜僼僩僂僃傾
* 傪夵曄偟偨傕偺傪娷傓丏埲壓摨偠乯傪巊梡丒暋惢丒夵曄丒嵞攝晍乮埲壓丆
* 棙梡偲屇傇乯偡傞偙偲傪柍彏偱嫋戻偡傞丏
* (1) 杮僜僼僩僂僃傾傪僜乕僗僐乕僪偺宍偱棙梡偡傞応崌偵偼丆忋婰偺挊嶌
* 尃昞帵丆偙偺棙梡忦審偍傛傃壓婰偺柍曐徹婯掕偑丆偦偺傑傑偺宍偱僜乕
* 僗僐乕僪拞偵娷傑傟偰偄傞偙偲丏
* (2) 杮僜僼僩僂僃傾傪丆儔僀僽儔儕宍幃側偳丆懠偺僜僼僩僂僃傾奐敪偵巊
* 梡偱偒傞宍偱嵞攝晍偡傞応崌偵偼丆嵞攝晍偵敽偆僪僉儏儊儞僩乮棙梡
* 幰儅僯儏傾儖側偳乯偵丆忋婰偺挊嶌尃昞帵丆偙偺棙梡忦審偍傛傃壓婰
* 偺柍曐徹婯掕傪宖嵹偡傞偙偲丏
* (3) 杮僜僼僩僂僃傾傪丆婡婍偵慻傒崬傓側偳丆懠偺僜僼僩僂僃傾奐敪偵巊
* 梡偱偒側偄宍偱嵞攝晍偡傞応崌偵偼丆師偺偄偢傟偐偺忦審傪枮偨偡偙
* 偲丏
* (a) 嵞攝晍偵敽偆僪僉儏儊儞僩乮棙梡幰儅僯儏傾儖側偳乯偵丆忋婰偺挊
* 嶌尃昞帵丆偙偺棙梡忦審偍傛傃壓婰偺柍曐徹婯掕傪宖嵹偡傞偙偲丏
* (b) 嵞攝晍偺宍懺傪丆暿偵掕傔傞曽朄偵傛偭偰丆TOPPERS僾儘僕僃僋僩偵
* 曬崘偡傞偙偲丏
* (4) 杮僜僼僩僂僃傾偺棙梡偵傛傝捈愙揑傑偨偼娫愙揑偵惗偠傞偄偐側傞懝
* 奞偐傜傕丆忋婰挊嶌尃幰偍傛傃TOPPERS僾儘僕僃僋僩傪柶愑偡傞偙偲丏
*
* 杮僜僼僩僂僃傾偼丆柍曐徹偱採嫙偝傟偰偄傞傕偺偱偁傞丏忋婰挊嶌尃幰偍
* 傛傃TOPPERS僾儘僕僃僋僩偼丆杮僜僼僩僂僃傾偵娭偟偰丆偦偺揔梡壜擻惈傕
* 娷傔偰丆偄偐側傞曐徹傕峴傢側偄丏傑偨丆杮僜僼僩僂僃傾偺棙梡偵傛傝捈
* 愙揑傑偨偼娫愙揑偵惗偠偨偄偐側傞懝奞偵娭偟偰傕丆偦偺愑擟傪晧傢側偄丏
*
* @(#) $Id: eventflag.c,v 1.8 2003/06/04 01:46:16 hiro Exp $
*/
/*
* 僀儀儞僩僼儔僌婡擻
*/
#include "jsp_kernel.h"
#include "check.h"
#include "task.h"
#include "wait.h"
#include "eventflag.h"
/*
* 僀儀儞僩僼儔僌ID偺嵟戝抣乮kernel_cfg.c乯
*/
extern const ID tmax_flgid;
/*
* 僀儀儞僩僼儔僌弶婜壔僽儘僢僋偺僄儕傾乮kernel_cfg.c乯
*/
extern const FLGINIB flginib_table[];
/*
* 僀儀儞僩僼儔僌娗棟僽儘僢僋偺僄儕傾乮kernel_cfg.c乯
*/
extern FLGCB flgcb_table[];
/*
* 僀儀儞僩僼儔僌偺悢
*/
#define TNUM_FLG ((UINT)(tmax_flgid - TMIN_FLGID + 1))
/*
* 僀儀儞僩僼儔僌ID偐傜僀儀儞僩僼儔僌娗棟僽儘僢僋傪庢傝弌偡偨傔偺儅僋儘
*/
#define INDEX_FLG(flgid) ((UINT)((flgid) - TMIN_FLGID))
#define get_flgcb(flgid) (&(flgcb_table[INDEX_FLG(flgid)]))
/*
* 僀儀儞僩僼儔僌懸偪忣曬僽儘僢僋偺掕媊
*
* flgptn 偼丆waiptn 偍傛傃 wfmode 偲摨帪偵巊偆偙偲偼側偄偨傔丆union
* 傪巊偊偽儊儌儕傪愡栺偡傞偙偲偑壜擻偱偁傞丏
*/
typedef struct eventflag_waiting_information {
WINFO winfo; /* 昗弨偺懸偪忣曬僽儘僢僋 */
WOBJCB *wobjcb; /* 懸偪僆僽僕僃僋僩偺娗棟僽儘僢僋 */
FLGPTN waiptn; /* 懸偪僷僞乕儞 */
MODE wfmode; /* 懸偪儌乕僪 */
FLGPTN flgptn; /* 懸偪夝彍帪偺僷僞乕儞 */
} WINFO_FLG;
/*
* 僀儀儞僩僼儔僌婡擻偺弶婜壔
*/
#ifdef __flgini
void
eventflag_initialize(void)
{
UINT i;
FLGCB *flgcb;
for (flgcb = flgcb_table, i = 0; i < TNUM_FLG; flgcb++, i++) {
queue_initialize(&(flgcb->wait_queue));
flgcb->flginib = &(flginib_table[i]);
flgcb->flgptn = flgcb->flginib->iflgptn;
}
}
#endif /* __flgini */
/*
* 僀儀儞僩僼儔僌懸偪夝彍忦審偺僠僃僢僋
*/
#ifdef __flgcnd
BOOL
eventflag_cond(FLGCB *flgcb, FLGPTN waiptn, MODE wfmode, FLGPTN *p_flgptn)
{
if ((wfmode & TWF_ORW) != 0 ? (flgcb->flgptn & waiptn) != 0
: (flgcb->flgptn & waiptn) == waiptn) {
*p_flgptn = flgcb->flgptn;
if ((flgcb->flginib->flgatr & TA_CLR) != 0) {
flgcb->flgptn = 0;
}
return(TRUE);
}
return(FALSE);
}
#endif /* __flgcnd */
/*
* 僀儀儞僩僼儔僌偺僙僢僩
*/
#ifdef __set_flg
SYSCALL ER
set_flg(ID flgid, FLGPTN setptn)
{
FLGCB *flgcb;
TCB *tcb;
WINFO_FLG *winfo;
ER ercd;
LOG_SET_FLG_ENTER(flgid, setptn);
CHECK_TSKCTX_UNL();
CHECK_FLGID(flgid);
flgcb = get_flgcb(flgid);
t_lock_cpu();
flgcb->flgptn |= setptn;
if (!(queue_empty(&(flgcb->wait_queue)))) {
tcb = (TCB *)(flgcb->wait_queue.next);
winfo = (WINFO_FLG *)(tcb->winfo);
if (eventflag_cond(flgcb, winfo->waiptn,
winfo->wfmode, &(winfo->flgptn))) {
queue_delete(&(tcb->task_queue));
if (wait_complete(tcb)) {
// dispatch(); // Modified by Chen Yi @srf
}
}
}
ercd = E_OK;
t_unlock_cpu();
exit:
LOG_SET_FLG_LEAVE(ercd);
return(ercd);
}
#endif /* __set_flg */
/*
* 僀儀儞僩僼儔僌偺僙僢僩乮旕僞僗僋僐儞僥僉僗僩梡乯
*/
#ifdef __iset_flg
SYSCALL ER
iset_flg(ID flgid, FLGPTN setptn)
{
FLGCB *flgcb;
TCB *tcb;
WINFO_FLG *winfo;
ER ercd;
LOG_ISET_FLG_ENTER(flgid, setptn);
CHECK_INTCTX_UNL();
CHECK_FLGID(flgid);
flgcb = get_flgcb(flgid);
i_lock_cpu();
flgcb->flgptn |= setptn;
if (!(queue_empty(&(flgcb->wait_queue)))) {
tcb = (TCB *)(flgcb->wait_queue.next);
winfo = (WINFO_FLG *)(tcb->winfo);
if (eventflag_cond(flgcb, winfo->waiptn,
winfo->wfmode, &(winfo->flgptn))) {
queue_delete(&(tcb->task_queue));
if (wait_complete(tcb)) {
reqflg = TRUE;
}
}
}
ercd = E_OK;
i_unlock_cpu();
exit:
LOG_ISET_FLG_LEAVE(ercd);
return(ercd);
}
#endif /* __iset_flg */
/*
* 僀儀儞僩僼儔僌偺僋儕傾
*/
#ifdef __clr_flg
SYSCALL ER
clr_flg(ID flgid, FLGPTN clrptn)
{
FLGCB *flgcb;
ER ercd;
LOG_CLR_FLG_ENTER(flgid, clrptn);
CHECK_TSKCTX_UNL();
CHECK_FLGID(flgid);
flgcb = get_flgcb(flgid);
t_lock_cpu();
flgcb->flgptn &= clrptn;
ercd = E_OK;
t_unlock_cpu();
exit:
LOG_CLR_FLG_LEAVE(ercd);
return(ercd);
}
#endif /* __clr_flg */
/*
* 僀儀儞僩僼儔僌懸偪
*/
#ifdef __wai_flg
SYSCALL ER
wai_flg(ID flgid, FLGPTN waiptn, MODE wfmode, FLGPTN *p_flgptn)
{
FLGCB *flgcb;
WINFO_FLG winfo;
ER ercd;
LOG_WAI_FLG_ENTER(flgid, waiptn, wfmode, p_flgptn);
CHECK_DISPATCH();
CHECK_FLGID(flgid);
CHECK_PAR(waiptn != 0);
CHECK_PAR((wfmode & ~TWF_ORW) == 0);
flgcb = get_flgcb(flgid);
t_lock_cpu();
if (!(queue_empty(&(flgcb->wait_queue)))) {
ercd = E_ILUSE;
}
else if (eventflag_cond(flgcb, waiptn, wfmode, p_flgptn)) {
ercd = E_OK;
}
else {
winfo.waiptn = waiptn;
winfo.wfmode = wfmode;
wobj_make_wait((WOBJCB *) flgcb, (WINFO_WOBJ *) &winfo);
//dispatch();
t_unlock_cpu();
exit_to_dispatch(); // Modified by Chen Yi @srf
ercd = winfo.winfo.wercd;
if (ercd == E_OK) {
*p_flgptn = winfo.flgptn;
}
}
t_unlock_cpu();
exit:
LOG_WAI_FLG_LEAVE(ercd, *p_flgptn);
return(ercd);
}
#endif /* __wai_flg */
/*
* 僀儀儞僩僼儔僌懸偪乮億乕儕儞僌乯
*/
#ifdef __pol_flg
SYSCALL ER
pol_flg(ID flgid, FLGPTN waiptn, MODE wfmode, FLGPTN *p_flgptn)
{
FLGCB *flgcb;
ER ercd;
LOG_POL_FLG_ENTER(flgid, waiptn, wfmode, p_flgptn);
CHECK_TSKCTX_UNL();
CHECK_FLGID(flgid);
CHECK_PAR(waiptn != 0);
CHECK_PAR((wfmode & ~TWF_ORW) == 0);
flgcb = get_flgcb(flgid);
t_lock_cpu();
if (!(queue_empty(&(flgcb->wait_queue)))) {
ercd = E_ILUSE;
}
else if (eventflag_cond(flgcb, waiptn, wfmode, p_flgptn)) {
ercd = E_OK;
}
else {
ercd = E_TMOUT;
}
t_unlock_cpu();
exit:
LOG_POL_FLG_LEAVE(ercd, *p_flgptn);
return(ercd);
}
#endif /* __pol_flg */
/*
* 僀儀儞僩僼儔僌懸偪乮僞僀儉傾僂僩偁傝乯
*/
#ifdef __twai_flg
SYSCALL ER
twai_flg(ID flgid, FLGPTN waiptn, MODE wfmode, FLGPTN *p_flgptn, TMO tmout)
{
FLGCB *flgcb;
WINFO_FLG winfo;
TMEVTB tmevtb;
ER ercd;
LOG_TWAI_FLG_ENTER(flgid, waiptn, wfmode, p_flgptn, tmout);
CHECK_DISPATCH();
CHECK_FLGID(flgid);
CHECK_PAR(waiptn != 0);
CHECK_PAR((wfmode & ~TWF_ORW) == 0);
CHECK_TMOUT(tmout);
flgcb = get_flgcb(flgid);
t_lock_cpu();
if (!(queue_empty(&(flgcb->wait_queue)))) {
ercd = E_ILUSE;
}
else if (eventflag_cond(flgcb, waiptn, wfmode, p_flgptn)) {
ercd = E_OK;
}
else if (tmout == TMO_POL) {
ercd = E_TMOUT;
}
else {
winfo.waiptn = waiptn;
winfo.wfmode = wfmode;
wobj_make_wait_tmout((WOBJCB *) flgcb, (WINFO_WOBJ *) &winfo,
&tmevtb, tmout);
//dispatch();
t_unlock_cpu();
exit_to_dispatch(); // Modifeid by Chen Yi @srf
ercd = winfo.winfo.wercd;
if (ercd == E_OK) {
*p_flgptn = winfo.flgptn;
}
}
t_unlock_cpu();
exit:
LOG_TWAI_FLG_LEAVE(ercd, *p_flgptn);
return(ercd);
}
#endif /* __twai_flg */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -