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