diff options
Diffstat (limited to 'drivers/net/skfp/rmt.c')
-rw-r--r-- | drivers/net/skfp/rmt.c | 654 |
1 files changed, 0 insertions, 654 deletions
diff --git a/drivers/net/skfp/rmt.c b/drivers/net/skfp/rmt.c deleted file mode 100644 index ef8d567..0000000 --- a/drivers/net/skfp/rmt.c +++ /dev/null @@ -1,654 +0,0 @@ -/****************************************************************************** - * - * (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 lint -static 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_3 -extern 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 ! - */ - if ((tx = sm_mac_get_tx_state(smc)) == 4 || tx == 5) { - DB_RMTN(2,"RMT : DETECT && TRT_EXPIRED && T4/T5\n",0,0); - smc->r.bn_flag = TRUE ; - /* - * If one of the upstream stations beaconed - * and the link to the upstream neighbor is - * lost we need to restart the stuck timer to - * check the "stuck beacon" condition. - */ - start_rmt_timer1(smc,smc->s.rmt_t_stuck, - RM_TIMEOUT_T_STUCK) ; - } - /* - * We do NOT need to clear smc->r.bn_flag in case of - * not being in state T4 or T5, because the flag - * must be cleared in order to get in this condition. - */ - - DB_RMTN(2, - "RMT : sm_mac_get_tx_state() = %d (bn_flag = %d)\n", - tx,smc->r.bn_flag) ; - } - /*RM34a*/ - else if (cmd == RM_MY_CLAIM && smc->r.timer0_exp) { - rmt_new_dup_actions(smc) ; - GO_STATE(RM4_NON_OP_DUP) ; - break ; - } - /*RM34b*/ - else if (cmd == RM_MY_BEACON && smc->r.timer0_exp) { - rmt_new_dup_actions(smc) ; - GO_STATE(RM4_NON_OP_DUP) ; - break ; - } - /*RM34c*/ - else if (cmd == RM_VALID_CLAIM) { - rmt_new_dup_actions(smc) ; - GO_STATE(RM4_NON_OP_DUP) ; - break ; - } - /*RM36*/ - else if (cmd == RM_TIMEOUT_T_STUCK && - smc->r.rm_join && smc->r.bn_flag) { - GO_STATE(RM6_DIRECTED) ; - break ; - } - break ; - case ACTIONS(RM4_NON_OP_DUP) : - start_rmt_timer0(smc,smc->s.rmt_t_announce,RM_TIMEOUT_ANNOUNCE); - 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 : RM4_NON_OP_DUP\n",0,0) ; - ACTIONS_DONE() ; - break ; - case RM4_NON_OP_DUP : - if (cmd == RM_TIMEOUT_POLL) { - start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); - sm_mac_check_beacon_claim(smc) ; - break ; - } - /*RM41*/ - if (!smc->r.da_flag) { - GO_STATE(RM1_NON_OP) ; - break ; - } - /*RM44a*/ - else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && - smc->r.bn_flag) { - smc->r.bn_flag = FALSE ; - } - /*RM44b*/ - 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 ! - */ - if ((tx = sm_mac_get_tx_state(smc)) == 4 || tx == 5) { - DB_RMTN(2,"RMT : NOPDUP && TRT_EXPIRED && T4/T5\n",0,0); - smc->r.bn_flag = TRUE ; - /* - * If one of the upstream stations beaconed - * and the link to the upstream neighbor is - * lost we need to restart the stuck timer to - * check the "stuck beacon" condition. - */ - start_rmt_timer1(smc,smc->s.rmt_t_stuck, - RM_TIMEOUT_T_STUCK) ; - } - /* - * We do NOT need to clear smc->r.bn_flag in case of - * not being in state T4 or T5, because the flag - * must be cleared in order to get in this condition. - */ - - DB_RMTN(2, - "RMT : sm_mac_get_tx_state() = %d (bn_flag = %d)\n", - tx,smc->r.bn_flag) ; - } - /*RM44c*/ - else if (cmd == RM_TIMEOUT_ANNOUNCE && !smc->r.bn_flag) { - rmt_dup_actions(smc) ; - } - /*RM45*/ - else if (cmd == RM_RING_OP) { - smc->r.no_flag = FALSE ; - GO_STATE(RM5_RING_OP_DUP) ; - break ; - } - /*RM46*/ - else if (cmd == RM_TIMEOUT_T_STUCK && - smc->r.rm_join && smc->r.bn_flag) { - GO_STATE(RM6_DIRECTED) ; - break ; - } - break ; - case ACTIONS(RM5_RING_OP_DUP) : - stop_rmt_timer0(smc) ; - stop_rmt_timer1(smc) ; - stop_rmt_timer2(smc) ; - DB_RMTN(1,"RMT : RM5_RING_OP_DUP\n",0,0) ; - ACTIONS_DONE() ; - break; - case RM5_RING_OP_DUP : - /*RM52*/ - if (smc->r.dup_addr_test == DA_PASSED) { - smc->r.da_flag = FALSE ; - GO_STATE(RM2_RING_OP) ; - break ; - } - /*RM54*/ - else if (cmd == RM_RING_NON_OP) { - smc->r.jm_flag = FALSE ; - smc->r.bn_flag = FALSE ; - GO_STATE(RM4_NON_OP_DUP) ; - break ; - } - break ; - case ACTIONS(RM6_DIRECTED) : - start_rmt_timer0(smc,smc->s.rmt_t_direct,RM_TIMEOUT_T_DIRECT) ; - stop_rmt_timer1(smc) ; - start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL) ; - sm_ma_control(smc,MA_DIRECTED) ; - RS_SET(smc,RS_BEACON) ; - DB_RMTN(1,"RMT : RM6_DIRECTED\n",0,0) ; - ACTIONS_DONE() ; - break ; - case RM6_DIRECTED : - /*RM63*/ - if (cmd == RM_TIMEOUT_POLL) { - start_rmt_timer2(smc,smc->s.rmt_t_poll,RM_TIMEOUT_POLL); - sm_mac_check_beacon_claim(smc) ; -#ifndef SUPERNET_3 - /* Because of problems with the Supernet II chip set - * sending of Directed Beacon will stop after 165ms - * therefore restart_trt_for_dbcn(smc) will be called - * to prevent this. - */ - restart_trt_for_dbcn(smc) ; -#endif /*SUPERNET_3*/ - break ; - } - if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && - !smc->r.da_flag) { - smc->r.bn_flag = FALSE ; - GO_STATE(RM3_DETECT) ; - break ; - } - /*RM64*/ - else if ((cmd == RM_MY_BEACON || cmd == RM_OTHER_BEACON) && - smc->r.da_flag) { - smc->r.bn_flag = FALSE ; - GO_STATE(RM4_NON_OP_DUP) ; - break ; - } - /*RM67*/ - else if (cmd == RM_TIMEOUT_T_DIRECT) { - GO_STATE(RM7_TRACE) ; - break ; - } - break ; - case ACTIONS(RM7_TRACE) : - stop_rmt_timer0(smc) ; - stop_rmt_timer1(smc) ; - stop_rmt_timer2(smc) ; - smc->e.trace_prop |= ENTITY_BIT(ENTITY_MAC) ; - queue_event(smc,EVENT_ECM,EC_TRACE_PROP) ; - DB_RMTN(1,"RMT : RM7_TRACE\n",0,0) ; - ACTIONS_DONE() ; - break ; - case RM7_TRACE : - break ; - default: - SMT_PANIC(smc,SMT_E0122, SMT_E0122_MSG) ; - break; - } -} - -/* - * (jd) RMT duplicate address actions - * leave the ring or reinsert just as configured - */ -static void rmt_dup_actions(struct s_smc *smc) -{ - if (smc->r.jm_flag) { - } - else { - if (smc->s.rmt_dup_mac_behavior) { - SMT_ERR_LOG(smc,SMT_E0138, SMT_E0138_MSG) ; - rmt_reinsert_actions(smc) ; - } - else { - SMT_ERR_LOG(smc,SMT_E0135, SMT_E0135_MSG) ; - rmt_leave_actions(smc) ; - } - } -} - -/* - * Reconnect to the Ring - */ -static void rmt_reinsert_actions(struct s_smc *smc) -{ - queue_event(smc,EVENT_ECM,EC_DISCONNECT) ; - queue_event(smc,EVENT_ECM,EC_CONNECT) ; -} - -/* - * duplicate address detected - */ -static void rmt_new_dup_actions(struct s_smc *smc) -{ - smc->r.da_flag = TRUE ; - smc->r.bn_flag = FALSE ; - smc->r.jm_flag = FALSE ; - /* - * we have three options : change address, jam or leave - * we leave the ring as default - * Optionally it's possible to reinsert after leaving the Ring - * but this will not conform with SMT Spec. - */ - if (smc->s.rmt_dup_mac_behavior) { - SMT_ERR_LOG(smc,SMT_E0138, SMT_E0138_MSG) ; - rmt_reinsert_actions(smc) ; - } - else { - SMT_ERR_LOG(smc,SMT_E0135, SMT_E0135_MSG) ; - rmt_leave_actions(smc) ; - } -} - - -/* - * leave the ring - */ -static void rmt_leave_actions(struct s_smc *smc) -{ - queue_event(smc,EVENT_ECM,EC_DISCONNECT) ; - /* - * Note: Do NOT try again later. (with please reconnect) - * The station must be left from the ring! - */ -} - -/* - * SMT timer interface - * start RMT timer 0 - */ -static void start_rmt_timer0(struct s_smc *smc, u_long value, int event) -{ - smc->r.timer0_exp = FALSE ; /* clear timer event flag */ - smt_timer_start(smc,&smc->r.rmt_timer0,value,EV_TOKEN(EVENT_RMT,event)); -} - -/* - * SMT timer interface - * start RMT timer 1 - */ -static void start_rmt_timer1(struct s_smc *smc, u_long value, int event) -{ - smc->r.timer1_exp = FALSE ; /* clear timer event flag */ - smt_timer_start(smc,&smc->r.rmt_timer1,value,EV_TOKEN(EVENT_RMT,event)); -} - -/* - * SMT timer interface - * start RMT timer 2 - */ -static void start_rmt_timer2(struct s_smc *smc, u_long value, int event) -{ - smc->r.timer2_exp = FALSE ; /* clear timer event flag */ - smt_timer_start(smc,&smc->r.rmt_timer2,value,EV_TOKEN(EVENT_RMT,event)); -} - -/* - * SMT timer interface - * stop RMT timer 0 - */ -static void stop_rmt_timer0(struct s_smc *smc) -{ - if (smc->r.rmt_timer0.tm_active) - smt_timer_stop(smc,&smc->r.rmt_timer0) ; -} - -/* - * SMT timer interface - * stop RMT timer 1 - */ -static void stop_rmt_timer1(struct s_smc *smc) -{ - if (smc->r.rmt_timer1.tm_active) - smt_timer_stop(smc,&smc->r.rmt_timer1) ; -} - -/* - * SMT timer interface - * stop RMT timer 2 - */ -static void stop_rmt_timer2(struct s_smc *smc) -{ - if (smc->r.rmt_timer2.tm_active) - smt_timer_stop(smc,&smc->r.rmt_timer2) ; -} - |