| /****************************************************************************** | 
 |  * | 
 |  *	(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) ; | 
 | } |