xref: /openbmc/linux/drivers/net/fddi/skfp/ecm.c (revision 8730046c)
1 /******************************************************************************
2  *
3  *	(C)Copyright 1998,1999 SysKonnect,
4  *	a business unit of Schneider & Koch & Co. Datensysteme GmbH.
5  *
6  *	See the file "skfddi.c" for further information.
7  *
8  *	This program is free software; you can redistribute it and/or modify
9  *	it under the terms of the GNU General Public License as published by
10  *	the Free Software Foundation; either version 2 of the License, or
11  *	(at your option) any later version.
12  *
13  *	The information in this file is provided "AS IS" without warranty.
14  *
15  ******************************************************************************/
16 
17 /*
18 	SMT ECM
19 	Entity Coordination Management
20 	Hardware independent state machine
21 */
22 
23 /*
24  * Hardware independent state machine implemantation
25  * The following external SMT functions are referenced :
26  *
27  * 		queue_event()
28  * 		smt_timer_start()
29  * 		smt_timer_stop()
30  *
31  * 	The following external HW dependent functions are referenced :
32  * 		sm_pm_bypass_req()
33  * 		sm_pm_ls_latch()
34  * 		sm_pm_get_ls()
35  *
36  * 	The following HW dependent events are required :
37  *		NONE
38  *
39  */
40 
41 #include "h/types.h"
42 #include "h/fddi.h"
43 #include "h/smc.h"
44 
45 #define KERNEL
46 #include "h/smtstate.h"
47 
48 #ifndef	lint
49 static const char ID_sccs[] = "@(#)ecm.c	2.7 99/08/05 (C) SK " ;
50 #endif
51 
52 /*
53  * FSM Macros
54  */
55 #define AFLAG	0x10
56 #define GO_STATE(x)	(smc->mib.fddiSMTECMState = (x)|AFLAG)
57 #define ACTIONS_DONE()	(smc->mib.fddiSMTECMState &= ~AFLAG)
58 #define ACTIONS(x)	(x|AFLAG)
59 
60 #define EC0_OUT		0			/* not inserted */
61 #define EC1_IN		1			/* inserted */
62 #define EC2_TRACE	2			/* tracing */
63 #define EC3_LEAVE	3			/* leaving the ring */
64 #define EC4_PATH_TEST	4			/* performing path test */
65 #define EC5_INSERT	5			/* bypass being turned on */
66 #define EC6_CHECK	6			/* checking bypass */
67 #define EC7_DEINSERT	7			/* bypass being turnde off */
68 
69 #ifdef	DEBUG
70 /*
71  * symbolic state names
72  */
73 static const char * const ecm_states[] = {
74 	"EC0_OUT","EC1_IN","EC2_TRACE","EC3_LEAVE","EC4_PATH_TEST",
75 	"EC5_INSERT","EC6_CHECK","EC7_DEINSERT"
76 } ;
77 
78 /*
79  * symbolic event names
80  */
81 static const char * const ecm_events[] = {
82 	"NONE","EC_CONNECT","EC_DISCONNECT","EC_TRACE_PROP","EC_PATH_TEST",
83 	"EC_TIMEOUT_TD","EC_TIMEOUT_TMAX",
84 	"EC_TIMEOUT_IMAX","EC_TIMEOUT_INMAX","EC_TEST_DONE"
85 } ;
86 #endif
87 
88 /*
89  * all Globals  are defined in smc.h
90  * struct s_ecm
91  */
92 
93 /*
94  * function declarations
95  */
96 
97 static void ecm_fsm(struct s_smc *smc, int cmd);
98 static void start_ecm_timer(struct s_smc *smc, u_long value, int event);
99 static void stop_ecm_timer(struct s_smc *smc);
100 static void prop_actions(struct s_smc *smc);
101 
102 /*
103 	init ECM state machine
104 	clear all ECM vars and flags
105 */
106 void ecm_init(struct s_smc *smc)
107 {
108 	smc->e.path_test = PT_PASSED ;
109 	smc->e.trace_prop = 0 ;
110 	smc->e.sb_flag = 0 ;
111 	smc->mib.fddiSMTECMState = ACTIONS(EC0_OUT) ;
112 	smc->e.ecm_line_state = FALSE ;
113 }
114 
115 /*
116 	ECM state machine
117 	called by dispatcher
118 
119 	do
120 		display state change
121 		process event
122 	until SM is stable
123 */
124 void ecm(struct s_smc *smc, int event)
125 {
126 	int	state ;
127 
128 	do {
129 		DB_ECM("ECM : state %s%s",
130 			(smc->mib.fddiSMTECMState & AFLAG) ? "ACTIONS " : "",
131 			ecm_states[smc->mib.fddiSMTECMState & ~AFLAG]) ;
132 		DB_ECM(" event %s\n",ecm_events[event],0) ;
133 		state = smc->mib.fddiSMTECMState ;
134 		ecm_fsm(smc,event) ;
135 		event = 0 ;
136 	} while (state != smc->mib.fddiSMTECMState) ;
137 	ecm_state_change(smc,(int)smc->mib.fddiSMTECMState) ;
138 }
139 
140 /*
141 	process ECM event
142 */
143 static void ecm_fsm(struct s_smc *smc, int cmd)
144 {
145 	int ls_a ;			/* current line state PHY A */
146 	int ls_b ;			/* current line state PHY B */
147 	int	p ;			/* ports */
148 
149 
150 	smc->mib.fddiSMTBypassPresent = sm_pm_bypass_present(smc) ;
151 	if (cmd == EC_CONNECT)
152 		smc->mib.fddiSMTRemoteDisconnectFlag = FALSE ;
153 
154 	/* For AIX event notification: */
155 	/* Is a disconnect  command remotely issued ? */
156 	if (cmd == EC_DISCONNECT &&
157 		smc->mib.fddiSMTRemoteDisconnectFlag == TRUE)
158 		AIX_EVENT (smc, (u_long) CIO_HARD_FAIL, (u_long)
159 			FDDI_REMOTE_DISCONNECT, smt_get_event_word(smc),
160 			smt_get_error_word(smc) );
161 
162 	/*jd 05-Aug-1999 Bug #10419 "Port Disconnect fails at Dup MAc Cond."*/
163 	if (cmd == EC_CONNECT) {
164 		smc->e.DisconnectFlag = FALSE ;
165 	}
166 	else if (cmd == EC_DISCONNECT) {
167 		smc->e.DisconnectFlag = TRUE ;
168 	}
169 
170 	switch(smc->mib.fddiSMTECMState) {
171 	case ACTIONS(EC0_OUT) :
172 		/*
173 		 * We do not perform a path test
174 		 */
175 		smc->e.path_test = PT_PASSED ;
176 		smc->e.ecm_line_state = FALSE ;
177 		stop_ecm_timer(smc) ;
178 		ACTIONS_DONE() ;
179 		break ;
180 	case EC0_OUT:
181 		/*EC01*/
182 		if (cmd == EC_CONNECT && !smc->mib.fddiSMTBypassPresent
183 			&& smc->e.path_test==PT_PASSED) {
184 			GO_STATE(EC1_IN) ;
185 			break ;
186 		}
187 		/*EC05*/
188 		else if (cmd == EC_CONNECT && (smc->e.path_test==PT_PASSED) &&
189 			smc->mib.fddiSMTBypassPresent &&
190 			(smc->s.sas == SMT_DAS)) {
191 			GO_STATE(EC5_INSERT) ;
192 			break ;
193 		}
194 		break;
195 	case ACTIONS(EC1_IN) :
196 		stop_ecm_timer(smc) ;
197 		smc->e.trace_prop = 0 ;
198 		sm_ma_control(smc,MA_TREQ) ;
199 		for (p = 0 ; p < NUMPHYS ; p++)
200 			if (smc->mib.p[p].fddiPORTHardwarePresent)
201 				queue_event(smc,EVENT_PCMA+p,PC_START) ;
202 		ACTIONS_DONE() ;
203 		break ;
204 	case EC1_IN:
205 		/*EC12*/
206 		if (cmd == EC_TRACE_PROP) {
207 			prop_actions(smc) ;
208 			GO_STATE(EC2_TRACE) ;
209 			break ;
210 		}
211 		/*EC13*/
212 		else if (cmd == EC_DISCONNECT) {
213 			GO_STATE(EC3_LEAVE) ;
214 			break ;
215 		}
216 		break;
217 	case ACTIONS(EC2_TRACE) :
218 		start_ecm_timer(smc,MIB2US(smc->mib.fddiSMTTrace_MaxExpiration),
219 			EC_TIMEOUT_TMAX) ;
220 		ACTIONS_DONE() ;
221 		break ;
222 	case EC2_TRACE :
223 		/*EC22*/
224 		if (cmd == EC_TRACE_PROP) {
225 			prop_actions(smc) ;
226 			GO_STATE(EC2_TRACE) ;
227 			break ;
228 		}
229 		/*EC23a*/
230 		else if (cmd == EC_DISCONNECT) {
231 			smc->e.path_test = PT_EXITING ;
232 			GO_STATE(EC3_LEAVE) ;
233 			break ;
234 		}
235 		/*EC23b*/
236 		else if (smc->e.path_test == PT_PENDING) {
237 			GO_STATE(EC3_LEAVE) ;
238 			break ;
239 		}
240 		/*EC23c*/
241 		else if (cmd == EC_TIMEOUT_TMAX) {
242 			/* Trace_Max is expired */
243 			/* -> send AIX_EVENT */
244 			AIX_EVENT(smc, (u_long) FDDI_RING_STATUS,
245 				(u_long) FDDI_SMT_ERROR, (u_long)
246 				FDDI_TRACE_MAX, smt_get_error_word(smc));
247 			smc->e.path_test = PT_PENDING ;
248 			GO_STATE(EC3_LEAVE) ;
249 			break ;
250 		}
251 		break ;
252 	case ACTIONS(EC3_LEAVE) :
253 		start_ecm_timer(smc,smc->s.ecm_td_min,EC_TIMEOUT_TD) ;
254 		for (p = 0 ; p < NUMPHYS ; p++)
255 			queue_event(smc,EVENT_PCMA+p,PC_STOP) ;
256 		ACTIONS_DONE() ;
257 		break ;
258 	case EC3_LEAVE:
259 		/*EC30*/
260 		if (cmd == EC_TIMEOUT_TD && !smc->mib.fddiSMTBypassPresent &&
261 			(smc->e.path_test != PT_PENDING)) {
262 			GO_STATE(EC0_OUT) ;
263 			break ;
264 		}
265 		/*EC34*/
266 		else if (cmd == EC_TIMEOUT_TD &&
267 			(smc->e.path_test == PT_PENDING)) {
268 			GO_STATE(EC4_PATH_TEST) ;
269 			break ;
270 		}
271 		/*EC31*/
272 		else if (cmd == EC_CONNECT && smc->e.path_test == PT_PASSED) {
273 			GO_STATE(EC1_IN) ;
274 			break ;
275 		}
276 		/*EC33*/
277 		else if (cmd == EC_DISCONNECT &&
278 			smc->e.path_test == PT_PENDING) {
279 			smc->e.path_test = PT_EXITING ;
280 			/*
281 			 * stay in state - state will be left via timeout
282 			 */
283 		}
284 		/*EC37*/
285 		else if (cmd == EC_TIMEOUT_TD &&
286 			smc->mib.fddiSMTBypassPresent &&
287 			smc->e.path_test != PT_PENDING) {
288 			GO_STATE(EC7_DEINSERT) ;
289 			break ;
290 		}
291 		break ;
292 	case ACTIONS(EC4_PATH_TEST) :
293 		stop_ecm_timer(smc) ;
294 		smc->e.path_test = PT_TESTING ;
295 		start_ecm_timer(smc,smc->s.ecm_test_done,EC_TEST_DONE) ;
296 		/* now perform path test ... just a simulation */
297 		ACTIONS_DONE() ;
298 		break ;
299 	case EC4_PATH_TEST :
300 		/* path test done delay */
301 		if (cmd == EC_TEST_DONE)
302 			smc->e.path_test = PT_PASSED ;
303 
304 		if (smc->e.path_test == PT_FAILED)
305 			RS_SET(smc,RS_PATHTEST) ;
306 
307 		/*EC40a*/
308 		if (smc->e.path_test == PT_FAILED &&
309 			!smc->mib.fddiSMTBypassPresent) {
310 			GO_STATE(EC0_OUT) ;
311 			break ;
312 		}
313 		/*EC40b*/
314 		else if (cmd == EC_DISCONNECT &&
315 			!smc->mib.fddiSMTBypassPresent) {
316 			GO_STATE(EC0_OUT) ;
317 			break ;
318 		}
319 		/*EC41*/
320 		else if (smc->e.path_test == PT_PASSED) {
321 			GO_STATE(EC1_IN) ;
322 			break ;
323 		}
324 		/*EC47a*/
325 		else if (smc->e.path_test == PT_FAILED &&
326 			smc->mib.fddiSMTBypassPresent) {
327 			GO_STATE(EC7_DEINSERT) ;
328 			break ;
329 		}
330 		/*EC47b*/
331 		else if (cmd == EC_DISCONNECT &&
332 			smc->mib.fddiSMTBypassPresent) {
333 			GO_STATE(EC7_DEINSERT) ;
334 			break ;
335 		}
336 		break ;
337 	case ACTIONS(EC5_INSERT) :
338 		sm_pm_bypass_req(smc,BP_INSERT);
339 		start_ecm_timer(smc,smc->s.ecm_in_max,EC_TIMEOUT_INMAX) ;
340 		ACTIONS_DONE() ;
341 		break ;
342 	case EC5_INSERT :
343 		/*EC56*/
344 		if (cmd == EC_TIMEOUT_INMAX) {
345 			GO_STATE(EC6_CHECK) ;
346 			break ;
347 		}
348 		/*EC57*/
349 		else if (cmd == EC_DISCONNECT) {
350 			GO_STATE(EC7_DEINSERT) ;
351 			break ;
352 		}
353 		break ;
354 	case ACTIONS(EC6_CHECK) :
355 		/*
356 		 * in EC6_CHECK, we *POLL* the line state !
357 		 * check whether both bypass switches have switched.
358 		 */
359 		start_ecm_timer(smc,smc->s.ecm_check_poll,0) ;
360 		smc->e.ecm_line_state = TRUE ;	/* flag to pcm: report Q/HLS */
361 		(void) sm_pm_ls_latch(smc,PA,1) ; /* enable line state latch */
362 		(void) sm_pm_ls_latch(smc,PB,1) ; /* enable line state latch */
363 		ACTIONS_DONE() ;
364 		break ;
365 	case EC6_CHECK :
366 		ls_a = sm_pm_get_ls(smc,PA) ;
367 		ls_b = sm_pm_get_ls(smc,PB) ;
368 
369 		/*EC61*/
370 		if (((ls_a == PC_QLS) || (ls_a == PC_HLS)) &&
371 		    ((ls_b == PC_QLS) || (ls_b == PC_HLS)) ) {
372 			smc->e.sb_flag = FALSE ;
373 			smc->e.ecm_line_state = FALSE ;
374 			GO_STATE(EC1_IN) ;
375 			break ;
376 		}
377 		/*EC66*/
378 		else if (!smc->e.sb_flag &&
379 			 (((ls_a == PC_ILS) && (ls_b == PC_QLS)) ||
380 			  ((ls_a == PC_QLS) && (ls_b == PC_ILS)))){
381 			smc->e.sb_flag = TRUE ;
382 			DB_ECMN(1,"ECM : EC6_CHECK - stuck bypass\n",0,0) ;
383 			AIX_EVENT(smc, (u_long) FDDI_RING_STATUS, (u_long)
384 				FDDI_SMT_ERROR, (u_long) FDDI_BYPASS_STUCK,
385 				smt_get_error_word(smc));
386 		}
387 		/*EC67*/
388 		else if (cmd == EC_DISCONNECT) {
389 			smc->e.ecm_line_state = FALSE ;
390 			GO_STATE(EC7_DEINSERT) ;
391 			break ;
392 		}
393 		else {
394 			/*
395 			 * restart poll
396 			 */
397 			start_ecm_timer(smc,smc->s.ecm_check_poll,0) ;
398 		}
399 		break ;
400 	case ACTIONS(EC7_DEINSERT) :
401 		sm_pm_bypass_req(smc,BP_DEINSERT);
402 		start_ecm_timer(smc,smc->s.ecm_i_max,EC_TIMEOUT_IMAX) ;
403 		ACTIONS_DONE() ;
404 		break ;
405 	case EC7_DEINSERT:
406 		/*EC70*/
407 		if (cmd == EC_TIMEOUT_IMAX) {
408 			GO_STATE(EC0_OUT) ;
409 			break ;
410 		}
411 		/*EC75*/
412 		else if (cmd == EC_CONNECT && smc->e.path_test == PT_PASSED) {
413 			GO_STATE(EC5_INSERT) ;
414 			break ;
415 		}
416 		break;
417 	default:
418 		SMT_PANIC(smc,SMT_E0107, SMT_E0107_MSG) ;
419 		break;
420 	}
421 }
422 
423 #ifndef	CONCENTRATOR
424 /*
425  * trace propagation actions for SAS & DAS
426  */
427 static void prop_actions(struct s_smc *smc)
428 {
429 	int	port_in = 0 ;
430 	int	port_out = 0 ;
431 
432 	RS_SET(smc,RS_EVENT) ;
433 	switch (smc->s.sas) {
434 	case SMT_SAS :
435 		port_in = port_out = pcm_get_s_port(smc) ;
436 		break ;
437 	case SMT_DAS :
438 		port_in = cfm_get_mac_input(smc) ;	/* PA or PB */
439 		port_out = cfm_get_mac_output(smc) ;	/* PA or PB */
440 		break ;
441 	case SMT_NAC :
442 		SMT_PANIC(smc,SMT_E0108, SMT_E0108_MSG) ;
443 		return ;
444 	}
445 
446 	DB_ECM("ECM : prop_actions - trace_prop %d\n", smc->e.trace_prop,0) ;
447 	DB_ECM("ECM : prop_actions - in %d out %d\n", port_in,port_out) ;
448 
449 	if (smc->e.trace_prop & ENTITY_BIT(ENTITY_MAC)) {
450 		/* trace initiatior */
451 		DB_ECM("ECM : initiate TRACE on PHY %c\n",'A'+port_in-PA,0) ;
452 		queue_event(smc,EVENT_PCM+port_in,PC_TRACE) ;
453 	}
454 	else if ((smc->e.trace_prop & ENTITY_BIT(ENTITY_PHY(PA))) &&
455 		port_out != PA) {
456 		/* trace propagate upstream */
457 		DB_ECM("ECM : propagate TRACE on PHY B\n",0,0) ;
458 		queue_event(smc,EVENT_PCMB,PC_TRACE) ;
459 	}
460 	else if ((smc->e.trace_prop & ENTITY_BIT(ENTITY_PHY(PB))) &&
461 		port_out != PB) {
462 		/* trace propagate upstream */
463 		DB_ECM("ECM : propagate TRACE on PHY A\n",0,0) ;
464 		queue_event(smc,EVENT_PCMA,PC_TRACE) ;
465 	}
466 	else {
467 		/* signal trace termination */
468 		DB_ECM("ECM : TRACE terminated\n",0,0) ;
469 		smc->e.path_test = PT_PENDING ;
470 	}
471 	smc->e.trace_prop = 0 ;
472 }
473 #else
474 /*
475  * trace propagation actions for Concentrator
476  */
477 static void prop_actions(struct s_smc *smc)
478 {
479 	int	initiator ;
480 	int	upstream ;
481 	int	p ;
482 
483 	RS_SET(smc,RS_EVENT) ;
484 	while (smc->e.trace_prop) {
485 		DB_ECM("ECM : prop_actions - trace_prop %d\n",
486 			smc->e.trace_prop,0) ;
487 
488 		if (smc->e.trace_prop & ENTITY_BIT(ENTITY_MAC)) {
489 			initiator = ENTITY_MAC ;
490 			smc->e.trace_prop &= ~ENTITY_BIT(ENTITY_MAC) ;
491 			DB_ECM("ECM: MAC initiates trace\n",0,0) ;
492 		}
493 		else {
494 			for (p = NUMPHYS-1 ; p >= 0 ; p--) {
495 				if (smc->e.trace_prop &
496 					ENTITY_BIT(ENTITY_PHY(p)))
497 					break ;
498 			}
499 			initiator = ENTITY_PHY(p) ;
500 			smc->e.trace_prop &= ~ENTITY_BIT(ENTITY_PHY(p)) ;
501 		}
502 		upstream = cem_get_upstream(smc,initiator) ;
503 
504 		if (upstream == ENTITY_MAC) {
505 			/* signal trace termination */
506 			DB_ECM("ECM : TRACE terminated\n",0,0) ;
507 			smc->e.path_test = PT_PENDING ;
508 		}
509 		else {
510 			/* trace propagate upstream */
511 			DB_ECM("ECM : propagate TRACE on PHY %d\n",upstream,0) ;
512 			queue_event(smc,EVENT_PCM+upstream,PC_TRACE) ;
513 		}
514 	}
515 }
516 #endif
517 
518 
519 /*
520  * SMT timer interface
521  *	start ECM timer
522  */
523 static void start_ecm_timer(struct s_smc *smc, u_long value, int event)
524 {
525 	smt_timer_start(smc,&smc->e.ecm_timer,value,EV_TOKEN(EVENT_ECM,event));
526 }
527 
528 /*
529  * SMT timer interface
530  *	stop ECM timer
531  */
532 static void stop_ecm_timer(struct s_smc *smc)
533 {
534 	if (smc->e.ecm_timer.tm_active)
535 		smt_timer_stop(smc,&smc->e.ecm_timer) ;
536 }
537