|  | /****************************************************************************** | 
|  | * | 
|  | *	(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 ECM | 
|  | Entity Coordination Management | 
|  | Hardware independent state machine | 
|  | */ | 
|  |  | 
|  | /* | 
|  | * 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_pm_bypass_req() | 
|  | * 		sm_pm_ls_latch() | 
|  | * 		sm_pm_get_ls() | 
|  | * | 
|  | * 	The following HW dependent events are required : | 
|  | *		NONE | 
|  | * | 
|  | */ | 
|  |  | 
|  | #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[] = "@(#)ecm.c	2.7 99/08/05 (C) SK " ; | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | * FSM Macros | 
|  | */ | 
|  | #define AFLAG	0x10 | 
|  | #define GO_STATE(x)	(smc->mib.fddiSMTECMState = (x)|AFLAG) | 
|  | #define ACTIONS_DONE()	(smc->mib.fddiSMTECMState &= ~AFLAG) | 
|  | #define ACTIONS(x)	(x|AFLAG) | 
|  |  | 
|  | #define EC0_OUT		0			/* not inserted */ | 
|  | #define EC1_IN		1			/* inserted */ | 
|  | #define EC2_TRACE	2			/* tracing */ | 
|  | #define EC3_LEAVE	3			/* leaving the ring */ | 
|  | #define EC4_PATH_TEST	4			/* performing path test */ | 
|  | #define EC5_INSERT	5			/* bypass being turned on */ | 
|  | #define EC6_CHECK	6			/* checking bypass */ | 
|  | #define EC7_DEINSERT	7			/* bypass being turnde off */ | 
|  |  | 
|  | #ifdef	DEBUG | 
|  | /* | 
|  | * symbolic state names | 
|  | */ | 
|  | static const char * const ecm_states[] = { | 
|  | "EC0_OUT","EC1_IN","EC2_TRACE","EC3_LEAVE","EC4_PATH_TEST", | 
|  | "EC5_INSERT","EC6_CHECK","EC7_DEINSERT" | 
|  | } ; | 
|  |  | 
|  | /* | 
|  | * symbolic event names | 
|  | */ | 
|  | static const char * const ecm_events[] = { | 
|  | "NONE","EC_CONNECT","EC_DISCONNECT","EC_TRACE_PROP","EC_PATH_TEST", | 
|  | "EC_TIMEOUT_TD","EC_TIMEOUT_TMAX", | 
|  | "EC_TIMEOUT_IMAX","EC_TIMEOUT_INMAX","EC_TEST_DONE" | 
|  | } ; | 
|  | #endif | 
|  |  | 
|  | /* | 
|  | * all Globals  are defined in smc.h | 
|  | * struct s_ecm | 
|  | */ | 
|  |  | 
|  | /* | 
|  | * function declarations | 
|  | */ | 
|  |  | 
|  | static void ecm_fsm(struct s_smc *smc, int cmd); | 
|  | static void start_ecm_timer(struct s_smc *smc, u_long value, int event); | 
|  | static void stop_ecm_timer(struct s_smc *smc); | 
|  | static void prop_actions(struct s_smc *smc); | 
|  |  | 
|  | /* | 
|  | init ECM state machine | 
|  | clear all ECM vars and flags | 
|  | */ | 
|  | void ecm_init(struct s_smc *smc) | 
|  | { | 
|  | smc->e.path_test = PT_PASSED ; | 
|  | smc->e.trace_prop = 0 ; | 
|  | smc->e.sb_flag = 0 ; | 
|  | smc->mib.fddiSMTECMState = ACTIONS(EC0_OUT) ; | 
|  | smc->e.ecm_line_state = FALSE ; | 
|  | } | 
|  |  | 
|  | /* | 
|  | ECM state machine | 
|  | called by dispatcher | 
|  |  | 
|  | do | 
|  | display state change | 
|  | process event | 
|  | until SM is stable | 
|  | */ | 
|  | void ecm(struct s_smc *smc, int event) | 
|  | { | 
|  | int	state ; | 
|  |  | 
|  | do { | 
|  | DB_ECM("ECM : state %s%s", | 
|  | (smc->mib.fddiSMTECMState & AFLAG) ? "ACTIONS " : "", | 
|  | ecm_states[smc->mib.fddiSMTECMState & ~AFLAG]) ; | 
|  | DB_ECM(" event %s\n",ecm_events[event],0) ; | 
|  | state = smc->mib.fddiSMTECMState ; | 
|  | ecm_fsm(smc,event) ; | 
|  | event = 0 ; | 
|  | } while (state != smc->mib.fddiSMTECMState) ; | 
|  | ecm_state_change(smc,(int)smc->mib.fddiSMTECMState) ; | 
|  | } | 
|  |  | 
|  | /* | 
|  | process ECM event | 
|  | */ | 
|  | static void ecm_fsm(struct s_smc *smc, int cmd) | 
|  | { | 
|  | int ls_a ;			/* current line state PHY A */ | 
|  | int ls_b ;			/* current line state PHY B */ | 
|  | int	p ;			/* ports */ | 
|  |  | 
|  |  | 
|  | smc->mib.fddiSMTBypassPresent = sm_pm_bypass_present(smc) ; | 
|  | if (cmd == EC_CONNECT) | 
|  | smc->mib.fddiSMTRemoteDisconnectFlag = FALSE ; | 
|  |  | 
|  | /* For AIX event notification: */ | 
|  | /* Is a disconnect  command remotely issued ? */ | 
|  | if (cmd == EC_DISCONNECT && | 
|  | smc->mib.fddiSMTRemoteDisconnectFlag == TRUE) | 
|  | AIX_EVENT (smc, (u_long) CIO_HARD_FAIL, (u_long) | 
|  | FDDI_REMOTE_DISCONNECT, smt_get_event_word(smc), | 
|  | smt_get_error_word(smc) ); | 
|  |  | 
|  | /*jd 05-Aug-1999 Bug #10419 "Port Disconnect fails at Dup MAc Cond."*/ | 
|  | if (cmd == EC_CONNECT) { | 
|  | smc->e.DisconnectFlag = FALSE ; | 
|  | } | 
|  | else if (cmd == EC_DISCONNECT) { | 
|  | smc->e.DisconnectFlag = TRUE ; | 
|  | } | 
|  |  | 
|  | switch(smc->mib.fddiSMTECMState) { | 
|  | case ACTIONS(EC0_OUT) : | 
|  | /* | 
|  | * We do not perform a path test | 
|  | */ | 
|  | smc->e.path_test = PT_PASSED ; | 
|  | smc->e.ecm_line_state = FALSE ; | 
|  | stop_ecm_timer(smc) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC0_OUT: | 
|  | /*EC01*/ | 
|  | if (cmd == EC_CONNECT && !smc->mib.fddiSMTBypassPresent | 
|  | && smc->e.path_test==PT_PASSED) { | 
|  | GO_STATE(EC1_IN) ; | 
|  | break ; | 
|  | } | 
|  | /*EC05*/ | 
|  | else if (cmd == EC_CONNECT && (smc->e.path_test==PT_PASSED) && | 
|  | smc->mib.fddiSMTBypassPresent && | 
|  | (smc->s.sas == SMT_DAS)) { | 
|  | GO_STATE(EC5_INSERT) ; | 
|  | break ; | 
|  | } | 
|  | break; | 
|  | case ACTIONS(EC1_IN) : | 
|  | stop_ecm_timer(smc) ; | 
|  | smc->e.trace_prop = 0 ; | 
|  | sm_ma_control(smc,MA_TREQ) ; | 
|  | for (p = 0 ; p < NUMPHYS ; p++) | 
|  | if (smc->mib.p[p].fddiPORTHardwarePresent) | 
|  | queue_event(smc,EVENT_PCMA+p,PC_START) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC1_IN: | 
|  | /*EC12*/ | 
|  | if (cmd == EC_TRACE_PROP) { | 
|  | prop_actions(smc) ; | 
|  | GO_STATE(EC2_TRACE) ; | 
|  | break ; | 
|  | } | 
|  | /*EC13*/ | 
|  | else if (cmd == EC_DISCONNECT) { | 
|  | GO_STATE(EC3_LEAVE) ; | 
|  | break ; | 
|  | } | 
|  | break; | 
|  | case ACTIONS(EC2_TRACE) : | 
|  | start_ecm_timer(smc,MIB2US(smc->mib.fddiSMTTrace_MaxExpiration), | 
|  | EC_TIMEOUT_TMAX) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC2_TRACE : | 
|  | /*EC22*/ | 
|  | if (cmd == EC_TRACE_PROP) { | 
|  | prop_actions(smc) ; | 
|  | GO_STATE(EC2_TRACE) ; | 
|  | break ; | 
|  | } | 
|  | /*EC23a*/ | 
|  | else if (cmd == EC_DISCONNECT) { | 
|  | smc->e.path_test = PT_EXITING ; | 
|  | GO_STATE(EC3_LEAVE) ; | 
|  | break ; | 
|  | } | 
|  | /*EC23b*/ | 
|  | else if (smc->e.path_test == PT_PENDING) { | 
|  | GO_STATE(EC3_LEAVE) ; | 
|  | break ; | 
|  | } | 
|  | /*EC23c*/ | 
|  | else if (cmd == EC_TIMEOUT_TMAX) { | 
|  | /* Trace_Max is expired */ | 
|  | /* -> send AIX_EVENT */ | 
|  | AIX_EVENT(smc, (u_long) FDDI_RING_STATUS, | 
|  | (u_long) FDDI_SMT_ERROR, (u_long) | 
|  | FDDI_TRACE_MAX, smt_get_error_word(smc)); | 
|  | smc->e.path_test = PT_PENDING ; | 
|  | GO_STATE(EC3_LEAVE) ; | 
|  | break ; | 
|  | } | 
|  | break ; | 
|  | case ACTIONS(EC3_LEAVE) : | 
|  | start_ecm_timer(smc,smc->s.ecm_td_min,EC_TIMEOUT_TD) ; | 
|  | for (p = 0 ; p < NUMPHYS ; p++) | 
|  | queue_event(smc,EVENT_PCMA+p,PC_STOP) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC3_LEAVE: | 
|  | /*EC30*/ | 
|  | if (cmd == EC_TIMEOUT_TD && !smc->mib.fddiSMTBypassPresent && | 
|  | (smc->e.path_test != PT_PENDING)) { | 
|  | GO_STATE(EC0_OUT) ; | 
|  | break ; | 
|  | } | 
|  | /*EC34*/ | 
|  | else if (cmd == EC_TIMEOUT_TD && | 
|  | (smc->e.path_test == PT_PENDING)) { | 
|  | GO_STATE(EC4_PATH_TEST) ; | 
|  | break ; | 
|  | } | 
|  | /*EC31*/ | 
|  | else if (cmd == EC_CONNECT && smc->e.path_test == PT_PASSED) { | 
|  | GO_STATE(EC1_IN) ; | 
|  | break ; | 
|  | } | 
|  | /*EC33*/ | 
|  | else if (cmd == EC_DISCONNECT && | 
|  | smc->e.path_test == PT_PENDING) { | 
|  | smc->e.path_test = PT_EXITING ; | 
|  | /* | 
|  | * stay in state - state will be left via timeout | 
|  | */ | 
|  | } | 
|  | /*EC37*/ | 
|  | else if (cmd == EC_TIMEOUT_TD && | 
|  | smc->mib.fddiSMTBypassPresent && | 
|  | smc->e.path_test != PT_PENDING) { | 
|  | GO_STATE(EC7_DEINSERT) ; | 
|  | break ; | 
|  | } | 
|  | break ; | 
|  | case ACTIONS(EC4_PATH_TEST) : | 
|  | stop_ecm_timer(smc) ; | 
|  | smc->e.path_test = PT_TESTING ; | 
|  | start_ecm_timer(smc,smc->s.ecm_test_done,EC_TEST_DONE) ; | 
|  | /* now perform path test ... just a simulation */ | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC4_PATH_TEST : | 
|  | /* path test done delay */ | 
|  | if (cmd == EC_TEST_DONE) | 
|  | smc->e.path_test = PT_PASSED ; | 
|  |  | 
|  | if (smc->e.path_test == PT_FAILED) | 
|  | RS_SET(smc,RS_PATHTEST) ; | 
|  |  | 
|  | /*EC40a*/ | 
|  | if (smc->e.path_test == PT_FAILED && | 
|  | !smc->mib.fddiSMTBypassPresent) { | 
|  | GO_STATE(EC0_OUT) ; | 
|  | break ; | 
|  | } | 
|  | /*EC40b*/ | 
|  | else if (cmd == EC_DISCONNECT && | 
|  | !smc->mib.fddiSMTBypassPresent) { | 
|  | GO_STATE(EC0_OUT) ; | 
|  | break ; | 
|  | } | 
|  | /*EC41*/ | 
|  | else if (smc->e.path_test == PT_PASSED) { | 
|  | GO_STATE(EC1_IN) ; | 
|  | break ; | 
|  | } | 
|  | /*EC47a*/ | 
|  | else if (smc->e.path_test == PT_FAILED && | 
|  | smc->mib.fddiSMTBypassPresent) { | 
|  | GO_STATE(EC7_DEINSERT) ; | 
|  | break ; | 
|  | } | 
|  | /*EC47b*/ | 
|  | else if (cmd == EC_DISCONNECT && | 
|  | smc->mib.fddiSMTBypassPresent) { | 
|  | GO_STATE(EC7_DEINSERT) ; | 
|  | break ; | 
|  | } | 
|  | break ; | 
|  | case ACTIONS(EC5_INSERT) : | 
|  | sm_pm_bypass_req(smc,BP_INSERT); | 
|  | start_ecm_timer(smc,smc->s.ecm_in_max,EC_TIMEOUT_INMAX) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC5_INSERT : | 
|  | /*EC56*/ | 
|  | if (cmd == EC_TIMEOUT_INMAX) { | 
|  | GO_STATE(EC6_CHECK) ; | 
|  | break ; | 
|  | } | 
|  | /*EC57*/ | 
|  | else if (cmd == EC_DISCONNECT) { | 
|  | GO_STATE(EC7_DEINSERT) ; | 
|  | break ; | 
|  | } | 
|  | break ; | 
|  | case ACTIONS(EC6_CHECK) : | 
|  | /* | 
|  | * in EC6_CHECK, we *POLL* the line state ! | 
|  | * check whether both bypass switches have switched. | 
|  | */ | 
|  | start_ecm_timer(smc,smc->s.ecm_check_poll,0) ; | 
|  | smc->e.ecm_line_state = TRUE ;	/* flag to pcm: report Q/HLS */ | 
|  | (void) sm_pm_ls_latch(smc,PA,1) ; /* enable line state latch */ | 
|  | (void) sm_pm_ls_latch(smc,PB,1) ; /* enable line state latch */ | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC6_CHECK : | 
|  | ls_a = sm_pm_get_ls(smc,PA) ; | 
|  | ls_b = sm_pm_get_ls(smc,PB) ; | 
|  |  | 
|  | /*EC61*/ | 
|  | if (((ls_a == PC_QLS) || (ls_a == PC_HLS)) && | 
|  | ((ls_b == PC_QLS) || (ls_b == PC_HLS)) ) { | 
|  | smc->e.sb_flag = FALSE ; | 
|  | smc->e.ecm_line_state = FALSE ; | 
|  | GO_STATE(EC1_IN) ; | 
|  | break ; | 
|  | } | 
|  | /*EC66*/ | 
|  | else if (!smc->e.sb_flag && | 
|  | (((ls_a == PC_ILS) && (ls_b == PC_QLS)) || | 
|  | ((ls_a == PC_QLS) && (ls_b == PC_ILS)))){ | 
|  | smc->e.sb_flag = TRUE ; | 
|  | DB_ECMN(1,"ECM : EC6_CHECK - stuck bypass\n",0,0) ; | 
|  | AIX_EVENT(smc, (u_long) FDDI_RING_STATUS, (u_long) | 
|  | FDDI_SMT_ERROR, (u_long) FDDI_BYPASS_STUCK, | 
|  | smt_get_error_word(smc)); | 
|  | } | 
|  | /*EC67*/ | 
|  | else if (cmd == EC_DISCONNECT) { | 
|  | smc->e.ecm_line_state = FALSE ; | 
|  | GO_STATE(EC7_DEINSERT) ; | 
|  | break ; | 
|  | } | 
|  | else { | 
|  | /* | 
|  | * restart poll | 
|  | */ | 
|  | start_ecm_timer(smc,smc->s.ecm_check_poll,0) ; | 
|  | } | 
|  | break ; | 
|  | case ACTIONS(EC7_DEINSERT) : | 
|  | sm_pm_bypass_req(smc,BP_DEINSERT); | 
|  | start_ecm_timer(smc,smc->s.ecm_i_max,EC_TIMEOUT_IMAX) ; | 
|  | ACTIONS_DONE() ; | 
|  | break ; | 
|  | case EC7_DEINSERT: | 
|  | /*EC70*/ | 
|  | if (cmd == EC_TIMEOUT_IMAX) { | 
|  | GO_STATE(EC0_OUT) ; | 
|  | break ; | 
|  | } | 
|  | /*EC75*/ | 
|  | else if (cmd == EC_CONNECT && smc->e.path_test == PT_PASSED) { | 
|  | GO_STATE(EC5_INSERT) ; | 
|  | break ; | 
|  | } | 
|  | break; | 
|  | default: | 
|  | SMT_PANIC(smc,SMT_E0107, SMT_E0107_MSG) ; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | #ifndef	CONCENTRATOR | 
|  | /* | 
|  | * trace propagation actions for SAS & DAS | 
|  | */ | 
|  | static void prop_actions(struct s_smc *smc) | 
|  | { | 
|  | int	port_in = 0 ; | 
|  | int	port_out = 0 ; | 
|  |  | 
|  | RS_SET(smc,RS_EVENT) ; | 
|  | switch (smc->s.sas) { | 
|  | case SMT_SAS : | 
|  | port_in = port_out = pcm_get_s_port(smc) ; | 
|  | break ; | 
|  | case SMT_DAS : | 
|  | port_in = cfm_get_mac_input(smc) ;	/* PA or PB */ | 
|  | port_out = cfm_get_mac_output(smc) ;	/* PA or PB */ | 
|  | break ; | 
|  | case SMT_NAC : | 
|  | SMT_PANIC(smc,SMT_E0108, SMT_E0108_MSG) ; | 
|  | return ; | 
|  | } | 
|  |  | 
|  | DB_ECM("ECM : prop_actions - trace_prop %d\n", smc->e.trace_prop,0) ; | 
|  | DB_ECM("ECM : prop_actions - in %d out %d\n", port_in,port_out) ; | 
|  |  | 
|  | if (smc->e.trace_prop & ENTITY_BIT(ENTITY_MAC)) { | 
|  | /* trace initiatior */ | 
|  | DB_ECM("ECM : initiate TRACE on PHY %c\n",'A'+port_in-PA,0) ; | 
|  | queue_event(smc,EVENT_PCM+port_in,PC_TRACE) ; | 
|  | } | 
|  | else if ((smc->e.trace_prop & ENTITY_BIT(ENTITY_PHY(PA))) && | 
|  | port_out != PA) { | 
|  | /* trace propagate upstream */ | 
|  | DB_ECM("ECM : propagate TRACE on PHY B\n",0,0) ; | 
|  | queue_event(smc,EVENT_PCMB,PC_TRACE) ; | 
|  | } | 
|  | else if ((smc->e.trace_prop & ENTITY_BIT(ENTITY_PHY(PB))) && | 
|  | port_out != PB) { | 
|  | /* trace propagate upstream */ | 
|  | DB_ECM("ECM : propagate TRACE on PHY A\n",0,0) ; | 
|  | queue_event(smc,EVENT_PCMA,PC_TRACE) ; | 
|  | } | 
|  | else { | 
|  | /* signal trace termination */ | 
|  | DB_ECM("ECM : TRACE terminated\n",0,0) ; | 
|  | smc->e.path_test = PT_PENDING ; | 
|  | } | 
|  | smc->e.trace_prop = 0 ; | 
|  | } | 
|  | #else | 
|  | /* | 
|  | * trace propagation actions for Concentrator | 
|  | */ | 
|  | static void prop_actions(struct s_smc *smc) | 
|  | { | 
|  | int	initiator ; | 
|  | int	upstream ; | 
|  | int	p ; | 
|  |  | 
|  | RS_SET(smc,RS_EVENT) ; | 
|  | while (smc->e.trace_prop) { | 
|  | DB_ECM("ECM : prop_actions - trace_prop %d\n", | 
|  | smc->e.trace_prop,0) ; | 
|  |  | 
|  | if (smc->e.trace_prop & ENTITY_BIT(ENTITY_MAC)) { | 
|  | initiator = ENTITY_MAC ; | 
|  | smc->e.trace_prop &= ~ENTITY_BIT(ENTITY_MAC) ; | 
|  | DB_ECM("ECM: MAC initiates trace\n",0,0) ; | 
|  | } | 
|  | else { | 
|  | for (p = NUMPHYS-1 ; p >= 0 ; p--) { | 
|  | if (smc->e.trace_prop & | 
|  | ENTITY_BIT(ENTITY_PHY(p))) | 
|  | break ; | 
|  | } | 
|  | initiator = ENTITY_PHY(p) ; | 
|  | smc->e.trace_prop &= ~ENTITY_BIT(ENTITY_PHY(p)) ; | 
|  | } | 
|  | upstream = cem_get_upstream(smc,initiator) ; | 
|  |  | 
|  | if (upstream == ENTITY_MAC) { | 
|  | /* signal trace termination */ | 
|  | DB_ECM("ECM : TRACE terminated\n",0,0) ; | 
|  | smc->e.path_test = PT_PENDING ; | 
|  | } | 
|  | else { | 
|  | /* trace propagate upstream */ | 
|  | DB_ECM("ECM : propagate TRACE on PHY %d\n",upstream,0) ; | 
|  | queue_event(smc,EVENT_PCM+upstream,PC_TRACE) ; | 
|  | } | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  |  | 
|  | /* | 
|  | * SMT timer interface | 
|  | *	start ECM timer | 
|  | */ | 
|  | static void start_ecm_timer(struct s_smc *smc, u_long value, int event) | 
|  | { | 
|  | smt_timer_start(smc,&smc->e.ecm_timer,value,EV_TOKEN(EVENT_ECM,event)); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * SMT timer interface | 
|  | *	stop ECM timer | 
|  | */ | 
|  | static void stop_ecm_timer(struct s_smc *smc) | 
|  | { | 
|  | if (smc->e.ecm_timer.tm_active) | 
|  | smt_timer_stop(smc,&smc->e.ecm_timer) ; | 
|  | } |