537 lines
		
	
	
	
		
			13 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			537 lines
		
	
	
	
		
			13 KiB
			
		
	
	
	
		
			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 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) ; | ||
|  | } |