📄 rmt.c
字号:
/****************************************************************************** * * (C)Copyright 1998,1999 SysKonnect, * a business unit of Schneider & Koch & Co. Datensysteme GmbH. * * See the file "skfddi.c" for further information. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * The information in this file is provided "AS IS" without warranty. * ******************************************************************************//* SMT RMT Ring Management*//* * Hardware independent state machine implemantation * The following external SMT functions are referenced : * * queue_event() * smt_timer_start() * smt_timer_stop() * * The following external HW dependent functions are referenced : * sm_ma_control() * sm_mac_check_beacon_claim() * * The following HW dependent events are required : * RM_RING_OP * RM_RING_NON_OP * RM_MY_BEACON * RM_OTHER_BEACON * RM_MY_CLAIM * RM_TRT_EXP * RM_VALID_CLAIM * */#include "h/types.h"#include "h/fddi.h"#include "h/smc.h"#define KERNEL#include "h/smtstate.h"#ifndef lintstatic const char ID_sccs[] = "@(#)rmt.c 2.13 99/07/02 (C) SK " ;#endif/* * FSM Macros */#define AFLAG 0x10#define GO_STATE(x) (smc->mib.m[MAC0].fddiMACRMTState = (x)|AFLAG)#define ACTIONS_DONE() (smc->mib.m[MAC0].fddiMACRMTState &= ~AFLAG)#define ACTIONS(x) (x|AFLAG)#define RM0_ISOLATED 0#define RM1_NON_OP 1 /* not operational */#define RM2_RING_OP 2 /* ring operational */#define RM3_DETECT 3 /* detect dupl addresses */#define RM4_NON_OP_DUP 4 /* dupl. addr detected */#define RM5_RING_OP_DUP 5 /* ring oper. with dupl. addr */#define RM6_DIRECTED 6 /* sending directed beacons */#define RM7_TRACE 7 /* trace initiated */#ifdef DEBUG/* * symbolic state names */static const char * const rmt_states[] = { "RM0_ISOLATED","RM1_NON_OP","RM2_RING_OP","RM3_DETECT", "RM4_NON_OP_DUP","RM5_RING_OP_DUP","RM6_DIRECTED", "RM7_TRACE"} ;/* * symbolic event names */static const char * const rmt_events[] = { "NONE","RM_RING_OP","RM_RING_NON_OP","RM_MY_BEACON", "RM_OTHER_BEACON","RM_MY_CLAIM","RM_TRT_EXP","RM_VALID_CLAIM", "RM_JOIN","RM_LOOP","RM_DUP_ADDR","RM_ENABLE_FLAG", "RM_TIMEOUT_NON_OP","RM_TIMEOUT_T_STUCK", "RM_TIMEOUT_ANNOUNCE","RM_TIMEOUT_T_DIRECT", "RM_TIMEOUT_D_MAX","RM_TIMEOUT_POLL","RM_TX_STATE_CHANGE"} ;#endif/* * Globals * in struct s_rmt *//* * function declarations */static void rmt_fsm(struct s_smc *smc, int cmd);static void start_rmt_timer0(struct s_smc *smc, u_long value, int event);static void start_rmt_timer1(struct s_smc *smc, u_long value, int event);static void start_rmt_timer2(struct s_smc *smc, u_long value, int event);static void stop_rmt_timer0(struct s_smc *smc);static void stop_rmt_timer1(struct s_smc *smc);static void stop_rmt_timer2(struct s_smc *smc);static void rmt_dup_actions(struct s_smc *smc);static void rmt_reinsert_actions(struct s_smc *smc);static void rmt_leave_actions(struct s_smc *smc);static void rmt_new_dup_actions(struct s_smc *smc);#ifndef SUPERNET_3extern void restart_trt_for_dbcn() ;#endif /*SUPERNET_3*//* init RMT state machine clear all RMT vars and flags*/void rmt_init(struct s_smc *smc){ smc->mib.m[MAC0].fddiMACRMTState = ACTIONS(RM0_ISOLATED) ; smc->r.dup_addr_test = DA_NONE ; smc->r.da_flag = 0 ; smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; smc->r.sm_ma_avail = FALSE ; smc->r.loop_avail = 0 ; smc->r.bn_flag = 0 ; smc->r.jm_flag = 0 ; smc->r.no_flag = TRUE ;}/* RMT state machine called by dispatcher do display state change process event until SM is stable*/void rmt(struct s_smc *smc, int event){ int state ; do { DB_RMT("RMT : state %s%s", (smc->mib.m[MAC0].fddiMACRMTState & AFLAG) ? "ACTIONS " : "", rmt_states[smc->mib.m[MAC0].fddiMACRMTState & ~AFLAG]) ; DB_RMT(" event %s\n",rmt_events[event],0) ; state = smc->mib.m[MAC0].fddiMACRMTState ; rmt_fsm(smc,event) ; event = 0 ; } while (state != smc->mib.m[MAC0].fddiMACRMTState) ; rmt_state_change(smc,(int)smc->mib.m[MAC0].fddiMACRMTState) ;}/* process RMT event*/static void rmt_fsm(struct s_smc *smc, int cmd){ /* * RM00-RM70 : from all states */ if (!smc->r.rm_join && !smc->r.rm_loop && smc->mib.m[MAC0].fddiMACRMTState != ACTIONS(RM0_ISOLATED) && smc->mib.m[MAC0].fddiMACRMTState != RM0_ISOLATED) { RS_SET(smc,RS_NORINGOP) ; rmt_indication(smc,0) ; GO_STATE(RM0_ISOLATED) ; return ; } switch(smc->mib.m[MAC0].fddiMACRMTState) { case ACTIONS(RM0_ISOLATED) : stop_rmt_timer0(smc) ; stop_rmt_timer1(smc) ; stop_rmt_timer2(smc) ; /* * Disable MAC. */ sm_ma_control(smc,MA_OFFLINE) ; smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; smc->r.loop_avail = FALSE ; smc->r.sm_ma_avail = FALSE ; smc->r.no_flag = TRUE ; DB_RMTN(1,"RMT : ISOLATED\n",0,0) ; ACTIONS_DONE() ; break ; case RM0_ISOLATED : /*RM01*/ if (smc->r.rm_join || smc->r.rm_loop) { /* * According to the standard the MAC must be reset * here. The FORMAC will be initialized and Claim * and Beacon Frames will be uploaded to the MAC. * So any change of Treq will take effect NOW. */ sm_ma_control(smc,MA_RESET) ; GO_STATE(RM1_NON_OP) ; break ; } break ; case ACTIONS(RM1_NON_OP) : start_rmt_timer0(smc,smc->s.rmt_t_non_op,RM_TIMEOUT_NON_OP) ; stop_rmt_timer1(smc) ; stop_rmt_timer2(smc) ; sm_ma_control(smc,MA_BEACON) ; DB_RMTN(1,"RMT : RING DOWN\n",0,0) ; RS_SET(smc,RS_NORINGOP) ; smc->r.sm_ma_avail = FALSE ; rmt_indication(smc,0) ; ACTIONS_DONE() ; break ; case RM1_NON_OP : /*RM12*/ if (cmd == RM_RING_OP) { RS_SET(smc,RS_RINGOPCHANGE) ; GO_STATE(RM2_RING_OP) ; break ; } /*RM13*/ else if (cmd == RM_TIMEOUT_NON_OP) { smc->r.bn_flag = FALSE ; smc->r.no_flag = TRUE ; GO_STATE(RM3_DETECT) ; break ; } break ; case ACTIONS(RM2_RING_OP) : stop_rmt_timer0(smc) ; stop_rmt_timer1(smc) ; stop_rmt_timer2(smc) ; smc->r.no_flag = FALSE ; if (smc->r.rm_loop) smc->r.loop_avail = TRUE ; if (smc->r.rm_join) { smc->r.sm_ma_avail = TRUE ; if (smc->mib.m[MAC0].fddiMACMA_UnitdataEnable) smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = TRUE ; else smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; } DB_RMTN(1,"RMT : RING UP\n",0,0) ; RS_CLEAR(smc,RS_NORINGOP) ; RS_SET(smc,RS_RINGOPCHANGE) ; rmt_indication(smc,1) ; smt_stat_counter(smc,0) ; ACTIONS_DONE() ; break ; case RM2_RING_OP : /*RM21*/ if (cmd == RM_RING_NON_OP) { smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; smc->r.loop_avail = FALSE ; RS_SET(smc,RS_RINGOPCHANGE) ; GO_STATE(RM1_NON_OP) ; break ; } /*RM22a*/ else if (cmd == RM_ENABLE_FLAG) { if (smc->mib.m[MAC0].fddiMACMA_UnitdataEnable) smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = TRUE ; else smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; } /*RM25*/ else if (smc->r.dup_addr_test == DA_FAILED) { smc->mib.m[MAC0].fddiMACMA_UnitdataAvailable = FALSE ; smc->r.loop_avail = FALSE ; smc->r.da_flag = TRUE ; GO_STATE(RM5_RING_OP_DUP) ; break ; } break ; case ACTIONS(RM3_DETECT) : start_rmt_timer0(smc,smc->s.mac_d_max*2,RM_TIMEOUT_D_MAX) ; start_rmt_timer1(smc,smc->s.rmt_t_stuck,RM_TIMEOUT_T_STUCK) ; start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL) ; sm_mac_check_beacon_claim(smc) ; DB_RMTN(1,"RMT : RM3_DETECT\n",0,0) ; ACTIONS_DONE() ; break ; case RM3_DETECT : if (cmd == RM_TIMEOUT_POLL) { start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); sm_mac_check_beacon_claim(smc) ; break ; } if (cmd == RM_TIMEOUT_D_MAX) { smc->r.timer0_exp = TRUE ; } /* *jd(22-Feb-1999) * We need a time ">= 2*mac_d_max" since we had finished * Claim or Beacon state. So we will restart timer0 at * every state change. */ if (cmd == RM_TX_STATE_CHANGE) { start_rmt_timer0(smc, smc->s.mac_d_max*2, RM_TIMEOUT_D_MAX) ; } /*RM32*/ if (cmd == RM_RING_OP) { GO_STATE(RM2_RING_OP) ; break ; } /*RM33a*/ else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && smc->r.bn_flag) { smc->r.bn_flag = FALSE ; } /*RM33b*/ else if (cmd == RM_TRT_EXP && !smc->r.bn_flag) { int tx ; /* * set bn_flag only if in state T4 or T5: * only if we're the beaconer should we start the * trace ! */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -